-(function () {
-
- var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};
-
- function getDefaultExportFromCjs (x) {
- return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
- }
-
- function createCommonjsModule(fn, basedir, module) {
- return module = {
- path: basedir,
- 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');
- }
-
- var check = function (it) {
- return it && it.Math == Math && it;
- };
-
- // https://github.com/zloirock/core-js/issues/86#issuecomment-115759028
- var global_1 =
- // eslint-disable-next-line no-undef
- check(typeof globalThis == 'object' && globalThis) ||
- check(typeof window == 'object' && window) ||
- check(typeof self == 'object' && self) ||
- check(typeof commonjsGlobal == 'object' && commonjsGlobal) ||
- // eslint-disable-next-line no-new-func
- Function('return this')();
-
- var fails = function (exec) {
- try {
- return !!exec();
- } catch (error) {
- return true;
- }
- };
-
- // Thank's IE8 for his funny defineProperty
- var descriptors = !fails(function () {
- return Object.defineProperty({}, 1, { get: function () { return 7; } })[1] != 7;
- });
-
- var nativePropertyIsEnumerable = {}.propertyIsEnumerable;
- var getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
-
- // Nashorn ~ JDK8 bug
- var NASHORN_BUG = getOwnPropertyDescriptor && !nativePropertyIsEnumerable.call({ 1: 2 }, 1);
-
- // `Object.prototype.propertyIsEnumerable` method implementation
- // https://tc39.github.io/ecma262/#sec-object.prototype.propertyisenumerable
- var f = NASHORN_BUG ? function propertyIsEnumerable(V) {
- var descriptor = getOwnPropertyDescriptor(this, V);
- return !!descriptor && descriptor.enumerable;
- } : nativePropertyIsEnumerable;
-
- var objectPropertyIsEnumerable = {
- f: f
- };
-
- var createPropertyDescriptor = function (bitmap, value) {
- return {
- enumerable: !(bitmap & 1),
- configurable: !(bitmap & 2),
- writable: !(bitmap & 4),
- value: value
- };
- };
-
- var toString = {}.toString;
-
- var classofRaw = function (it) {
- return toString.call(it).slice(8, -1);
- };
-
- var split = ''.split;
-
- // fallback for non-array-like ES3 and non-enumerable old V8 strings
- var indexedObject = fails(function () {
- // throws an error in rhino, see https://github.com/mozilla/rhino/issues/346
- // eslint-disable-next-line no-prototype-builtins
- return !Object('z').propertyIsEnumerable(0);
- }) ? function (it) {
- return classofRaw(it) == 'String' ? split.call(it, '') : Object(it);
- } : Object;
-
- // `RequireObjectCoercible` abstract operation
- // https://tc39.github.io/ecma262/#sec-requireobjectcoercible
- var requireObjectCoercible = function (it) {
- if (it == undefined) throw TypeError("Can't call method on " + it);
- return it;
- };
-
- // toObject with fallback for non-array-like ES3 strings
-
-
-
- var toIndexedObject = function (it) {
- return indexedObject(requireObjectCoercible(it));
- };
-
- var isObject = function (it) {
- return typeof it === 'object' ? it !== null : typeof it === 'function';
- };
-
- // `ToPrimitive` abstract operation
- // https://tc39.github.io/ecma262/#sec-toprimitive
- // 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;
- 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;
- throw TypeError("Can't convert object to primitive value");
- };
-
- var hasOwnProperty = {}.hasOwnProperty;
-
- var has = function (it, key) {
- return hasOwnProperty.call(it, key);
- };
-
- var document$1 = global_1.document;
- // typeof document.createElement is 'object' in old IE
- var EXISTS = isObject(document$1) && isObject(document$1.createElement);
-
- var documentCreateElement = function (it) {
- return EXISTS ? document$1.createElement(it) : {};
- };
-
- // Thank's IE8 for his funny defineProperty
- var ie8DomDefine = !descriptors && !fails(function () {
- return Object.defineProperty(documentCreateElement('div'), 'a', {
- get: function () { return 7; }
- }).a != 7;
- });
-
- var nativeGetOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
-
- // `Object.getOwnPropertyDescriptor` method
- // https://tc39.github.io/ecma262/#sec-object.getownpropertydescriptor
- var f$1 = descriptors ? nativeGetOwnPropertyDescriptor : function getOwnPropertyDescriptor(O, P) {
- O = toIndexedObject(O);
- P = toPrimitive(P, true);
- if (ie8DomDefine) try {
- return nativeGetOwnPropertyDescriptor(O, P);
- } catch (error) { /* empty */ }
- if (has(O, P)) return createPropertyDescriptor(!objectPropertyIsEnumerable.f.call(O, P), O[P]);
- };
-
- var objectGetOwnPropertyDescriptor = {
- f: f$1
- };
-
- var anObject = function (it) {
- if (!isObject(it)) {
- throw TypeError(String(it) + ' is not an object');
- } return it;
- };
-
- var nativeDefineProperty = Object.defineProperty;
-
- // `Object.defineProperty` method
- // https://tc39.github.io/ecma262/#sec-object.defineproperty
- var f$2 = descriptors ? nativeDefineProperty : function defineProperty(O, P, Attributes) {
- anObject(O);
- P = toPrimitive(P, true);
- anObject(Attributes);
- if (ie8DomDefine) try {
- return nativeDefineProperty(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;
- return O;
- };
-
- var objectDefineProperty = {
- f: f$2
- };
-
- var createNonEnumerableProperty = descriptors ? function (object, key, value) {
- return objectDefineProperty.f(object, key, createPropertyDescriptor(1, value));
- } : function (object, key, value) {
- object[key] = value;
- return object;
- };
-
- var setGlobal = function (key, value) {
- try {
- createNonEnumerableProperty(global_1, key, value);
- } catch (error) {
- global_1[key] = value;
- } return value;
- };
-
- var SHARED = '__core-js_shared__';
- var store = global_1[SHARED] || setGlobal(SHARED, {});
-
- var sharedStore = store;
-
- var functionToString = Function.toString;
-
- // this helper broken in `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 nativeWeakMap = typeof WeakMap === 'function' && /native code/.test(inspectSource(WeakMap));
-
- var isPure = false;
-
- var shared = createCommonjsModule(function (module) {
- (module.exports = function (key, value) {
- return sharedStore[key] || (sharedStore[key] = value !== undefined ? value : {});
- })('versions', []).push({
- version: '3.6.5',
- mode: 'global',
- copyright: '© 2020 Denis Pushkarev (zloirock.ru)'
- });
- });
-
- var id = 0;
- var postfix = Math.random();
-
- var uid = function (key) {
- return 'Symbol(' + String(key === undefined ? '' : key) + ')_' + (++id + postfix).toString(36);
- };
-
- var keys = shared('keys');
-
- var sharedKey = function (key) {
- return keys[key] || (keys[key] = uid(key));
- };
-
- var hiddenKeys = {};
-
- var WeakMap$1 = global_1.WeakMap;
- var set, get, has$1;
-
- var enforce = function (it) {
- return has$1(it) ? get(it) : set(it, {});
- };
-
- var getterFor = function (TYPE) {
- return function (it) {
- var state;
- if (!isObject(it) || (state = get(it)).type !== TYPE) {
- throw TypeError('Incompatible receiver, ' + TYPE + ' required');
- } return state;
- };
- };
-
- if (nativeWeakMap) {
- var store$1 = new WeakMap$1();
- var wmget = store$1.get;
- var wmhas = store$1.has;
- var wmset = store$1.set;
- set = function (it, metadata) {
- wmset.call(store$1, it, metadata);
- return metadata;
- };
- get = function (it) {
- return wmget.call(store$1, it) || {};
- };
- has$1 = function (it) {
- return wmhas.call(store$1, it);
- };
- } else {
- var STATE = sharedKey('state');
- hiddenKeys[STATE] = true;
- set = function (it, metadata) {
- createNonEnumerableProperty(it, STATE, metadata);
- return metadata;
- };
- get = function (it) {
- return has(it, STATE) ? it[STATE] : {};
- };
- has$1 = function (it) {
- return has(it, STATE);
- };
- }
-
- var internalState = {
- set: set,
- get: get,
- has: has$1,
- enforce: enforce,
- getterFor: getterFor
- };
-
- var redefine = createCommonjsModule(function (module) {
- var getInternalState = internalState.get;
- var enforceInternalState = internalState.enforce;
- var TEMPLATE = String(String).split('String');
-
- (module.exports = function (O, key, value, options) {
- var unsafe = options ? !!options.unsafe : false;
- var simple = options ? !!options.enumerable : false;
- var noTargetGet = options ? !!options.noTargetGet : false;
- if (typeof value == 'function') {
- if (typeof key == 'string' && !has(value, 'name')) createNonEnumerableProperty(value, 'name', key);
- enforceInternalState(value).source = TEMPLATE.join(typeof key == 'string' ? key : '');
- }
- if (O === global_1) {
- if (simple) O[key] = value;
- else setGlobal(key, value);
- return;
- } else if (!unsafe) {
- delete O[key];
- } else if (!noTargetGet && O[key]) {
- simple = true;
- }
- if (simple) O[key] = value;
- else createNonEnumerableProperty(O, key, value);
- // add fake Function#toString for correct work wrapped methods / constructors with methods like LoDash isNative
- })(Function.prototype, 'toString', function toString() {
- return typeof this == 'function' && getInternalState(this).source || inspectSource(this);
- });
- });
-
- var path = global_1;
-
- var aFunction = 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];
- };
-
- var ceil = Math.ceil;
- var floor = Math.floor;
-
- // `ToInteger` abstract operation
- // https://tc39.github.io/ecma262/#sec-tointeger
- var toInteger = function (argument) {
- return isNaN(argument = +argument) ? 0 : (argument > 0 ? floor : ceil)(argument);
- };
-
- var min = Math.min;
-
- // `ToLength` abstract operation
- // https://tc39.github.io/ecma262/#sec-tolength
- var toLength = function (argument) {
- return argument > 0 ? min(toInteger(argument), 0x1FFFFFFFFFFFFF) : 0; // 2 ** 53 - 1 == 9007199254740991
- };
-
- var max = Math.max;
- var min$1 = 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);
- };
-
- // `Array.prototype.{ indexOf, includes }` methods implementation
- var createMethod = function (IS_INCLUDES) {
- return function ($this, el, fromIndex) {
- var O = toIndexedObject($this);
- var length = toLength(O.length);
- var index = toAbsoluteIndex(fromIndex, length);
- var value;
- // Array#includes uses SameValueZero equality algorithm
- // eslint-disable-next-line no-self-compare
- if (IS_INCLUDES && el != el) while (length > index) {
- value = O[index++];
- // eslint-disable-next-line no-self-compare
- if (value != value) return true;
- // Array#indexOf ignores holes, Array#includes - not
- } else for (;length > index; index++) {
- if ((IS_INCLUDES || index in O) && O[index] === el) return IS_INCLUDES || index || 0;
- } return !IS_INCLUDES && -1;
- };
- };
-
- var arrayIncludes = {
- // `Array.prototype.includes` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.includes
- includes: createMethod(true),
- // `Array.prototype.indexOf` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.indexof
- indexOf: createMethod(false)
- };
-
- var indexOf = arrayIncludes.indexOf;
-
-
- var objectKeysInternal = function (object, names) {
- var O = toIndexedObject(object);
- var i = 0;
- var result = [];
- var key;
- for (key in O) !has(hiddenKeys, key) && has(O, key) && result.push(key);
- // Don't enum bug & hidden keys
- while (names.length > i) if (has(O, key = names[i++])) {
- ~indexOf(result, key) || result.push(key);
- }
- return result;
- };
-
- // IE8- don't enum bug keys
- var enumBugKeys = [
- 'constructor',
- 'hasOwnProperty',
- 'isPrototypeOf',
- 'propertyIsEnumerable',
- 'toLocaleString',
- 'toString',
- 'valueOf'
- ];
-
- var hiddenKeys$1 = enumBugKeys.concat('length', 'prototype');
-
- // `Object.getOwnPropertyNames` method
- // https://tc39.github.io/ecma262/#sec-object.getownpropertynames
- var f$3 = Object.getOwnPropertyNames || function getOwnPropertyNames(O) {
- return objectKeysInternal(O, hiddenKeys$1);
- };
-
- var objectGetOwnPropertyNames = {
- f: f$3
- };
-
- var f$4 = Object.getOwnPropertySymbols;
-
- var objectGetOwnPropertySymbols = {
- f: f$4
- };
-
- // all object keys, includes non-enumerable and symbols
- var ownKeys = getBuiltIn('Reflect', 'ownKeys') || function ownKeys(it) {
- var keys = objectGetOwnPropertyNames.f(anObject(it));
- var getOwnPropertySymbols = objectGetOwnPropertySymbols.f;
- return getOwnPropertySymbols ? keys.concat(getOwnPropertySymbols(it)) : keys;
- };
-
- var copyConstructorProperties = function (target, source) {
- var keys = ownKeys(source);
- var defineProperty = objectDefineProperty.f;
- 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));
- }
- };
-
- var replacement = /#|\.prototype\./;
-
- var isForced = function (feature, detection) {
- var value = data[normalize(feature)];
- return value == POLYFILL ? true
- : value == NATIVE ? false
- : typeof detection == 'function' ? fails(detection)
- : !!detection;
- };
-
- var normalize = isForced.normalize = function (string) {
- return String(string).replace(replacement, '.').toLowerCase();
- };
-
- var data = isForced.data = {};
- var NATIVE = isForced.NATIVE = 'N';
- var POLYFILL = isForced.POLYFILL = 'P';
-
- var isForced_1 = isForced;
-
- var getOwnPropertyDescriptor$1 = objectGetOwnPropertyDescriptor.f;
-
-
-
-
-
-
- /*
- options.target - name of the target object
- options.global - target is the global object
- options.stat - export as static methods of target
- options.proto - export as prototype methods of target
- options.real - real prototype method for the `pure` version
- options.forced - export even if the native feature is available
- options.bind - bind methods to the target, required for the `pure` version
- options.wrap - wrap constructors to preventing global pollution, required for the `pure` version
- options.unsafe - use the simple assignment of property instead of delete + defineProperty
- options.sham - add a flag to not completely full polyfills
- options.enumerable - export as enumerable property
- options.noTargetGet - prevent calling a getter on target
- */
- var _export = function (options, source) {
- var TARGET = options.target;
- var GLOBAL = options.global;
- var STATIC = options.stat;
- var FORCED, target, key, targetProperty, sourceProperty, descriptor;
- if (GLOBAL) {
- target = global_1;
- } else if (STATIC) {
- target = global_1[TARGET] || setGlobal(TARGET, {});
- } else {
- target = (global_1[TARGET] || {}).prototype;
- }
- if (target) for (key in source) {
- sourceProperty = source[key];
- if (options.noTargetGet) {
- descriptor = getOwnPropertyDescriptor$1(target, key);
- targetProperty = descriptor && descriptor.value;
- } else targetProperty = target[key];
- FORCED = isForced_1(GLOBAL ? key : TARGET + (STATIC ? '.' : '#') + key, options.forced);
- // contained in target
- if (!FORCED && targetProperty !== undefined) {
- if (typeof sourceProperty === typeof targetProperty) continue;
- copyConstructorProperties(sourceProperty, targetProperty);
- }
- // add a flag to not completely full polyfills
- if (options.sham || (targetProperty && targetProperty.sham)) {
- createNonEnumerableProperty(sourceProperty, 'sham', true);
- }
- // extend global
- redefine(target, key, sourceProperty, options);
- }
- };
-
- // `Date.now` method
- // https://tc39.github.io/ecma262/#sec-date.now
- _export({ target: 'Date', stat: true }, {
- now: function now() {
- return new Date().getTime();
- }
- });
-
- var DatePrototype = Date.prototype;
- var INVALID_DATE = 'Invalid Date';
- var TO_STRING = 'toString';
- var nativeDateToString = DatePrototype[TO_STRING];
- var getTime = DatePrototype.getTime;
-
- // `Date.prototype.toString` method
- // https://tc39.github.io/ecma262/#sec-date.prototype.tostring
- if (new Date(NaN) + '' != INVALID_DATE) {
- redefine(DatePrototype, TO_STRING, function toString() {
- var value = getTime.call(this);
- // eslint-disable-next-line no-self-compare
- return value === value ? nativeDateToString.call(this) : INVALID_DATE;
- });
- }
-
- var nativeSymbol = !!Object.getOwnPropertySymbols && !fails(function () {
- // Chrome 38 Symbol has incorrect toString conversion
- // eslint-disable-next-line no-undef
- return !String(Symbol());
- });
-
- var useSymbolAsUid = nativeSymbol
- // eslint-disable-next-line no-undef
- && !Symbol.sham
- // eslint-disable-next-line no-undef
- && typeof Symbol.iterator == 'symbol';
-
- // `IsArray` abstract operation
- // https://tc39.github.io/ecma262/#sec-isarray
- var isArray = Array.isArray || function isArray(arg) {
- return classofRaw(arg) == 'Array';
- };
-
- // `ToObject` abstract operation
- // https://tc39.github.io/ecma262/#sec-toobject
- var toObject = function (argument) {
- return Object(requireObjectCoercible(argument));
- };
-
- // `Object.keys` method
- // https://tc39.github.io/ecma262/#sec-object.keys
- var objectKeys = Object.keys || function keys(O) {
- return objectKeysInternal(O, enumBugKeys);
- };
-
- // `Object.defineProperties` method
- // https://tc39.github.io/ecma262/#sec-object.defineproperties
- var objectDefineProperties = descriptors ? Object.defineProperties : function defineProperties(O, Properties) {
- anObject(O);
- var keys = objectKeys(Properties);
- var length = keys.length;
- var index = 0;
- var key;
- while (length > index) objectDefineProperty.f(O, key = keys[index++], Properties[key]);
- return O;
- };
-
- var html = getBuiltIn('document', 'documentElement');
-
- var GT = '>';
- var LT = '<';
- var PROTOTYPE = 'prototype';
- var SCRIPT = 'script';
- var IE_PROTO = sharedKey('IE_PROTO');
-
- var EmptyConstructor = function () { /* empty */ };
-
- var scriptTag = function (content) {
- return LT + SCRIPT + GT + content + LT + '/' + SCRIPT + GT;
- };
-
- // Create object with fake `null` prototype: use ActiveX Object with cleared prototype
- var NullProtoObjectViaActiveX = function (activeXDocument) {
- activeXDocument.write(scriptTag(''));
- activeXDocument.close();
- var temp = activeXDocument.parentWindow.Object;
- activeXDocument = null; // avoid memory leak
- return temp;
- };
-
- // Create object with fake `null` prototype: use iframe Object with cleared prototype
- var NullProtoObjectViaIFrame = function () {
- // Thrash, waste and sodomy: IE GC bug
- var iframe = documentCreateElement('iframe');
- var JS = 'java' + SCRIPT + ':';
- var iframeDocument;
- iframe.style.display = 'none';
- html.appendChild(iframe);
- // https://github.com/zloirock/core-js/issues/475
- iframe.src = String(JS);
- iframeDocument = iframe.contentWindow.document;
- iframeDocument.open();
- iframeDocument.write(scriptTag('document.F=Object'));
- iframeDocument.close();
- return iframeDocument.F;
- };
-
- // Check for document.domain and active x support
- // No need to use active x approach when document.domain is not set
- // see https://github.com/es-shims/es5-shim/issues/150
- // variation of https://github.com/kitcambridge/es5-shim/commit/4f738ac066346
- // avoid IE GC bug
- var activeXDocument;
- var NullProtoObject = function () {
- try {
- /* global ActiveXObject */
- activeXDocument = document.domain && new ActiveXObject('htmlfile');
- } catch (error) { /* ignore */ }
- NullProtoObject = activeXDocument ? NullProtoObjectViaActiveX(activeXDocument) : NullProtoObjectViaIFrame();
- var length = enumBugKeys.length;
- while (length--) delete NullProtoObject[PROTOTYPE][enumBugKeys[length]];
- return NullProtoObject();
- };
-
- hiddenKeys[IE_PROTO] = true;
-
- // `Object.create` method
- // https://tc39.github.io/ecma262/#sec-object.create
- var objectCreate = Object.create || function create(O, Properties) {
- var result;
- if (O !== null) {
- EmptyConstructor[PROTOTYPE] = anObject(O);
- result = new EmptyConstructor();
- EmptyConstructor[PROTOTYPE] = null;
- // add "__proto__" for Object.getPrototypeOf polyfill
- result[IE_PROTO] = O;
- } else result = NullProtoObject();
- return Properties === undefined ? result : objectDefineProperties(result, Properties);
- };
-
- var nativeGetOwnPropertyNames = objectGetOwnPropertyNames.f;
-
- var toString$1 = {}.toString;
-
- var windowNames = typeof window == 'object' && window && Object.getOwnPropertyNames
- ? Object.getOwnPropertyNames(window) : [];
-
- var getWindowNames = function (it) {
- try {
- return nativeGetOwnPropertyNames(it);
- } catch (error) {
- return windowNames.slice();
- }
- };
-
- // fallback for IE11 buggy Object.getOwnPropertyNames with iframe and window
- var f$5 = function getOwnPropertyNames(it) {
- return windowNames && toString$1.call(it) == '[object Window]'
- ? getWindowNames(it)
- : nativeGetOwnPropertyNames(toIndexedObject(it));
- };
-
- var objectGetOwnPropertyNamesExternal = {
- f: f$5
- };
-
- var WellKnownSymbolsStore = shared('wks');
- var Symbol$1 = global_1.Symbol;
- var createWellKnownSymbol = useSymbolAsUid ? Symbol$1 : Symbol$1 && Symbol$1.withoutSetter || uid;
-
- var wellKnownSymbol = function (name) {
- if (!has(WellKnownSymbolsStore, name)) {
- if (nativeSymbol && has(Symbol$1, name)) WellKnownSymbolsStore[name] = Symbol$1[name];
- else WellKnownSymbolsStore[name] = createWellKnownSymbol('Symbol.' + name);
- } return WellKnownSymbolsStore[name];
- };
-
- var f$6 = wellKnownSymbol;
-
- var wellKnownSymbolWrapped = {
- f: f$6
- };
-
- var defineProperty = objectDefineProperty.f;
-
- var defineWellKnownSymbol = function (NAME) {
- var Symbol = path.Symbol || (path.Symbol = {});
- if (!has(Symbol, NAME)) defineProperty(Symbol, NAME, {
- value: wellKnownSymbolWrapped.f(NAME)
- });
- };
-
- var defineProperty$1 = objectDefineProperty.f;
-
-
-
- var TO_STRING_TAG = wellKnownSymbol('toStringTag');
-
- var setToStringTag = function (it, TAG, STATIC) {
- if (it && !has(it = STATIC ? it : it.prototype, TO_STRING_TAG)) {
- defineProperty$1(it, TO_STRING_TAG, { configurable: true, value: TAG });
- }
- };
-
- var aFunction$1 = 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);
- if (that === undefined) return fn;
- switch (length) {
- case 0: return function () {
- return fn.call(that);
- };
- case 1: return function (a) {
- return fn.call(that, a);
- };
- case 2: return function (a, b) {
- return fn.call(that, a, b);
- };
- case 3: return function (a, b, c) {
- return fn.call(that, a, b, c);
- };
- }
- return function (/* ...args */) {
- return fn.apply(that, arguments);
- };
- };
-
- var SPECIES = wellKnownSymbol('species');
-
- // `ArraySpeciesCreate` abstract operation
- // https://tc39.github.io/ecma262/#sec-arrayspeciescreate
- var arraySpeciesCreate = function (originalArray, length) {
- var C;
- if (isArray(originalArray)) {
- C = originalArray.constructor;
- // cross-realm fallback
- if (typeof C == 'function' && (C === Array || isArray(C.prototype))) C = undefined;
- else if (isObject(C)) {
- C = C[SPECIES];
- 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 }` methods implementation
- var createMethod$1 = function (TYPE) {
- var IS_MAP = TYPE == 1;
- var IS_FILTER = TYPE == 2;
- var IS_SOME = TYPE == 3;
- var IS_EVERY = TYPE == 4;
- var IS_FIND_INDEX = TYPE == 6;
- var NO_HOLES = TYPE == 5 || IS_FIND_INDEX;
- return function ($this, callbackfn, that, specificCreate) {
- var O = toObject($this);
- var self = indexedObject(O);
- var boundFunction = functionBindContext(callbackfn, that, 3);
- var length = toLength(self.length);
- var index = 0;
- var create = specificCreate || arraySpeciesCreate;
- var target = IS_MAP ? create($this, length) : IS_FILTER ? create($this, 0) : undefined;
- var value, result;
- for (;length > index; index++) if (NO_HOLES || index in self) {
- value = self[index];
- result = boundFunction(value, index, O);
- if (TYPE) {
- if (IS_MAP) target[index] = result; // map
- else if (result) switch (TYPE) {
- case 3: return true; // some
- case 5: return value; // find
- case 6: return index; // findIndex
- case 2: push.call(target, value); // filter
- } else if (IS_EVERY) return false; // every
- }
- }
- return IS_FIND_INDEX ? -1 : IS_SOME || IS_EVERY ? IS_EVERY : target;
- };
- };
-
- var arrayIteration = {
- // `Array.prototype.forEach` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.foreach
- forEach: createMethod$1(0),
- // `Array.prototype.map` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.map
- map: createMethod$1(1),
- // `Array.prototype.filter` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.filter
- filter: createMethod$1(2),
- // `Array.prototype.some` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.some
- some: createMethod$1(3),
- // `Array.prototype.every` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.every
- every: createMethod$1(4),
- // `Array.prototype.find` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.find
- find: createMethod$1(5),
- // `Array.prototype.findIndex` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.findIndex
- findIndex: createMethod$1(6)
- };
-
- var $forEach = arrayIteration.forEach;
-
- var HIDDEN = sharedKey('hidden');
- var SYMBOL = 'Symbol';
- var PROTOTYPE$1 = 'prototype';
- var TO_PRIMITIVE = wellKnownSymbol('toPrimitive');
- var setInternalState = internalState.set;
- var getInternalState = internalState.getterFor(SYMBOL);
- var ObjectPrototype = Object[PROTOTYPE$1];
- var $Symbol = global_1.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 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;
- // 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; }
- })).a != 7;
- }) ? function (O, P, Attributes) {
- var ObjectPrototypeDescriptor = nativeGetOwnPropertyDescriptor$1(ObjectPrototype, P);
- if (ObjectPrototypeDescriptor) delete ObjectPrototype[P];
- nativeDefineProperty$1(O, P, Attributes);
- if (ObjectPrototypeDescriptor && O !== ObjectPrototype) {
- nativeDefineProperty$1(ObjectPrototype, P, ObjectPrototypeDescriptor);
- }
- } : nativeDefineProperty$1;
-
- var wrap = function (tag, description) {
- var symbol = AllSymbols[tag] = objectCreate($Symbol[PROTOTYPE$1]);
- setInternalState(symbol, {
- type: SYMBOL,
- tag: tag,
- description: description
- });
- if (!descriptors) symbol.description = description;
- return symbol;
- };
-
- var isSymbol = useSymbolAsUid ? function (it) {
- return typeof it == 'symbol';
- } : function (it) {
- return Object(it) instanceof $Symbol;
- };
-
- var $defineProperty = function defineProperty(O, P, Attributes) {
- if (O === ObjectPrototype) $defineProperty(ObjectPrototypeSymbols, P, Attributes);
- anObject(O);
- var key = toPrimitive(P, true);
- anObject(Attributes);
- if (has(AllSymbols, key)) {
- if (!Attributes.enumerable) {
- if (!has(O, HIDDEN)) nativeDefineProperty$1(O, HIDDEN, createPropertyDescriptor(1, {}));
- O[HIDDEN][key] = true;
- } else {
- if (has(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);
- };
-
- var $defineProperties = function defineProperties(O, Properties) {
- anObject(O);
- var properties = toIndexedObject(Properties);
- var keys = objectKeys(properties).concat($getOwnPropertySymbols(properties));
- $forEach(keys, function (key) {
- if (!descriptors || $propertyIsEnumerable.call(properties, key)) $defineProperty(O, key, properties[key]);
- });
- return O;
- };
-
- var $create = function create(O, Properties) {
- return Properties === undefined ? objectCreate(O) : $defineProperties(objectCreate(O), Properties);
- };
-
- var $propertyIsEnumerable = function propertyIsEnumerable(V) {
- var P = toPrimitive(V, true);
- var enumerable = nativePropertyIsEnumerable$1.call(this, P);
- if (this === ObjectPrototype && 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 $getOwnPropertyDescriptor = function getOwnPropertyDescriptor(O, P) {
- var it = toIndexedObject(O);
- var key = toPrimitive(P, true);
- if (it === ObjectPrototype && has(AllSymbols, key) && !has(ObjectPrototypeSymbols, key)) return;
- var descriptor = nativeGetOwnPropertyDescriptor$1(it, key);
- if (descriptor && has(AllSymbols, key) && !(has(it, HIDDEN) && it[HIDDEN][key])) {
- descriptor.enumerable = true;
- }
- return descriptor;
- };
-
- var $getOwnPropertyNames = function getOwnPropertyNames(O) {
- var names = nativeGetOwnPropertyNames$1(toIndexedObject(O));
- var result = [];
- $forEach(names, function (key) {
- if (!has(AllSymbols, key) && !has(hiddenKeys, key)) result.push(key);
- });
- return result;
- };
-
- var $getOwnPropertySymbols = function getOwnPropertySymbols(O) {
- var IS_OBJECT_PROTOTYPE = O === ObjectPrototype;
- var names = nativeGetOwnPropertyNames$1(IS_OBJECT_PROTOTYPE ? ObjectPrototypeSymbols : toIndexedObject(O));
- var result = [];
- $forEach(names, function (key) {
- if (has(AllSymbols, key) && (!IS_OBJECT_PROTOTYPE || has(ObjectPrototype, key))) {
- result.push(AllSymbols[key]);
- }
- });
- return result;
- };
-
- // `Symbol` constructor
- // https://tc39.github.io/ecma262/#sec-symbol-constructor
- if (!nativeSymbol) {
- $Symbol = function Symbol() {
- if (this instanceof $Symbol) throw TypeError('Symbol is not a constructor');
- var description = !arguments.length || arguments[0] === undefined ? undefined : String(arguments[0]);
- var tag = uid(description);
- var setter = function (value) {
- if (this === ObjectPrototype) setter.call(ObjectPrototypeSymbols, value);
- if (has(this, HIDDEN) && has(this[HIDDEN], tag)) this[HIDDEN][tag] = false;
- setSymbolDescriptor(this, tag, createPropertyDescriptor(1, value));
- };
- if (descriptors && USE_SETTER) setSymbolDescriptor(ObjectPrototype, tag, { configurable: true, set: setter });
- return wrap(tag, description);
- };
-
- redefine($Symbol[PROTOTYPE$1], 'toString', function toString() {
- return getInternalState(this).tag;
- });
-
- redefine($Symbol, 'withoutSetter', function (description) {
- return wrap(uid(description), description);
- });
-
- objectPropertyIsEnumerable.f = $propertyIsEnumerable;
- objectDefineProperty.f = $defineProperty;
- objectGetOwnPropertyDescriptor.f = $getOwnPropertyDescriptor;
- objectGetOwnPropertyNames.f = objectGetOwnPropertyNamesExternal.f = $getOwnPropertyNames;
- objectGetOwnPropertySymbols.f = $getOwnPropertySymbols;
-
- wellKnownSymbolWrapped.f = function (name) {
- return wrap(wellKnownSymbol(name), name);
- };
-
- if (descriptors) {
- // https://github.com/tc39/proposal-Symbol-description
- nativeDefineProperty$1($Symbol[PROTOTYPE$1], 'description', {
- configurable: true,
- get: function description() {
- return getInternalState(this).description;
- }
- });
- {
- redefine(ObjectPrototype, 'propertyIsEnumerable', $propertyIsEnumerable, { unsafe: true });
- }
- }
- }
-
- _export({ global: true, wrap: true, forced: !nativeSymbol, sham: !nativeSymbol }, {
- Symbol: $Symbol
- });
-
- $forEach(objectKeys(WellKnownSymbolsStore$1), function (name) {
- defineWellKnownSymbol(name);
- });
-
- _export({ target: SYMBOL, stat: true, forced: !nativeSymbol }, {
- // `Symbol.for` method
- // https://tc39.github.io/ecma262/#sec-symbol.for
- 'for': function (key) {
- var string = String(key);
- if (has(StringToSymbolRegistry, string)) return StringToSymbolRegistry[string];
- var symbol = $Symbol(string);
- StringToSymbolRegistry[string] = symbol;
- SymbolToStringRegistry[symbol] = string;
- return symbol;
- },
- // `Symbol.keyFor` method
- // https://tc39.github.io/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];
- },
- useSetter: function () { USE_SETTER = true; },
- useSimple: function () { USE_SETTER = false; }
- });
-
- _export({ target: 'Object', stat: true, forced: !nativeSymbol, sham: !descriptors }, {
- // `Object.create` method
- // https://tc39.github.io/ecma262/#sec-object.create
- create: $create,
- // `Object.defineProperty` method
- // https://tc39.github.io/ecma262/#sec-object.defineproperty
- defineProperty: $defineProperty,
- // `Object.defineProperties` method
- // https://tc39.github.io/ecma262/#sec-object.defineproperties
- defineProperties: $defineProperties,
- // `Object.getOwnPropertyDescriptor` method
- // https://tc39.github.io/ecma262/#sec-object.getownpropertydescriptors
- getOwnPropertyDescriptor: $getOwnPropertyDescriptor
- });
-
- _export({ target: 'Object', stat: true, forced: !nativeSymbol }, {
- // `Object.getOwnPropertyNames` method
- // https://tc39.github.io/ecma262/#sec-object.getownpropertynames
- getOwnPropertyNames: $getOwnPropertyNames,
- // `Object.getOwnPropertySymbols` method
- // https://tc39.github.io/ecma262/#sec-object.getownpropertysymbols
- getOwnPropertySymbols: $getOwnPropertySymbols
- });
-
- // Chrome 38 and 39 `Object.getOwnPropertySymbols` fails on primitives
- // https://bugs.chromium.org/p/v8/issues/detail?id=3443
- _export({ target: 'Object', stat: true, forced: fails(function () { objectGetOwnPropertySymbols.f(1); }) }, {
- getOwnPropertySymbols: function getOwnPropertySymbols(it) {
- return objectGetOwnPropertySymbols.f(toObject(it));
- }
- });
-
- // `JSON.stringify` method behavior with symbols
- // https://tc39.github.io/ecma262/#sec-json.stringify
- if ($stringify) {
- var FORCED_JSON_STRINGIFY = !nativeSymbol || fails(function () {
- var symbol = $Symbol();
- // MS Edge converts symbol values to JSON as {}
- return $stringify([symbol]) != '[null]'
- // WebKit converts symbol values to JSON as null
- || $stringify({ a: symbol }) != '{}'
- // V8 throws on boxed symbols
- || $stringify(Object(symbol)) != '{}';
- });
-
- _export({ target: 'JSON', stat: true, forced: FORCED_JSON_STRINGIFY }, {
- // eslint-disable-next-line no-unused-vars
- stringify: function stringify(it, replacer, space) {
- var args = [it];
- var index = 1;
- 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 (!isArray(replacer)) replacer = function (key, value) {
- if (typeof $replacer == 'function') value = $replacer.call(this, key, value);
- if (!isSymbol(value)) return value;
- };
- args[1] = replacer;
- return $stringify.apply(null, args);
- }
- });
- }
-
- // `Symbol.prototype[@@toPrimitive]` method
- // https://tc39.github.io/ecma262/#sec-symbol.prototype-@@toprimitive
- if (!$Symbol[PROTOTYPE$1][TO_PRIMITIVE]) {
- createNonEnumerableProperty($Symbol[PROTOTYPE$1], TO_PRIMITIVE, $Symbol[PROTOTYPE$1].valueOf);
- }
- // `Symbol.prototype[@@toStringTag]` property
- // https://tc39.github.io/ecma262/#sec-symbol.prototype-@@tostringtag
- setToStringTag($Symbol, SYMBOL);
-
- hiddenKeys[HIDDEN] = true;
-
- var defineProperty$2 = objectDefineProperty.f;
-
-
- var NativeSymbol = global_1.Symbol;
-
- if (descriptors && typeof NativeSymbol == 'function' && (!('description' in NativeSymbol.prototype) ||
- // Safari 12 bug
- NativeSymbol().description !== undefined
- )) {
- var EmptyStringDescriptionStore = {};
- // wrap Symbol constructor for correct work with undefined description
- var SymbolWrapper = function Symbol() {
- var description = arguments.length < 1 || arguments[0] === undefined ? undefined : String(arguments[0]);
- var result = this instanceof SymbolWrapper
- ? new NativeSymbol(description)
- // in Edge 13, String(Symbol(undefined)) === 'Symbol(undefined)'
- : description === undefined ? NativeSymbol() : NativeSymbol(description);
- if (description === '') EmptyStringDescriptionStore[result] = true;
- return result;
- };
- copyConstructorProperties(SymbolWrapper, NativeSymbol);
- var symbolPrototype = SymbolWrapper.prototype = NativeSymbol.prototype;
- symbolPrototype.constructor = SymbolWrapper;
-
- var symbolToString = symbolPrototype.toString;
- var native = String(NativeSymbol('test')) == 'Symbol(test)';
- var regexp = /^Symbol\((.*)\)[^)]+$/;
- defineProperty$2(symbolPrototype, 'description', {
- configurable: true,
- get: function description() {
- var symbol = isObject(this) ? this.valueOf() : this;
- var string = symbolToString.call(symbol);
- if (has(EmptyStringDescriptionStore, symbol)) return '';
- var desc = native ? string.slice(7, -1) : string.replace(regexp, '$1');
- return desc === '' ? undefined : desc;
- }
- });
-
- _export({ global: true, forced: true }, {
- Symbol: SymbolWrapper
- });
- }
-
- // `Symbol.iterator` well-known symbol
- // https://tc39.github.io/ecma262/#sec-symbol.iterator
- defineWellKnownSymbol('iterator');
-
- var arrayMethodIsStrict = function (METHOD_NAME, argument) {
- var method = [][METHOD_NAME];
- return !!method && fails(function () {
- // eslint-disable-next-line no-useless-call,no-throw-literal
- method.call(null, argument || function () { throw 1; }, 1);
- });
- };
-
- var defineProperty$3 = Object.defineProperty;
- var cache = {};
-
- var thrower = function (it) { throw it; };
-
- var arrayMethodUsesToLength = function (METHOD_NAME, options) {
- if (has(cache, METHOD_NAME)) return cache[METHOD_NAME];
- if (!options) options = {};
- var method = [][METHOD_NAME];
- var ACCESSORS = has(options, 'ACCESSORS') ? options.ACCESSORS : false;
- var argument0 = has(options, 0) ? options[0] : thrower;
- var argument1 = has(options, 1) ? options[1] : undefined;
-
- return cache[METHOD_NAME] = !!method && !fails(function () {
- if (ACCESSORS && !descriptors) return true;
- var O = { length: -1 };
-
- if (ACCESSORS) defineProperty$3(O, 1, { enumerable: true, get: thrower });
- else O[1] = 1;
-
- method.call(O, argument0, argument1);
- });
- };
-
- var $forEach$1 = arrayIteration.forEach;
-
-
-
- var STRICT_METHOD = arrayMethodIsStrict('forEach');
- var USES_TO_LENGTH = arrayMethodUsesToLength('forEach');
-
- // `Array.prototype.forEach` method implementation
- // https://tc39.github.io/ecma262/#sec-array.prototype.foreach
- var arrayForEach = (!STRICT_METHOD || !USES_TO_LENGTH) ? function forEach(callbackfn /* , thisArg */) {
- return $forEach$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- } : [].forEach;
-
- // `Array.prototype.forEach` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.foreach
- _export({ target: 'Array', proto: true, forced: [].forEach != arrayForEach }, {
- forEach: arrayForEach
- });
-
- var $indexOf = arrayIncludes.indexOf;
-
-
-
- var nativeIndexOf = [].indexOf;
-
- var NEGATIVE_ZERO = !!nativeIndexOf && 1 / [1].indexOf(1, -0) < 0;
- var STRICT_METHOD$1 = arrayMethodIsStrict('indexOf');
- var USES_TO_LENGTH$1 = arrayMethodUsesToLength('indexOf', { ACCESSORS: true, 1: 0 });
-
- // `Array.prototype.indexOf` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.indexof
- _export({ target: 'Array', proto: true, forced: NEGATIVE_ZERO || !STRICT_METHOD$1 || !USES_TO_LENGTH$1 }, {
- indexOf: function indexOf(searchElement /* , fromIndex = 0 */) {
- return NEGATIVE_ZERO
- // convert -0 to +0
- ? nativeIndexOf.apply(this, arguments) || 0
- : $indexOf(this, searchElement, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- // `Array.isArray` method
- // https://tc39.github.io/ecma262/#sec-array.isarray
- _export({ target: 'Array', stat: true }, {
- isArray: isArray
- });
-
- var UNSCOPABLES = wellKnownSymbol('unscopables');
- var ArrayPrototype = Array.prototype;
-
- // Array.prototype[@@unscopables]
- // https://tc39.github.io/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 iterators = {};
-
- var correctPrototypeGetter = !fails(function () {
- function F() { /* empty */ }
- F.prototype.constructor = null;
- return Object.getPrototypeOf(new F()) !== F.prototype;
- });
-
- var IE_PROTO$1 = sharedKey('IE_PROTO');
- var ObjectPrototype$1 = Object.prototype;
-
- // `Object.getPrototypeOf` method
- // https://tc39.github.io/ecma262/#sec-object.getprototypeof
- var objectGetPrototypeOf = correctPrototypeGetter ? Object.getPrototypeOf : function (O) {
- O = toObject(O);
- if (has(O, IE_PROTO$1)) return O[IE_PROTO$1];
- if (typeof O.constructor == 'function' && O instanceof O.constructor) {
- return O.constructor.prototype;
- } return O instanceof Object ? ObjectPrototype$1 : null;
- };
-
- var ITERATOR = wellKnownSymbol('iterator');
- var BUGGY_SAFARI_ITERATORS = false;
-
- var returnThis = function () { return this; };
-
- // `%IteratorPrototype%` object
- // https://tc39.github.io/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;
- }
- }
-
- if (IteratorPrototype == undefined) 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
- };
-
- var IteratorPrototype$1 = iteratorsCore.IteratorPrototype;
-
-
-
-
-
- var returnThis$1 = function () { return this; };
-
- var createIteratorConstructor = function (IteratorConstructor, NAME, next) {
- var TO_STRING_TAG = NAME + ' Iterator';
- IteratorConstructor.prototype = objectCreate(IteratorPrototype$1, { next: createPropertyDescriptor(1, next) });
- setToStringTag(IteratorConstructor, TO_STRING_TAG, false);
- iterators[TO_STRING_TAG] = returnThis$1;
- return IteratorConstructor;
- };
-
- var aPossiblePrototype = function (it) {
- if (!isObject(it) && it !== null) {
- throw TypeError("Can't set " + String(it) + ' as a prototype');
- } return it;
- };
-
- // `Object.setPrototypeOf` method
- // https://tc39.github.io/ecma262/#sec-object.setprototypeof
- // Works with __proto__ only. Old v8 can't work with null proto objects.
- /* eslint-disable no-proto */
- var objectSetPrototypeOf = Object.setPrototypeOf || ('__proto__' in {} ? function () {
- var CORRECT_SETTER = false;
- var test = {};
- var setter;
- try {
- setter = Object.getOwnPropertyDescriptor(Object.prototype, '__proto__').set;
- setter.call(test, []);
- CORRECT_SETTER = test instanceof Array;
- } catch (error) { /* empty */ }
- return function setPrototypeOf(O, proto) {
- anObject(O);
- aPossiblePrototype(proto);
- if (CORRECT_SETTER) setter.call(O, proto);
- else O.__proto__ = proto;
- return O;
- };
- }() : undefined);
-
- var IteratorPrototype$2 = iteratorsCore.IteratorPrototype;
- var BUGGY_SAFARI_ITERATORS$1 = iteratorsCore.BUGGY_SAFARI_ITERATORS;
- var ITERATOR$1 = wellKnownSymbol('iterator');
- var KEYS = 'keys';
- var VALUES = 'values';
- var ENTRIES = 'entries';
-
- var returnThis$2 = 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];
- switch (KIND) {
- case KEYS: return function keys() { return new IteratorConstructor(this, KIND); };
- case VALUES: return function values() { return new IteratorConstructor(this, KIND); };
- case ENTRIES: return function entries() { return new IteratorConstructor(this, KIND); };
- } return function () { return new IteratorConstructor(this); };
- };
-
- var TO_STRING_TAG = NAME + ' Iterator';
- var INCORRECT_VALUES_NAME = false;
- var IterablePrototype = Iterable.prototype;
- var nativeIterator = IterablePrototype[ITERATOR$1]
- || IterablePrototype['@@iterator']
- || DEFAULT && IterablePrototype[DEFAULT];
- var defaultIterator = !BUGGY_SAFARI_ITERATORS$1 && 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 (objectSetPrototypeOf) {
- objectSetPrototypeOf(CurrentIteratorPrototype, IteratorPrototype$2);
- } else if (typeof CurrentIteratorPrototype[ITERATOR$1] != 'function') {
- createNonEnumerableProperty(CurrentIteratorPrototype, ITERATOR$1, returnThis$2);
- }
- }
- // Set @@toStringTag to native iterators
- setToStringTag(CurrentIteratorPrototype, TO_STRING_TAG, true);
- }
- }
-
- // fix Array#{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);
- }
- iterators[NAME] = defaultIterator;
-
- // export additional methods
- if (DEFAULT) {
- methods = {
- values: getIterationMethod(VALUES),
- keys: IS_SET ? defaultIterator : getIterationMethod(KEYS),
- entries: getIterationMethod(ENTRIES)
- };
- if (FORCED) for (KEY in methods) {
- if (BUGGY_SAFARI_ITERATORS$1 || 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);
- }
-
- return methods;
- };
-
- var ARRAY_ITERATOR = 'Array Iterator';
- var setInternalState$1 = internalState.set;
- var getInternalState$1 = internalState.getterFor(ARRAY_ITERATOR);
-
- // `Array.prototype.entries` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.entries
- // `Array.prototype.keys` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.keys
- // `Array.prototype.values` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.values
- // `Array.prototype[@@iterator]` method
- // https://tc39.github.io/ecma262/#sec-array.prototype-@@iterator
- // `CreateArrayIterator` internal method
- // https://tc39.github.io/ecma262/#sec-createarrayiterator
- var es_array_iterator = defineIterator(Array, 'Array', function (iterated, kind) {
- setInternalState$1(this, {
- type: ARRAY_ITERATOR,
- target: toIndexedObject(iterated), // target
- index: 0, // next index
- kind: kind // kind
- });
- // `%ArrayIteratorPrototype%.next` method
- // https://tc39.github.io/ecma262/#sec-%arrayiteratorprototype%.next
- }, function () {
- var state = getInternalState$1(this);
- var target = state.target;
- var kind = state.kind;
- var index = state.index++;
- if (!target || index >= target.length) {
- state.target = undefined;
- return { value: undefined, done: true };
- }
- if (kind == 'keys') return { value: index, done: false };
- if (kind == 'values') return { value: target[index], done: false };
- return { value: [index, target[index]], done: false };
- }, 'values');
-
- // argumentsList[@@iterator] is %ArrayProto_values%
- // https://tc39.github.io/ecma262/#sec-createunmappedargumentsobject
- // https://tc39.github.io/ecma262/#sec-createmappedargumentsobject
- iterators.Arguments = iterators.Array;
-
- // https://tc39.github.io/ecma262/#sec-array.prototype-@@unscopables
- addToUnscopables('keys');
- addToUnscopables('values');
- addToUnscopables('entries');
-
- var nativeJoin = [].join;
-
- var ES3_STRINGS = indexedObject != Object;
- var STRICT_METHOD$2 = arrayMethodIsStrict('join', ',');
-
- // `Array.prototype.join` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.join
- _export({ target: 'Array', proto: true, forced: ES3_STRINGS || !STRICT_METHOD$2 }, {
- join: function join(separator) {
- return nativeJoin.call(toIndexedObject(this), separator === undefined ? ',' : separator);
- }
- });
-
- var engineUserAgent = getBuiltIn('navigator', 'userAgent') || '';
-
- var process$1 = global_1.process;
- var versions = process$1 && process$1.versions;
- var v8 = versions && versions.v8;
- var match, version;
-
- if (v8) {
- match = v8.split('.');
- version = 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];
- }
- }
-
- var engineV8Version = version && +version;
-
- var SPECIES$1 = wellKnownSymbol('species');
-
- var arrayMethodHasSpeciesSupport = function (METHOD_NAME) {
- // We can't use this feature detection in V8 since it causes
- // deoptimization and serious performance degradation
- // https://github.com/zloirock/core-js/issues/677
- return engineV8Version >= 51 || !fails(function () {
- var array = [];
- var constructor = array.constructor = {};
- constructor[SPECIES$1] = function () {
- return { foo: 1 };
- };
- return array[METHOD_NAME](Boolean).foo !== 1;
- });
- };
-
- var $map = arrayIteration.map;
-
-
-
- var HAS_SPECIES_SUPPORT = arrayMethodHasSpeciesSupport('map');
- // FF49- issue
- var USES_TO_LENGTH$2 = arrayMethodUsesToLength('map');
-
- // `Array.prototype.map` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.map
- // with adding support of @@species
- _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT || !USES_TO_LENGTH$2 }, {
- map: function map(callbackfn /* , thisArg */) {
- return $map(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- var createProperty = function (object, key, value) {
- var propertyKey = toPrimitive(key);
- if (propertyKey in object) objectDefineProperty.f(object, propertyKey, createPropertyDescriptor(0, value));
- else object[propertyKey] = value;
- };
-
- var HAS_SPECIES_SUPPORT$1 = arrayMethodHasSpeciesSupport('slice');
- var USES_TO_LENGTH$3 = arrayMethodUsesToLength('slice', { ACCESSORS: true, 0: 0, 1: 2 });
-
- var SPECIES$2 = wellKnownSymbol('species');
- var nativeSlice = [].slice;
- var max$1 = Math.max;
-
- // `Array.prototype.slice` method
- // https://tc39.github.io/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 || !USES_TO_LENGTH$3 }, {
- slice: function slice(start, end) {
- var O = toIndexedObject(this);
- var length = toLength(O.length);
- var k = toAbsoluteIndex(start, length);
- var fin = toAbsoluteIndex(end === undefined ? length : end, length);
- // inline `ArraySpeciesCreate` for usage native `Array#slice` where it's possible
- var Constructor, result, n;
- if (isArray(O)) {
- Constructor = O.constructor;
- // cross-realm fallback
- if (typeof Constructor == 'function' && (Constructor === Array || isArray(Constructor.prototype))) {
- Constructor = undefined;
- } else if (isObject(Constructor)) {
- Constructor = Constructor[SPECIES$2];
- 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));
- for (n = 0; k < fin; k++, n++) if (k in O) createProperty(result, n, O[k]);
- result.length = n;
- return result;
- }
- });
-
- var arrayBufferNative = typeof ArrayBuffer !== 'undefined' && typeof DataView !== 'undefined';
-
- var redefineAll = function (target, src, options) {
- for (var key in src) redefine(target, key, src[key], options);
- return target;
- };
-
- var anInstance = function (it, Constructor, name) {
- if (!(it instanceof Constructor)) {
- throw TypeError('Incorrect ' + (name ? name + ' ' : '') + 'invocation');
- } return it;
- };
-
- // `ToIndex` abstract operation
- // https://tc39.github.io/ecma262/#sec-toindex
- var toIndex = function (it) {
- if (it === undefined) return 0;
- var number = toInteger(it);
- var length = toLength(number);
- if (number !== length) throw RangeError('Wrong length or index');
- return length;
- };
-
- // IEEE754 conversions based on https://github.com/feross/ieee754
- // eslint-disable-next-line no-shadow-restricted-names
- var Infinity$1 = 1 / 0;
- var abs = Math.abs;
- var pow = Math.pow;
- var floor$1 = Math.floor;
- var log = Math.log;
- var LN2 = Math.LN2;
-
- var pack = function (number, mantissaLength, bytes) {
- var buffer = new Array(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 sign = number < 0 || number === 0 && 1 / number < 0 ? 1 : 0;
- var index = 0;
- var exponent, mantissa, c;
- number = abs(number);
- // eslint-disable-next-line no-self-compare
- if (number != number || number === Infinity$1) {
- // eslint-disable-next-line no-self-compare
- mantissa = number != number ? 1 : 0;
- exponent = eMax;
- } else {
- exponent = floor$1(log(number) / LN2);
- if (number * (c = pow(2, -exponent)) < 1) {
- exponent--;
- c *= 2;
- }
- if (exponent + eBias >= 1) {
- number += rt / c;
- } else {
- number += rt * pow(2, 1 - eBias);
- }
- if (number * c >= 2) {
- exponent++;
- c /= 2;
- }
- if (exponent + eBias >= eMax) {
- mantissa = 0;
- exponent = eMax;
- } else if (exponent + eBias >= 1) {
- mantissa = (number * c - 1) * pow(2, mantissaLength);
- exponent = exponent + eBias;
- } else {
- mantissa = number * pow(2, eBias - 1) * pow(2, mantissaLength);
- exponent = 0;
- }
- }
- for (; mantissaLength >= 8; buffer[index++] = mantissa & 255, mantissa /= 256, mantissaLength -= 8);
- exponent = exponent << mantissaLength | mantissa;
- exponentLength += mantissaLength;
- for (; exponentLength > 0; buffer[index++] = exponent & 255, exponent /= 256, exponentLength -= 8);
- buffer[--index] |= sign * 128;
- return buffer;
- };
-
- var unpack = function (buffer, mantissaLength) {
- var bytes = buffer.length;
- var exponentLength = bytes * 8 - mantissaLength - 1;
- var eMax = (1 << exponentLength) - 1;
- var eBias = eMax >> 1;
- var nBits = exponentLength - 7;
- var index = bytes - 1;
- var sign = buffer[index--];
- var exponent = sign & 127;
- var mantissa;
- sign >>= 7;
- for (; nBits > 0; exponent = exponent * 256 + buffer[index], index--, nBits -= 8);
- mantissa = exponent & (1 << -nBits) - 1;
- exponent >>= -nBits;
- nBits += mantissaLength;
- for (; nBits > 0; mantissa = mantissa * 256 + buffer[index], index--, nBits -= 8);
- if (exponent === 0) {
- exponent = 1 - eBias;
- } else if (exponent === eMax) {
- return mantissa ? NaN : sign ? -Infinity$1 : Infinity$1;
- } else {
- mantissa = mantissa + pow(2, mantissaLength);
- exponent = exponent - eBias;
- } return (sign ? -1 : 1) * mantissa * pow(2, exponent - mantissaLength);
- };
-
- var ieee754 = {
- pack: pack,
- unpack: unpack
- };
-
- // `Array.prototype.fill` method implementation
- // https://tc39.github.io/ecma262/#sec-array.prototype.fill
- var arrayFill = function fill(value /* , start = 0, end = @length */) {
- var O = toObject(this);
- var length = toLength(O.length);
- var argumentsLength = arguments.length;
- var index = toAbsoluteIndex(argumentsLength > 1 ? arguments[1] : undefined, length);
- var end = argumentsLength > 2 ? arguments[2] : undefined;
- var endPos = end === undefined ? length : toAbsoluteIndex(end, length);
- while (endPos > index) O[index++] = value;
- return O;
- };
-
- var getOwnPropertyNames = objectGetOwnPropertyNames.f;
- var defineProperty$4 = objectDefineProperty.f;
-
-
-
-
- var getInternalState$2 = internalState.get;
- var setInternalState$2 = internalState.set;
- var ARRAY_BUFFER = 'ArrayBuffer';
- var DATA_VIEW = 'DataView';
- var PROTOTYPE$2 = '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 packIEEE754 = ieee754.pack;
- var unpackIEEE754 = ieee754.unpack;
-
- var packInt8 = function (number) {
- return [number & 0xFF];
- };
-
- var packInt16 = function (number) {
- return [number & 0xFF, number >> 8 & 0xFF];
- };
-
- var packInt32 = function (number) {
- return [number & 0xFF, number >> 8 & 0xFF, number >> 16 & 0xFF, number >> 24 & 0xFF];
- };
-
- var unpackInt32 = function (buffer) {
- return buffer[3] << 24 | buffer[2] << 16 | buffer[1] << 8 | buffer[0];
- };
-
- var packFloat32 = function (number) {
- return packIEEE754(number, 23, 4);
- };
-
- var packFloat64 = function (number) {
- return packIEEE754(number, 52, 8);
- };
-
- var addGetter = function (Constructor, key) {
- defineProperty$4(Constructor[PROTOTYPE$2], key, { get: function () { return getInternalState$2(this)[key]; } });
- };
-
- var get$1 = function (view, count, index, isLittleEndian) {
- var intIndex = toIndex(index);
- var store = getInternalState$2(view);
- if (intIndex + count > store.byteLength) throw RangeError$1(WRONG_INDEX);
- 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 intIndex = toIndex(index);
- var store = getInternalState$2(view);
- if (intIndex + count > store.byteLength) throw RangeError$1(WRONG_INDEX);
- 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);
- var byteLength = toIndex(length);
- setInternalState$2(this, {
- bytes: arrayFill.call(new Array(byteLength), 0),
- byteLength: byteLength
- });
- if (!descriptors) this.byteLength = byteLength;
- };
-
- $DataView = function DataView(buffer, byteOffset, byteLength) {
- anInstance(this, $DataView, DATA_VIEW);
- anInstance(buffer, $ArrayBuffer, DATA_VIEW);
- 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$2(this, {
- buffer: buffer,
- byteLength: byteLength,
- byteOffset: offset
- });
- if (!descriptors) {
- this.buffer = buffer;
- this.byteLength = byteLength;
- this.byteOffset = offset;
- }
- };
-
- if (descriptors) {
- addGetter($ArrayBuffer, 'byteLength');
- addGetter($DataView, 'buffer');
- addGetter($DataView, 'byteLength');
- addGetter($DataView, 'byteOffset');
- }
-
- redefineAll($DataView[PROTOTYPE$2], {
- getInt8: function getInt8(byteOffset) {
- return get$1(this, 1, byteOffset)[0] << 24 >> 24;
- },
- getUint8: function getUint8(byteOffset) {
- return get$1(this, 1, byteOffset)[0];
- },
- getInt16: function getInt16(byteOffset /* , littleEndian */) {
- var bytes = get$1(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);
- return bytes[1] << 8 | bytes[0];
- },
- getInt32: function getInt32(byteOffset /* , littleEndian */) {
- return unpackInt32(get$1(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;
- },
- getFloat32: function getFloat32(byteOffset /* , littleEndian */) {
- return unpackIEEE754(get$1(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);
- },
- setInt8: function setInt8(byteOffset, value) {
- set$1(this, 1, byteOffset, packInt8, value);
- },
- setUint8: function setUint8(byteOffset, value) {
- set$1(this, 1, byteOffset, packInt8, value);
- },
- setInt16: function setInt16(byteOffset, value /* , littleEndian */) {
- set$1(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);
- },
- setInt32: function setInt32(byteOffset, value /* , littleEndian */) {
- set$1(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);
- },
- setFloat32: function setFloat32(byteOffset, value /* , littleEndian */) {
- set$1(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);
- }
- });
- } else {
- if (!fails(function () {
- NativeArrayBuffer(1);
- }) || !fails(function () {
- new NativeArrayBuffer(-1); // eslint-disable-line no-new
- }) || fails(function () {
- new NativeArrayBuffer(); // eslint-disable-line no-new
- new NativeArrayBuffer(1.5); // eslint-disable-line no-new
- new NativeArrayBuffer(NaN); // eslint-disable-line no-new
- return NativeArrayBuffer.name != ARRAY_BUFFER;
- })) {
- $ArrayBuffer = function ArrayBuffer(length) {
- anInstance(this, $ArrayBuffer);
- return new NativeArrayBuffer(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]);
- }
- }
- 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);
- }
-
- // iOS Safari 7.x bug
- var testView = new $DataView(new $ArrayBuffer(2));
- var nativeSetInt8 = $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);
- },
- setUint8: function setUint8(byteOffset, value) {
- nativeSetInt8.call(this, byteOffset, value << 24 >> 24);
- }
- }, { unsafe: true });
- }
-
- setToStringTag($ArrayBuffer, ARRAY_BUFFER);
- setToStringTag($DataView, DATA_VIEW);
-
- var arrayBuffer = {
- ArrayBuffer: $ArrayBuffer,
- DataView: $DataView
- };
-
- var SPECIES$3 = wellKnownSymbol('species');
-
- var setSpecies = function (CONSTRUCTOR_NAME) {
- var Constructor = getBuiltIn(CONSTRUCTOR_NAME);
- var defineProperty = objectDefineProperty.f;
-
- if (descriptors && Constructor && !Constructor[SPECIES$3]) {
- defineProperty(Constructor, SPECIES$3, {
- configurable: true,
- get: function () { return this; }
- });
- }
- };
-
- var ARRAY_BUFFER$1 = 'ArrayBuffer';
- var ArrayBuffer$1 = arrayBuffer[ARRAY_BUFFER$1];
- var NativeArrayBuffer$1 = global_1[ARRAY_BUFFER$1];
-
- // `ArrayBuffer` constructor
- // https://tc39.github.io/ecma262/#sec-arraybuffer-constructor
- _export({ global: true, forced: NativeArrayBuffer$1 !== ArrayBuffer$1 }, {
- ArrayBuffer: ArrayBuffer$1
- });
-
- setSpecies(ARRAY_BUFFER$1);
-
- var TO_STRING_TAG$1 = wellKnownSymbol('toStringTag');
- var test = {};
-
- test[TO_STRING_TAG$1] = 'z';
-
- var toStringTagSupport = String(test) === '[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;
- };
-
- var defineProperty$5 = objectDefineProperty.f;
-
-
-
-
-
- var Int8Array$1 = global_1.Int8Array;
- var Int8ArrayPrototype = Int8Array$1 && Int8Array$1.prototype;
- var Uint8ClampedArray = global_1.Uint8ClampedArray;
- var Uint8ClampedArrayPrototype = Uint8ClampedArray && Uint8ClampedArray.prototype;
- var TypedArray = Int8Array$1 && objectGetPrototypeOf(Int8Array$1);
- var TypedArrayPrototype = Int8ArrayPrototype && objectGetPrototypeOf(Int8ArrayPrototype);
- var ObjectPrototype$3 = Object.prototype;
- var isPrototypeOf = ObjectPrototype$3.isPrototypeOf;
-
- var TO_STRING_TAG$3 = 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 TYPED_ARRAY_TAG_REQIRED = false;
- var NAME;
-
- var TypedArrayConstructorsList = {
- Int8Array: 1,
- Uint8Array: 1,
- Uint8ClampedArray: 1,
- Int16Array: 2,
- Uint16Array: 2,
- Int32Array: 4,
- Uint32Array: 4,
- Float32Array: 4,
- Float64Array: 8
- };
-
- var isView = function isView(it) {
- var klass = classof(it);
- return klass === 'DataView' || has(TypedArrayConstructorsList, klass);
- };
-
- var isTypedArray = function (it) {
- return isObject(it) && has(TypedArrayConstructorsList, classof(it));
- };
-
- var aTypedArray = function (it) {
- if (isTypedArray(it)) return it;
- throw TypeError('Target is not a typed array');
- };
-
- var aTypedArrayConstructor = 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];
- 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) {
- if (!descriptors) return;
- if (forced) for (var ARRAY in TypedArrayConstructorsList) {
- var TypedArrayConstructor = global_1[ARRAY];
- if (TypedArrayConstructor && has(TypedArrayConstructor.prototype, KEY)) {
- delete TypedArrayConstructor.prototype[KEY];
- }
- }
- if (!TypedArrayPrototype[KEY] || forced) {
- redefine(TypedArrayPrototype, KEY, forced ? property
- : NATIVE_ARRAY_BUFFER_VIEWS && Int8ArrayPrototype[KEY] || property);
- }
- };
-
- var exportTypedArrayStaticMethod = 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)) {
- delete TypedArrayConstructor[KEY];
- }
- }
- 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);
- } catch (error) { /* empty */ }
- } else return;
- }
- for (ARRAY in TypedArrayConstructorsList) {
- TypedArrayConstructor = global_1[ARRAY];
- if (TypedArrayConstructor && (!TypedArrayConstructor[KEY] || forced)) {
- redefine(TypedArrayConstructor, KEY, property);
- }
- }
- };
-
- for (NAME in TypedArrayConstructorsList) {
- if (!global_1[NAME]) NATIVE_ARRAY_BUFFER_VIEWS = false;
- }
-
- // WebKit bug - typed arrays constructors prototype is Object.prototype
- if (!NATIVE_ARRAY_BUFFER_VIEWS || typeof TypedArray != 'function' || TypedArray === Function.prototype) {
- // eslint-disable-next-line no-shadow
- 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 || !TypedArrayPrototype || TypedArrayPrototype === ObjectPrototype$3) {
- TypedArrayPrototype = TypedArray.prototype;
- if (NATIVE_ARRAY_BUFFER_VIEWS) for (NAME in TypedArrayConstructorsList) {
- if (global_1[NAME]) objectSetPrototypeOf(global_1[NAME].prototype, TypedArrayPrototype);
- }
- }
-
- // WebKit bug - one more object in Uint8ClampedArray prototype chain
- if (NATIVE_ARRAY_BUFFER_VIEWS && objectGetPrototypeOf(Uint8ClampedArrayPrototype) !== TypedArrayPrototype) {
- objectSetPrototypeOf(Uint8ClampedArrayPrototype, TypedArrayPrototype);
- }
-
- if (descriptors && !has(TypedArrayPrototype, TO_STRING_TAG$3)) {
- TYPED_ARRAY_TAG_REQIRED = true;
- defineProperty$5(TypedArrayPrototype, TO_STRING_TAG$3, { get: function () {
- return isObject(this) ? this[TYPED_ARRAY_TAG] : undefined;
- } });
- for (NAME in TypedArrayConstructorsList) if (global_1[NAME]) {
- createNonEnumerableProperty(global_1[NAME], TYPED_ARRAY_TAG, NAME);
- }
- }
-
- var arrayBufferViewCore = {
- NATIVE_ARRAY_BUFFER_VIEWS: NATIVE_ARRAY_BUFFER_VIEWS,
- TYPED_ARRAY_TAG: TYPED_ARRAY_TAG_REQIRED && TYPED_ARRAY_TAG,
- aTypedArray: aTypedArray,
- aTypedArrayConstructor: aTypedArrayConstructor,
- exportTypedArrayMethod: exportTypedArrayMethod,
- exportTypedArrayStaticMethod: exportTypedArrayStaticMethod,
- isView: isView,
- isTypedArray: isTypedArray,
- TypedArray: TypedArray,
- TypedArrayPrototype: TypedArrayPrototype
- };
-
- var NATIVE_ARRAY_BUFFER_VIEWS$1 = arrayBufferViewCore.NATIVE_ARRAY_BUFFER_VIEWS;
-
- // `ArrayBuffer.isView` method
- // https://tc39.github.io/ecma262/#sec-arraybuffer.isview
- _export({ target: 'ArrayBuffer', stat: true, forced: !NATIVE_ARRAY_BUFFER_VIEWS$1 }, {
- isView: arrayBufferViewCore.isView
- });
-
- var SPECIES$4 = wellKnownSymbol('species');
-
- // `SpeciesConstructor` abstract operation
- // https://tc39.github.io/ecma262/#sec-speciesconstructor
- var speciesConstructor = function (O, defaultConstructor) {
- var C = anObject(O).constructor;
- var S;
- return C === undefined || (S = anObject(C)[SPECIES$4]) == undefined ? defaultConstructor : aFunction$1(S);
- };
-
- var ArrayBuffer$2 = arrayBuffer.ArrayBuffer;
- var DataView$1 = arrayBuffer.DataView;
- var nativeArrayBufferSlice = ArrayBuffer$2.prototype.slice;
-
- var INCORRECT_SLICE = fails(function () {
- return !new ArrayBuffer$2(2).slice(1, undefined).byteLength;
- });
-
- // `ArrayBuffer.prototype.slice` method
- // https://tc39.github.io/ecma262/#sec-arraybuffer.prototype.slice
- _export({ target: 'ArrayBuffer', proto: true, unsafe: true, forced: INCORRECT_SLICE }, {
- slice: function slice(start, end) {
- if (nativeArrayBufferSlice !== undefined && end === undefined) {
- return nativeArrayBufferSlice.call(anObject(this), start); // FF fix
- }
- var length = anObject(this).byteLength;
- var first = toAbsoluteIndex(start, length);
- var fin = toAbsoluteIndex(end === undefined ? length : end, length);
- var result = new (speciesConstructor(this, ArrayBuffer$2))(toLength(fin - first));
- var viewSource = new DataView$1(this);
- var viewTarget = new DataView$1(result);
- var index = 0;
- while (first < fin) {
- viewTarget.setUint8(index++, viewSource.getUint8(first++));
- } return result;
- }
- });
-
- // `DataView` constructor
- // https://tc39.github.io/ecma262/#sec-dataview-constructor
- _export({ global: true, forced: !arrayBufferNative }, {
- DataView: arrayBuffer.DataView
- });
-
- var defineProperty$6 = objectDefineProperty.f;
-
- var FunctionPrototype = Function.prototype;
- var FunctionPrototypeToString = FunctionPrototype.toString;
- var nameRE = /^\s*function ([^ (]*)/;
- var NAME$1 = 'name';
-
- // Function instances `.name` property
- // https://tc39.github.io/ecma262/#sec-function-instances-name
- if (descriptors && !(NAME$1 in FunctionPrototype)) {
- defineProperty$6(FunctionPrototype, NAME$1, {
- configurable: true,
- get: function () {
- try {
- return FunctionPrototypeToString.call(this).match(nameRE)[1];
- } catch (error) {
- return '';
- }
- }
- });
- }
-
- // `Object.create` method
- // https://tc39.github.io/ecma262/#sec-object.create
- _export({ target: 'Object', stat: true, sham: !descriptors }, {
- create: objectCreate
- });
-
- var nativeGetOwnPropertyNames$2 = objectGetOwnPropertyNamesExternal.f;
-
- var FAILS_ON_PRIMITIVES = fails(function () { return !Object.getOwnPropertyNames(1); });
-
- // `Object.getOwnPropertyNames` method
- // https://tc39.github.io/ecma262/#sec-object.getownpropertynames
- _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES }, {
- getOwnPropertyNames: nativeGetOwnPropertyNames$2
- });
-
- // `Object.prototype.toString` method implementation
- // https://tc39.github.io/ecma262/#sec-object.prototype.tostring
- var objectToString = toStringTagSupport ? {}.toString : function toString() {
- return '[object ' + classof(this) + ']';
- };
-
- // `Object.prototype.toString` method
- // https://tc39.github.io/ecma262/#sec-object.prototype.tostring
- if (!toStringTagSupport) {
- redefine(Object.prototype, 'toString', objectToString, { unsafe: true });
- }
-
- var nativePromiseConstructor = global_1.Promise;
-
- var ITERATOR$2 = wellKnownSymbol('iterator');
- var ArrayPrototype$1 = Array.prototype;
-
- // check on default Array iterator
- var isArrayIteratorMethod = function (it) {
- return it !== undefined && (iterators.Array === it || ArrayPrototype$1[ITERATOR$2] === it);
- };
-
- var ITERATOR$3 = wellKnownSymbol('iterator');
-
- var getIteratorMethod = function (it) {
- if (it != undefined) return it[ITERATOR$3]
- || it['@@iterator']
- || iterators[classof(it)];
- };
-
- // 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) {
- var returnMethod = iterator['return'];
- if (returnMethod !== undefined) anObject(returnMethod.call(iterator));
- throw error;
- }
- };
-
- var iterate_1 = createCommonjsModule(function (module) {
- var Result = function (stopped, result) {
- this.stopped = stopped;
- this.result = result;
- };
-
- var iterate = module.exports = function (iterable, fn, that, AS_ENTRIES, IS_ITERATOR) {
- var boundFunction = functionBindContext(fn, that, AS_ENTRIES ? 2 : 1);
- var iterator, iterFn, index, length, result, next, step;
-
- if (IS_ITERATOR) {
- iterator = iterable;
- } else {
- iterFn = getIteratorMethod(iterable);
- if (typeof iterFn != 'function') throw TypeError('Target is not iterable');
- // optimisation for array iterators
- if (isArrayIteratorMethod(iterFn)) {
- for (index = 0, length = toLength(iterable.length); length > index; index++) {
- result = AS_ENTRIES
- ? boundFunction(anObject(step = iterable[index])[0], step[1])
- : boundFunction(iterable[index]);
- if (result && result instanceof Result) return result;
- } return new Result(false);
- }
- iterator = iterFn.call(iterable);
- }
-
- next = iterator.next;
- while (!(step = next.call(iterator)).done) {
- result = callWithSafeIterationClosing(iterator, boundFunction, step.value, AS_ENTRIES);
- if (typeof result == 'object' && result && result instanceof Result) return result;
- } return new Result(false);
- };
-
- iterate.stop = function (result) {
- return new Result(true, result);
- };
- });
-
- var ITERATOR$4 = wellKnownSymbol('iterator');
- var SAFE_CLOSING = false;
-
- try {
- var called = 0;
- var iteratorWithReturn = {
- next: function () {
- return { done: !!called++ };
- },
- 'return': function () {
- SAFE_CLOSING = true;
- }
- };
- iteratorWithReturn[ITERATOR$4] = function () {
- return this;
- };
- // eslint-disable-next-line no-throw-literal
- Array.from(iteratorWithReturn, function () { throw 2; });
- } catch (error) { /* empty */ }
-
- var checkCorrectnessOfIteration = function (exec, SKIP_CLOSING) {
- if (!SKIP_CLOSING && !SAFE_CLOSING) return false;
- var ITERATION_SUPPORT = false;
- try {
- var object = {};
- object[ITERATOR$4] = function () {
- return {
- next: function () {
- return { done: ITERATION_SUPPORT = true };
- }
- };
- };
- exec(object);
- } catch (error) { /* empty */ }
- return ITERATION_SUPPORT;
- };
-
- var engineIsIos = /(iphone|ipod|ipad).*applewebkit/i.test(engineUserAgent);
-
- 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 counter = 0;
- var queue = {};
- var ONREADYSTATECHANGE = 'onreadystatechange';
- var defer, channel, port;
-
- var run = function (id) {
- // eslint-disable-next-line no-prototype-builtins
- if (queue.hasOwnProperty(id)) {
- var fn = queue[id];
- delete queue[id];
- fn();
- }
- };
-
- var runner = function (id) {
- return function () {
- run(id);
- };
- };
-
- var listener = function (event) {
- run(event.data);
- };
-
- var post = function (id) {
- // old engines have not location.origin
- global_1.postMessage(id + '', location$1.protocol + '//' + location$1.host);
- };
-
- // Node.js 0.9+ & IE10+ has setImmediate, otherwise:
- if (!set$2 || !clear) {
- set$2 = function setImmediate(fn) {
- var args = [];
- var i = 1;
- while (arguments.length > i) args.push(arguments[i++]);
- queue[++counter] = function () {
- // eslint-disable-next-line no-new-func
- (typeof fn == 'function' ? fn : Function(fn)).apply(undefined, args);
- };
- defer(counter);
- return counter;
- };
- clear = function clearImmediate(id) {
- delete queue[id];
- };
- // Node.js 0.8-
- if (classofRaw(process$2) == 'process') {
- defer = function (id) {
- process$2.nextTick(runner(id));
- };
- // Sphere (JS game engine) Dispatch API
- } else if (Dispatch && Dispatch.now) {
- defer = function (id) {
- Dispatch.now(runner(id));
- };
- // Browsers with MessageChannel, includes WebWorkers
- // except iOS - https://github.com/zloirock/core-js/issues/624
- } else if (MessageChannel && !engineIsIos) {
- channel = new MessageChannel();
- port = channel.port2;
- channel.port1.onmessage = listener;
- defer = functionBindContext(port.postMessage, port, 1);
- // Browsers with postMessage, skip WebWorkers
- // IE8 has postMessage, but it's sync & typeof its postMessage is 'object'
- } else if (
- global_1.addEventListener &&
- typeof postMessage == 'function' &&
- !global_1.importScripts &&
- !fails(post) &&
- location$1.protocol !== 'file:'
- ) {
- defer = post;
- global_1.addEventListener('message', listener, false);
- // IE8-
- } else if (ONREADYSTATECHANGE in documentCreateElement('script')) {
- defer = function (id) {
- html.appendChild(documentCreateElement('script'))[ONREADYSTATECHANGE] = function () {
- html.removeChild(this);
- run(id);
- };
- };
- // Rest old browsers
- } else {
- defer = function (id) {
- setTimeout(runner(id), 0);
- };
- }
- }
-
- var task = {
- set: set$2,
- clear: clear
- };
-
- var getOwnPropertyDescriptor$2 = objectGetOwnPropertyDescriptor.f;
-
- var macrotask = task.set;
-
-
- var MutationObserver = global_1.MutationObserver || global_1.WebKitMutationObserver;
- var process$3 = global_1.process;
- var Promise$1 = global_1.Promise;
- var IS_NODE = classofRaw(process$3) == 'process';
- // Node.js 11 shows ExperimentalWarning on getting `queueMicrotask`
- var queueMicrotaskDescriptor = getOwnPropertyDescriptor$2(global_1, 'queueMicrotask');
- var queueMicrotask = queueMicrotaskDescriptor && queueMicrotaskDescriptor.value;
-
- var flush, head, last, notify, toggle, node, promise, then;
-
- // modern engines have queueMicrotask method
- if (!queueMicrotask) {
- flush = function () {
- var parent, fn;
- if (IS_NODE && (parent = process$3.domain)) parent.exit();
- while (head) {
- fn = head.fn;
- head = head.next;
- try {
- fn();
- } catch (error) {
- if (head) notify();
- else last = undefined;
- throw error;
- }
- } last = undefined;
- if (parent) parent.enter();
- };
-
- // Node.js
- if (IS_NODE) {
- notify = function () {
- process$3.nextTick(flush);
- };
- // browsers with MutationObserver, except iOS - https://github.com/zloirock/core-js/issues/339
- } else if (MutationObserver && !engineIsIos) {
- toggle = true;
- node = document.createTextNode('');
- new MutationObserver(flush).observe(node, { characterData: true });
- notify = 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);
- then = promise.then;
- notify = function () {
- then.call(promise, flush);
- };
- // for other environments - macrotask based on:
- // - setImmediate
- // - MessageChannel
- // - window.postMessag
- // - onreadystatechange
- // - setTimeout
- } else {
- notify = function () {
- // strange IE + webpack dev server bug - use .call(global)
- macrotask.call(global_1, flush);
- };
- }
- }
-
- var microtask = queueMicrotask || function (fn) {
- var task = { fn: fn, next: undefined };
- if (last) last.next = task;
- if (!head) {
- head = task;
- notify();
- } last = task;
- };
-
- var PromiseCapability = function (C) {
- var resolve, reject;
- this.promise = new C(function ($$resolve, $$reject) {
- if (resolve !== undefined || reject !== undefined) throw TypeError('Bad Promise constructor');
- resolve = $$resolve;
- reject = $$reject;
- });
- this.resolve = aFunction$1(resolve);
- this.reject = aFunction$1(reject);
- };
-
- // 25.4.1.5 NewPromiseCapability(C)
- var f$7 = function (C) {
- return new PromiseCapability(C);
- };
-
- var newPromiseCapability = {
- f: f$7
- };
-
- var promiseResolve = function (C, x) {
- anObject(C);
- if (isObject(x) && x.constructor === C) return x;
- var promiseCapability = newPromiseCapability.f(C);
- var resolve = promiseCapability.resolve;
- resolve(x);
- return promiseCapability.promise;
- };
-
- var hostReportErrors = function (a, b) {
- var console = global_1.console;
- if (console && console.error) {
- arguments.length === 1 ? console.error(a) : console.error(a, b);
- }
- };
-
- var perform = function (exec) {
- try {
- return { error: false, value: exec() };
- } catch (error) {
- return { error: true, value: error };
- }
- };
-
- var task$1 = task.set;
-
-
-
-
-
-
-
-
-
-
- var SPECIES$5 = wellKnownSymbol('species');
- var PROMISE = 'Promise';
- var getInternalState$3 = internalState.get;
- var setInternalState$3 = internalState.set;
- var getInternalPromiseState = internalState.getterFor(PROMISE);
- var PromiseConstructor = nativePromiseConstructor;
- var TypeError$1 = global_1.TypeError;
- var document$2 = global_1.document;
- var process$4 = global_1.process;
- var $fetch = getBuiltIn('fetch');
- var newPromiseCapability$1 = newPromiseCapability.f;
- var newGenericPromiseCapability = newPromiseCapability$1;
- var IS_NODE$1 = classofRaw(process$4) == 'process';
- var DISPATCH_EVENT = !!(document$2 && document$2.createEvent && global_1.dispatchEvent);
- var UNHANDLED_REJECTION = 'unhandledrejection';
- var REJECTION_HANDLED = 'rejectionhandled';
- var PENDING = 0;
- var FULFILLED = 1;
- var REJECTED = 2;
- var HANDLED = 1;
- var UNHANDLED = 2;
- 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 (!IS_NODE$1 && typeof PromiseRejectionEvent != 'function') 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;
- // Detect correctness of subclassing with @@species support
- var promise = PromiseConstructor.resolve(1);
- var FakePromise = function (exec) {
- exec(function () { /* empty */ }, function () { /* empty */ });
- };
- var constructor = promise.constructor = {};
- constructor[SPECIES$5] = FakePromise;
- return !(promise.then(function () { /* empty */ }) instanceof FakePromise);
- });
-
- var INCORRECT_ITERATION = FORCED || !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;
- };
-
- var notify$1 = function (promise, state, isReject) {
- if (state.notified) return;
- state.notified = true;
- var chain = state.reactions;
- microtask(function () {
- var value = state.value;
- var ok = state.state == FULFILLED;
- var index = 0;
- // variable length - can't use forEach
- while (chain.length > index) {
- var reaction = chain[index++];
- var handler = ok ? reaction.ok : reaction.fail;
- var resolve = reaction.resolve;
- var reject = reaction.reject;
- var domain = reaction.domain;
- var result, then, exited;
- try {
- if (handler) {
- if (!ok) {
- if (state.rejection === UNHANDLED) onHandleUnhandled(promise, state);
- state.rejection = HANDLED;
- }
- if (handler === true) result = value;
- else {
- if (domain) domain.enter();
- result = handler(value); // can throw
- if (domain) {
- domain.exit();
- exited = true;
- }
- }
- if (result === reaction.promise) {
- reject(TypeError$1('Promise-chain cycle'));
- } else if (then = isThenable(result)) {
- then.call(result, resolve, reject);
- } else resolve(result);
- } else reject(value);
- } catch (error) {
- if (domain && !exited) domain.exit();
- reject(error);
- }
- }
- state.reactions = [];
- state.notified = false;
- if (isReject && !state.rejection) onUnhandled(promise, state);
- });
- };
-
- var dispatchEvent = function (name, promise, reason) {
- var event, handler;
- if (DISPATCH_EVENT) {
- event = document$2.createEvent('Event');
- event.promise = promise;
- event.reason = reason;
- event.initEvent(name, false, true);
- global_1.dispatchEvent(event);
- } else event = { promise: promise, reason: reason };
- if (handler = global_1['on' + name]) handler(event);
- else if (name === UNHANDLED_REJECTION) hostReportErrors('Unhandled promise rejection', reason);
- };
-
- var onUnhandled = function (promise, state) {
- task$1.call(global_1, function () {
- var value = state.value;
- var IS_UNHANDLED = isUnhandled(state);
- var result;
- if (IS_UNHANDLED) {
- result = perform(function () {
- if (IS_NODE$1) {
- process$4.emit('unhandledRejection', value, promise);
- } else dispatchEvent(UNHANDLED_REJECTION, promise, value);
- });
- // Browsers should not trigger `rejectionHandled` event if it was handled here, NodeJS - should
- state.rejection = IS_NODE$1 || isUnhandled(state) ? UNHANDLED : HANDLED;
- if (result.error) throw result.value;
- }
- });
- };
-
- var isUnhandled = function (state) {
- return state.rejection !== HANDLED && !state.parent;
- };
-
- var onHandleUnhandled = function (promise, state) {
- task$1.call(global_1, function () {
- if (IS_NODE$1) {
- process$4.emit('rejectionHandled', promise);
- } else dispatchEvent(REJECTION_HANDLED, promise, state.value);
- });
- };
-
- var bind = function (fn, promise, state, unwrap) {
- return function (value) {
- fn(promise, state, value, unwrap);
- };
- };
-
- var internalReject = function (promise, state, value, unwrap) {
- if (state.done) return;
- state.done = true;
- if (unwrap) state = unwrap;
- state.value = value;
- state.state = REJECTED;
- notify$1(promise, state, true);
- };
-
- var internalResolve = function (promise, state, value, unwrap) {
- if (state.done) return;
- state.done = true;
- if (unwrap) state = unwrap;
- try {
- if (promise === value) throw TypeError$1("Promise can't be resolved itself");
- var then = isThenable(value);
- if (then) {
- microtask(function () {
- var wrapper = { done: false };
- try {
- then.call(value,
- bind(internalResolve, promise, wrapper, state),
- bind(internalReject, promise, wrapper, state)
- );
- } catch (error) {
- internalReject(promise, wrapper, error, state);
- }
- });
- } else {
- state.value = value;
- state.state = FULFILLED;
- notify$1(promise, state, false);
- }
- } catch (error) {
- internalReject(promise, { done: false }, error, state);
- }
- };
-
- // constructor polyfill
- if (FORCED) {
- // 25.4.3.1 Promise(executor)
- PromiseConstructor = function Promise(executor) {
- anInstance(this, PromiseConstructor, PROMISE);
- aFunction$1(executor);
- Internal.call(this);
- var state = getInternalState$3(this);
- try {
- executor(bind(internalResolve, this, state), bind(internalReject, this, state));
- } catch (error) {
- internalReject(this, state, error);
- }
- };
- // eslint-disable-next-line no-unused-vars
- Internal = function Promise(executor) {
- setInternalState$3(this, {
- type: PROMISE,
- done: false,
- notified: false,
- parent: false,
- reactions: [],
- rejection: false,
- state: PENDING,
- value: undefined
- });
- };
- Internal.prototype = redefineAll(PromiseConstructor.prototype, {
- // `Promise.prototype.then` method
- // https://tc39.github.io/ecma262/#sec-promise.prototype.then
- then: function then(onFulfilled, onRejected) {
- var state = getInternalPromiseState(this);
- var reaction = newPromiseCapability$1(speciesConstructor(this, PromiseConstructor));
- reaction.ok = typeof onFulfilled == 'function' ? onFulfilled : true;
- reaction.fail = typeof onRejected == 'function' && onRejected;
- reaction.domain = IS_NODE$1 ? process$4.domain : undefined;
- state.parent = true;
- state.reactions.push(reaction);
- if (state.state != PENDING) notify$1(this, state, false);
- return reaction.promise;
- },
- // `Promise.prototype.catch` method
- // https://tc39.github.io/ecma262/#sec-promise.prototype.catch
- 'catch': function (onRejected) {
- return this.then(undefined, onRejected);
- }
- });
- OwnPromiseCapability = function () {
- var promise = new Internal();
- var state = getInternalState$3(promise);
- this.promise = promise;
- this.resolve = bind(internalResolve, promise, state);
- this.reject = bind(internalReject, promise, state);
- };
- newPromiseCapability.f = newPromiseCapability$1 = function (C) {
- return C === PromiseConstructor || C === PromiseWrapper
- ? new OwnPromiseCapability(C)
- : newGenericPromiseCapability(C);
- };
-
- if ( typeof nativePromiseConstructor == 'function') {
- nativeThen = nativePromiseConstructor.prototype.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 });
-
- // wrap fetch result
- if (typeof $fetch == 'function') _export({ global: true, enumerable: true, forced: true }, {
- // eslint-disable-next-line no-unused-vars
- fetch: function fetch(input /* , init */) {
- return promiseResolve(PromiseConstructor, $fetch.apply(global_1, arguments));
- }
- });
- }
- }
-
- _export({ global: true, wrap: true, forced: FORCED }, {
- Promise: PromiseConstructor
- });
-
- setToStringTag(PromiseConstructor, PROMISE, false);
- setSpecies(PROMISE);
-
- PromiseWrapper = getBuiltIn(PROMISE);
-
- // statics
- _export({ target: PROMISE, stat: true, forced: FORCED }, {
- // `Promise.reject` method
- // https://tc39.github.io/ecma262/#sec-promise.reject
- reject: function reject(r) {
- var capability = newPromiseCapability$1(this);
- capability.reject.call(undefined, r);
- return capability.promise;
- }
- });
-
- _export({ target: PROMISE, stat: true, forced: FORCED }, {
- // `Promise.resolve` method
- // https://tc39.github.io/ecma262/#sec-promise.resolve
- resolve: function resolve(x) {
- return promiseResolve( this, x);
- }
- });
-
- _export({ target: PROMISE, stat: true, forced: INCORRECT_ITERATION }, {
- // `Promise.all` method
- // https://tc39.github.io/ecma262/#sec-promise.all
- all: function all(iterable) {
- var C = this;
- var capability = newPromiseCapability$1(C);
- var resolve = capability.resolve;
- var reject = capability.reject;
- var result = perform(function () {
- var $promiseResolve = aFunction$1(C.resolve);
- var values = [];
- var counter = 0;
- var remaining = 1;
- iterate_1(iterable, function (promise) {
- var index = counter++;
- var alreadyCalled = false;
- values.push(undefined);
- remaining++;
- $promiseResolve.call(C, promise).then(function (value) {
- if (alreadyCalled) return;
- alreadyCalled = true;
- values[index] = value;
- --remaining || resolve(values);
- }, reject);
- });
- --remaining || resolve(values);
- });
- if (result.error) reject(result.value);
- return capability.promise;
- },
- // `Promise.race` method
- // https://tc39.github.io/ecma262/#sec-promise.race
- race: function race(iterable) {
- var C = this;
- var capability = newPromiseCapability$1(C);
- var reject = capability.reject;
- var result = perform(function () {
- var $promiseResolve = aFunction$1(C.resolve);
- iterate_1(iterable, function (promise) {
- $promiseResolve.call(C, promise).then(capability.resolve, reject);
- });
- });
- if (result.error) reject(result.value);
- return capability.promise;
- }
- });
-
- // `RegExp.prototype.flags` getter implementation
- // https://tc39.github.io/ecma262/#sec-get-regexp.prototype.flags
- var regexpFlags = function () {
- var that = anObject(this);
- var result = '';
- if (that.global) result += 'g';
- if (that.ignoreCase) result += 'i';
- if (that.multiline) result += 'm';
- if (that.dotAll) result += 's';
- if (that.unicode) result += 'u';
- if (that.sticky) result += 'y';
- return result;
- };
-
- // babel-minify transpiles RegExp('a', 'y') -> /a/y and it causes SyntaxError,
- // so we use an intermediate function.
- function RE(s, f) {
- return RegExp(s, f);
- }
-
- var UNSUPPORTED_Y = fails(function () {
- // babel-minify transpiles RegExp('a', 'y') -> /a/y and it causes SyntaxError
- var re = RE('a', 'y');
- re.lastIndex = 2;
- return re.exec('abcd') != null;
- });
-
- var BROKEN_CARET = fails(function () {
- // https://bugzilla.mozilla.org/show_bug.cgi?id=773687
- var re = RE('^r', 'gy');
- re.lastIndex = 2;
- return re.exec('str') != null;
- });
-
- var regexpStickyHelpers = {
- UNSUPPORTED_Y: UNSUPPORTED_Y,
- BROKEN_CARET: BROKEN_CARET
- };
-
- 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 patchedExec = nativeExec;
-
- var UPDATES_LAST_INDEX_WRONG = (function () {
- var re1 = /a/;
- var re2 = /b*/g;
- nativeExec.call(re1, 'a');
- nativeExec.call(re2, 'a');
- return re1.lastIndex !== 0 || re2.lastIndex !== 0;
- })();
-
- var UNSUPPORTED_Y$1 = regexpStickyHelpers.UNSUPPORTED_Y || regexpStickyHelpers.BROKEN_CARET;
-
- // nonparticipating capturing group, copied from es5-shim's String#split patch.
- var NPCG_INCLUDED = /()??/.exec('')[1] !== undefined;
-
- var PATCH = UPDATES_LAST_INDEX_WRONG || NPCG_INCLUDED || UNSUPPORTED_Y$1;
-
- if (PATCH) {
- patchedExec = function exec(str) {
- var re = this;
- var lastIndex, reCopy, match, i;
- var sticky = UNSUPPORTED_Y$1 && re.sticky;
- var flags = regexpFlags.call(re);
- var source = re.source;
- var charsAdded = 0;
- var strCopy = str;
-
- if (sticky) {
- flags = flags.replace('y', '');
- if (flags.indexOf('g') === -1) {
- flags += 'g';
- }
-
- strCopy = String(str).slice(re.lastIndex);
- // Support anchored sticky behavior.
- if (re.lastIndex > 0 && (!re.multiline || re.multiline && str[re.lastIndex - 1] !== '\n')) {
- source = '(?: ' + source + ')';
- strCopy = ' ' + strCopy;
- charsAdded++;
- }
- // ^(? + rx + ) is needed, in combination with some str slicing, to
- // simulate the 'y' flag.
- reCopy = new RegExp('^(?:' + source + ')', flags);
- }
-
- if (NPCG_INCLUDED) {
- reCopy = new RegExp('^' + source + '$(?!\\s)', flags);
- }
- if (UPDATES_LAST_INDEX_WRONG) lastIndex = re.lastIndex;
-
- match = nativeExec.call(sticky ? reCopy : re, strCopy);
-
- if (sticky) {
- if (match) {
- match.input = match.input.slice(charsAdded);
- match[0] = match[0].slice(charsAdded);
- match.index = re.lastIndex;
- re.lastIndex += match[0].length;
- } else re.lastIndex = 0;
- } else if (UPDATES_LAST_INDEX_WRONG && match) {
- re.lastIndex = re.global ? match.index + match[0].length : lastIndex;
- }
- if (NPCG_INCLUDED && match && match.length > 1) {
- // Fix browsers whose `exec` methods don't consistently return `undefined`
- // for NPCG, like IE8. NOTE: This doesn' work for /(.?)?/
- nativeReplace.call(match[0], reCopy, function () {
- for (i = 1; i < arguments.length - 2; i++) {
- if (arguments[i] === undefined) match[i] = undefined;
- }
- });
- }
-
- return match;
- };
- }
-
- var regexpExec = patchedExec;
-
- _export({ target: 'RegExp', proto: true, forced: /./.exec !== regexpExec }, {
- exec: regexpExec
- });
-
- var TO_STRING$1 = 'toString';
- var RegExpPrototype = RegExp.prototype;
- var nativeToString = RegExpPrototype[TO_STRING$1];
-
- 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;
-
- // `RegExp.prototype.toString` method
- // https://tc39.github.io/ecma262/#sec-regexp.prototype.tostring
- if (NOT_GENERIC || INCORRECT_NAME) {
- redefine(RegExp.prototype, TO_STRING$1, 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);
- return '/' + p + '/' + f;
- }, { unsafe: true });
- }
-
- // `String.prototype.{ codePointAt, at }` methods implementation
- var createMethod$2 = 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.github.io/ecma262/#sec-string.prototype.codepointat
- codeAt: createMethod$2(false),
- // `String.prototype.at` method
- // https://github.com/mathiasbynens/String.prototype.at
- charAt: createMethod$2(true)
- };
-
- var charAt = stringMultibyte.charAt;
-
-
-
- var STRING_ITERATOR = 'String Iterator';
- var setInternalState$4 = internalState.set;
- var getInternalState$4 = internalState.getterFor(STRING_ITERATOR);
-
- // `String.prototype[@@iterator]` method
- // https://tc39.github.io/ecma262/#sec-string.prototype-@@iterator
- defineIterator(String, 'String', function (iterated) {
- setInternalState$4(this, {
- type: STRING_ITERATOR,
- string: String(iterated),
- index: 0
- });
- // `%StringIteratorPrototype%.next` method
- // https://tc39.github.io/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(string, index);
- state.index += point.length;
- return { value: point, done: false };
- });
-
- // TODO: Remove from `core-js@4` since it's moved to entry points
-
-
-
-
-
-
-
- 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 () {
- 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 fixRegexpWellKnownSymbolLogic = function (KEY, length, exec, sham) {
- var SYMBOL = wellKnownSymbol(KEY);
-
- var DELEGATES_TO_SYMBOL = !fails(function () {
- // String methods call symbol-named RegEp methods
- var O = {};
- O[SYMBOL] = function () { return 7; };
- return ''[KEY](O) != 7;
- });
-
- var DELEGATES_TO_EXEC = DELEGATES_TO_SYMBOL && !fails(function () {
- // Symbol-named RegExp methods call .exec
- var execCalled = false;
- var re = /a/;
-
- if (KEY === 'split') {
- // We can't use real regex here since it causes deoptimization
- // and serious performance degradation in V8
- // https://github.com/zloirock/core-js/issues/306
- re = {};
- // 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.flags = '';
- re[SYMBOL] = /./[SYMBOL];
- }
-
- re.exec = function () { execCalled = true; return null; };
-
- re[SYMBOL]('');
- return !execCalled;
- });
-
- 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)
- ) {
- var nativeRegExpMethod = /./[SYMBOL];
- var methods = exec(SYMBOL, ''[KEY], function (nativeMethod, regexp, str, arg2, forceStringMethod) {
- if (regexp.exec === regexpExec) {
- if (DELEGATES_TO_SYMBOL && !forceStringMethod) {
- // The native String method already delegates to @@method (this
- // polyfilled function), leasing to infinite recursion.
- // We avoid it by directly calling the native @@method method.
- return { done: true, value: nativeRegExpMethod.call(regexp, str, arg2) };
- }
- 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); }
- );
- }
-
- if (sham) createNonEnumerableProperty(RegExp.prototype[SYMBOL], 'sham', true);
- };
-
- var charAt$1 = stringMultibyte.charAt;
-
- // `AdvanceStringIndex` abstract operation
- // https://tc39.github.io/ecma262/#sec-advancestringindex
- var advanceStringIndex = function (S, index, unicode) {
- return index + (unicode ? charAt$1(S, index).length : 1);
- };
-
- // `RegExpExec` abstract operation
- // https://tc39.github.io/ecma262/#sec-regexpexec
- var regexpExecAbstract = function (R, S) {
- var exec = R.exec;
- if (typeof exec === 'function') {
- var result = exec.call(R, S);
- if (typeof result !== 'object') {
- throw TypeError('RegExp exec method returned something other than an Object or null');
- }
- return result;
- }
-
- if (classofRaw(R) !== 'RegExp') {
- throw TypeError('RegExp#exec called on incompatible receiver');
- }
-
- return regexpExec.call(R, S);
- };
-
- var max$2 = Math.max;
- var min$2 = Math.min;
- var floor$2 = Math.floor;
- var SUBSTITUTION_SYMBOLS = /\$([$&'`]|\d\d?|<[^>]*>)/g;
- var SUBSTITUTION_SYMBOLS_NO_NAMED = /\$([$&'`]|\d\d?)/g;
-
- var maybeToString = function (it) {
- return it === undefined ? it : String(it);
- };
-
- // @@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;
- var UNSAFE_SUBSTITUTE = REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE ? '$' : '$0';
-
- return [
- // `String.prototype.replace` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.replace
- function replace(searchValue, replaceValue) {
- var O = requireObjectCoercible(this);
- var replacer = searchValue == undefined ? undefined : searchValue[REPLACE];
- return replacer !== undefined
- ? replacer.call(searchValue, O, replaceValue)
- : nativeReplace.call(String(O), searchValue, replaceValue);
- },
- // `RegExp.prototype[@@replace]` method
- // https://tc39.github.io/ecma262/#sec-regexp.prototype-@@replace
- function (regexp, replaceValue) {
- if (
- (!REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE && REPLACE_KEEPS_$0) ||
- (typeof replaceValue === 'string' && replaceValue.indexOf(UNSAFE_SUBSTITUTE) === -1)
- ) {
- var res = maybeCallNative(nativeReplace, regexp, this, replaceValue);
- if (res.done) return res.value;
- }
-
- var rx = anObject(regexp);
- var S = String(this);
-
- var functionalReplace = typeof replaceValue === 'function';
- if (!functionalReplace) replaceValue = String(replaceValue);
-
- var global = rx.global;
- if (global) {
- var fullUnicode = rx.unicode;
- rx.lastIndex = 0;
- }
- var results = [];
- while (true) {
- var result = regexpExecAbstract(rx, S);
- if (result === null) break;
-
- results.push(result);
- if (!global) break;
-
- var matchStr = String(result[0]);
- if (matchStr === '') rx.lastIndex = advanceStringIndex(S, toLength(rx.lastIndex), fullUnicode);
- }
-
- var accumulatedResult = '';
- var nextSourcePosition = 0;
- for (var i = 0; i < results.length; i++) {
- result = results[i];
-
- var matched = String(result[0]);
- var position = max$2(min$2(toInteger(result.index), S.length), 0);
- var captures = [];
- // NOTE: This is equivalent to
- // captures = result.slice(1).map(maybeToString)
- // but for some reason `nativeSlice.call(result, 1, result.length)` (called in
- // the slice polyfill when slicing native arrays) "doesn't work" in safari 9 and
- // causes a crash (https://pastebin.com/N21QzeQA) when trying to debug it.
- for (var j = 1; j < result.length; j++) captures.push(maybeToString(result[j]));
- var namedCaptures = result.groups;
- if (functionalReplace) {
- var replacerArgs = [matched].concat(captures, position, S);
- if (namedCaptures !== undefined) replacerArgs.push(namedCaptures);
- var replacement = String(replaceValue.apply(undefined, replacerArgs));
- } else {
- replacement = getSubstitution(matched, S, position, captures, namedCaptures, replaceValue);
- }
- if (position >= nextSourcePosition) {
- accumulatedResult += S.slice(nextSourcePosition, position) + replacement;
- nextSourcePosition = position + matched.length;
- }
- }
- return accumulatedResult + S.slice(nextSourcePosition);
- }
- ];
-
- // https://tc39.github.io/ecma262/#sec-getsubstitution
- function getSubstitution(matched, str, position, captures, namedCaptures, replacement) {
- var tailPos = position + matched.length;
- var m = captures.length;
- var symbols = SUBSTITUTION_SYMBOLS_NO_NAMED;
- if (namedCaptures !== undefined) {
- namedCaptures = toObject(namedCaptures);
- symbols = SUBSTITUTION_SYMBOLS;
- }
- return nativeReplace.call(replacement, symbols, function (match, ch) {
- var capture;
- switch (ch.charAt(0)) {
- case '$': return '$';
- case '&': return matched;
- case '`': return str.slice(0, position);
- case "'": return str.slice(tailPos);
- case '<':
- capture = namedCaptures[ch.slice(1, -1)];
- break;
- default: // \d\d?
- var n = +ch;
- if (n === 0) return match;
- if (n > m) {
- var f = floor$2(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;
- }
- capture = captures[n - 1];
- }
- return capture === undefined ? '' : capture;
- });
- }
- });
-
- var MATCH = wellKnownSymbol('match');
-
- // `IsRegExp` abstract operation
- // https://tc39.github.io/ecma262/#sec-isregexp
- var isRegexp = function (it) {
- var isRegExp;
- return isObject(it) && ((isRegExp = it[MATCH]) !== undefined ? !!isRegExp : classofRaw(it) == 'RegExp');
- };
-
- var arrayPush = [].push;
- var min$3 = 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'); });
-
- // @@split logic
- fixRegexpWellKnownSymbolLogic('split', 2, function (SPLIT, nativeSplit, maybeCallNative) {
- var internalSplit;
- if (
- 'abbc'.split(/(b)*/)[1] == 'c' ||
- 'test'.split(/(?:)/, -1).length != 4 ||
- 'ab'.split(/(?:ab)*/).length != 2 ||
- '.'.split(/(.?)(.?)/).length != 4 ||
- '.'.split(/()()/).length > 1 ||
- ''.split(/.?/).length
- ) {
- // based on es5-shim implementation, need to rework it
- internalSplit = function (separator, limit) {
- var string = String(requireObjectCoercible(this));
- var lim = limit === undefined ? MAX_UINT32 : limit >>> 0;
- if (lim === 0) return [];
- if (separator === undefined) return [string];
- // If `separator` is not a regex, use native split
- if (!isRegexp(separator)) {
- return nativeSplit.call(string, separator, lim);
- }
- var output = [];
- var flags = (separator.ignoreCase ? 'i' : '') +
- (separator.multiline ? 'm' : '') +
- (separator.unicode ? 'u' : '') +
- (separator.sticky ? 'y' : '');
- var lastLastIndex = 0;
- // Make `global` and avoid `lastIndex` issues by working with a copy
- var separatorCopy = new RegExp(separator.source, flags + 'g');
- var match, lastIndex, lastLength;
- while (match = regexpExec.call(separatorCopy, string)) {
- lastIndex = separatorCopy.lastIndex;
- if (lastIndex > lastLastIndex) {
- output.push(string.slice(lastLastIndex, match.index));
- if (match.length > 1 && match.index < string.length) arrayPush.apply(output, match.slice(1));
- lastLength = match[0].length;
- lastLastIndex = lastIndex;
- if (output.length >= lim) break;
- }
- if (separatorCopy.lastIndex === match.index) separatorCopy.lastIndex++; // Avoid an infinite loop
- }
- if (lastLastIndex === string.length) {
- if (lastLength || !separatorCopy.test('')) output.push('');
- } else output.push(string.slice(lastLastIndex));
- return output.length > lim ? output.slice(0, lim) : output;
- };
- // Chakra, V8
- } else if ('0'.split(undefined, 0).length) {
- internalSplit = function (separator, limit) {
- return separator === undefined && limit === 0 ? [] : nativeSplit.call(this, separator, limit);
- };
- } else internalSplit = nativeSplit;
-
- return [
- // `String.prototype.split` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.split
- function split(separator, limit) {
- var O = requireObjectCoercible(this);
- var splitter = separator == undefined ? undefined : separator[SPLIT];
- return splitter !== undefined
- ? splitter.call(separator, O, limit)
- : internalSplit.call(String(O), separator, limit);
- },
- // `RegExp.prototype[@@split]` method
- // https://tc39.github.io/ecma262/#sec-regexp.prototype-@@split
- //
- // 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);
- if (res.done) return res.value;
-
- var rx = anObject(regexp);
- var S = String(this);
- var C = speciesConstructor(rx, RegExp);
-
- var unicodeMatching = rx.unicode;
- var flags = (rx.ignoreCase ? 'i' : '') +
- (rx.multiline ? 'm' : '') +
- (rx.unicode ? 'u' : '') +
- (SUPPORTS_Y ? 'y' : 'g');
-
- // ^(? + 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 lim = limit === undefined ? MAX_UINT32 : limit >>> 0;
- if (lim === 0) return [];
- if (S.length === 0) return regexpExecAbstract(splitter, S) === null ? [S] : [];
- var p = 0;
- 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));
- var e;
- if (
- z === null ||
- (e = min$3(toLength(splitter.lastIndex + (SUPPORTS_Y ? 0 : q)), S.length)) === p
- ) {
- q = advanceStringIndex(S, q, unicodeMatching);
- } else {
- A.push(S.slice(p, q));
- if (A.length === lim) return A;
- for (var i = 1; i <= z.length - 1; i++) {
- A.push(z[i]);
- if (A.length === lim) return A;
- }
- q = p = e;
- }
- }
- A.push(S.slice(p));
- return A;
- }
- ];
- }, !SUPPORTS_Y);
-
- // a string of all valid unicode whitespaces
- // eslint-disable-next-line max-len
- var whitespaces = '\u0009\u000A\u000B\u000C\u000D\u0020\u00A0\u1680\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200A\u202F\u205F\u3000\u2028\u2029\uFEFF';
-
- var whitespace = '[' + whitespaces + ']';
- var ltrim = RegExp('^' + whitespace + whitespace + '*');
- var rtrim = RegExp(whitespace + whitespace + '*$');
-
- // `String.prototype.{ trim, trimStart, trimEnd, trimLeft, trimRight }` methods implementation
- var createMethod$3 = function (TYPE) {
- return function ($this) {
- var string = String(requireObjectCoercible($this));
- if (TYPE & 1) string = string.replace(ltrim, '');
- if (TYPE & 2) string = string.replace(rtrim, '');
- return string;
- };
- };
-
- var stringTrim = {
- // `String.prototype.{ trimLeft, trimStart }` methods
- // https://tc39.github.io/ecma262/#sec-string.prototype.trimstart
- start: createMethod$3(1),
- // `String.prototype.{ trimRight, trimEnd }` methods
- // https://tc39.github.io/ecma262/#sec-string.prototype.trimend
- end: createMethod$3(2),
- // `String.prototype.trim` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.trim
- trim: createMethod$3(3)
- };
-
- var non = '\u200B\u0085\u180E';
-
- // check that a method works with the correct list
- // of whitespaces and has a correct name
- var stringTrimForced = function (METHOD_NAME) {
- return fails(function () {
- return !!whitespaces[METHOD_NAME]() || non[METHOD_NAME]() != non || whitespaces[METHOD_NAME].name !== METHOD_NAME;
- });
- };
-
- var $trim = stringTrim.trim;
-
-
- // `String.prototype.trim` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.trim
- _export({ target: 'String', proto: true, forced: stringTrimForced('trim') }, {
- trim: function trim() {
- return $trim(this);
- }
- });
-
- /* eslint-disable no-new */
-
-
-
- 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 () {
- Int8Array$2(1);
- }) || !fails(function () {
- new Int8Array$2(-1);
- }) || !checkCorrectnessOfIteration(function (iterable) {
- new Int8Array$2();
- new Int8Array$2(null);
- new Int8Array$2(1.5);
- 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;
- });
-
- var toPositiveInteger = function (it) {
- var result = toInteger(it);
- if (result < 0) throw RangeError("The argument can't be less than 0");
- return result;
- };
-
- var toOffset = function (it, BYTES) {
- var offset = toPositiveInteger(it);
- if (offset % BYTES) throw RangeError('Wrong offset');
- return offset;
- };
-
- var aTypedArrayConstructor$1 = arrayBufferViewCore.aTypedArrayConstructor;
-
- var typedArrayFrom = function from(source /* , mapfn, thisArg */) {
- var O = toObject(source);
- var argumentsLength = arguments.length;
- var mapfn = argumentsLength > 1 ? arguments[1] : undefined;
- var mapping = mapfn !== undefined;
- var iteratorMethod = getIteratorMethod(O);
- var i, length, result, step, iterator, next;
- if (iteratorMethod != undefined && !isArrayIteratorMethod(iteratorMethod)) {
- iterator = iteratorMethod.call(O);
- next = iterator.next;
- O = [];
- while (!(step = next.call(iterator)).done) {
- O.push(step.value);
- }
- }
- if (mapping && argumentsLength > 2) {
- mapfn = functionBindContext(mapfn, arguments[2], 2);
- }
- length = toLength(O.length);
- result = new (aTypedArrayConstructor$1(this))(length);
- for (i = 0; length > i; i++) {
- result[i] = mapping ? mapfn(O[i], i) : O[i];
- }
- return result;
- };
-
- // makes subclassing work correct for wrapped built-ins
- var inheritIfRequired = function ($this, dummy, Wrapper) {
- var NewTarget, NewTargetPrototype;
- if (
- // it can work only with native `setPrototypeOf`
- objectSetPrototypeOf &&
- // 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) &&
- NewTargetPrototype !== Wrapper.prototype
- ) objectSetPrototypeOf($this, NewTargetPrototype);
- return $this;
- };
-
- var typedArrayConstructor = createCommonjsModule(function (module) {
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- var getOwnPropertyNames = objectGetOwnPropertyNames.f;
-
- var forEach = arrayIteration.forEach;
-
-
-
-
-
-
- var getInternalState = internalState.get;
- var setInternalState = internalState.set;
- var nativeDefineProperty = objectDefineProperty.f;
- var nativeGetOwnPropertyDescriptor = objectGetOwnPropertyDescriptor.f;
- var round = Math.round;
- var RangeError = global_1.RangeError;
- var ArrayBuffer = arrayBuffer.ArrayBuffer;
- var DataView = arrayBuffer.DataView;
- var NATIVE_ARRAY_BUFFER_VIEWS = arrayBufferViewCore.NATIVE_ARRAY_BUFFER_VIEWS;
- var TYPED_ARRAY_TAG = arrayBufferViewCore.TYPED_ARRAY_TAG;
- var TypedArray = arrayBufferViewCore.TypedArray;
- var TypedArrayPrototype = arrayBufferViewCore.TypedArrayPrototype;
- var aTypedArrayConstructor = arrayBufferViewCore.aTypedArrayConstructor;
- var isTypedArray = arrayBufferViewCore.isTypedArray;
- var BYTES_PER_ELEMENT = 'BYTES_PER_ELEMENT';
- var WRONG_LENGTH = 'Wrong length';
-
- var fromList = function (C, list) {
- var index = 0;
- var length = list.length;
- var result = new (aTypedArrayConstructor(C))(length);
- while (length > index) result[index] = list[index++];
- return result;
- };
-
- var addGetter = function (it, key) {
- nativeDefineProperty(it, key, { get: function () {
- return getInternalState(this)[key];
- } });
- };
-
- var isArrayBuffer = function (it) {
- var klass;
- return it instanceof ArrayBuffer || (klass = classof(it)) == 'ArrayBuffer' || klass == 'SharedArrayBuffer';
- };
-
- var isTypedArrayIndex = function (target, key) {
- return isTypedArray(target)
- && typeof key != 'symbol'
- && key in target
- && String(+key) == String(key);
- };
-
- var wrappedGetOwnPropertyDescriptor = function getOwnPropertyDescriptor(target, key) {
- return isTypedArrayIndex(target, key = toPrimitive(key, true))
- ? createPropertyDescriptor(2, target[key])
- : nativeGetOwnPropertyDescriptor(target, key);
- };
-
- 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')
- // TODO: add validation descriptor w/o calling accessors
- && !descriptor.configurable
- && (!has(descriptor, 'writable') || descriptor.writable)
- && (!has(descriptor, 'enumerable') || descriptor.enumerable)
- ) {
- target[key] = descriptor.value;
- return target;
- } return nativeDefineProperty(target, key, descriptor);
- };
-
- if (descriptors) {
- if (!NATIVE_ARRAY_BUFFER_VIEWS) {
- objectGetOwnPropertyDescriptor.f = wrappedGetOwnPropertyDescriptor;
- objectDefineProperty.f = wrappedDefineProperty;
- addGetter(TypedArrayPrototype, 'buffer');
- addGetter(TypedArrayPrototype, 'byteOffset');
- addGetter(TypedArrayPrototype, 'byteLength');
- addGetter(TypedArrayPrototype, 'length');
- }
-
- _export({ target: 'Object', stat: true, forced: !NATIVE_ARRAY_BUFFER_VIEWS }, {
- getOwnPropertyDescriptor: wrappedGetOwnPropertyDescriptor,
- defineProperty: wrappedDefineProperty
- });
-
- module.exports = function (TYPE, wrapper, CLAMPED) {
- var BYTES = TYPE.match(/\d+$/)[0] / 8;
- var CONSTRUCTOR_NAME = TYPE + (CLAMPED ? 'Clamped' : '') + 'Array';
- var GETTER = 'get' + TYPE;
- var SETTER = 'set' + TYPE;
- var NativeTypedArrayConstructor = global_1[CONSTRUCTOR_NAME];
- var TypedArrayConstructor = NativeTypedArrayConstructor;
- var TypedArrayConstructorPrototype = TypedArrayConstructor && TypedArrayConstructor.prototype;
- var exported = {};
-
- var getter = function (that, index) {
- var data = getInternalState(that);
- return data.view[GETTER](index * BYTES + data.byteOffset, true);
- };
-
- var setter = function (that, index, value) {
- var data = getInternalState(that);
- if (CLAMPED) value = (value = round(value)) < 0 ? 0 : value > 0xFF ? 0xFF : value & 0xFF;
- data.view[SETTER](index * BYTES + data.byteOffset, value, true);
- };
-
- var addElement = function (that, index) {
- nativeDefineProperty(that, index, {
- get: function () {
- return getter(this, index);
- },
- set: function (value) {
- return setter(this, index, value);
- },
- enumerable: true
- });
- };
-
- if (!NATIVE_ARRAY_BUFFER_VIEWS) {
- TypedArrayConstructor = wrapper(function (that, data, offset, $length) {
- anInstance(that, TypedArrayConstructor, CONSTRUCTOR_NAME);
- var index = 0;
- var byteOffset = 0;
- var buffer, byteLength, length;
- if (!isObject(data)) {
- length = toIndex(data);
- byteLength = length * BYTES;
- buffer = new ArrayBuffer(byteLength);
- } else if (isArrayBuffer(data)) {
- buffer = data;
- byteOffset = toOffset(offset, BYTES);
- var $len = data.byteLength;
- if ($length === undefined) {
- if ($len % BYTES) throw RangeError(WRONG_LENGTH);
- byteLength = $len - byteOffset;
- if (byteLength < 0) throw RangeError(WRONG_LENGTH);
- } else {
- byteLength = toLength($length) * BYTES;
- if (byteLength + byteOffset > $len) throw RangeError(WRONG_LENGTH);
- }
- length = byteLength / BYTES;
- } else if (isTypedArray(data)) {
- return fromList(TypedArrayConstructor, data);
- } else {
- return typedArrayFrom.call(TypedArrayConstructor, data);
- }
- setInternalState(that, {
- buffer: buffer,
- byteOffset: byteOffset,
- byteLength: byteLength,
- length: length,
- view: new DataView(buffer)
- });
- while (index < length) addElement(that, index++);
- });
-
- if (objectSetPrototypeOf) objectSetPrototypeOf(TypedArrayConstructor, TypedArray);
- TypedArrayConstructorPrototype = TypedArrayConstructor.prototype = objectCreate(TypedArrayPrototype);
- } else if (typedArrayConstructorsRequireWrappers) {
- TypedArrayConstructor = wrapper(function (dummy, data, typedArrayOffset, $length) {
- anInstance(dummy, TypedArrayConstructor, CONSTRUCTOR_NAME);
- return inheritIfRequired(function () {
- if (!isObject(data)) return new NativeTypedArrayConstructor(toIndex(data));
- if (isArrayBuffer(data)) return $length !== undefined
- ? new NativeTypedArrayConstructor(data, toOffset(typedArrayOffset, BYTES), $length)
- : typedArrayOffset !== undefined
- ? new NativeTypedArrayConstructor(data, toOffset(typedArrayOffset, BYTES))
- : new NativeTypedArrayConstructor(data);
- if (isTypedArray(data)) return fromList(TypedArrayConstructor, data);
- return typedArrayFrom.call(TypedArrayConstructor, data);
- }(), dummy, TypedArrayConstructor);
- });
-
- if (objectSetPrototypeOf) objectSetPrototypeOf(TypedArrayConstructor, TypedArray);
- forEach(getOwnPropertyNames(NativeTypedArrayConstructor), function (key) {
- if (!(key in TypedArrayConstructor)) {
- createNonEnumerableProperty(TypedArrayConstructor, key, NativeTypedArrayConstructor[key]);
- }
- });
- TypedArrayConstructor.prototype = TypedArrayConstructorPrototype;
- }
-
- if (TypedArrayConstructorPrototype.constructor !== TypedArrayConstructor) {
- createNonEnumerableProperty(TypedArrayConstructorPrototype, 'constructor', TypedArrayConstructor);
- }
-
- if (TYPED_ARRAY_TAG) {
- createNonEnumerableProperty(TypedArrayConstructorPrototype, TYPED_ARRAY_TAG, CONSTRUCTOR_NAME);
- }
-
- exported[CONSTRUCTOR_NAME] = TypedArrayConstructor;
-
- _export({
- global: true, forced: TypedArrayConstructor != NativeTypedArrayConstructor, sham: !NATIVE_ARRAY_BUFFER_VIEWS
- }, exported);
-
- if (!(BYTES_PER_ELEMENT in TypedArrayConstructor)) {
- createNonEnumerableProperty(TypedArrayConstructor, BYTES_PER_ELEMENT, BYTES);
- }
-
- if (!(BYTES_PER_ELEMENT in TypedArrayConstructorPrototype)) {
- createNonEnumerableProperty(TypedArrayConstructorPrototype, BYTES_PER_ELEMENT, BYTES);
- }
-
- setSpecies(CONSTRUCTOR_NAME);
- };
- } else module.exports = function () { /* empty */ };
- });
-
- // `Uint8Array` constructor
- // https://tc39.github.io/ecma262/#sec-typedarray-objects
- typedArrayConstructor('Uint8', function (init) {
- return function Uint8Array(data, byteOffset, length) {
- return init(this, data, byteOffset, length);
- };
- });
-
- var min$4 = Math.min;
-
- // `Array.prototype.copyWithin` method implementation
- // https://tc39.github.io/ecma262/#sec-array.prototype.copywithin
- 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$4((end === undefined ? len : toAbsoluteIndex(end, len)) - from, len - to);
- var inc = 1;
- if (from < to && to < from + count) {
- inc = -1;
- from += count - 1;
- to += count - 1;
- }
- while (count-- > 0) {
- if (from in O) O[to] = O[from];
- else delete O[to];
- to += inc;
- from += inc;
- } return O;
- };
-
- var aTypedArray$1 = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$1 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.copyWithin` method
- // https://tc39.github.io/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);
- });
-
- var $every = arrayIteration.every;
-
- var aTypedArray$2 = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$2 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.every` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.every
- exportTypedArrayMethod$2('every', function every(callbackfn /* , thisArg */) {
- return $every(aTypedArray$2(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- });
-
- var aTypedArray$3 = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$3 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.fill` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.fill
- // eslint-disable-next-line no-unused-vars
- exportTypedArrayMethod$3('fill', function fill(value /* , start, end */) {
- return arrayFill.apply(aTypedArray$3(this), arguments);
- });
-
- var $filter = arrayIteration.filter;
-
-
- var aTypedArray$4 = arrayBufferViewCore.aTypedArray;
- var aTypedArrayConstructor$2 = arrayBufferViewCore.aTypedArrayConstructor;
- var exportTypedArrayMethod$4 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.filter` method
- // https://tc39.github.io/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);
- var C = speciesConstructor(this, this.constructor);
- var index = 0;
- var length = list.length;
- var result = new (aTypedArrayConstructor$2(C))(length);
- while (length > index) result[index] = list[index++];
- return result;
- });
-
- var $find = arrayIteration.find;
-
- var aTypedArray$5 = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$5 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.find` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.find
- exportTypedArrayMethod$5('find', function find(predicate /* , thisArg */) {
- return $find(aTypedArray$5(this), predicate, arguments.length > 1 ? arguments[1] : undefined);
- });
-
- var $findIndex = arrayIteration.findIndex;
-
- var aTypedArray$6 = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$6 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.findIndex` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.findindex
- exportTypedArrayMethod$6('findIndex', function findIndex(predicate /* , thisArg */) {
- return $findIndex(aTypedArray$6(this), predicate, arguments.length > 1 ? arguments[1] : undefined);
- });
-
- var $forEach$2 = arrayIteration.forEach;
-
- var aTypedArray$7 = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$7 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.forEach` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.foreach
- exportTypedArrayMethod$7('forEach', function forEach(callbackfn /* , thisArg */) {
- $forEach$2(aTypedArray$7(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- });
-
- var $includes = arrayIncludes.includes;
-
- var aTypedArray$8 = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$8 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.includes` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.includes
- exportTypedArrayMethod$8('includes', function includes(searchElement /* , fromIndex */) {
- return $includes(aTypedArray$8(this), searchElement, arguments.length > 1 ? arguments[1] : undefined);
- });
-
- var $indexOf$1 = arrayIncludes.indexOf;
-
- var aTypedArray$9 = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$9 = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.indexOf` method
- // https://tc39.github.io/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);
- });
-
- var ITERATOR$5 = wellKnownSymbol('iterator');
- var Uint8Array$1 = global_1.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$5];
-
- var CORRECT_ITER_NAME = !!nativeTypedArrayIterator
- && (nativeTypedArrayIterator.name == 'values' || nativeTypedArrayIterator.name == undefined);
-
- var typedArrayValues = function values() {
- return arrayValues.call(aTypedArray$a(this));
- };
-
- // `%TypedArray%.prototype.entries` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.entries
- exportTypedArrayMethod$a('entries', function entries() {
- return arrayEntries.call(aTypedArray$a(this));
- });
- // `%TypedArray%.prototype.keys` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.keys
- exportTypedArrayMethod$a('keys', function keys() {
- return arrayKeys.call(aTypedArray$a(this));
- });
- // `%TypedArray%.prototype.values` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.values
- exportTypedArrayMethod$a('values', typedArrayValues, !CORRECT_ITER_NAME);
- // `%TypedArray%.prototype[@@iterator]` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype-@@iterator
- exportTypedArrayMethod$a(ITERATOR$5, typedArrayValues, !CORRECT_ITER_NAME);
-
- var aTypedArray$b = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$b = arrayBufferViewCore.exportTypedArrayMethod;
- var $join = [].join;
-
- // `%TypedArray%.prototype.join` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.join
- // eslint-disable-next-line no-unused-vars
- exportTypedArrayMethod$b('join', function join(separator) {
- return $join.apply(aTypedArray$b(this), arguments);
- });
-
- var min$5 = Math.min;
- var nativeLastIndexOf = [].lastIndexOf;
- var NEGATIVE_ZERO$1 = !!nativeLastIndexOf && 1 / [1].lastIndexOf(1, -0) < 0;
- var STRICT_METHOD$3 = arrayMethodIsStrict('lastIndexOf');
- // For preventing possible almost infinite loop in non-standard implementations, test the forward version of the method
- var USES_TO_LENGTH$4 = arrayMethodUsesToLength('indexOf', { ACCESSORS: true, 1: 0 });
- var FORCED$1 = NEGATIVE_ZERO$1 || !STRICT_METHOD$3 || !USES_TO_LENGTH$4;
-
- // `Array.prototype.lastIndexOf` method implementation
- // https://tc39.github.io/ecma262/#sec-array.prototype.lastindexof
- var arrayLastIndexOf = FORCED$1 ? function lastIndexOf(searchElement /* , fromIndex = @[*-1] */) {
- // convert -0 to +0
- if (NEGATIVE_ZERO$1) return nativeLastIndexOf.apply(this, arguments) || 0;
- var O = toIndexedObject(this);
- var length = toLength(O.length);
- var index = length - 1;
- if (arguments.length > 1) index = min$5(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;
-
- var aTypedArray$c = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$c = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.lastIndexOf` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.lastindexof
- // eslint-disable-next-line no-unused-vars
- exportTypedArrayMethod$c('lastIndexOf', function lastIndexOf(searchElement /* , fromIndex */) {
- return arrayLastIndexOf.apply(aTypedArray$c(this), arguments);
- });
-
- var $map$1 = arrayIteration.map;
-
-
- var aTypedArray$d = arrayBufferViewCore.aTypedArray;
- var aTypedArrayConstructor$3 = arrayBufferViewCore.aTypedArrayConstructor;
- var exportTypedArrayMethod$d = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.map` method
- // https://tc39.github.io/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);
- });
- });
-
- // `Array.prototype.{ reduce, reduceRight }` methods implementation
- var createMethod$4 = function (IS_RIGHT) {
- return function (that, callbackfn, argumentsLength, memo) {
- aFunction$1(callbackfn);
- var O = toObject(that);
- var self = indexedObject(O);
- var length = toLength(O.length);
- var index = IS_RIGHT ? length - 1 : 0;
- var i = IS_RIGHT ? -1 : 1;
- if (argumentsLength < 2) while (true) {
- if (index in self) {
- memo = self[index];
- index += i;
- break;
- }
- index += i;
- if (IS_RIGHT ? index < 0 : length <= index) {
- throw TypeError('Reduce of empty array with no initial value');
- }
- }
- for (;IS_RIGHT ? index >= 0 : length > index; index += i) if (index in self) {
- memo = callbackfn(memo, self[index], index, O);
- }
- return memo;
- };
- };
-
- var arrayReduce = {
- // `Array.prototype.reduce` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.reduce
- left: createMethod$4(false),
- // `Array.prototype.reduceRight` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.reduceright
- right: createMethod$4(true)
- };
-
- var $reduce = arrayReduce.left;
-
- var aTypedArray$e = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$e = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.reduce` method
- // https://tc39.github.io/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);
- });
-
- var $reduceRight = arrayReduce.right;
-
- var aTypedArray$f = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$f = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.reduceRicht` method
- // https://tc39.github.io/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);
- });
-
- var aTypedArray$g = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$g = arrayBufferViewCore.exportTypedArrayMethod;
- var floor$3 = Math.floor;
-
- // `%TypedArray%.prototype.reverse` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.reverse
- exportTypedArrayMethod$g('reverse', function reverse() {
- var that = this;
- var length = aTypedArray$g(that).length;
- var middle = floor$3(length / 2);
- var index = 0;
- var value;
- while (index < middle) {
- value = that[index];
- that[index++] = that[--length];
- that[length] = value;
- } return that;
- });
-
- var aTypedArray$h = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$h = arrayBufferViewCore.exportTypedArrayMethod;
-
- var FORCED$2 = fails(function () {
- // eslint-disable-next-line no-undef
- new Int8Array(1).set({});
- });
-
- // `%TypedArray%.prototype.set` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.set
- exportTypedArrayMethod$h('set', function set(arrayLike /* , offset */) {
- aTypedArray$h(this);
- var offset = toOffset(arguments.length > 1 ? arguments[1] : undefined, 1);
- var length = this.length;
- var src = toObject(arrayLike);
- var len = toLength(src.length);
- var index = 0;
- if (len + offset > length) throw RangeError('Wrong length');
- while (index < len) this[offset + index] = src[index++];
- }, FORCED$2);
-
- var aTypedArray$i = arrayBufferViewCore.aTypedArray;
- var aTypedArrayConstructor$4 = arrayBufferViewCore.aTypedArrayConstructor;
- var exportTypedArrayMethod$i = arrayBufferViewCore.exportTypedArrayMethod;
- var $slice = [].slice;
-
- var FORCED$3 = fails(function () {
- // eslint-disable-next-line no-undef
- new Int8Array(1).slice();
- });
-
- // `%TypedArray%.prototype.slice` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.slice
- exportTypedArrayMethod$i('slice', function slice(start, end) {
- var list = $slice.call(aTypedArray$i(this), start, end);
- var C = speciesConstructor(this, this.constructor);
- var index = 0;
- var length = list.length;
- var result = new (aTypedArrayConstructor$4(C))(length);
- while (length > index) result[index] = list[index++];
- return result;
- }, FORCED$3);
-
- var $some = arrayIteration.some;
-
- var aTypedArray$j = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$j = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.some` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.some
- exportTypedArrayMethod$j('some', function some(callbackfn /* , thisArg */) {
- return $some(aTypedArray$j(this), callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- });
-
- var aTypedArray$k = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$k = arrayBufferViewCore.exportTypedArrayMethod;
- var $sort = [].sort;
-
- // `%TypedArray%.prototype.sort` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.sort
- exportTypedArrayMethod$k('sort', function sort(comparefn) {
- return $sort.call(aTypedArray$k(this), comparefn);
- });
-
- var aTypedArray$l = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$l = arrayBufferViewCore.exportTypedArrayMethod;
-
- // `%TypedArray%.prototype.subarray` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.subarray
- exportTypedArrayMethod$l('subarray', function subarray(begin, end) {
- var O = aTypedArray$l(this);
- var length = O.length;
- var beginIndex = toAbsoluteIndex(begin, length);
- return new (speciesConstructor(O, O.constructor))(
- O.buffer,
- O.byteOffset + beginIndex * O.BYTES_PER_ELEMENT,
- toLength((end === undefined ? length : toAbsoluteIndex(end, length)) - beginIndex)
- );
- });
-
- var Int8Array$3 = global_1.Int8Array;
- var aTypedArray$m = arrayBufferViewCore.aTypedArray;
- var exportTypedArrayMethod$m = arrayBufferViewCore.exportTypedArrayMethod;
- var $toLocaleString = [].toLocaleString;
- var $slice$1 = [].slice;
-
- // iOS Safari 6.x fails here
- var TO_LOCALE_STRING_BUG = !!Int8Array$3 && fails(function () {
- $toLocaleString.call(new Int8Array$3(1));
- });
-
- var FORCED$4 = fails(function () {
- return [1, 2].toLocaleString() != new Int8Array$3([1, 2]).toLocaleString();
- }) || !fails(function () {
- Int8Array$3.prototype.toLocaleString.call([1, 2]);
- });
-
- // `%TypedArray%.prototype.toLocaleString` method
- // https://tc39.github.io/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);
-
- var exportTypedArrayMethod$n = arrayBufferViewCore.exportTypedArrayMethod;
-
-
-
- var Uint8Array$2 = global_1.Uint8Array;
- var Uint8ArrayPrototype = Uint8Array$2 && Uint8Array$2.prototype || {};
- var arrayToString = [].toString;
- var arrayJoin = [].join;
-
- if (fails(function () { arrayToString.call({}); })) {
- arrayToString = function toString() {
- return arrayJoin.call(this);
- };
- }
-
- var IS_NOT_ARRAY_METHOD = Uint8ArrayPrototype.toString != arrayToString;
-
- // `%TypedArray%.prototype.toString` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.prototype.tostring
- exportTypedArrayMethod$n('toString', arrayToString, IS_NOT_ARRAY_METHOD);
-
- // iterable DOM collections
- // flag - `iterable` interface - 'entries', 'keys', 'values', 'forEach' methods
- var domIterables = {
- CSSRuleList: 0,
- CSSStyleDeclaration: 0,
- CSSValueList: 0,
- ClientRectList: 0,
- DOMRectList: 0,
- DOMStringList: 0,
- DOMTokenList: 1,
- DataTransferItemList: 0,
- FileList: 0,
- HTMLAllCollection: 0,
- HTMLCollection: 0,
- HTMLFormElement: 0,
- HTMLSelectElement: 0,
- MediaList: 0,
- MimeTypeArray: 0,
- NamedNodeMap: 0,
- NodeList: 1,
- PaintRequestList: 0,
- Plugin: 0,
- PluginArray: 0,
- SVGLengthList: 0,
- SVGNumberList: 0,
- SVGPathSegList: 0,
- SVGPointList: 0,
- SVGStringList: 0,
- SVGTransformList: 0,
- SourceBufferList: 0,
- StyleSheetList: 0,
- TextTrackCueList: 0,
- TextTrackList: 0,
- TouchList: 0
- };
-
- for (var COLLECTION_NAME in domIterables) {
- var Collection = global_1[COLLECTION_NAME];
- var CollectionPrototype = Collection && Collection.prototype;
- // some Chrome versions have non-configurable methods on DOMTokenList
- if (CollectionPrototype && CollectionPrototype.forEach !== arrayForEach) try {
- createNonEnumerableProperty(CollectionPrototype, 'forEach', arrayForEach);
- } catch (error) {
- CollectionPrototype.forEach = arrayForEach;
- }
- }
-
- var ITERATOR$6 = wellKnownSymbol('iterator');
- var TO_STRING_TAG$4 = wellKnownSymbol('toStringTag');
- var ArrayValues = es_array_iterator.values;
-
- for (var COLLECTION_NAME$1 in domIterables) {
- var Collection$1 = global_1[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$1[ITERATOR$6] !== ArrayValues) try {
- createNonEnumerableProperty(CollectionPrototype$1, ITERATOR$6, ArrayValues);
- } catch (error) {
- CollectionPrototype$1[ITERATOR$6] = ArrayValues;
- }
- if (!CollectionPrototype$1[TO_STRING_TAG$4]) {
- createNonEnumerableProperty(CollectionPrototype$1, TO_STRING_TAG$4, COLLECTION_NAME$1);
- }
- if (domIterables[COLLECTION_NAME$1]) for (var METHOD_NAME in es_array_iterator) {
- // some Chrome versions have non-configurable methods on DOMTokenList
- if (CollectionPrototype$1[METHOD_NAME] !== es_array_iterator[METHOD_NAME]) try {
- createNonEnumerableProperty(CollectionPrototype$1, METHOD_NAME, es_array_iterator[METHOD_NAME]);
- } catch (error) {
- CollectionPrototype$1[METHOD_NAME] = es_array_iterator[METHOD_NAME];
- }
- }
- }
- }
-
- var slice = [].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;
- return scheduler(boundArgs ? function () {
- // eslint-disable-next-line no-new-func
- (typeof handler == 'function' ? handler : Function(handler)).apply(this, args);
- } : handler, timeout);
- };
- };
-
- // ie9- setTimeout & setInterval additional parameters fix
- // https://html.spec.whatwg.org/multipage/timers-and-user-prompts.html#timers
- _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),
- // `setInterval` method
- // https://html.spec.whatwg.org/multipage/timers-and-user-prompts.html#dom-setinterval
- setInterval: wrap$1(global_1.setInterval)
- });
-
- var ITERATOR$7 = wellKnownSymbol('iterator');
-
- var nativeUrl = !fails(function () {
- var url = new URL('b?a=1&b=2&c=3', 'http://a');
- var searchParams = url.searchParams;
- var result = '';
- url.pathname = 'c%20d';
- searchParams.forEach(function (value, key) {
- searchParams['delete']('b');
- result += key + value;
- });
- return (isPure && !url.toJSON)
- || !searchParams.sort
- || url.href !== 'http://a/c%20d?a=1&c=3'
- || searchParams.get('c') !== '3'
- || String(new URLSearchParams('?a=1')) !== 'a=1'
- || !searchParams[ITERATOR$7]
- // throws in Edge
- || new URL('https://a@b').username !== 'a'
- || new URLSearchParams(new URLSearchParams('a=b')).get('a') !== 'b'
- // not punycoded in Edge
- || new URL('http://тест').host !== 'xn--e1aybc'
- // not escaped in Chrome 62-
- || new URL('http://a#б').hash !== '#%D0%B1'
- // fails in Chrome 66-
- || result !== 'a1c3'
- // throws in Safari
- || new URL('http://x', undefined).host !== 'x';
- });
-
- var nativeAssign = Object.assign;
- var defineProperty$7 = Object.defineProperty;
-
- // `Object.assign` method
- // https://tc39.github.io/ecma262/#sec-object.assign
- var objectAssign = !nativeAssign || fails(function () {
- // should have correct order of operations (Edge bug)
- if (descriptors && nativeAssign({ b: 1 }, nativeAssign(defineProperty$7({}, 'a', {
- enumerable: true,
- get: function () {
- defineProperty$7(this, 'b', {
- value: 3,
- enumerable: false
- });
- }
- }), { b: 2 })).b !== 1) return true;
- // should work with symbols and should have deterministic property order (V8 bug)
- var A = {};
- var B = {};
- // eslint-disable-next-line no-undef
- 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;
- }) ? function assign(target, source) { // eslint-disable-line no-unused-vars
- var T = toObject(target);
- var argumentsLength = arguments.length;
- var index = 1;
- var getOwnPropertySymbols = objectGetOwnPropertySymbols.f;
- var propertyIsEnumerable = objectPropertyIsEnumerable.f;
- while (argumentsLength > index) {
- var S = indexedObject(arguments[index++]);
- var keys = getOwnPropertySymbols ? objectKeys(S).concat(getOwnPropertySymbols(S)) : objectKeys(S);
- var length = keys.length;
- var j = 0;
- var key;
- while (length > j) {
- key = keys[j++];
- if (!descriptors || propertyIsEnumerable.call(S, key)) T[key] = S[key];
- }
- } return T;
- } : nativeAssign;
-
- // `Array.from` method implementation
- // https://tc39.github.io/ecma262/#sec-array.from
- var arrayFrom = function from(arrayLike /* , mapfn = undefined, thisArg = undefined */) {
- var O = toObject(arrayLike);
- var C = typeof this == 'function' ? this : Array;
- var argumentsLength = arguments.length;
- var mapfn = argumentsLength > 1 ? arguments[1] : undefined;
- var mapping = mapfn !== undefined;
- var iteratorMethod = getIteratorMethod(O);
- var index = 0;
- var length, result, step, iterator, next, value;
- if (mapping) mapfn = functionBindContext(mapfn, argumentsLength > 2 ? arguments[2] : undefined, 2);
- // if the target is not iterable or it's an array with the default iterator - use a simple case
- if (iteratorMethod != undefined && !(C == Array && isArrayIteratorMethod(iteratorMethod))) {
- iterator = iteratorMethod.call(O);
- next = iterator.next;
- result = new C();
- for (;!(step = next.call(iterator)).done; index++) {
- value = mapping ? callWithSafeIterationClosing(iterator, mapfn, [step.value, index], true) : step.value;
- createProperty(result, index, value);
- }
- } else {
- length = toLength(O.length);
- result = new C(length);
- for (;length > index; index++) {
- value = mapping ? mapfn(O[index], index) : O[index];
- createProperty(result, index, value);
- }
- }
- result.length = index;
- return result;
- };
-
- // based on https://github.com/bestiejs/punycode.js/blob/master/punycode.js
- var maxInt = 2147483647; // aka. 0x7FFFFFFF or 2^31-1
- var base = 36;
- var tMin = 1;
- var tMax = 26;
- var skew = 38;
- var damp = 700;
- var initialBias = 72;
- var initialN = 128; // 0x80
- var delimiter = '-'; // '\x2D'
- var regexNonASCII = /[^\0-\u007E]/; // non-ASCII chars
- var regexSeparators = /[.\u3002\uFF0E\uFF61]/g; // RFC 3490 separators
- var OVERFLOW_ERROR = 'Overflow: input needs wider integers to process';
- var baseMinusTMin = base - tMin;
- var floor$4 = Math.floor;
- var stringFromCharCode = String.fromCharCode;
-
- /**
- * Creates an array containing the numeric code points of each Unicode
- * character in the string. While JavaScript uses UCS-2 internally,
- * this function will convert a pair of surrogate halves (each of which
- * UCS-2 exposes as separate characters) into a single code point,
- * matching UTF-16.
- */
- var ucs2decode = function (string) {
- var output = [];
- var counter = 0;
- var length = string.length;
- while (counter < length) {
- var value = string.charCodeAt(counter++);
- if (value >= 0xD800 && value <= 0xDBFF && counter < length) {
- // It's a high surrogate, and there is a next character.
- var extra = string.charCodeAt(counter++);
- if ((extra & 0xFC00) == 0xDC00) { // Low surrogate.
- output.push(((value & 0x3FF) << 10) + (extra & 0x3FF) + 0x10000);
- } else {
- // It's an unmatched surrogate; only append this code unit, in case the
- // next code unit is the high surrogate of a surrogate pair.
- output.push(value);
- counter--;
- }
- } else {
- output.push(value);
- }
- }
- return output;
- };
-
- /**
- * Converts a digit/integer into a basic code point.
- */
- var digitToBasic = function (digit) {
- // 0..25 map to ASCII a..z or A..Z
- // 26..35 map to ASCII 0..9
- return digit + 22 + 75 * (digit < 26);
- };
-
- /**
- * Bias adaptation function as per section 3.4 of RFC 3492.
- * https://tools.ietf.org/html/rfc3492#section-3.4
- */
- var adapt = function (delta, numPoints, firstTime) {
- var k = 0;
- delta = firstTime ? floor$4(delta / damp) : delta >> 1;
- delta += floor$4(delta / numPoints);
- for (; delta > baseMinusTMin * tMax >> 1; k += base) {
- delta = floor$4(delta / baseMinusTMin);
- }
- return floor$4(k + (baseMinusTMin + 1) * delta / (delta + skew));
- };
-
- /**
- * Converts a string of Unicode symbols (e.g. a domain name label) to a
- * Punycode string of ASCII-only symbols.
- */
- // eslint-disable-next-line max-statements
- var encode = function (input) {
- var output = [];
-
- // Convert the input in UCS-2 to an array of Unicode code points.
- input = ucs2decode(input);
-
- // Cache the length.
- var inputLength = input.length;
-
- // Initialize the state.
- var n = initialN;
- var delta = 0;
- var bias = initialBias;
- var i, currentValue;
-
- // Handle the basic code points.
- for (i = 0; i < input.length; i++) {
- currentValue = input[i];
- if (currentValue < 0x80) {
- output.push(stringFromCharCode(currentValue));
- }
- }
-
- var basicLength = output.length; // number of basic code points.
- var handledCPCount = basicLength; // number of code points that have been handled;
-
- // Finish the basic string with a delimiter unless it's empty.
- if (basicLength) {
- output.push(delimiter);
- }
-
- // Main encoding loop:
- while (handledCPCount < inputLength) {
- // All non-basic code points < n have been handled already. Find the next larger one:
- var m = maxInt;
- for (i = 0; i < input.length; i++) {
- currentValue = input[i];
- if (currentValue >= n && currentValue < m) {
- m = currentValue;
- }
- }
-
- // Increase `delta` enough to advance the decoder's <n,i> state to <m,0>, but guard against overflow.
- var handledCPCountPlusOne = handledCPCount + 1;
- if (m - n > floor$4((maxInt - delta) / handledCPCountPlusOne)) {
- throw RangeError(OVERFLOW_ERROR);
- }
-
- delta += (m - n) * handledCPCountPlusOne;
- n = m;
-
- for (i = 0; i < input.length; i++) {
- currentValue = input[i];
- if (currentValue < n && ++delta > maxInt) {
- throw RangeError(OVERFLOW_ERROR);
- }
- if (currentValue == n) {
- // Represent delta as a generalized variable-length integer.
- var q = delta;
- for (var k = base; /* no condition */; k += base) {
- var t = k <= bias ? tMin : (k >= bias + tMax ? tMax : k - bias);
- if (q < t) break;
- var qMinusT = q - t;
- var baseMinusT = base - t;
- output.push(stringFromCharCode(digitToBasic(t + qMinusT % baseMinusT)));
- q = floor$4(qMinusT / baseMinusT);
- }
-
- output.push(stringFromCharCode(digitToBasic(q)));
- bias = adapt(delta, handledCPCountPlusOne, handledCPCount == basicLength);
- delta = 0;
- ++handledCPCount;
- }
- }
-
- ++delta;
- ++n;
- }
- return output.join('');
- };
-
- var stringPunycodeToAscii = function (input) {
- var encoded = [];
- var labels = input.toLowerCase().replace(regexSeparators, '\u002E').split('.');
- var i, label;
- for (i = 0; i < labels.length; i++) {
- label = labels[i];
- encoded.push(regexNonASCII.test(label) ? 'xn--' + encode(label) : label);
- }
- return encoded.join('.');
- };
-
- var getIterator = function (it) {
- var iteratorMethod = getIteratorMethod(it);
- if (typeof iteratorMethod != 'function') {
- throw TypeError(String(it) + ' is not iterable');
- } return anObject(iteratorMethod.call(it));
- };
-
- // TODO: in core-js@4, move /modules/ dependencies to public entries for better optimization by tools like `preset-env`
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- var $fetch$1 = getBuiltIn('fetch');
- var Headers = getBuiltIn('Headers');
- var ITERATOR$8 = wellKnownSymbol('iterator');
- var URL_SEARCH_PARAMS = 'URLSearchParams';
- var URL_SEARCH_PARAMS_ITERATOR = URL_SEARCH_PARAMS + 'Iterator';
- var setInternalState$5 = internalState.set;
- var getInternalParamsState = internalState.getterFor(URL_SEARCH_PARAMS);
- var getInternalIteratorState = internalState.getterFor(URL_SEARCH_PARAMS_ITERATOR);
-
- var plus = /\+/g;
- var sequences = Array(4);
-
- var percentSequence = function (bytes) {
- return sequences[bytes - 1] || (sequences[bytes - 1] = RegExp('((?:%[\\da-f]{2}){' + bytes + '})', 'gi'));
- };
-
- var percentDecode = function (sequence) {
- try {
- return decodeURIComponent(sequence);
- } catch (error) {
- return sequence;
- }
- };
-
- var deserialize = function (it) {
- var result = it.replace(plus, ' ');
- var bytes = 4;
- try {
- return decodeURIComponent(result);
- } catch (error) {
- while (bytes) {
- result = result.replace(percentSequence(bytes--), percentDecode);
- }
- return result;
- }
- };
-
- var find = /[!'()~]|%20/g;
-
- var replace = {
- '!': '%21',
- "'": '%27',
- '(': '%28',
- ')': '%29',
- '~': '%7E',
- '%20': '+'
- };
-
- var replacer = function (match) {
- return replace[match];
- };
-
- var serialize = function (it) {
- return encodeURIComponent(it).replace(find, replacer);
- };
-
- var parseSearchParams = function (result, query) {
- if (query) {
- var attributes = query.split('&');
- var index = 0;
- var attribute, entry;
- while (index < attributes.length) {
- attribute = attributes[index++];
- if (attribute.length) {
- entry = attribute.split('=');
- result.push({
- key: deserialize(entry.shift()),
- value: deserialize(entry.join('='))
- });
- }
- }
- }
- };
-
- var updateSearchParams = function (query) {
- this.entries.length = 0;
- parseSearchParams(this.entries, query);
- };
-
- var validateArgumentsLength = function (passed, required) {
- if (passed < required) throw TypeError('Not enough arguments');
- };
-
- var URLSearchParamsIterator = createIteratorConstructor(function Iterator(params, kind) {
- setInternalState$5(this, {
- type: URL_SEARCH_PARAMS_ITERATOR,
- iterator: getIterator(getInternalParamsState(params).entries),
- kind: kind
- });
- }, 'Iterator', function next() {
- var state = getInternalIteratorState(this);
- var kind = state.kind;
- var step = state.iterator.next();
- var entry = step.value;
- if (!step.done) {
- step.value = kind === 'keys' ? entry.key : kind === 'values' ? entry.value : [entry.key, entry.value];
- } return step;
- });
-
- // `URLSearchParams` constructor
- // https://url.spec.whatwg.org/#interface-urlsearchparams
- var URLSearchParamsConstructor = function URLSearchParams(/* init */) {
- anInstance(this, URLSearchParamsConstructor, URL_SEARCH_PARAMS);
- var init = arguments.length > 0 ? arguments[0] : undefined;
- var that = this;
- var entries = [];
- var iteratorMethod, iterator, next, step, entryIterator, entryNext, first, second, key;
-
- setInternalState$5(that, {
- type: URL_SEARCH_PARAMS,
- entries: entries,
- updateURL: function () { /* empty */ },
- updateSearchParams: updateSearchParams
- });
-
- if (init !== undefined) {
- if (isObject(init)) {
- iteratorMethod = getIteratorMethod(init);
- if (typeof iteratorMethod === 'function') {
- iterator = iteratorMethod.call(init);
- next = iterator.next;
- while (!(step = next.call(iterator)).done) {
- entryIterator = getIterator(anObject(step.value));
- entryNext = entryIterator.next;
- if (
- (first = entryNext.call(entryIterator)).done ||
- (second = entryNext.call(entryIterator)).done ||
- !entryNext.call(entryIterator).done
- ) 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 {
- parseSearchParams(entries, typeof init === 'string' ? init.charAt(0) === '?' ? init.slice(1) : init : init + '');
- }
- }
- };
-
- var URLSearchParamsPrototype = URLSearchParamsConstructor.prototype;
-
- redefineAll(URLSearchParamsPrototype, {
- // `URLSearchParams.prototype.appent` method
- // https://url.spec.whatwg.org/#dom-urlsearchparams-append
- append: function append(name, value) {
- validateArgumentsLength(arguments.length, 2);
- var state = getInternalParamsState(this);
- state.entries.push({ key: name + '', value: value + '' });
- state.updateURL();
- },
- // `URLSearchParams.prototype.delete` method
- // https://url.spec.whatwg.org/#dom-urlsearchparams-delete
- 'delete': function (name) {
- validateArgumentsLength(arguments.length, 1);
- var state = getInternalParamsState(this);
- var entries = state.entries;
- var key = name + '';
- var index = 0;
- while (index < entries.length) {
- if (entries[index].key === key) entries.splice(index, 1);
- else index++;
- }
- state.updateURL();
- },
- // `URLSearchParams.prototype.get` method
- // https://url.spec.whatwg.org/#dom-urlsearchparams-get
- get: function get(name) {
- validateArgumentsLength(arguments.length, 1);
- var entries = getInternalParamsState(this).entries;
- var key = name + '';
- var index = 0;
- for (; index < entries.length; index++) {
- if (entries[index].key === key) return entries[index].value;
- }
- return null;
- },
- // `URLSearchParams.prototype.getAll` method
- // https://url.spec.whatwg.org/#dom-urlsearchparams-getall
- getAll: function getAll(name) {
- validateArgumentsLength(arguments.length, 1);
- var entries = getInternalParamsState(this).entries;
- var key = name + '';
- var result = [];
- var index = 0;
- for (; index < entries.length; index++) {
- if (entries[index].key === key) result.push(entries[index].value);
- }
- return result;
- },
- // `URLSearchParams.prototype.has` method
- // https://url.spec.whatwg.org/#dom-urlsearchparams-has
- has: function has(name) {
- validateArgumentsLength(arguments.length, 1);
- var entries = getInternalParamsState(this).entries;
- var key = name + '';
- var index = 0;
- while (index < entries.length) {
- if (entries[index++].key === key) return true;
- }
- return false;
- },
- // `URLSearchParams.prototype.set` method
- // https://url.spec.whatwg.org/#dom-urlsearchparams-set
- set: function set(name, value) {
- validateArgumentsLength(arguments.length, 1);
- var state = getInternalParamsState(this);
- var entries = state.entries;
- var found = false;
- var key = name + '';
- var val = value + '';
- var index = 0;
- var entry;
- for (; index < entries.length; index++) {
- entry = entries[index];
- if (entry.key === key) {
- if (found) entries.splice(index--, 1);
- else {
- found = true;
- entry.value = val;
- }
- }
- }
- if (!found) entries.push({ key: key, value: val });
- state.updateURL();
- },
- // `URLSearchParams.prototype.sort` method
- // https://url.spec.whatwg.org/#dom-urlsearchparams-sort
- sort: function sort() {
- var state = getInternalParamsState(this);
- var entries = state.entries;
- // Array#sort is not stable in some engines
- var slice = entries.slice();
- var entry, entriesIndex, sliceIndex;
- entries.length = 0;
- for (sliceIndex = 0; sliceIndex < slice.length; sliceIndex++) {
- entry = slice[sliceIndex];
- for (entriesIndex = 0; entriesIndex < sliceIndex; entriesIndex++) {
- if (entries[entriesIndex].key > entry.key) {
- entries.splice(entriesIndex, 0, entry);
- break;
- }
- }
- if (entriesIndex === sliceIndex) entries.push(entry);
- }
- state.updateURL();
- },
- // `URLSearchParams.prototype.forEach` method
- forEach: function forEach(callback /* , thisArg */) {
- var entries = getInternalParamsState(this).entries;
- var boundFunction = functionBindContext(callback, arguments.length > 1 ? arguments[1] : undefined, 3);
- var index = 0;
- var entry;
- while (index < entries.length) {
- entry = entries[index++];
- boundFunction(entry.value, entry.key, this);
- }
- },
- // `URLSearchParams.prototype.keys` method
- keys: function keys() {
- return new URLSearchParamsIterator(this, 'keys');
- },
- // `URLSearchParams.prototype.values` method
- values: function values() {
- return new URLSearchParamsIterator(this, 'values');
- },
- // `URLSearchParams.prototype.entries` method
- entries: function entries() {
- return new URLSearchParamsIterator(this, 'entries');
- }
- }, { enumerable: true });
-
- // `URLSearchParams.prototype[@@iterator]` method
- redefine(URLSearchParamsPrototype, ITERATOR$8, URLSearchParamsPrototype.entries);
-
- // `URLSearchParams.prototype.toString` method
- // https://url.spec.whatwg.org/#urlsearchparams-stringification-behavior
- redefine(URLSearchParamsPrototype, 'toString', function toString() {
- var entries = getInternalParamsState(this).entries;
- var result = [];
- var index = 0;
- var entry;
- while (index < entries.length) {
- entry = entries[index++];
- result.push(serialize(entry.key) + '=' + serialize(entry.value));
- } return result.join('&');
- }, { enumerable: true });
-
- setToStringTag(URLSearchParamsConstructor, URL_SEARCH_PARAMS);
-
- _export({ global: true, forced: !nativeUrl }, {
- URLSearchParams: URLSearchParamsConstructor
- });
-
- // 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') {
- _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)) {
- body = init.body;
- if (classof(body) === URL_SEARCH_PARAMS) {
- headers = init.headers ? new Headers(init.headers) : new Headers();
- if (!headers.has('content-type')) {
- headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');
- }
- init = objectCreate(init, {
- body: createPropertyDescriptor(0, String(body)),
- headers: createPropertyDescriptor(0, headers)
- });
- }
- }
- args.push(init);
- } return $fetch$1.apply(this, args);
- }
- });
- }
-
- var web_urlSearchParams = {
- URLSearchParams: URLSearchParamsConstructor,
- getState: getInternalParamsState
- };
-
- // TODO: in core-js@4, move /modules/ dependencies to public entries for better optimization by tools like `preset-env`
-
-
-
-
-
-
-
-
-
-
-
- var codeAt = stringMultibyte.codeAt;
-
-
-
-
-
- var NativeURL = global_1.URL;
- var URLSearchParams$1 = web_urlSearchParams.URLSearchParams;
- var getInternalSearchParamsState = web_urlSearchParams.getState;
- var setInternalState$6 = internalState.set;
- var getInternalURLState = internalState.getterFor('URL');
- var floor$5 = Math.floor;
- var pow$1 = Math.pow;
-
- var INVALID_AUTHORITY = 'Invalid authority';
- var INVALID_SCHEME = 'Invalid scheme';
- var INVALID_HOST = 'Invalid host';
- var INVALID_PORT = 'Invalid port';
-
- var ALPHA = /[A-Za-z]/;
- var ALPHANUMERIC = /[\d+-.A-Za-z]/;
- var DIGIT = /\d/;
- var HEX_START = /^(0x|0X)/;
- var OCT = /^[0-7]+$/;
- var DEC = /^\d+$/;
- var HEX = /^[\dA-Fa-f]+$/;
- // eslint-disable-next-line no-control-regex
- var FORBIDDEN_HOST_CODE_POINT = /[\u0000\u0009\u000A\u000D #%/:?@[\\]]/;
- // eslint-disable-next-line no-control-regex
- var FORBIDDEN_HOST_CODE_POINT_EXCLUDING_PERCENT = /[\u0000\u0009\u000A\u000D #/:?@[\\]]/;
- // eslint-disable-next-line no-control-regex
- var LEADING_AND_TRAILING_C0_CONTROL_OR_SPACE = /^[\u0000-\u001F ]+|[\u0000-\u001F ]+$/g;
- // eslint-disable-next-line no-control-regex
- var TAB_AND_NEW_LINE = /[\u0009\u000A\u000D]/g;
- var EOF;
-
- var parseHost = function (url, input) {
- var result, codePoints, index;
- if (input.charAt(0) == '[') {
- if (input.charAt(input.length - 1) != ']') return INVALID_HOST;
- result = parseIPv6(input.slice(1, -1));
- if (!result) return INVALID_HOST;
- url.host = result;
- // opaque host
- } else if (!isSpecial(url)) {
- if (FORBIDDEN_HOST_CODE_POINT_EXCLUDING_PERCENT.test(input)) return INVALID_HOST;
- result = '';
- codePoints = arrayFrom(input);
- for (index = 0; index < codePoints.length; index++) {
- result += percentEncode(codePoints[index], C0ControlPercentEncodeSet);
- }
- url.host = result;
- } else {
- input = stringPunycodeToAscii(input);
- if (FORBIDDEN_HOST_CODE_POINT.test(input)) return INVALID_HOST;
- result = parseIPv4(input);
- if (result === null) return INVALID_HOST;
- url.host = result;
- }
- };
-
- var parseIPv4 = function (input) {
- var parts = input.split('.');
- var partsLength, numbers, index, part, radix, number, ipv4;
- if (parts.length && parts[parts.length - 1] == '') {
- parts.pop();
- }
- partsLength = parts.length;
- if (partsLength > 4) return input;
- numbers = [];
- for (index = 0; index < partsLength; index++) {
- part = parts[index];
- if (part == '') return input;
- radix = 10;
- if (part.length > 1 && part.charAt(0) == '0') {
- radix = HEX_START.test(part) ? 16 : 8;
- part = part.slice(radix == 8 ? 1 : 2);
- }
- if (part === '') {
- number = 0;
- } else {
- if (!(radix == 10 ? DEC : radix == 8 ? OCT : HEX).test(part)) return input;
- number = parseInt(part, radix);
- }
- numbers.push(number);
- }
- for (index = 0; index < partsLength; index++) {
- number = numbers[index];
- if (index == partsLength - 1) {
- if (number >= pow$1(256, 5 - partsLength)) return null;
- } else if (number > 255) return null;
- }
- ipv4 = numbers.pop();
- for (index = 0; index < numbers.length; index++) {
- ipv4 += numbers[index] * pow$1(256, 3 - index);
- }
- return ipv4;
- };
-
- // eslint-disable-next-line max-statements
- var parseIPv6 = function (input) {
- var address = [0, 0, 0, 0, 0, 0, 0, 0];
- var pieceIndex = 0;
- var compress = null;
- var pointer = 0;
- var value, length, numbersSeen, ipv4Piece, number, swaps, swap;
-
- var char = function () {
- return input.charAt(pointer);
- };
-
- if (char() == ':') {
- if (input.charAt(1) != ':') return;
- pointer += 2;
- pieceIndex++;
- compress = pieceIndex;
- }
- while (char()) {
- if (pieceIndex == 8) return;
- if (char() == ':') {
- if (compress !== null) return;
- pointer++;
- pieceIndex++;
- compress = pieceIndex;
- continue;
- }
- value = length = 0;
- while (length < 4 && HEX.test(char())) {
- value = value * 16 + parseInt(char(), 16);
- pointer++;
- length++;
- }
- if (char() == '.') {
- if (length == 0) return;
- pointer -= length;
- if (pieceIndex > 6) return;
- numbersSeen = 0;
- while (char()) {
- ipv4Piece = null;
- if (numbersSeen > 0) {
- if (char() == '.' && numbersSeen < 4) pointer++;
- else return;
- }
- if (!DIGIT.test(char())) return;
- while (DIGIT.test(char())) {
- number = parseInt(char(), 10);
- if (ipv4Piece === null) ipv4Piece = number;
- else if (ipv4Piece == 0) return;
- else ipv4Piece = ipv4Piece * 10 + number;
- if (ipv4Piece > 255) return;
- pointer++;
- }
- address[pieceIndex] = address[pieceIndex] * 256 + ipv4Piece;
- numbersSeen++;
- if (numbersSeen == 2 || numbersSeen == 4) pieceIndex++;
- }
- if (numbersSeen != 4) return;
- break;
- } else if (char() == ':') {
- pointer++;
- if (!char()) return;
- } else if (char()) return;
- address[pieceIndex++] = value;
- }
- if (compress !== null) {
- swaps = pieceIndex - compress;
- pieceIndex = 7;
- while (pieceIndex != 0 && swaps > 0) {
- swap = address[pieceIndex];
- address[pieceIndex--] = address[compress + swaps - 1];
- address[compress + --swaps] = swap;
- }
- } else if (pieceIndex != 8) return;
- return address;
- };
-
- var findLongestZeroSequence = function (ipv6) {
- var maxIndex = null;
- var maxLength = 1;
- var currStart = null;
- var currLength = 0;
- var index = 0;
- for (; index < 8; index++) {
- if (ipv6[index] !== 0) {
- if (currLength > maxLength) {
- maxIndex = currStart;
- maxLength = currLength;
- }
- currStart = null;
- currLength = 0;
- } else {
- if (currStart === null) currStart = index;
- ++currLength;
- }
- }
- if (currLength > maxLength) {
- maxIndex = currStart;
- maxLength = currLength;
- }
- return maxIndex;
- };
-
- var serializeHost = function (host) {
- var result, index, compress, ignore0;
- // ipv4
- if (typeof host == 'number') {
- result = [];
- for (index = 0; index < 4; index++) {
- result.unshift(host % 256);
- host = floor$5(host / 256);
- } return result.join('.');
- // ipv6
- } else if (typeof host == 'object') {
- result = '';
- compress = findLongestZeroSequence(host);
- for (index = 0; index < 8; index++) {
- if (ignore0 && host[index] === 0) continue;
- if (ignore0) ignore0 = false;
- if (compress === index) {
- result += index ? ':' : '::';
- ignore0 = true;
- } else {
- result += host[index].toString(16);
- if (index < 7) result += ':';
- }
- }
- return '[' + result + ']';
- } return host;
- };
-
- var C0ControlPercentEncodeSet = {};
- var fragmentPercentEncodeSet = objectAssign({}, C0ControlPercentEncodeSet, {
- ' ': 1, '"': 1, '<': 1, '>': 1, '`': 1
- });
- var pathPercentEncodeSet = objectAssign({}, fragmentPercentEncodeSet, {
- '#': 1, '?': 1, '{': 1, '}': 1
- });
- var userinfoPercentEncodeSet = objectAssign({}, pathPercentEncodeSet, {
- '/': 1, ':': 1, ';': 1, '=': 1, '@': 1, '[': 1, '\\': 1, ']': 1, '^': 1, '|': 1
- });
-
- var percentEncode = function (char, set) {
- var code = codeAt(char, 0);
- return code > 0x20 && code < 0x7F && !has(set, char) ? char : encodeURIComponent(char);
- };
-
- var specialSchemes = {
- ftp: 21,
- file: null,
- http: 80,
- https: 443,
- ws: 80,
- wss: 443
- };
-
- var isSpecial = function (url) {
- return has(specialSchemes, url.scheme);
- };
-
- var includesCredentials = function (url) {
- return url.username != '' || url.password != '';
- };
-
- var cannotHaveUsernamePasswordPort = function (url) {
- return !url.host || url.cannotBeABaseURL || url.scheme == 'file';
- };
-
- var isWindowsDriveLetter = function (string, normalized) {
- var second;
- return string.length == 2 && ALPHA.test(string.charAt(0))
- && ((second = string.charAt(1)) == ':' || (!normalized && second == '|'));
- };
-
- var startsWithWindowsDriveLetter = function (string) {
- var third;
- return string.length > 1 && isWindowsDriveLetter(string.slice(0, 2)) && (
- string.length == 2 ||
- ((third = string.charAt(2)) === '/' || third === '\\' || third === '?' || third === '#')
- );
- };
-
- var shortenURLsPath = function (url) {
- var path = url.path;
- var pathSize = path.length;
- if (pathSize && (url.scheme != 'file' || pathSize != 1 || !isWindowsDriveLetter(path[0], true))) {
- path.pop();
- }
- };
-
- var isSingleDot = function (segment) {
- return segment === '.' || segment.toLowerCase() === '%2e';
- };
-
- var isDoubleDot = function (segment) {
- segment = segment.toLowerCase();
- return segment === '..' || segment === '%2e.' || segment === '.%2e' || segment === '%2e%2e';
- };
-
- // States:
- var SCHEME_START = {};
- var SCHEME = {};
- var NO_SCHEME = {};
- var SPECIAL_RELATIVE_OR_AUTHORITY = {};
- var PATH_OR_AUTHORITY = {};
- var RELATIVE = {};
- var RELATIVE_SLASH = {};
- var SPECIAL_AUTHORITY_SLASHES = {};
- var SPECIAL_AUTHORITY_IGNORE_SLASHES = {};
- var AUTHORITY = {};
- var HOST = {};
- var HOSTNAME = {};
- var PORT = {};
- var FILE = {};
- var FILE_SLASH = {};
- var FILE_HOST = {};
- var PATH_START = {};
- var PATH = {};
- var CANNOT_BE_A_BASE_URL_PATH = {};
- var QUERY = {};
- var FRAGMENT = {};
-
- // eslint-disable-next-line max-statements
- var parseURL = function (url, input, stateOverride, base) {
- var state = stateOverride || SCHEME_START;
- var pointer = 0;
- var buffer = '';
- var seenAt = false;
- var seenBracket = false;
- var seenPasswordToken = false;
- var codePoints, char, bufferCodePoints, failure;
-
- if (!stateOverride) {
- url.scheme = '';
- url.username = '';
- url.password = '';
- url.host = null;
- url.port = null;
- url.path = [];
- url.query = null;
- url.fragment = null;
- url.cannotBeABaseURL = false;
- input = input.replace(LEADING_AND_TRAILING_C0_CONTROL_OR_SPACE, '');
- }
-
- input = input.replace(TAB_AND_NEW_LINE, '');
-
- codePoints = arrayFrom(input);
-
- while (pointer <= codePoints.length) {
- char = codePoints[pointer];
- switch (state) {
- case SCHEME_START:
- if (char && ALPHA.test(char)) {
- buffer += char.toLowerCase();
- state = SCHEME;
- } else if (!stateOverride) {
- state = NO_SCHEME;
- continue;
- } else return INVALID_SCHEME;
- break;
-
- case SCHEME:
- if (char && (ALPHANUMERIC.test(char) || char == '+' || char == '-' || char == '.')) {
- buffer += char.toLowerCase();
- } else if (char == ':') {
- if (stateOverride && (
- (isSpecial(url) != has(specialSchemes, buffer)) ||
- (buffer == 'file' && (includesCredentials(url) || url.port !== null)) ||
- (url.scheme == 'file' && !url.host)
- )) return;
- url.scheme = buffer;
- if (stateOverride) {
- if (isSpecial(url) && specialSchemes[url.scheme] == url.port) url.port = null;
- return;
- }
- buffer = '';
- if (url.scheme == 'file') {
- state = FILE;
- } else if (isSpecial(url) && base && base.scheme == url.scheme) {
- state = SPECIAL_RELATIVE_OR_AUTHORITY;
- } else if (isSpecial(url)) {
- state = SPECIAL_AUTHORITY_SLASHES;
- } else if (codePoints[pointer + 1] == '/') {
- state = PATH_OR_AUTHORITY;
- pointer++;
- } else {
- url.cannotBeABaseURL = true;
- url.path.push('');
- state = CANNOT_BE_A_BASE_URL_PATH;
- }
- } else if (!stateOverride) {
- buffer = '';
- state = NO_SCHEME;
- pointer = 0;
- continue;
- } else return INVALID_SCHEME;
- break;
-
- case NO_SCHEME:
- if (!base || (base.cannotBeABaseURL && char != '#')) return INVALID_SCHEME;
- if (base.cannotBeABaseURL && char == '#') {
- url.scheme = base.scheme;
- url.path = base.path.slice();
- url.query = base.query;
- url.fragment = '';
- url.cannotBeABaseURL = true;
- state = FRAGMENT;
- break;
- }
- state = base.scheme == 'file' ? FILE : RELATIVE;
- continue;
-
- case SPECIAL_RELATIVE_OR_AUTHORITY:
- if (char == '/' && codePoints[pointer + 1] == '/') {
- state = SPECIAL_AUTHORITY_IGNORE_SLASHES;
- pointer++;
- } else {
- state = RELATIVE;
- continue;
- } break;
-
- case PATH_OR_AUTHORITY:
- if (char == '/') {
- state = AUTHORITY;
- break;
- } else {
- state = PATH;
- continue;
- }
-
- case RELATIVE:
- url.scheme = base.scheme;
- if (char == EOF) {
- url.username = base.username;
- url.password = base.password;
- url.host = base.host;
- url.port = base.port;
- url.path = base.path.slice();
- url.query = base.query;
- } else if (char == '/' || (char == '\\' && isSpecial(url))) {
- state = RELATIVE_SLASH;
- } else if (char == '?') {
- url.username = base.username;
- url.password = base.password;
- url.host = base.host;
- url.port = base.port;
- url.path = base.path.slice();
- url.query = '';
- state = QUERY;
- } else if (char == '#') {
- url.username = base.username;
- url.password = base.password;
- url.host = base.host;
- url.port = base.port;
- url.path = base.path.slice();
- url.query = base.query;
- url.fragment = '';
- state = FRAGMENT;
- } else {
- url.username = base.username;
- url.password = base.password;
- url.host = base.host;
- url.port = base.port;
- url.path = base.path.slice();
- url.path.pop();
- state = PATH;
- continue;
- } break;
-
- case RELATIVE_SLASH:
- if (isSpecial(url) && (char == '/' || char == '\\')) {
- state = SPECIAL_AUTHORITY_IGNORE_SLASHES;
- } else if (char == '/') {
- state = AUTHORITY;
- } else {
- url.username = base.username;
- url.password = base.password;
- url.host = base.host;
- url.port = base.port;
- state = PATH;
- continue;
- } break;
-
- case SPECIAL_AUTHORITY_SLASHES:
- state = SPECIAL_AUTHORITY_IGNORE_SLASHES;
- if (char != '/' || buffer.charAt(pointer + 1) != '/') continue;
- pointer++;
- break;
-
- case SPECIAL_AUTHORITY_IGNORE_SLASHES:
- if (char != '/' && char != '\\') {
- state = AUTHORITY;
- continue;
- } break;
-
- case AUTHORITY:
- if (char == '@') {
- if (seenAt) buffer = '%40' + buffer;
- seenAt = true;
- bufferCodePoints = arrayFrom(buffer);
- for (var i = 0; i < bufferCodePoints.length; i++) {
- var codePoint = bufferCodePoints[i];
- if (codePoint == ':' && !seenPasswordToken) {
- seenPasswordToken = true;
- continue;
- }
- var encodedCodePoints = percentEncode(codePoint, userinfoPercentEncodeSet);
- if (seenPasswordToken) url.password += encodedCodePoints;
- else url.username += encodedCodePoints;
- }
- buffer = '';
- } else if (
- char == EOF || char == '/' || char == '?' || char == '#' ||
- (char == '\\' && isSpecial(url))
- ) {
- if (seenAt && buffer == '') return INVALID_AUTHORITY;
- pointer -= arrayFrom(buffer).length + 1;
- buffer = '';
- state = HOST;
- } else buffer += char;
- break;
-
- case HOST:
- case HOSTNAME:
- if (stateOverride && url.scheme == 'file') {
- state = FILE_HOST;
- continue;
- } else if (char == ':' && !seenBracket) {
- if (buffer == '') return INVALID_HOST;
- failure = parseHost(url, buffer);
- if (failure) return failure;
- buffer = '';
- state = PORT;
- if (stateOverride == HOSTNAME) return;
- } else if (
- char == EOF || char == '/' || char == '?' || char == '#' ||
- (char == '\\' && isSpecial(url))
- ) {
- if (isSpecial(url) && buffer == '') return INVALID_HOST;
- if (stateOverride && buffer == '' && (includesCredentials(url) || url.port !== null)) return;
- failure = parseHost(url, buffer);
- if (failure) return failure;
- buffer = '';
- state = PATH_START;
- if (stateOverride) return;
- continue;
- } else {
- if (char == '[') seenBracket = true;
- else if (char == ']') seenBracket = false;
- buffer += char;
- } break;
-
- case PORT:
- if (DIGIT.test(char)) {
- buffer += char;
- } else if (
- char == EOF || char == '/' || char == '?' || char == '#' ||
- (char == '\\' && isSpecial(url)) ||
- stateOverride
- ) {
- if (buffer != '') {
- var port = parseInt(buffer, 10);
- if (port > 0xFFFF) return INVALID_PORT;
- url.port = (isSpecial(url) && port === specialSchemes[url.scheme]) ? null : port;
- buffer = '';
- }
- if (stateOverride) return;
- state = PATH_START;
- continue;
- } else return INVALID_PORT;
- break;
-
- case FILE:
- url.scheme = 'file';
- if (char == '/' || char == '\\') state = FILE_SLASH;
- else if (base && base.scheme == 'file') {
- if (char == EOF) {
- url.host = base.host;
- url.path = base.path.slice();
- url.query = base.query;
- } else if (char == '?') {
- url.host = base.host;
- url.path = base.path.slice();
- url.query = '';
- state = QUERY;
- } else if (char == '#') {
- url.host = base.host;
- url.path = base.path.slice();
- url.query = base.query;
- url.fragment = '';
- state = FRAGMENT;
- } else {
- if (!startsWithWindowsDriveLetter(codePoints.slice(pointer).join(''))) {
- url.host = base.host;
- url.path = base.path.slice();
- shortenURLsPath(url);
- }
- state = PATH;
- continue;
- }
- } else {
- state = PATH;
- continue;
- } break;
-
- case FILE_SLASH:
- if (char == '/' || char == '\\') {
- state = FILE_HOST;
- break;
- }
- if (base && base.scheme == 'file' && !startsWithWindowsDriveLetter(codePoints.slice(pointer).join(''))) {
- if (isWindowsDriveLetter(base.path[0], true)) url.path.push(base.path[0]);
- else url.host = base.host;
- }
- state = PATH;
- continue;
-
- case FILE_HOST:
- if (char == EOF || char == '/' || char == '\\' || char == '?' || char == '#') {
- if (!stateOverride && isWindowsDriveLetter(buffer)) {
- state = PATH;
- } else if (buffer == '') {
- url.host = '';
- if (stateOverride) return;
- state = PATH_START;
- } else {
- failure = parseHost(url, buffer);
- if (failure) return failure;
- if (url.host == 'localhost') url.host = '';
- if (stateOverride) return;
- buffer = '';
- state = PATH_START;
- } continue;
- } else buffer += char;
- break;
-
- case PATH_START:
- if (isSpecial(url)) {
- state = PATH;
- if (char != '/' && char != '\\') continue;
- } else if (!stateOverride && char == '?') {
- url.query = '';
- state = QUERY;
- } else if (!stateOverride && char == '#') {
- url.fragment = '';
- state = FRAGMENT;
- } else if (char != EOF) {
- state = PATH;
- if (char != '/') continue;
- } break;
-
- case PATH:
- if (
- char == EOF || char == '/' ||
- (char == '\\' && isSpecial(url)) ||
- (!stateOverride && (char == '?' || char == '#'))
- ) {
- if (isDoubleDot(buffer)) {
- shortenURLsPath(url);
- if (char != '/' && !(char == '\\' && isSpecial(url))) {
- url.path.push('');
- }
- } else if (isSingleDot(buffer)) {
- if (char != '/' && !(char == '\\' && isSpecial(url))) {
- url.path.push('');
- }
- } else {
- if (url.scheme == 'file' && !url.path.length && isWindowsDriveLetter(buffer)) {
- if (url.host) url.host = '';
- buffer = buffer.charAt(0) + ':'; // normalize windows drive letter
- }
- url.path.push(buffer);
- }
- buffer = '';
- if (url.scheme == 'file' && (char == EOF || char == '?' || char == '#')) {
- while (url.path.length > 1 && url.path[0] === '') {
- url.path.shift();
- }
- }
- if (char == '?') {
- url.query = '';
- state = QUERY;
- } else if (char == '#') {
- url.fragment = '';
- state = FRAGMENT;
- }
- } else {
- buffer += percentEncode(char, pathPercentEncodeSet);
- } break;
-
- case CANNOT_BE_A_BASE_URL_PATH:
- if (char == '?') {
- url.query = '';
- state = QUERY;
- } else if (char == '#') {
- url.fragment = '';
- state = FRAGMENT;
- } else if (char != EOF) {
- url.path[0] += percentEncode(char, C0ControlPercentEncodeSet);
- } break;
-
- case QUERY:
- if (!stateOverride && char == '#') {
- url.fragment = '';
- state = FRAGMENT;
- } else if (char != EOF) {
- if (char == "'" && isSpecial(url)) url.query += '%27';
- else if (char == '#') url.query += '%23';
- else url.query += percentEncode(char, C0ControlPercentEncodeSet);
- } break;
-
- case FRAGMENT:
- if (char != EOF) url.fragment += percentEncode(char, fragmentPercentEncodeSet);
- break;
- }
-
- pointer++;
- }
- };
-
- // `URL` constructor
- // https://url.spec.whatwg.org/#url-class
- var URLConstructor = function URL(url /* , base */) {
- 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 baseState, failure;
- if (base !== undefined) {
- if (base instanceof URLConstructor) baseState = getInternalURLState(base);
- else {
- failure = parseURL(baseState = {}, String(base));
- if (failure) throw TypeError(failure);
- }
- }
- failure = parseURL(state, urlString, null, baseState);
- if (failure) throw TypeError(failure);
- var searchParams = state.searchParams = new URLSearchParams$1();
- var searchParamsState = getInternalSearchParamsState(searchParams);
- searchParamsState.updateSearchParams(state.query);
- searchParamsState.updateURL = function () {
- state.query = String(searchParams) || null;
- };
- if (!descriptors) {
- that.href = serializeURL.call(that);
- that.origin = getOrigin.call(that);
- that.protocol = getProtocol.call(that);
- that.username = getUsername.call(that);
- that.password = getPassword.call(that);
- that.host = getHost.call(that);
- that.hostname = getHostname.call(that);
- that.port = getPort.call(that);
- that.pathname = getPathname.call(that);
- that.search = getSearch.call(that);
- that.searchParams = getSearchParams.call(that);
- that.hash = getHash.call(that);
- }
- };
-
- var URLPrototype = URLConstructor.prototype;
-
- var serializeURL = function () {
- var url = getInternalURLState(this);
- var scheme = url.scheme;
- var username = url.username;
- var password = url.password;
- var host = url.host;
- var port = url.port;
- var path = url.path;
- var query = url.query;
- var fragment = url.fragment;
- var output = scheme + ':';
- if (host !== null) {
- output += '//';
- if (includesCredentials(url)) {
- output += username + (password ? ':' + password : '') + '@';
- }
- output += serializeHost(host);
- if (port !== null) output += ':' + port;
- } else if (scheme == 'file') output += '//';
- output += url.cannotBeABaseURL ? path[0] : path.length ? '/' + path.join('/') : '';
- if (query !== null) output += '?' + query;
- if (fragment !== null) output += '#' + fragment;
- return output;
- };
-
- var getOrigin = function () {
- var url = getInternalURLState(this);
- var scheme = url.scheme;
- var port = url.port;
- if (scheme == 'blob') try {
- return new URL(scheme.path[0]).origin;
- } catch (error) {
- return 'null';
- }
- if (scheme == 'file' || !isSpecial(url)) return 'null';
- return scheme + '://' + serializeHost(url.host) + (port !== null ? ':' + port : '');
- };
-
- var getProtocol = function () {
- return getInternalURLState(this).scheme + ':';
- };
-
- var getUsername = function () {
- return getInternalURLState(this).username;
- };
-
- var getPassword = function () {
- return getInternalURLState(this).password;
- };
-
- var getHost = function () {
- var url = getInternalURLState(this);
- var host = url.host;
- var port = url.port;
- return host === null ? ''
- : port === null ? serializeHost(host)
- : serializeHost(host) + ':' + port;
- };
-
- var getHostname = function () {
- var host = getInternalURLState(this).host;
- return host === null ? '' : serializeHost(host);
- };
-
- var getPort = function () {
- var port = getInternalURLState(this).port;
- return port === null ? '' : String(port);
- };
-
- var getPathname = function () {
- var url = getInternalURLState(this);
- var path = url.path;
- return url.cannotBeABaseURL ? path[0] : path.length ? '/' + path.join('/') : '';
- };
-
- var getSearch = function () {
- var query = getInternalURLState(this).query;
- return query ? '?' + query : '';
- };
-
- var getSearchParams = function () {
- return getInternalURLState(this).searchParams;
- };
-
- var getHash = function () {
- var fragment = getInternalURLState(this).fragment;
- return fragment ? '#' + fragment : '';
- };
-
- var accessorDescriptor = function (getter, setter) {
- return { get: getter, set: setter, configurable: true, enumerable: true };
- };
-
- if (descriptors) {
- objectDefineProperties(URLPrototype, {
- // `URL.prototype.href` accessors pair
- // https://url.spec.whatwg.org/#dom-url-href
- href: accessorDescriptor(serializeURL, function (href) {
- var url = getInternalURLState(this);
- var urlString = String(href);
- var failure = parseURL(url, urlString);
- if (failure) throw TypeError(failure);
- getInternalSearchParamsState(url.searchParams).updateSearchParams(url.query);
- }),
- // `URL.prototype.origin` getter
- // https://url.spec.whatwg.org/#dom-url-origin
- origin: accessorDescriptor(getOrigin),
- // `URL.prototype.protocol` accessors pair
- // https://url.spec.whatwg.org/#dom-url-protocol
- protocol: accessorDescriptor(getProtocol, function (protocol) {
- var url = getInternalURLState(this);
- parseURL(url, String(protocol) + ':', SCHEME_START);
- }),
- // `URL.prototype.username` accessors pair
- // https://url.spec.whatwg.org/#dom-url-username
- username: accessorDescriptor(getUsername, function (username) {
- var url = getInternalURLState(this);
- var codePoints = arrayFrom(String(username));
- if (cannotHaveUsernamePasswordPort(url)) return;
- url.username = '';
- for (var i = 0; i < codePoints.length; i++) {
- url.username += percentEncode(codePoints[i], userinfoPercentEncodeSet);
- }
- }),
- // `URL.prototype.password` accessors pair
- // https://url.spec.whatwg.org/#dom-url-password
- password: accessorDescriptor(getPassword, function (password) {
- var url = getInternalURLState(this);
- var codePoints = arrayFrom(String(password));
- if (cannotHaveUsernamePasswordPort(url)) return;
- url.password = '';
- for (var i = 0; i < codePoints.length; i++) {
- url.password += percentEncode(codePoints[i], userinfoPercentEncodeSet);
- }
- }),
- // `URL.prototype.host` accessors pair
- // https://url.spec.whatwg.org/#dom-url-host
- host: accessorDescriptor(getHost, function (host) {
- var url = getInternalURLState(this);
- if (url.cannotBeABaseURL) return;
- parseURL(url, String(host), HOST);
- }),
- // `URL.prototype.hostname` accessors pair
- // https://url.spec.whatwg.org/#dom-url-hostname
- hostname: accessorDescriptor(getHostname, function (hostname) {
- var url = getInternalURLState(this);
- if (url.cannotBeABaseURL) return;
- parseURL(url, String(hostname), HOSTNAME);
- }),
- // `URL.prototype.port` accessors pair
- // https://url.spec.whatwg.org/#dom-url-port
- port: accessorDescriptor(getPort, function (port) {
- var url = getInternalURLState(this);
- if (cannotHaveUsernamePasswordPort(url)) return;
- port = String(port);
- if (port == '') url.port = null;
- else parseURL(url, port, PORT);
- }),
- // `URL.prototype.pathname` accessors pair
- // https://url.spec.whatwg.org/#dom-url-pathname
- pathname: accessorDescriptor(getPathname, function (pathname) {
- var url = getInternalURLState(this);
- if (url.cannotBeABaseURL) return;
- url.path = [];
- parseURL(url, pathname + '', PATH_START);
- }),
- // `URL.prototype.search` accessors pair
- // https://url.spec.whatwg.org/#dom-url-search
- search: accessorDescriptor(getSearch, function (search) {
- var url = getInternalURLState(this);
- search = String(search);
- if (search == '') {
- url.query = null;
- } else {
- if ('?' == search.charAt(0)) search = search.slice(1);
- url.query = '';
- parseURL(url, search, QUERY);
- }
- getInternalSearchParamsState(url.searchParams).updateSearchParams(url.query);
- }),
- // `URL.prototype.searchParams` getter
- // https://url.spec.whatwg.org/#dom-url-searchparams
- searchParams: accessorDescriptor(getSearchParams),
- // `URL.prototype.hash` accessors pair
- // https://url.spec.whatwg.org/#dom-url-hash
- hash: accessorDescriptor(getHash, function (hash) {
- var url = getInternalURLState(this);
- hash = String(hash);
- if (hash == '') {
- url.fragment = null;
- return;
- }
- if ('#' == hash.charAt(0)) hash = hash.slice(1);
- url.fragment = '';
- parseURL(url, hash, FRAGMENT);
- })
- });
- }
-
- // `URL.prototype.toJSON` method
- // https://url.spec.whatwg.org/#dom-url-tojson
- redefine(URLPrototype, 'toJSON', function toJSON() {
- return serializeURL.call(this);
- }, { enumerable: true });
-
- // `URL.prototype.toString` method
- // https://url.spec.whatwg.org/#URL-stringification-behavior
- redefine(URLPrototype, 'toString', function toString() {
- return serializeURL.call(this);
- }, { enumerable: true });
-
- if (NativeURL) {
- var nativeCreateObjectURL = NativeURL.createObjectURL;
- var nativeRevokeObjectURL = NativeURL.revokeObjectURL;
- // `URL.createObjectURL` method
- // https://developer.mozilla.org/en-US/docs/Web/API/URL/createObjectURL
- // eslint-disable-next-line no-unused-vars
- if (nativeCreateObjectURL) redefine(URLConstructor, 'createObjectURL', function createObjectURL(blob) {
- return nativeCreateObjectURL.apply(NativeURL, arguments);
- });
- // `URL.revokeObjectURL` method
- // https://developer.mozilla.org/en-US/docs/Web/API/URL/revokeObjectURL
- // eslint-disable-next-line no-unused-vars
- if (nativeRevokeObjectURL) redefine(URLConstructor, 'revokeObjectURL', function revokeObjectURL(url) {
- return nativeRevokeObjectURL.apply(NativeURL, arguments);
- });
- }
-
- setToStringTag(URLConstructor, 'URL');
-
- _export({ global: true, forced: !nativeUrl, sham: !descriptors }, {
- URL: URLConstructor
- });
-
- function _typeof(obj) {
- "@babel/helpers - typeof";
-
- if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") {
- _typeof = function (obj) {
- return typeof obj;
- };
- } else {
- _typeof = function (obj) {
- return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj;
- };
- }
-
- return _typeof(obj);
- }
-
- function _classCallCheck(instance, Constructor) {
- if (!(instance instanceof Constructor)) {
- throw new TypeError("Cannot call a class as a function");
- }
- }
-
- 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);
- }
- }
-
- function _createClass(Constructor, protoProps, staticProps) {
- if (protoProps) _defineProperties(Constructor.prototype, protoProps);
- if (staticProps) _defineProperties(Constructor, staticProps);
- return Constructor;
- }
-
- function _defineProperty(obj, key, value) {
- if (key in obj) {
- Object.defineProperty(obj, key, {
- value: value,
- enumerable: true,
- configurable: true,
- writable: true
- });
- } else {
- obj[key] = value;
- }
-
- return obj;
- }
-
- function _slicedToArray(arr, i) {
- return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest();
- }
-
- function _toConsumableArray(arr) {
- return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread();
- }
-
- function _arrayWithoutHoles(arr) {
- if (Array.isArray(arr)) return _arrayLikeToArray(arr);
- }
-
- function _arrayWithHoles(arr) {
- if (Array.isArray(arr)) return arr;
- }
-
- function _iterableToArray(iter) {
- if (typeof Symbol !== "undefined" && Symbol.iterator in Object(iter)) return Array.from(iter);
- }
-
- function _iterableToArrayLimit(arr, i) {
- if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return;
- var _arr = [];
- var _n = true;
- var _d = false;
- var _e = undefined;
-
- try {
- for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) {
- _arr.push(_s.value);
-
- if (i && _arr.length === i) break;
- }
- } catch (err) {
- _d = true;
- _e = err;
- } finally {
- try {
- if (!_n && _i["return"] != null) _i["return"]();
- } finally {
- if (_d) throw _e;
- }
- }
-
- return _arr;
- }
-
- function _unsupportedIterableToArray(o, minLen) {
- if (!o) return;
- if (typeof o === "string") return _arrayLikeToArray(o, minLen);
- var n = Object.prototype.toString.call(o).slice(8, -1);
- if (n === "Object" && o.constructor) n = o.constructor.name;
- if (n === "Map" || n === "Set") return Array.from(o);
- if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen);
- }
-
- function _arrayLikeToArray(arr, len) {
- if (len == null || len > arr.length) len = arr.length;
-
- for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i];
-
- return arr2;
- }
-
- function _nonIterableSpread() {
- throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
- }
-
- function _nonIterableRest() {
- throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
- }
-
- function _createForOfIteratorHelper(o, allowArrayLike) {
- var it;
-
- if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) {
- if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") {
- if (it) o = it;
- var i = 0;
-
- var F = function () {};
-
- return {
- s: F,
- n: function () {
- if (i >= o.length) return {
- done: true
- };
- return {
- done: false,
- value: o[i++]
- };
- },
- e: function (e) {
- throw e;
- },
- f: F
- };
- }
-
- throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
- }
-
- var normalCompletion = true,
- didErr = false,
- err;
- return {
- s: function () {
- it = o[Symbol.iterator]();
- },
- n: function () {
- var step = it.next();
- normalCompletion = step.done;
- return step;
- },
- e: function (e) {
- didErr = true;
- err = e;
- },
- f: function () {
- try {
- if (!normalCompletion && it.return != null) it.return();
- } finally {
- if (didErr) throw err;
- }
- }
- };
- }
-
- var global$1 = typeof globalThis !== 'undefined' && globalThis || typeof self !== 'undefined' && self || typeof global$1 !== 'undefined' && global$1;
- var support = {
- searchParams: 'URLSearchParams' in global$1,
- iterable: 'Symbol' in global$1 && 'iterator' in Symbol,
- blob: 'FileReader' in global$1 && 'Blob' in global$1 && function () {
- try {
- new Blob();
- return true;
- } catch (e) {
- return false;
- }
- }(),
- formData: 'FormData' in global$1,
- arrayBuffer: 'ArrayBuffer' in global$1
- };
-
- function isDataView(obj) {
- return obj && DataView.prototype.isPrototypeOf(obj);
- }
-
- if (support.arrayBuffer) {
- var viewClasses = ['[object Int8Array]', '[object Uint8Array]', '[object Uint8ClampedArray]', '[object Int16Array]', '[object Uint16Array]', '[object Int32Array]', '[object Uint32Array]', '[object Float32Array]', '[object Float64Array]'];
-
- var isArrayBufferView = ArrayBuffer.isView || function (obj) {
- return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1;
- };
- }
-
- function normalizeName(name) {
- if (typeof name !== 'string') {
- name = String(name);
- }
-
- if (/[^a-z0-9\-#$%&'*+.^_`|~!]/i.test(name) || name === '') {
- throw new TypeError('Invalid character in header field name');
- }
-
- return name.toLowerCase();
- }
-
- function normalizeValue(value) {
- if (typeof value !== 'string') {
- value = String(value);
- }
-
- return value;
- } // Build a destructive iterator for the value list
-
-
- function iteratorFor(items) {
- var iterator = {
- next: function next() {
- var value = items.shift();
- return {
- done: value === undefined,
- value: value
- };
- }
- };
-
- if (support.iterable) {
- iterator[Symbol.iterator] = function () {
- return iterator;
- };
- }
-
- return iterator;
- }
-
- function Headers$1(headers) {
- this.map = {};
-
- if (headers instanceof Headers$1) {
- headers.forEach(function (value, name) {
- this.append(name, value);
- }, this);
- } else if (Array.isArray(headers)) {
- headers.forEach(function (header) {
- this.append(header[0], header[1]);
- }, this);
- } else if (headers) {
- Object.getOwnPropertyNames(headers).forEach(function (name) {
- this.append(name, headers[name]);
- }, this);
- }
- }
-
- Headers$1.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) {
- delete this.map[normalizeName(name)];
- };
-
- Headers$1.prototype.get = function (name) {
- name = normalizeName(name);
- return this.has(name) ? this.map[name] : null;
- };
-
- Headers$1.prototype.has = function (name) {
- return this.map.hasOwnProperty(normalizeName(name));
- };
-
- Headers$1.prototype.set = function (name, value) {
- this.map[normalizeName(name)] = normalizeValue(value);
- };
-
- Headers$1.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 () {
- var items = [];
- this.forEach(function (value, name) {
- items.push(name);
- });
- return iteratorFor(items);
- };
-
- Headers$1.prototype.values = function () {
- var items = [];
- this.forEach(function (value) {
- items.push(value);
- });
- return iteratorFor(items);
- };
-
- Headers$1.prototype.entries = function () {
- var items = [];
- this.forEach(function (value, name) {
- items.push([name, value]);
- });
- return iteratorFor(items);
- };
-
- if (support.iterable) {
- Headers$1.prototype[Symbol.iterator] = Headers$1.prototype.entries;
- }
-
- function consumed(body) {
- if (body.bodyUsed) {
- return Promise.reject(new TypeError('Already read'));
- }
-
- body.bodyUsed = true;
- }
-
- function fileReaderReady(reader) {
- return new Promise(function (resolve, reject) {
- reader.onload = function () {
- resolve(reader.result);
- };
-
- reader.onerror = function () {
- reject(reader.error);
- };
- });
- }
-
- function readBlobAsArrayBuffer(blob) {
- var reader = new FileReader();
- var promise = fileReaderReady(reader);
- reader.readAsArrayBuffer(blob);
- return promise;
- }
-
- function readBlobAsText(blob) {
- var reader = new FileReader();
- var promise = fileReaderReady(reader);
- reader.readAsText(blob);
- return promise;
- }
-
- function readArrayBufferAsText(buf) {
- var view = new Uint8Array(buf);
- var chars = new Array(view.length);
-
- for (var i = 0; i < view.length; i++) {
- chars[i] = String.fromCharCode(view[i]);
- }
-
- return chars.join('');
- }
-
- function bufferClone(buf) {
- if (buf.slice) {
- return buf.slice(0);
- } else {
- var view = new Uint8Array(buf.byteLength);
- view.set(new Uint8Array(buf));
- return view.buffer;
- }
- }
-
- function Body() {
- this.bodyUsed = false;
-
- this._initBody = function (body) {
- /*
- fetch-mock wraps the Response object in an ES6 Proxy to
- provide useful test harness features such as flush. However, on
- ES5 browsers without fetch or Proxy support pollyfills must be used;
- the proxy-pollyfill is unable to proxy an attribute unless it exists
- on the object before the Proxy is created. This change ensures
- Response.bodyUsed exists on the instance, while maintaining the
- semantic of setting Request.bodyUsed in the constructor before
- _initBody is called.
- */
- this.bodyUsed = this.bodyUsed;
- this._bodyInit = body;
-
- if (!body) {
- this._bodyText = '';
- } else if (typeof body === 'string') {
- this._bodyText = body;
- } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {
- this._bodyBlob = body;
- } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {
- this._bodyFormData = body;
- } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
- this._bodyText = body.toString();
- } else if (support.arrayBuffer && support.blob && isDataView(body)) {
- this._bodyArrayBuffer = bufferClone(body.buffer); // IE 10-11 can't handle a DataView body.
-
- this._bodyInit = new Blob([this._bodyArrayBuffer]);
- } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {
- this._bodyArrayBuffer = bufferClone(body);
- } else {
- this._bodyText = body = Object.prototype.toString.call(body);
- }
-
- if (!this.headers.get('content-type')) {
- if (typeof body === 'string') {
- this.headers.set('content-type', 'text/plain;charset=UTF-8');
- } else if (this._bodyBlob && this._bodyBlob.type) {
- this.headers.set('content-type', this._bodyBlob.type);
- } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
- this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');
- }
- }
- };
-
- if (support.blob) {
- this.blob = function () {
- var rejected = consumed(this);
-
- if (rejected) {
- return rejected;
- }
-
- if (this._bodyBlob) {
- return Promise.resolve(this._bodyBlob);
- } else if (this._bodyArrayBuffer) {
- return Promise.resolve(new Blob([this._bodyArrayBuffer]));
- } else if (this._bodyFormData) {
- throw new Error('could not read FormData body as blob');
- } else {
- return Promise.resolve(new Blob([this._bodyText]));
- }
- };
-
- this.arrayBuffer = function () {
- if (this._bodyArrayBuffer) {
- var isConsumed = consumed(this);
-
- if (isConsumed) {
- return isConsumed;
- }
-
- if (ArrayBuffer.isView(this._bodyArrayBuffer)) {
- return Promise.resolve(this._bodyArrayBuffer.buffer.slice(this._bodyArrayBuffer.byteOffset, this._bodyArrayBuffer.byteOffset + this._bodyArrayBuffer.byteLength));
- } else {
- return Promise.resolve(this._bodyArrayBuffer);
- }
- } else {
- return this.blob().then(readBlobAsArrayBuffer);
- }
- };
- }
-
- this.text = function () {
- var rejected = consumed(this);
-
- if (rejected) {
- return rejected;
- }
-
- if (this._bodyBlob) {
- return readBlobAsText(this._bodyBlob);
- } else if (this._bodyArrayBuffer) {
- return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer));
- } else if (this._bodyFormData) {
- throw new Error('could not read FormData body as text');
- } else {
- return Promise.resolve(this._bodyText);
- }
- };
-
- if (support.formData) {
- this.formData = function () {
- return this.text().then(decode);
- };
- }
-
- this.json = function () {
- return this.text().then(JSON.parse);
- };
-
- return this;
- } // HTTP methods whose capitalization should be normalized
-
-
- var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT'];
-
- function normalizeMethod(method) {
- var upcased = method.toUpperCase();
- return methods.indexOf(upcased) > -1 ? upcased : method;
- }
-
- function Request(input, options) {
- if (!(this instanceof Request)) {
- throw new TypeError('Please use the "new" operator, this DOM object constructor cannot be called as a function.');
- }
-
- options = options || {};
- var body = options.body;
-
- if (input instanceof Request) {
- if (input.bodyUsed) {
- throw new TypeError('Already read');
- }
-
- this.url = input.url;
- this.credentials = input.credentials;
-
- if (!options.headers) {
- this.headers = new Headers$1(input.headers);
- }
-
- this.method = input.method;
- this.mode = input.mode;
- this.signal = input.signal;
-
- if (!body && input._bodyInit != null) {
- body = input._bodyInit;
- input.bodyUsed = true;
- }
- } else {
- this.url = String(input);
- }
-
- this.credentials = options.credentials || this.credentials || 'same-origin';
-
- if (options.headers || !this.headers) {
- this.headers = new Headers$1(options.headers);
- }
-
- this.method = normalizeMethod(options.method || this.method || 'GET');
- this.mode = options.mode || this.mode || null;
- this.signal = options.signal || this.signal;
- this.referrer = null;
-
- if ((this.method === 'GET' || this.method === 'HEAD') && body) {
- throw new TypeError('Body not allowed for GET or HEAD requests');
- }
-
- this._initBody(body);
-
- if (this.method === 'GET' || this.method === 'HEAD') {
- if (options.cache === 'no-store' || options.cache === 'no-cache') {
- // Search for a '_' parameter in the query string
- var reParamSearch = /([?&])_=[^&]*/;
-
- if (reParamSearch.test(this.url)) {
- // If it already exists then set the value with the current time
- this.url = this.url.replace(reParamSearch, '$1_=' + new Date().getTime());
- } else {
- // Otherwise add a new '_' parameter to the end with the current time
- var reQueryString = /\?/;
- this.url += (reQueryString.test(this.url) ? '&' : '?') + '_=' + new Date().getTime();
- }
- }
- }
- }
-
- Request.prototype.clone = function () {
- return new Request(this, {
- body: this._bodyInit
- });
- };
-
- function decode(body) {
- var form = new FormData();
- body.trim().split('&').forEach(function (bytes) {
- if (bytes) {
- var split = bytes.split('=');
- var name = split.shift().replace(/\+/g, ' ');
- var value = split.join('=').replace(/\+/g, ' ');
- form.append(decodeURIComponent(name), decodeURIComponent(value));
- }
- });
- return form;
- }
-
- 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
- // https://tools.ietf.org/html/rfc7230#section-3.2
-
- var preProcessedHeaders = rawHeaders.replace(/\r?\n[\t ]+/g, ' ');
- preProcessedHeaders.split(/\r?\n/).forEach(function (line) {
- var parts = line.split(':');
- var key = parts.shift().trim();
-
- if (key) {
- var value = parts.join(':').trim();
- headers.append(key, value);
- }
- });
- return headers;
- }
-
- Body.call(Request.prototype);
- function Response(bodyInit, options) {
- if (!(this instanceof Response)) {
- throw new TypeError('Please use the "new" operator, this DOM object constructor cannot be called as a function.');
- }
-
- if (!options) {
- options = {};
- }
-
- this.type = 'default';
- this.status = options.status === undefined ? 200 : options.status;
- this.ok = this.status >= 200 && this.status < 300;
- this.statusText = 'statusText' in options ? options.statusText : '';
- this.headers = new Headers$1(options.headers);
- this.url = options.url || '';
-
- this._initBody(bodyInit);
- }
- Body.call(Response.prototype);
-
- Response.prototype.clone = function () {
- return new Response(this._bodyInit, {
- status: this.status,
- statusText: this.statusText,
- headers: new Headers$1(this.headers),
- url: this.url
- });
- };
-
- Response.error = function () {
- var response = new Response(null, {
- status: 0,
- statusText: ''
- });
- response.type = 'error';
- return response;
- };
-
- var redirectStatuses = [301, 302, 303, 307, 308];
-
- Response.redirect = function (url, status) {
- if (redirectStatuses.indexOf(status) === -1) {
- throw new RangeError('Invalid status code');
- }
-
- return new Response(null, {
- status: status,
- headers: {
- location: url
- }
- });
- };
-
- var DOMException$1 = global$1.DOMException;
-
- try {
- new DOMException$1();
- } catch (err) {
- DOMException$1 = 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;
- }
-
- function fetch$1(input, init) {
- return new Promise(function (resolve, reject) {
- var request = new Request(input, init);
-
- if (request.signal && request.signal.aborted) {
- return reject(new DOMException$1('Aborted', 'AbortError'));
- }
-
- var xhr = new XMLHttpRequest();
-
- function abortXhr() {
- xhr.abort();
- }
-
- xhr.onload = function () {
- var options = {
- status: xhr.status,
- statusText: xhr.statusText,
- headers: parseHeaders(xhr.getAllResponseHeaders() || '')
- };
- options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL');
- var body = 'response' in xhr ? xhr.response : xhr.responseText;
- setTimeout(function () {
- resolve(new Response(body, options));
- }, 0);
- };
-
- xhr.onerror = function () {
- setTimeout(function () {
- reject(new TypeError('Network request failed'));
- }, 0);
- };
-
- xhr.ontimeout = function () {
- setTimeout(function () {
- reject(new TypeError('Network request failed'));
- }, 0);
- };
-
- xhr.onabort = function () {
- setTimeout(function () {
- reject(new DOMException$1('Aborted', 'AbortError'));
- }, 0);
- };
-
- function fixUrl(url) {
- try {
- return url === '' && global$1.location.href ? global$1.location.href : url;
- } catch (e) {
- return url;
- }
- }
-
- xhr.open(request.method, fixUrl(request.url), true);
-
- if (request.credentials === 'include') {
- xhr.withCredentials = true;
- } else if (request.credentials === 'omit') {
- xhr.withCredentials = false;
- }
-
- if ('responseType' in xhr) {
- if (support.blob) {
- xhr.responseType = 'blob';
- } else if (support.arrayBuffer && request.headers.get('Content-Type') && request.headers.get('Content-Type').indexOf('application/octet-stream') !== -1) {
- xhr.responseType = 'arraybuffer';
- }
- }
-
- if (init && _typeof(init.headers) === 'object' && !(init.headers instanceof Headers$1)) {
- Object.getOwnPropertyNames(init.headers).forEach(function (name) {
- xhr.setRequestHeader(name, normalizeValue(init.headers[name]));
- });
- } else {
- request.headers.forEach(function (value, name) {
- xhr.setRequestHeader(name, value);
- });
- }
-
- if (request.signal) {
- request.signal.addEventListener('abort', abortXhr);
-
- xhr.onreadystatechange = function () {
- // DONE (success or failure)
- if (xhr.readyState === 4) {
- request.signal.removeEventListener('abort', abortXhr);
- }
- };
- }
-
- xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit);
- });
- }
- fetch$1.polyfill = true;
-
- if (!global$1.fetch) {
- global$1.fetch = fetch$1;
- global$1.Headers = Headers$1;
- global$1.Request = Request;
- global$1.Response = Response;
- }
-
- // `Symbol.toStringTag` well-known symbol
- // https://tc39.github.io/ecma262/#sec-symbol.tostringtag
- defineWellKnownSymbol('toStringTag');
-
- var HAS_SPECIES_SUPPORT$2 = arrayMethodHasSpeciesSupport('splice');
- var USES_TO_LENGTH$5 = arrayMethodUsesToLength('splice', { ACCESSORS: true, 0: 0, 1: 2 });
-
- var max$3 = Math.max;
- var min$6 = Math.min;
- var MAX_SAFE_INTEGER = 0x1FFFFFFFFFFFFF;
- var MAXIMUM_ALLOWED_LENGTH_EXCEEDED = 'Maximum allowed length exceeded';
-
- // `Array.prototype.splice` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.splice
- // with adding support of @@species
- _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT$2 || !USES_TO_LENGTH$5 }, {
- splice: function splice(start, deleteCount /* , ...items */) {
- var O = toObject(this);
- var len = toLength(O.length);
- var actualStart = toAbsoluteIndex(start, len);
- var argumentsLength = arguments.length;
- var insertCount, actualDeleteCount, A, k, from, to;
- if (argumentsLength === 0) {
- insertCount = actualDeleteCount = 0;
- } else if (argumentsLength === 1) {
- insertCount = 0;
- actualDeleteCount = len - actualStart;
- } else {
- insertCount = argumentsLength - 2;
- actualDeleteCount = min$6(max$3(toInteger(deleteCount), 0), len - actualStart);
- }
- if (len + insertCount - actualDeleteCount > MAX_SAFE_INTEGER) {
- throw TypeError(MAXIMUM_ALLOWED_LENGTH_EXCEEDED);
- }
- A = arraySpeciesCreate(O, actualDeleteCount);
- for (k = 0; k < actualDeleteCount; k++) {
- from = actualStart + k;
- if (from in O) createProperty(A, k, O[from]);
- }
- A.length = actualDeleteCount;
- if (insertCount < actualDeleteCount) {
- for (k = actualStart; k < len - actualDeleteCount; k++) {
- from = k + actualDeleteCount;
- to = k + insertCount;
- if (from in O) O[to] = O[from];
- else delete O[to];
- }
- for (k = len; k > len - actualDeleteCount + insertCount; k--) delete O[k - 1];
- } else if (insertCount > actualDeleteCount) {
- for (k = len - actualDeleteCount; k > actualStart; k--) {
- from = k + actualDeleteCount - 1;
- to = k + insertCount - 1;
- if (from in O) O[to] = O[from];
- else delete O[to];
- }
- }
- for (k = 0; k < insertCount; k++) {
- O[k + actualStart] = arguments[k + 2];
- }
- O.length = len - actualDeleteCount + insertCount;
- return A;
- }
- });
-
- // JSON[@@toStringTag] property
- // https://tc39.github.io/ecma262/#sec-json-@@tostringtag
- setToStringTag(global_1.JSON, 'JSON', true);
-
- // Math[@@toStringTag] property
- // https://tc39.github.io/ecma262/#sec-math-@@tostringtag
- setToStringTag(Math, 'Math', true);
-
- // `Object.defineProperty` method
- // https://tc39.github.io/ecma262/#sec-object.defineproperty
- _export({ target: 'Object', stat: true, forced: !descriptors, sham: !descriptors }, {
- defineProperty: objectDefineProperty.f
- });
-
- var nativeGetOwnPropertyDescriptor$2 = objectGetOwnPropertyDescriptor.f;
-
-
- var FAILS_ON_PRIMITIVES$1 = fails(function () { nativeGetOwnPropertyDescriptor$2(1); });
- var FORCED$5 = !descriptors || FAILS_ON_PRIMITIVES$1;
-
- // `Object.getOwnPropertyDescriptor` method
- // https://tc39.github.io/ecma262/#sec-object.getownpropertydescriptor
- _export({ target: 'Object', stat: true, forced: FORCED$5, sham: !descriptors }, {
- getOwnPropertyDescriptor: function getOwnPropertyDescriptor(it, key) {
- return nativeGetOwnPropertyDescriptor$2(toIndexedObject(it), key);
- }
- });
-
- var FAILS_ON_PRIMITIVES$2 = fails(function () { objectGetPrototypeOf(1); });
-
- // `Object.getPrototypeOf` method
- // https://tc39.github.io/ecma262/#sec-object.getprototypeof
- _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$2, sham: !correctPrototypeGetter }, {
- getPrototypeOf: function getPrototypeOf(it) {
- return objectGetPrototypeOf(toObject(it));
- }
- });
-
- // `Object.setPrototypeOf` method
- // https://tc39.github.io/ecma262/#sec-object.setprototypeof
- _export({ target: 'Object', stat: true }, {
- setPrototypeOf: objectSetPrototypeOf
- });
-
- var slice$1 = [].slice;
- var factories = {};
-
- var construct = function (C, argsLength, args) {
- if (!(argsLength in factories)) {
- for (var list = [], i = 0; i < argsLength; i++) list[i] = 'a[' + i + ']';
- // eslint-disable-next-line no-new-func
- factories[argsLength] = Function('C,a', 'return new C(' + list.join(',') + ')');
- } return factories[argsLength](C, args);
- };
-
- // `Function.prototype.bind` method implementation
- // https://tc39.github.io/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 boundFunction = function bound(/* args... */) {
- var args = partArgs.concat(slice$1.call(arguments));
- return this instanceof boundFunction ? construct(fn, args.length, args) : fn.apply(that, args);
- };
- if (isObject(fn.prototype)) boundFunction.prototype = fn.prototype;
- return boundFunction;
- };
-
- var nativeConstruct = getBuiltIn('Reflect', 'construct');
-
- // `Reflect.construct` method
- // https://tc39.github.io/ecma262/#sec-reflect.construct
- // MS Edge supports only 2 arguments and argumentsList argument is optional
- // FF Nightly sets third argument as `new.target`, but does not create `this` from it
- var NEW_TARGET_BUG = fails(function () {
- function F() { /* empty */ }
- return !(nativeConstruct(function () { /* empty */ }, [], F) instanceof F);
- });
- var ARGS_BUG = !fails(function () {
- nativeConstruct(function () { /* empty */ });
- });
- var FORCED$6 = NEW_TARGET_BUG || ARGS_BUG;
-
- _export({ target: 'Reflect', stat: true, forced: FORCED$6, sham: FORCED$6 }, {
- construct: function construct(Target, args /* , newTarget */) {
- aFunction$1(Target);
- anObject(args);
- var newTarget = arguments.length < 3 ? Target : aFunction$1(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
- switch (args.length) {
- case 0: return new Target();
- case 1: return new Target(args[0]);
- case 2: return new Target(args[0], args[1]);
- case 3: return new Target(args[0], args[1], args[2]);
- case 4: return new Target(args[0], args[1], args[2], args[3]);
- }
- // w/o altered newTarget, lot of arguments case
- var $args = [null];
- $args.push.apply($args, args);
- return new (functionBind.apply(Target, $args))();
- }
- // with altered newTarget, not support built-in constructors
- var proto = newTarget.prototype;
- var instance = objectCreate(isObject(proto) ? proto : Object.prototype);
- var result = Function.apply.call(Target, instance, args);
- return isObject(result) ? result : instance;
- }
- });
-
- // `Reflect.get` method
- // https://tc39.github.io/ecma262/#sec-reflect.get
- function get$2(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')
- ? descriptor.value
- : descriptor.get === undefined
- ? undefined
- : descriptor.get.call(receiver);
- if (isObject(prototype = objectGetPrototypeOf(target))) return get$2(prototype, propertyKey, receiver);
- }
-
- _export({ target: 'Reflect', stat: true }, {
- get: get$2
- });
-
- (function (factory) {
- factory();
- })(function () {
-
- function _classCallCheck(instance, Constructor) {
- if (!(instance instanceof Constructor)) {
- throw new TypeError("Cannot call a class as a function");
- }
- }
-
- 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);
- }
- }
-
- function _createClass(Constructor, protoProps, staticProps) {
- if (protoProps) _defineProperties(Constructor.prototype, protoProps);
- if (staticProps) _defineProperties(Constructor, staticProps);
- return Constructor;
- }
-
- function _inherits(subClass, superClass) {
- if (typeof superClass !== "function" && superClass !== null) {
- throw new TypeError("Super expression must either be null or a function");
- }
-
- subClass.prototype = Object.create(superClass && superClass.prototype, {
- constructor: {
- value: subClass,
- writable: true,
- configurable: true
- }
- });
- if (superClass) _setPrototypeOf(subClass, superClass);
- }
-
- function _getPrototypeOf(o) {
- _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf(o) {
- return o.__proto__ || Object.getPrototypeOf(o);
- };
- return _getPrototypeOf(o);
- }
-
- function _setPrototypeOf(o, p) {
- _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) {
- o.__proto__ = p;
- return o;
- };
-
- return _setPrototypeOf(o, p);
- }
-
- function _isNativeReflectConstruct() {
- if (typeof Reflect === "undefined" || !Reflect.construct) return false;
- if (Reflect.construct.sham) return false;
- if (typeof Proxy === "function") return true;
-
- try {
- Date.prototype.toString.call(Reflect.construct(Date, [], function () {}));
- return true;
- } catch (e) {
- return false;
- }
- }
-
- function _assertThisInitialized(self) {
- if (self === void 0) {
- throw new ReferenceError("this hasn't been initialised - super() hasn't been called");
- }
-
- return self;
- }
-
- function _possibleConstructorReturn(self, call) {
- if (call && (_typeof(call) === "object" || typeof call === "function")) {
- return call;
- }
-
- return _assertThisInitialized(self);
- }
-
- function _createSuper(Derived) {
- var hasNativeReflectConstruct = _isNativeReflectConstruct();
-
- return function _createSuperInternal() {
- var Super = _getPrototypeOf(Derived),
- result;
-
- if (hasNativeReflectConstruct) {
- var NewTarget = _getPrototypeOf(this).constructor;
-
- result = Reflect.construct(Super, arguments, NewTarget);
- } else {
- result = Super.apply(this, arguments);
- }
-
- return _possibleConstructorReturn(this, result);
- };
- }
-
- function _superPropBase(object, property) {
- while (!Object.prototype.hasOwnProperty.call(object, property)) {
- object = _getPrototypeOf(object);
- if (object === null) break;
- }
-
- return object;
- }
-
- function _get(target, property, receiver) {
- if (typeof Reflect !== "undefined" && Reflect.get) {
- _get = Reflect.get;
- } else {
- _get = function _get(target, property, receiver) {
- var base = _superPropBase(target, property);
-
- if (!base) return;
- var desc = Object.getOwnPropertyDescriptor(base, property);
-
- if (desc.get) {
- return desc.get.call(receiver);
- }
-
- return desc.value;
- };
- }
-
- return _get(target, property, receiver || target);
- }
-
- var Emitter = /*#__PURE__*/function () {
- function Emitter() {
- _classCallCheck(this, Emitter);
-
- Object.defineProperty(this, 'listeners', {
- value: {},
- writable: true,
- configurable: true
- });
- }
-
- _createClass(Emitter, [{
- key: "addEventListener",
- value: function addEventListener(type, callback) {
- if (!(type in this.listeners)) {
- this.listeners[type] = [];
- }
-
- this.listeners[type].push(callback);
- }
- }, {
- key: "removeEventListener",
- value: function removeEventListener(type, callback) {
- if (!(type in this.listeners)) {
- return;
- }
-
- var stack = this.listeners[type];
-
- for (var i = 0, l = stack.length; i < l; i++) {
- if (stack[i] === callback) {
- stack.splice(i, 1);
- return;
- }
- }
- }
- }, {
- key: "dispatchEvent",
- value: function dispatchEvent(event) {
- var _this = this;
-
- if (!(event.type in this.listeners)) {
- return;
- }
-
- var debounce = function debounce(callback) {
- setTimeout(function () {
- return callback.call(_this, event);
- });
- };
-
- var stack = this.listeners[event.type];
-
- for (var i = 0, l = stack.length; i < l; i++) {
- debounce(stack[i]);
- }
-
- return !event.defaultPrevented;
- }
- }]);
-
- return Emitter;
- }();
-
- var AbortSignal = /*#__PURE__*/function (_Emitter) {
- _inherits(AbortSignal, _Emitter);
-
- var _super = _createSuper(AbortSignal);
-
- function AbortSignal() {
- var _this2;
-
- _classCallCheck(this, AbortSignal);
-
- _this2 = _super.call(this); // Some versions of babel does not transpile super() correctly for IE <= 10, if the parent
- // constructor has failed to run, then "this.listeners" will still be undefined and then we call
- // the parent constructor directly instead as a workaround. For general details, see babel bug:
- // https://github.com/babel/babel/issues/3041
- // This hack was added as a fix for the issue described here:
- // https://github.com/Financial-Times/polyfill-library/pull/59#issuecomment-477558042
-
- if (!_this2.listeners) {
- Emitter.call(_assertThisInitialized(_this2));
- } // Compared to assignment, Object.defineProperty makes properties non-enumerable by default and
- // we want Object.keys(new AbortController().signal) to be [] for compat with the native impl
-
-
- Object.defineProperty(_assertThisInitialized(_this2), 'aborted', {
- value: false,
- writable: true,
- configurable: true
- });
- Object.defineProperty(_assertThisInitialized(_this2), 'onabort', {
- value: null,
- writable: true,
- configurable: true
- });
- return _this2;
- }
-
- _createClass(AbortSignal, [{
- key: "toString",
- value: function toString() {
- return '[object AbortSignal]';
- }
- }, {
- key: "dispatchEvent",
- value: function dispatchEvent(event) {
- if (event.type === 'abort') {
- this.aborted = true;
-
- if (typeof this.onabort === 'function') {
- this.onabort.call(this, event);
- }
- }
-
- _get(_getPrototypeOf(AbortSignal.prototype), "dispatchEvent", this).call(this, event);
- }
- }]);
-
- return AbortSignal;
- }(Emitter);
-
- var AbortController = /*#__PURE__*/function () {
- function AbortController() {
- _classCallCheck(this, AbortController); // Compared to assignment, Object.defineProperty makes properties non-enumerable by default and
- // we want Object.keys(new AbortController()) to be [] for compat with the native impl
-
-
- Object.defineProperty(this, 'signal', {
- value: new AbortSignal(),
- writable: true,
- configurable: true
- });
- }
-
- _createClass(AbortController, [{
- key: "abort",
- value: function abort() {
- var event;
-
- try {
- event = new Event('abort');
- } catch (e) {
- if (typeof document !== 'undefined') {
- if (!document.createEvent) {
- // For Internet Explorer 8:
- event = document.createEventObject();
- event.type = 'abort';
- } else {
- // For Internet Explorer 11:
- event = document.createEvent('Event');
- event.initEvent('abort', false, false);
- }
- } else {
- // Fallback where document isn't available:
- event = {
- type: 'abort',
- bubbles: false,
- cancelable: false
- };
- }
- }
-
- this.signal.dispatchEvent(event);
- }
- }, {
- key: "toString",
- value: function toString() {
- return '[object AbortController]';
- }
- }]);
-
- return AbortController;
- }();
-
- if (typeof Symbol !== 'undefined' && Symbol.toStringTag) {
- // These are necessary to make sure that we get correct output for:
- // Object.prototype.toString.call(new AbortController())
- AbortController.prototype[Symbol.toStringTag] = 'AbortController';
- AbortSignal.prototype[Symbol.toStringTag] = 'AbortSignal';
- }
-
- function polyfillNeeded(self) {
- if (self.__FORCE_INSTALL_ABORTCONTROLLER_POLYFILL) {
- console.log('__FORCE_INSTALL_ABORTCONTROLLER_POLYFILL=true is set, will force install polyfill');
- return true;
- } // Note that the "unfetch" minimal fetch polyfill defines fetch() without
- // defining window.Request, and this polyfill need to work on top of unfetch
- // so the below feature detection needs the !self.AbortController part.
- // The Request.prototype check is also needed because Safari versions 11.1.2
- // up to and including 12.1.x has a window.AbortController present but still
- // does NOT correctly implement abortable fetch:
- // https://bugs.webkit.org/show_bug.cgi?id=174980#c2
-
-
- return typeof self.Request === 'function' && !self.Request.prototype.hasOwnProperty('signal') || !self.AbortController;
- }
- /**
- * Note: the "fetch.Request" default value is available for fetch imported from
- * the "node-fetch" package and not in browsers. This is OK since browsers
- * will be importing umd-polyfill.js from that path "self" is passed the
- * decorator so the default value will not be used (because browsers that define
- * fetch also has Request). One quirky setup where self.fetch exists but
- * self.Request does not is when the "unfetch" minimal fetch polyfill is used
- * on top of IE11; for this case the browser will try to use the fetch.Request
- * default value which in turn will be undefined but then then "if (Request)"
- * will ensure that you get a patched fetch but still no Request (as expected).
- * @param {fetch, Request = fetch.Request}
- * @returns {fetch: abortableFetch, Request: AbortableRequest}
- */
-
-
- function abortableFetchDecorator(patchTargets) {
- if ('function' === typeof patchTargets) {
- patchTargets = {
- fetch: patchTargets
- };
- }
-
- var _patchTargets = patchTargets,
- fetch = _patchTargets.fetch,
- _patchTargets$Request = _patchTargets.Request,
- NativeRequest = _patchTargets$Request === void 0 ? fetch.Request : _patchTargets$Request,
- NativeAbortController = _patchTargets.AbortController,
- _patchTargets$__FORCE = _patchTargets.__FORCE_INSTALL_ABORTCONTROLLER_POLYFILL,
- __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL = _patchTargets$__FORCE === void 0 ? false : _patchTargets$__FORCE;
-
- if (!polyfillNeeded({
- fetch: fetch,
- Request: NativeRequest,
- AbortController: NativeAbortController,
- __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL: __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL
- })) {
- return {
- fetch: fetch,
- Request: Request
- };
- }
-
- var Request = NativeRequest; // Note that the "unfetch" minimal fetch polyfill defines fetch() without
- // defining window.Request, and this polyfill need to work on top of unfetch
- // hence we only patch it if it's available. Also we don't patch it if signal
- // is already available on the Request prototype because in this case support
- // is present and the patching below can cause a crash since it assigns to
- // request.signal which is technically a read-only property. This latter error
- // happens when you run the main5.js node-fetch example in the repo
- // "abortcontroller-polyfill-examples". The exact error is:
- // request.signal = init.signal;
- // ^
- // TypeError: Cannot set property signal of #<Request> which has only a getter
-
- if (Request && !Request.prototype.hasOwnProperty('signal') || __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL) {
- Request = function Request(input, init) {
- var signal;
-
- if (init && init.signal) {
- signal = init.signal; // Never pass init.signal to the native Request implementation when the polyfill has
- // been installed because if we're running on top of a browser with a
- // working native AbortController (i.e. the polyfill was installed due to
- // __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL being set), then passing our
- // fake AbortSignal to the native fetch will trigger:
- // TypeError: Failed to construct 'Request': member signal is not of type AbortSignal.
-
- delete init.signal;
- }
-
- var request = new NativeRequest(input, init);
-
- if (signal) {
- Object.defineProperty(request, 'signal', {
- writable: false,
- enumerable: false,
- configurable: true,
- value: signal
- });
- }
-
- return request;
- };
-
- Request.prototype = NativeRequest.prototype;
- }
-
- var realFetch = fetch;
-
- var abortableFetch = function abortableFetch(input, init) {
- var signal = Request && Request.prototype.isPrototypeOf(input) ? input.signal : init ? init.signal : undefined;
-
- if (signal) {
- var abortError;
-
- try {
- abortError = new DOMException('Aborted', 'AbortError');
- } catch (err) {
- // IE 11 does not support calling the DOMException constructor, use a
- // regular error object on it instead.
- abortError = new Error('Aborted');
- abortError.name = 'AbortError';
- } // Return early if already aborted, thus avoiding making an HTTP request
-
-
- if (signal.aborted) {
- return Promise.reject(abortError);
- } // Turn an event into a promise, reject it once `abort` is dispatched
-
-
- var cancellation = new Promise(function (_, reject) {
- signal.addEventListener('abort', function () {
- return reject(abortError);
- }, {
- once: true
- });
- });
-
- if (init && init.signal) {
- // Never pass .signal to the native implementation when the polyfill has
- // been installed because if we're running on top of a browser with a
- // working native AbortController (i.e. the polyfill was installed due to
- // __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL being set), then passing our
- // fake AbortSignal to the native fetch will trigger:
- // TypeError: Failed to execute 'fetch' on 'Window': member signal is not of type AbortSignal.
- delete init.signal;
- } // Return the fastest promise (don't need to wait for request to finish)
-
-
- return Promise.race([cancellation, realFetch(input, init)]);
- }
-
- return realFetch(input, init);
- };
-
- return {
- fetch: abortableFetch,
- Request: Request
- };
- }
-
- (function (self) {
- if (!polyfillNeeded(self)) {
- return;
- }
-
- if (!self.fetch) {
- console.warn('fetch() is not available, cannot install abortcontroller-polyfill');
- return;
- }
-
- var _abortableFetch = abortableFetchDecorator(self),
- fetch = _abortableFetch.fetch,
- Request = _abortableFetch.Request;
-
- self.fetch = fetch;
- self.Request = Request;
- Object.defineProperty(self, 'AbortController', {
- writable: true,
- enumerable: false,
- configurable: true,
- value: AbortController
- });
- Object.defineProperty(self, 'AbortSignal', {
- writable: true,
- enumerable: false,
- configurable: true,
- value: AbortSignal
- });
- })(typeof self !== 'undefined' ? self : commonjsGlobal);
- });
-
- function actionAddEntity(way) {
- return function (graph) {
- return graph.replace(way);
- };
- }
-
- var IS_CONCAT_SPREADABLE = wellKnownSymbol('isConcatSpreadable');
- var MAX_SAFE_INTEGER$1 = 0x1FFFFFFFFFFFFF;
- var MAXIMUM_ALLOWED_INDEX_EXCEEDED = 'Maximum allowed index exceeded';
-
- // We can't use this feature detection in V8 since it causes
- // deoptimization and serious performance degradation
- // https://github.com/zloirock/core-js/issues/679
- var IS_CONCAT_SPREADABLE_SUPPORT = engineV8Version >= 51 || !fails(function () {
- var array = [];
- array[IS_CONCAT_SPREADABLE] = false;
- return array.concat()[0] !== array;
- });
-
- var SPECIES_SUPPORT = arrayMethodHasSpeciesSupport('concat');
-
- var isConcatSpreadable = function (O) {
- if (!isObject(O)) return false;
- var spreadable = O[IS_CONCAT_SPREADABLE];
- return spreadable !== undefined ? !!spreadable : isArray(O);
- };
-
- var FORCED$7 = !IS_CONCAT_SPREADABLE_SUPPORT || !SPECIES_SUPPORT;
-
- // `Array.prototype.concat` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.concat
- // with adding support of @@isConcatSpreadable and @@species
- _export({ target: 'Array', proto: true, forced: FORCED$7 }, {
- concat: function concat(arg) { // eslint-disable-line no-unused-vars
- var O = toObject(this);
- var A = arraySpeciesCreate(O, 0);
- var n = 0;
- var i, k, length, len, E;
- for (i = -1, length = arguments.length; i < length; i++) {
- 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);
- 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);
- createProperty(A, n++, E);
- }
- }
- A.length = n;
- return A;
- }
- });
-
- // `Object.assign` method
- // https://tc39.github.io/ecma262/#sec-object.assign
- _export({ target: 'Object', stat: true, forced: Object.assign !== objectAssign }, {
- assign: objectAssign
- });
-
- var $filter$1 = arrayIteration.filter;
-
-
-
- var HAS_SPECIES_SUPPORT$3 = arrayMethodHasSpeciesSupport('filter');
- // Edge 14- issue
- var USES_TO_LENGTH$6 = arrayMethodUsesToLength('filter');
-
- // `Array.prototype.filter` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.filter
- // with adding support of @@species
- _export({ target: 'Array', proto: true, forced: !HAS_SPECIES_SUPPORT$3 || !USES_TO_LENGTH$6 }, {
- filter: function filter(callbackfn /* , thisArg */) {
- return $filter$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- var nativeReverse = [].reverse;
- var test$1 = [1, 2];
-
- // `Array.prototype.reverse` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.reverse
- // fix for Safari 12.0 bug
- // https://bugs.webkit.org/show_bug.cgi?id=188794
- _export({ target: 'Array', proto: true, forced: String(test$1) === String(test$1.reverse()) }, {
- reverse: function reverse() {
- // eslint-disable-next-line no-self-assign
- if (isArray(this)) this.length = this.length;
- return nativeReverse.call(this);
- }
- });
-
- var FAILS_ON_PRIMITIVES$3 = fails(function () { objectKeys(1); });
-
- // `Object.keys` method
- // https://tc39.github.io/ecma262/#sec-object.keys
- _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$3 }, {
- keys: function keys(it) {
- return objectKeys(toObject(it));
- }
- });
-
- var trim = stringTrim.trim;
-
-
- var $parseFloat = global_1.parseFloat;
- var FORCED$8 = 1 / $parseFloat(whitespaces + '-0') !== -Infinity;
-
- // `parseFloat` method
- // https://tc39.github.io/ecma262/#sec-parsefloat-string
- var numberParseFloat = FORCED$8 ? function parseFloat(string) {
- var trimmedString = trim(String(string));
- var result = $parseFloat(trimmedString);
- return result === 0 && trimmedString.charAt(0) == '-' ? -0 : result;
- } : $parseFloat;
-
- // `parseFloat` method
- // https://tc39.github.io/ecma262/#sec-parsefloat-string
- _export({ global: true, forced: parseFloat != numberParseFloat }, {
- parseFloat: numberParseFloat
- });
-
- /*
- Order the nodes of a way in reverse order and reverse any direction dependent tags
- other than `oneway`. (We assume that correcting a backwards oneway is the primary
- reason for reversing a way.)
-
- In addition, numeric-valued `incline` tags are negated.
-
- The JOSM implementation was used as a guide, but transformations that were of unclear benefit
- or adjusted tags that don't seem to be used in practice were omitted.
-
- References:
- http://wiki.openstreetmap.org/wiki/Forward_%26_backward,_left_%26_right
- http://wiki.openstreetmap.org/wiki/Key:direction#Steps
- http://wiki.openstreetmap.org/wiki/Key:incline
- http://wiki.openstreetmap.org/wiki/Route#Members
- http://josm.openstreetmap.de/browser/josm/trunk/src/org/openstreetmap/josm/corrector/ReverseWayTagCorrector.java
- http://wiki.openstreetmap.org/wiki/Tag:highway%3Dstop
- http://wiki.openstreetmap.org/wiki/Key:traffic_sign#On_a_way_or_area
- */
- function actionReverse(entityID, options) {
- var ignoreKey = /^.*(_|:)?(description|name|note|website|ref|source|comment|watch|attribution)(_|:)?/;
- var numeric = /^([+\-]?)(?=[\d.])/;
- var directionKey = /direction$/;
- var turn_lanes = /^turn:lanes:?/;
- var keyReplacements = [[/:right$/, ':left'], [/:left$/, ':right'], [/:forward$/, ':backward'], [/:backward$/, ':forward'], [/:right:/, ':left:'], [/:left:/, ':right:'], [/:forward:/, ':backward:'], [/:backward:/, ':forward:']];
- var valueReplacements = {
- left: 'right',
- right: 'left',
- up: 'down',
- down: 'up',
- forward: 'backward',
- backward: 'forward',
- forwards: 'backward',
- backwards: 'forward'
- };
- var roleReplacements = {
- forward: 'backward',
- backward: 'forward',
- forwards: 'backward',
- backwards: 'forward'
- };
- var onewayReplacements = {
- yes: '-1',
- '1': '-1',
- '-1': 'yes'
- };
- var compassReplacements = {
- N: 'S',
- NNE: 'SSW',
- NE: 'SW',
- ENE: 'WSW',
- E: 'W',
- ESE: 'WNW',
- SE: 'NW',
- SSE: 'NNW',
- S: 'N',
- SSW: 'NNE',
- SW: 'NE',
- WSW: 'ENE',
- W: 'E',
- WNW: 'ESE',
- NW: 'SE',
- NNW: 'SSE'
- };
-
- function reverseKey(key) {
- for (var i = 0; i < keyReplacements.length; ++i) {
- var replacement = keyReplacements[i];
-
- if (replacement[0].test(key)) {
- return key.replace(replacement[0], replacement[1]);
- }
- }
-
- return key;
- }
-
- function reverseValue(key, value, includeAbsolute) {
- if (ignoreKey.test(key)) return value; // Turn lanes are left/right to key (not way) direction - #5674
-
- if (turn_lanes.test(key)) {
- return value;
- } else if (key === 'incline' && numeric.test(value)) {
- return value.replace(numeric, function (_, sign) {
- return sign === '-' ? '' : '-';
- });
- } else if (options && options.reverseOneway && key === 'oneway') {
- return onewayReplacements[value] || value;
- } else if (includeAbsolute && directionKey.test(key)) {
- if (compassReplacements[value]) return compassReplacements[value];
- var degrees = parseFloat(value);
-
- if (typeof degrees === 'number' && !isNaN(degrees)) {
- if (degrees < 180) {
- degrees += 180;
- } else {
- degrees -= 180;
- }
-
- return degrees.toString();
- }
- }
-
- return valueReplacements[value] || value;
- } // Reverse the direction of tags attached to the nodes - #3076
-
-
- function reverseNodeTags(graph, nodeIDs) {
- for (var i = 0; i < nodeIDs.length; i++) {
- var node = graph.hasEntity(nodeIDs[i]);
- if (!node || !Object.keys(node.tags).length) continue;
- var tags = {};
-
- for (var key in node.tags) {
- tags[reverseKey(key)] = reverseValue(key, node.tags[key], node.id === entityID);
- }
-
- graph = graph.replace(node.update({
- tags: tags
- }));
- }
-
- return graph;
- }
-
- function reverseWay(graph, way) {
- var nodes = way.nodes.slice().reverse();
- var tags = {};
- var role;
-
- for (var key in way.tags) {
- tags[reverseKey(key)] = reverseValue(key, way.tags[key]);
- }
-
- graph.parentRelations(way).forEach(function (relation) {
- relation.members.forEach(function (member, index) {
- if (member.id === way.id && (role = roleReplacements[member.role])) {
- relation = relation.updateMember({
- role: role
- }, index);
- graph = graph.replace(relation);
- }
- });
- }); // Reverse any associated directions on nodes on the way and then replace
- // the way itself with the reversed node ids and updated way tags
-
- return reverseNodeTags(graph, nodes).replace(way.update({
- nodes: nodes,
- tags: tags
- }));
- }
-
- var action = function action(graph) {
- var entity = graph.entity(entityID);
-
- if (entity.type === 'way') {
- return reverseWay(graph, entity);
- }
-
- return reverseNodeTags(graph, [entityID]);
- };
-
- action.disabled = function (graph) {
- var entity = graph.hasEntity(entityID);
- if (!entity || entity.type === 'way') return false;
-
- for (var key in entity.tags) {
- var value = entity.tags[key];
-
- if (reverseKey(key) !== key || reverseValue(key, value, true) !== value) {
- return false;
- }
- }
-
- return 'nondirectional_node';
- };
-
- action.entityID = function () {
- return entityID;
- };
-
- return action;
- }
-
- function osmIsInterestingTag(key) {
- return key !== 'attribution' && key !== 'created_by' && key !== 'source' && key !== 'odbl' && key.indexOf('source:') !== 0 && key.indexOf('source_ref') !== 0 && // purposely exclude colon
- key.indexOf('tiger:') !== 0;
- }
- var osmAreaKeys = {};
- function osmSetAreaKeys(value) {
- osmAreaKeys = value;
- } // returns an object with the tag from `tags` that implies an area geometry, if any
-
- function osmTagSuggestingArea(tags) {
- if (tags.area === 'yes') return {
- area: 'yes'
- };
- if (tags.area === 'no') return null; // `highway` and `railway` are typically linear features, but there
- // are a few exceptions that should be treated as areas, even in the
- // absence of a proper `area=yes` or `areaKeys` tag.. see #4194
-
- var lineKeys = {
- highway: {
- rest_area: true,
- services: true
- },
- railway: {
- roundhouse: true,
- station: true,
- traverser: true,
- turntable: true,
- wash: true
- }
- };
- var returnTags = {};
-
- for (var key in tags) {
- if (key in osmAreaKeys && !(tags[key] in osmAreaKeys[key])) {
- returnTags[key] = tags[key];
- return returnTags;
- }
-
- if (key in lineKeys && tags[key] in lineKeys[key]) {
- returnTags[key] = tags[key];
- return returnTags;
- }
- }
-
- return null;
- } // Tags that indicate a node can be a standalone point
- // e.g. { amenity: { bar: true, parking: true, ... } ... }
-
- var osmPointTags = {};
- function osmSetPointTags(value) {
- osmPointTags = value;
- } // Tags that indicate a node can be part of a way
- // e.g. { amenity: { parking: true, ... }, highway: { stop: true ... } ... }
-
- var osmVertexTags = {};
- function osmSetVertexTags(value) {
- osmVertexTags = value;
- }
- function osmNodeGeometriesForTags(nodeTags) {
- var geometries = {};
-
- for (var key in nodeTags) {
- if (osmPointTags[key] && (osmPointTags[key]['*'] || osmPointTags[key][nodeTags[key]])) {
- geometries.point = true;
- }
-
- if (osmVertexTags[key] && (osmVertexTags[key]['*'] || osmVertexTags[key][nodeTags[key]])) {
- geometries.vertex = true;
- } // break early if both are already supported
-
-
- if (geometries.point && geometries.vertex) break;
- }
-
- return geometries;
- }
- var osmOneWayTags = {
- 'aerialway': {
- 'chair_lift': true,
- 'drag_lift': true,
- 'j-bar': true,
- 'magic_carpet': true,
- 'mixed_lift': true,
- 'platter': true,
- 'rope_tow': true,
- 't-bar': true,
- 'zip_line': true
- },
- 'highway': {
- 'motorway': true
- },
- 'junction': {
- 'circular': true,
- 'roundabout': true
- },
- 'man_made': {
- 'goods_conveyor': true,
- 'piste:halfpipe': true
- },
- 'piste:type': {
- 'downhill': true,
- 'sled': true,
- 'yes': true
- },
- 'waterway': {
- 'canal': true,
- 'ditch': true,
- 'drain': true,
- 'fish_pass': true,
- 'river': true,
- 'stream': true,
- 'tidal_channel': true
- }
- }; // solid and smooth surfaces akin to the assumed default road surface in OSM
-
- var osmPavedTags = {
- 'surface': {
- 'paved': true,
- 'asphalt': true,
- 'concrete': true,
- 'concrete:lanes': true,
- 'concrete:plates': true
- },
- 'tracktype': {
- 'grade1': true
- }
- }; // solid, if somewhat uncommon surfaces with a high range of smoothness
-
- var osmSemipavedTags = {
- 'surface': {
- 'cobblestone': true,
- 'cobblestone:flattened': true,
- 'unhewn_cobblestone': true,
- 'sett': true,
- 'paving_stones': true,
- 'metal': true,
- 'wood': true
- }
- };
- var osmRightSideIsInsideTags = {
- 'natural': {
- 'cliff': true,
- 'coastline': 'coastline'
- },
- 'barrier': {
- 'retaining_wall': true,
- 'kerb': true,
- 'guard_rail': true,
- 'city_wall': true
- },
- 'man_made': {
- 'embankment': true
- },
- 'waterway': {
- 'weir': true
- }
- }; // "highway" tag values for pedestrian or vehicle right-of-ways that make up the routable network
- // (does not include `raceway`)
-
- var osmRoutableHighwayTagValues = {
- motorway: true,
- trunk: true,
- primary: true,
- secondary: true,
- tertiary: true,
- residential: true,
- motorway_link: true,
- trunk_link: true,
- primary_link: true,
- secondary_link: true,
- tertiary_link: true,
- unclassified: true,
- road: true,
- service: true,
- track: true,
- living_street: true,
- bus_guideway: true,
- path: true,
- footway: true,
- cycleway: true,
- bridleway: true,
- pedestrian: true,
- corridor: true,
- steps: true
- }; // "highway" tag values that generally do not allow motor vehicles
-
- var osmPathHighwayTagValues = {
- path: true,
- footway: true,
- cycleway: true,
- bridleway: true,
- pedestrian: true,
- corridor: true,
- steps: true
- }; // "railway" tag values representing existing railroad tracks (purposely does not include 'abandoned')
-
- var osmRailwayTrackTagValues = {
- rail: true,
- light_rail: true,
- tram: true,
- subway: true,
- monorail: true,
- funicular: true,
- miniature: true,
- narrow_gauge: true,
- disused: true,
- preserved: true
- }; // "waterway" tag values for line features representing water flow
-
- var osmFlowingWaterwayTagValues = {
- canal: true,
- ditch: true,
- drain: true,
- fish_pass: true,
- river: true,
- stream: true,
- tidal_channel: true
- };
-
- var trim$1 = stringTrim.trim;
-
-
- var $parseInt = global_1.parseInt;
- var hex$1 = /^[+-]?0[Xx]/;
- var FORCED$9 = $parseInt(whitespaces + '08') !== 8 || $parseInt(whitespaces + '0x16') !== 22;
-
- // `parseInt` method
- // https://tc39.github.io/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));
- } : $parseInt;
-
- // `parseInt` method
- // https://tc39.github.io/ecma262/#sec-parseint-string-radix
- _export({ global: true, forced: parseInt != numberParseInt }, {
- parseInt: numberParseInt
- });
-
- var freezing = !fails(function () {
- return Object.isExtensible(Object.preventExtensions({}));
- });
-
- var internalMetadata = createCommonjsModule(function (module) {
- var defineProperty = objectDefineProperty.f;
-
-
-
- var METADATA = uid('meta');
- var id = 0;
-
- var isExtensible = Object.isExtensible || function () {
- return true;
- };
-
- var setMetadata = function (it) {
- defineProperty(it, METADATA, { value: {
- 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)) {
- // can't set metadata to uncaught frozen object
- if (!isExtensible(it)) return 'F';
- // not necessary to add metadata
- if (!create) return 'E';
- // add missing metadata
- setMetadata(it);
- // return object ID
- } return it[METADATA].objectID;
- };
-
- var getWeakData = function (it, create) {
- if (!has(it, METADATA)) {
- // can't set metadata to uncaught frozen object
- if (!isExtensible(it)) return true;
- // not necessary to add metadata
- if (!create) return false;
- // add missing metadata
- setMetadata(it);
- // return the store of weak collections IDs
- } return it[METADATA].weakData;
- };
-
- // add metadata on freeze-family methods calling
- var onFreeze = function (it) {
- if (freezing && meta.REQUIRED && isExtensible(it) && !has(it, METADATA)) setMetadata(it);
- return it;
- };
-
- var meta = module.exports = {
- REQUIRED: false,
- fastKey: fastKey,
- getWeakData: getWeakData,
- onFreeze: onFreeze
- };
-
- hiddenKeys[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 NativePrototype = NativeConstructor && NativeConstructor.prototype;
- var Constructor = NativeConstructor;
- var exported = {};
-
- var fixMethod = function (KEY) {
- var nativeMethod = NativePrototype[KEY];
- redefine(NativePrototype, KEY,
- KEY == 'add' ? function add(value) {
- 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);
- } : KEY == 'get' ? function get(key) {
- return IS_WEAK && !isObject(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);
- } : function set(key, value) {
- nativeMethod.call(this, key === 0 ? 0 : key, value);
- return this;
- }
- );
- };
-
- // eslint-disable-next-line max-len
- if (isForced_1(CONSTRUCTOR_NAME, typeof NativeConstructor != 'function' || !(IS_WEAK || NativePrototype.forEach && !fails(function () {
- new NativeConstructor().entries().next();
- })))) {
- // create collection constructor
- Constructor = common.getConstructor(wrapper, CONSTRUCTOR_NAME, IS_MAP, ADDER);
- internalMetadata.REQUIRED = true;
- } else if (isForced_1(CONSTRUCTOR_NAME, true)) {
- var instance = new Constructor();
- // early implementations not supports chaining
- var HASNT_CHAINING = instance[ADDER](IS_WEAK ? {} : -0, 1) != instance;
- // V8 ~ Chromium 40- weak-collections throws on primitives, but should return false
- var THROWS_ON_PRIMITIVES = fails(function () { instance.has(1); });
- // most early implementations doesn't supports iterables, most modern - not close it correctly
- // eslint-disable-next-line no-new
- var ACCEPT_ITERABLES = checkCorrectnessOfIteration(function (iterable) { new NativeConstructor(iterable); });
- // for early implementations -0 and +0 not the same
- var BUGGY_ZERO = !IS_WEAK && fails(function () {
- // V8 ~ Chromium 42- fails only with 5+ elements
- var $instance = new NativeConstructor();
- var index = 5;
- while (index--) $instance[ADDER](index, index);
- return !$instance.has(-0);
- });
-
- if (!ACCEPT_ITERABLES) {
- Constructor = wrapper(function (dummy, iterable) {
- anInstance(dummy, Constructor, CONSTRUCTOR_NAME);
- var that = inheritIfRequired(new NativeConstructor(), dummy, Constructor);
- if (iterable != undefined) iterate_1(iterable, that[ADDER], that, IS_MAP);
- return that;
- });
- Constructor.prototype = NativePrototype;
- NativePrototype.constructor = Constructor;
- }
-
- if (THROWS_ON_PRIMITIVES || BUGGY_ZERO) {
- fixMethod('delete');
- fixMethod('has');
- IS_MAP && fixMethod('get');
- }
-
- if (BUGGY_ZERO || HASNT_CHAINING) fixMethod(ADDER);
-
- // weak collections should not contains .clear method
- if (IS_WEAK && NativePrototype.clear) delete NativePrototype.clear;
- }
-
- exported[CONSTRUCTOR_NAME] = Constructor;
- _export({ global: true, forced: Constructor != NativeConstructor }, exported);
-
- setToStringTag(Constructor, CONSTRUCTOR_NAME);
-
- if (!IS_WEAK) common.setStrong(Constructor, CONSTRUCTOR_NAME, IS_MAP);
-
- return Constructor;
- };
-
- var defineProperty$8 = objectDefineProperty.f;
-
-
-
-
-
-
-
-
- var fastKey = internalMetadata.fastKey;
-
-
- var setInternalState$7 = 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, {
- type: CONSTRUCTOR_NAME,
- index: objectCreate(null),
- first: undefined,
- last: undefined,
- size: 0
- });
- if (!descriptors) that.size = 0;
- if (iterable != undefined) iterate_1(iterable, that[ADDER], that, IS_MAP);
- });
-
- var getInternalState = internalStateGetterFor(CONSTRUCTOR_NAME);
-
- var define = function (that, key, value) {
- var state = getInternalState(that);
- var entry = getEntry(that, key);
- var previous, index;
- // change existing entry
- if (entry) {
- entry.value = value;
- // create new entry
- } else {
- state.last = entry = {
- index: index = fastKey(key, true),
- key: key,
- value: value,
- previous: previous = state.last,
- next: undefined,
- removed: false
- };
- if (!state.first) state.first = entry;
- if (previous) previous.next = entry;
- if (descriptors) state.size++;
- else that.size++;
- // add to index
- if (index !== 'F') state.index[index] = entry;
- } return that;
- };
-
- var getEntry = function (that, key) {
- var state = getInternalState(that);
- // fast case
- var index = fastKey(key);
- var entry;
- if (index !== 'F') return state.index[index];
- // frozen object case
- for (entry = state.first; entry; entry = entry.next) {
- if (entry.key == key) return entry;
- }
- };
-
- redefineAll(C.prototype, {
- // 23.1.3.1 Map.prototype.clear()
- // 23.2.3.2 Set.prototype.clear()
- clear: function clear() {
- var that = this;
- var state = getInternalState(that);
- var data = state.index;
- var entry = state.first;
- while (entry) {
- entry.removed = true;
- if (entry.previous) entry.previous = entry.previous.next = undefined;
- delete data[entry.index];
- entry = entry.next;
- }
- state.first = state.last = undefined;
- 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)
- 'delete': function (key) {
- var that = this;
- var state = getInternalState(that);
- var entry = getEntry(that, key);
- if (entry) {
- var next = entry.next;
- var prev = entry.previous;
- delete state.index[entry.index];
- entry.removed = true;
- if (prev) prev.next = next;
- if (next) next.previous = prev;
- if (state.first == entry) state.first = next;
- if (state.last == entry) state.last = prev;
- if (descriptors) state.size--;
- 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)
- forEach: function forEach(callbackfn /* , that = undefined */) {
- var state = getInternalState(this);
- var boundFunction = functionBindContext(callbackfn, arguments.length > 1 ? arguments[1] : undefined, 3);
- var entry;
- while (entry = entry ? entry.next : state.first) {
- boundFunction(entry.value, entry.key, this);
- // revert to the last existing entry
- while (entry && entry.removed) entry = entry.previous;
- }
- },
- // 23.1.3.7 Map.prototype.has(key)
- // 23.2.3.7 Set.prototype.has(value)
- has: function has(key) {
- return !!getEntry(this, key);
- }
- });
-
- redefineAll(C.prototype, IS_MAP ? {
- // 23.1.3.6 Map.prototype.get(key)
- get: function get(key) {
- var entry = getEntry(this, key);
- return entry && entry.value;
- },
- // 23.1.3.9 Map.prototype.set(key, value)
- set: function set(key, value) {
- return define(this, key === 0 ? 0 : key, value);
- }
- } : {
- // 23.2.3.1 Set.prototype.add(value)
- add: function add(value) {
- return define(this, value = value === 0 ? 0 : value, value);
- }
- });
- if (descriptors) defineProperty$8(C.prototype, 'size', {
- get: function () {
- return getInternalState(this).size;
- }
- });
- return C;
- },
- setStrong: function (C, CONSTRUCTOR_NAME, IS_MAP) {
- 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
- defineIterator(C, CONSTRUCTOR_NAME, function (iterated, kind) {
- setInternalState$7(this, {
- type: ITERATOR_NAME,
- target: iterated,
- state: getInternalCollectionState(iterated),
- kind: kind,
- last: undefined
- });
- }, function () {
- var state = getInternalIteratorState(this);
- var kind = state.kind;
- var entry = state.last;
- // revert to the last existing entry
- while (entry && entry.removed) entry = entry.previous;
- // get next entry
- if (!state.target || !(state.last = entry = entry ? entry.next : state.state.first)) {
- // or finish the iteration
- state.target = undefined;
- return { value: undefined, done: true };
- }
- // return step by kind
- if (kind == 'keys') return { value: entry.key, done: false };
- if (kind == 'values') return { value: entry.value, done: false };
- 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
- setSpecies(CONSTRUCTOR_NAME);
- }
- };
-
- // `Set` constructor
- // https://tc39.github.io/ecma262/#sec-set-objects
- var es_set = collection('Set', function (init) {
- return function Set() { return init(this, arguments.length ? arguments[0] : undefined); };
- }, collectionStrong);
-
- function d3_ascending (a, b) {
- return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
- }
-
- function d3_bisector (f) {
- var delta = f;
- var compare = f;
-
- if (f.length === 1) {
- delta = function delta(d, x) {
- return f(d) - x;
- };
-
- compare = ascendingComparator(f);
- }
-
- function left(a, x, lo, hi) {
- if (lo == null) lo = 0;
- if (hi == null) hi = a.length;
-
- while (lo < hi) {
- var mid = lo + hi >>> 1;
- if (compare(a[mid], x) < 0) lo = mid + 1;else hi = mid;
- }
-
- return lo;
- }
-
- function right(a, x, lo, hi) {
- if (lo == null) lo = 0;
- if (hi == null) hi = a.length;
-
- while (lo < hi) {
- var mid = lo + hi >>> 1;
- if (compare(a[mid], x) > 0) hi = mid;else lo = mid + 1;
- }
-
- return lo;
- }
-
- function center(a, x, lo, hi) {
- if (lo == null) lo = 0;
- if (hi == null) hi = a.length;
- var i = left(a, x, lo, hi - 1);
- return i > lo && delta(a[i - 1], x) > -delta(a[i], x) ? i - 1 : i;
- }
-
- return {
- left: left,
- center: center,
- right: right
- };
- }
-
- function ascendingComparator(f) {
- return function (d, x) {
- return d3_ascending(f(d), x);
- };
- }
-
- // `Symbol.asyncIterator` well-known symbol
- // https://tc39.github.io/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.
- */
- var runtime = function (exports) {
-
- var Op = Object.prototype;
- var hasOwn = Op.hasOwnProperty;
- var undefined$1; // More compressible than void 0.
-
- var $Symbol = typeof Symbol === "function" ? Symbol : {};
- var iteratorSymbol = $Symbol.iterator || "@@iterator";
- var asyncIteratorSymbol = $Symbol.asyncIterator || "@@asyncIterator";
- var toStringTagSymbol = $Symbol.toStringTag || "@@toStringTag";
-
- function define(obj, key, value) {
- Object.defineProperty(obj, key, {
- value: value,
- enumerable: true,
- configurable: true,
- writable: true
- });
- return obj[key];
- }
-
- try {
- // IE 8 has a broken Object.defineProperty that only works on DOM objects.
- define({}, "");
- } catch (err) {
- define = function define(obj, key, value) {
- return obj[key] = value;
- };
- }
-
- function wrap(innerFn, outerFn, self, tryLocsList) {
- // If outerFn provided and outerFn.prototype is a Generator, then outerFn.prototype instanceof Generator.
- var protoGenerator = outerFn && outerFn.prototype instanceof Generator ? outerFn : Generator;
- var generator = Object.create(protoGenerator.prototype);
- var context = new Context(tryLocsList || []); // The ._invoke method unifies the implementations of the .next,
- // .throw, and .return methods.
-
- generator._invoke = makeInvokeMethod(innerFn, self, context);
- return generator;
- }
-
- exports.wrap = wrap; // Try/catch helper to minimize deoptimizations. Returns a completion
- // record like context.tryEntries[i].completion. This interface could
- // have been (and was previously) designed to take a closure to be
- // invoked without arguments, but in all the cases we care about we
- // already have an existing method we want to call, so there's no need
- // to create a new function object. We can even get away with assuming
- // the method takes exactly one argument, since that happens to be true
- // in every case, so we don't have to touch the arguments object. The
- // only additional allocation required is the completion record, which
- // has a stable shape and so hopefully should be cheap to allocate.
-
- function tryCatch(fn, obj, arg) {
- try {
- return {
- type: "normal",
- arg: fn.call(obj, arg)
- };
- } catch (err) {
- return {
- type: "throw",
- arg: err
- };
- }
- }
-
- var GenStateSuspendedStart = "suspendedStart";
- var GenStateSuspendedYield = "suspendedYield";
- var GenStateExecuting = "executing";
- var GenStateCompleted = "completed"; // Returning this object from the innerFn has the same effect as
- // breaking out of the dispatch switch statement.
-
- var ContinueSentinel = {}; // Dummy constructor functions that we use as the .constructor and
- // .constructor.prototype properties for functions that return Generator
- // objects. For full spec compliance, you may wish to configure your
- // minifier not to mangle the names of these two functions.
-
- function Generator() {}
-
- function GeneratorFunction() {}
-
- function GeneratorFunctionPrototype() {} // This is a polyfill for %IteratorPrototype% for environments that
- // don't natively support it.
-
-
- var IteratorPrototype = {};
-
- IteratorPrototype[iteratorSymbol] = function () {
- return this;
- };
-
- var getProto = Object.getPrototypeOf;
- var NativeIteratorPrototype = getProto && getProto(getProto(values([])));
-
- if (NativeIteratorPrototype && NativeIteratorPrototype !== Op && hasOwn.call(NativeIteratorPrototype, iteratorSymbol)) {
- // This environment has a native %IteratorPrototype%; use it instead
- // of the polyfill.
- IteratorPrototype = NativeIteratorPrototype;
- }
-
- var Gp = GeneratorFunctionPrototype.prototype = Generator.prototype = Object.create(IteratorPrototype);
- GeneratorFunction.prototype = Gp.constructor = GeneratorFunctionPrototype;
- GeneratorFunctionPrototype.constructor = GeneratorFunction;
- GeneratorFunction.displayName = define(GeneratorFunctionPrototype, toStringTagSymbol, "GeneratorFunction"); // Helper for defining the .next, .throw, and .return methods of the
- // Iterator interface in terms of a single ._invoke method.
-
- function defineIteratorMethods(prototype) {
- ["next", "throw", "return"].forEach(function (method) {
- define(prototype, method, function (arg) {
- return this._invoke(method, arg);
- });
- });
- }
-
- exports.isGeneratorFunction = function (genFun) {
- var ctor = typeof genFun === "function" && genFun.constructor;
- return ctor ? ctor === GeneratorFunction || // For the native GeneratorFunction constructor, the best we can
- // do is to check its .name property.
- (ctor.displayName || ctor.name) === "GeneratorFunction" : false;
- };
-
- exports.mark = function (genFun) {
- if (Object.setPrototypeOf) {
- Object.setPrototypeOf(genFun, GeneratorFunctionPrototype);
- } else {
- genFun.__proto__ = GeneratorFunctionPrototype;
- define(genFun, toStringTagSymbol, "GeneratorFunction");
- }
-
- genFun.prototype = Object.create(Gp);
- return genFun;
- }; // Within the body of any async function, `await x` is transformed to
- // `yield regeneratorRuntime.awrap(x)`, so that the runtime can test
- // `hasOwn.call(value, "__await")` to determine if the yielded value is
- // meant to be awaited.
-
-
- exports.awrap = function (arg) {
- return {
- __await: arg
- };
- };
-
- function AsyncIterator(generator, PromiseImpl) {
- function invoke(method, arg, resolve, reject) {
- var record = tryCatch(generator[method], generator, arg);
-
- if (record.type === "throw") {
- reject(record.arg);
- } else {
- var result = record.arg;
- var value = result.value;
-
- if (value && _typeof(value) === "object" && hasOwn.call(value, "__await")) {
- return PromiseImpl.resolve(value.__await).then(function (value) {
- invoke("next", value, resolve, reject);
- }, function (err) {
- invoke("throw", err, resolve, reject);
- });
- }
-
- return PromiseImpl.resolve(value).then(function (unwrapped) {
- // When a yielded Promise is resolved, its final value becomes
- // the .value of the Promise<{value,done}> result for the
- // current iteration.
- result.value = unwrapped;
- resolve(result);
- }, function (error) {
- // If a rejected Promise was yielded, throw the rejection back
- // into the async generator function so it can be handled there.
- return invoke("throw", error, resolve, reject);
- });
- }
- }
-
- var previousPromise;
-
- function enqueue(method, arg) {
- function callInvokeWithMethodAndArg() {
- return new PromiseImpl(function (resolve, reject) {
- invoke(method, arg, resolve, reject);
- });
- }
-
- return previousPromise = // If enqueue has been called before, then we want to wait until
- // all previous Promises have been resolved before calling invoke,
- // so that results are always delivered in the correct order. If
- // enqueue has not been called before, then it is important to
- // call invoke immediately, without waiting on a callback to fire,
- // so that the async generator function has the opportunity to do
- // any necessary setup in a predictable way. This predictability
- // is why the Promise constructor synchronously invokes its
- // executor callback, and why async functions synchronously
- // execute code before the first await. Since we implement simple
- // async functions in terms of async generators, it is especially
- // important to get this right, even though it requires care.
- previousPromise ? previousPromise.then(callInvokeWithMethodAndArg, // Avoid propagating failures to Promises returned by later
- // invocations of the iterator.
- callInvokeWithMethodAndArg) : callInvokeWithMethodAndArg();
- } // Define the unified helper method that is used to implement .next,
- // .throw, and .return (see defineIteratorMethods).
-
-
- this._invoke = enqueue;
- }
-
- defineIteratorMethods(AsyncIterator.prototype);
-
- AsyncIterator.prototype[asyncIteratorSymbol] = function () {
- return this;
- };
-
- exports.AsyncIterator = AsyncIterator; // Note that simple async functions are implemented on top of
- // AsyncIterator objects; they just return a Promise for the value of
- // the final result produced by the iterator.
-
- exports.async = function (innerFn, outerFn, self, tryLocsList, PromiseImpl) {
- if (PromiseImpl === void 0) PromiseImpl = Promise;
- var iter = new AsyncIterator(wrap(innerFn, outerFn, self, tryLocsList), PromiseImpl);
- return exports.isGeneratorFunction(outerFn) ? iter // If outerFn is a generator, return the full iterator.
- : iter.next().then(function (result) {
- return result.done ? result.value : iter.next();
- });
- };
-
- function makeInvokeMethod(innerFn, self, context) {
- var state = GenStateSuspendedStart;
- return function invoke(method, arg) {
- if (state === GenStateExecuting) {
- throw new Error("Generator is already running");
- }
-
- if (state === GenStateCompleted) {
- if (method === "throw") {
- throw arg;
- } // Be forgiving, per 25.3.3.3.3 of the spec:
- // https://people.mozilla.org/~jorendorff/es6-draft.html#sec-generatorresume
-
-
- return doneResult();
- }
-
- context.method = method;
- context.arg = arg;
-
- while (true) {
- var delegate = context.delegate;
-
- if (delegate) {
- var delegateResult = maybeInvokeDelegate(delegate, context);
-
- if (delegateResult) {
- if (delegateResult === ContinueSentinel) continue;
- return delegateResult;
- }
- }
-
- if (context.method === "next") {
- // Setting context._sent for legacy support of Babel's
- // function.sent implementation.
- context.sent = context._sent = context.arg;
- } else if (context.method === "throw") {
- if (state === GenStateSuspendedStart) {
- state = GenStateCompleted;
- throw context.arg;
- }
-
- context.dispatchException(context.arg);
- } else if (context.method === "return") {
- context.abrupt("return", context.arg);
- }
-
- state = GenStateExecuting;
- var record = tryCatch(innerFn, self, context);
-
- if (record.type === "normal") {
- // If an exception is thrown from innerFn, we leave state ===
- // GenStateExecuting and loop back for another invocation.
- state = context.done ? GenStateCompleted : GenStateSuspendedYield;
-
- if (record.arg === ContinueSentinel) {
- continue;
- }
-
- return {
- value: record.arg,
- done: context.done
- };
- } else if (record.type === "throw") {
- state = GenStateCompleted; // Dispatch the exception by looping back around to the
- // context.dispatchException(context.arg) call above.
-
- context.method = "throw";
- context.arg = record.arg;
- }
- }
- };
- } // Call delegate.iterator[context.method](context.arg) and handle the
- // result, either by returning a { value, done } result from the
- // delegate iterator, or by modifying context.method and context.arg,
- // setting context.delegate to null, and returning the ContinueSentinel.
-
-
- function maybeInvokeDelegate(delegate, context) {
- var method = delegate.iterator[context.method];
-
- if (method === undefined$1) {
- // A .throw or .return when the delegate iterator has no .throw
- // method always terminates the yield* loop.
- context.delegate = null;
-
- if (context.method === "throw") {
- // Note: ["return"] must be used for ES3 parsing compatibility.
- if (delegate.iterator["return"]) {
- // If the delegate iterator has a return method, give it a
- // chance to clean up.
- context.method = "return";
- context.arg = undefined$1;
- maybeInvokeDelegate(delegate, context);
-
- if (context.method === "throw") {
- // If maybeInvokeDelegate(context) changed context.method from
- // "return" to "throw", let that override the TypeError below.
- return ContinueSentinel;
- }
- }
-
- context.method = "throw";
- context.arg = new TypeError("The iterator does not provide a 'throw' method");
- }
-
- return ContinueSentinel;
- }
-
- var record = tryCatch(method, delegate.iterator, context.arg);
-
- if (record.type === "throw") {
- context.method = "throw";
- context.arg = record.arg;
- context.delegate = null;
- return ContinueSentinel;
- }
-
- var info = record.arg;
-
- if (!info) {
- context.method = "throw";
- context.arg = new TypeError("iterator result is not an object");
- context.delegate = null;
- return ContinueSentinel;
- }
-
- if (info.done) {
- // Assign the result of the finished delegate to the temporary
- // variable specified by delegate.resultName (see delegateYield).
- context[delegate.resultName] = info.value; // Resume execution at the desired location (see delegateYield).
-
- context.next = delegate.nextLoc; // If context.method was "throw" but the delegate handled the
- // exception, let the outer generator proceed normally. If
- // context.method was "next", forget context.arg since it has been
- // "consumed" by the delegate iterator. If context.method was
- // "return", allow the original .return call to continue in the
- // outer generator.
-
- if (context.method !== "return") {
- context.method = "next";
- context.arg = undefined$1;
- }
- } else {
- // Re-yield the result returned by the delegate method.
- return info;
- } // The delegate iterator is finished, so forget it and continue with
- // the outer generator.
-
-
- context.delegate = null;
- return ContinueSentinel;
- } // Define Generator.prototype.{next,throw,return} in terms of the
- // unified ._invoke helper method.
-
-
- defineIteratorMethods(Gp);
- define(Gp, toStringTagSymbol, "Generator"); // A Generator should always return itself as the iterator object when the
- // @@iterator function is called on it. Some browsers' implementations of the
- // iterator prototype chain incorrectly implement this, causing the Generator
- // object to not be returned from this call. This ensures that doesn't happen.
- // See https://github.com/facebook/regenerator/issues/274 for more details.
-
- Gp[iteratorSymbol] = function () {
- return this;
- };
-
- Gp.toString = function () {
- return "[object Generator]";
- };
-
- function pushTryEntry(locs) {
- var entry = {
- tryLoc: locs[0]
- };
-
- if (1 in locs) {
- entry.catchLoc = locs[1];
- }
-
- if (2 in locs) {
- entry.finallyLoc = locs[2];
- entry.afterLoc = locs[3];
- }
-
- this.tryEntries.push(entry);
- }
-
- function resetTryEntry(entry) {
- var record = entry.completion || {};
- record.type = "normal";
- delete record.arg;
- entry.completion = record;
- }
-
- function Context(tryLocsList) {
- // The root entry object (effectively a try statement without a catch
- // or a finally block) gives us a place to store values thrown from
- // locations where there is no enclosing try statement.
- this.tryEntries = [{
- tryLoc: "root"
- }];
- tryLocsList.forEach(pushTryEntry, this);
- this.reset(true);
- }
-
- exports.keys = function (object) {
- var keys = [];
-
- for (var key in object) {
- keys.push(key);
- }
-
- keys.reverse(); // Rather than returning an object with a next method, we keep
- // things simple and return the next function itself.
-
- return function next() {
- while (keys.length) {
- var key = keys.pop();
-
- if (key in object) {
- next.value = key;
- next.done = false;
- return next;
- }
- } // To avoid creating an additional object, we just hang the .value
- // and .done properties off the next function object itself. This
- // also ensures that the minifier will not anonymize the function.
-
-
- next.done = true;
- return next;
- };
- };
-
- function values(iterable) {
- if (iterable) {
- var iteratorMethod = iterable[iteratorSymbol];
-
- if (iteratorMethod) {
- return iteratorMethod.call(iterable);
- }
-
- if (typeof iterable.next === "function") {
- return iterable;
- }
-
- if (!isNaN(iterable.length)) {
- var i = -1,
- next = function next() {
- while (++i < iterable.length) {
- if (hasOwn.call(iterable, i)) {
- next.value = iterable[i];
- next.done = false;
- return next;
- }
- }
-
- next.value = undefined$1;
- next.done = true;
- return next;
- };
-
- return next.next = next;
- }
- } // Return an iterator with no values.
-
-
- return {
- next: doneResult
- };
- }
-
- exports.values = values;
-
- function doneResult() {
- return {
- value: undefined$1,
- done: true
- };
- }
-
- Context.prototype = {
- constructor: Context,
- reset: function reset(skipTempReset) {
- this.prev = 0;
- this.next = 0; // Resetting context._sent for legacy support of Babel's
- // function.sent implementation.
-
- this.sent = this._sent = undefined$1;
- this.done = false;
- this.delegate = null;
- this.method = "next";
- this.arg = undefined$1;
- this.tryEntries.forEach(resetTryEntry);
-
- if (!skipTempReset) {
- for (var name in this) {
- // Not sure about the optimal order of these conditions:
- if (name.charAt(0) === "t" && hasOwn.call(this, name) && !isNaN(+name.slice(1))) {
- this[name] = undefined$1;
- }
- }
- }
- },
- stop: function stop() {
- this.done = true;
- var rootEntry = this.tryEntries[0];
- var rootRecord = rootEntry.completion;
-
- if (rootRecord.type === "throw") {
- throw rootRecord.arg;
- }
-
- return this.rval;
- },
- dispatchException: function dispatchException(exception) {
- if (this.done) {
- throw exception;
- }
-
- var context = this;
-
- function handle(loc, caught) {
- record.type = "throw";
- record.arg = exception;
- context.next = loc;
-
- if (caught) {
- // If the dispatched exception was caught by a catch block,
- // then let that catch block handle the exception normally.
- context.method = "next";
- context.arg = undefined$1;
- }
-
- return !!caught;
- }
-
- for (var i = this.tryEntries.length - 1; i >= 0; --i) {
- var entry = this.tryEntries[i];
- var record = entry.completion;
-
- if (entry.tryLoc === "root") {
- // Exception thrown outside of any try block that could handle
- // it, so set the completion value of the entire function to
- // throw the exception.
- return handle("end");
- }
-
- if (entry.tryLoc <= this.prev) {
- var hasCatch = hasOwn.call(entry, "catchLoc");
- var hasFinally = hasOwn.call(entry, "finallyLoc");
-
- if (hasCatch && hasFinally) {
- if (this.prev < entry.catchLoc) {
- return handle(entry.catchLoc, true);
- } else if (this.prev < entry.finallyLoc) {
- return handle(entry.finallyLoc);
- }
- } else if (hasCatch) {
- if (this.prev < entry.catchLoc) {
- return handle(entry.catchLoc, true);
- }
- } else if (hasFinally) {
- if (this.prev < entry.finallyLoc) {
- return handle(entry.finallyLoc);
- }
- } else {
- throw new Error("try statement without catch or finally");
- }
- }
- }
- },
- abrupt: function abrupt(type, arg) {
- for (var i = this.tryEntries.length - 1; i >= 0; --i) {
- var entry = this.tryEntries[i];
-
- if (entry.tryLoc <= this.prev && hasOwn.call(entry, "finallyLoc") && this.prev < entry.finallyLoc) {
- var finallyEntry = entry;
- break;
- }
- }
-
- if (finallyEntry && (type === "break" || type === "continue") && finallyEntry.tryLoc <= arg && arg <= finallyEntry.finallyLoc) {
- // Ignore the finally entry if control is not jumping to a
- // location outside the try/catch block.
- finallyEntry = null;
- }
-
- var record = finallyEntry ? finallyEntry.completion : {};
- record.type = type;
- record.arg = arg;
-
- if (finallyEntry) {
- this.method = "next";
- this.next = finallyEntry.finallyLoc;
- return ContinueSentinel;
- }
-
- return this.complete(record);
- },
- complete: function complete(record, afterLoc) {
- if (record.type === "throw") {
- throw record.arg;
- }
-
- if (record.type === "break" || record.type === "continue") {
- this.next = record.arg;
- } else if (record.type === "return") {
- this.rval = this.arg = record.arg;
- this.method = "return";
- this.next = "end";
- } else if (record.type === "normal" && afterLoc) {
- this.next = afterLoc;
- }
-
- return ContinueSentinel;
- },
- finish: function finish(finallyLoc) {
- for (var i = this.tryEntries.length - 1; i >= 0; --i) {
- var entry = this.tryEntries[i];
-
- if (entry.finallyLoc === finallyLoc) {
- this.complete(entry.completion, entry.afterLoc);
- resetTryEntry(entry);
- return ContinueSentinel;
- }
- }
- },
- "catch": function _catch(tryLoc) {
- for (var i = this.tryEntries.length - 1; i >= 0; --i) {
- var entry = this.tryEntries[i];
-
- if (entry.tryLoc === tryLoc) {
- var record = entry.completion;
-
- if (record.type === "throw") {
- var thrown = record.arg;
- resetTryEntry(entry);
- }
-
- return thrown;
- }
- } // The context.catch method must only be called with a location
- // argument that corresponds to a known catch block.
-
-
- throw new Error("illegal catch attempt");
- },
- delegateYield: function delegateYield(iterable, resultName, nextLoc) {
- this.delegate = {
- iterator: values(iterable),
- resultName: resultName,
- nextLoc: nextLoc
- };
-
- if (this.method === "next") {
- // Deliberately forget the last sent value so that we don't
- // accidentally pass it on to the delegate.
- this.arg = undefined$1;
- }
-
- return ContinueSentinel;
- }
- }; // Regardless of whether this script is executing as a CommonJS module
- // or not, return the runtime object so that we can declare the variable
- // regeneratorRuntime in the outer scope, which allows this module to be
- // injected easily by `bin/regenerator --include-runtime script.js`.
-
- return exports;
- }( // If this script is executing as a CommonJS module, use module.exports
- // 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 );
-
- try {
- regeneratorRuntime = runtime;
- } catch (accidentalStrictMode) {
- // This module should not be running in strict mode, so the above
- // assignment should always work unless something is misconfigured. Just
- // in case runtime.js accidentally runs in strict mode, we can escape
- // strict mode using a global Function call. This could conceivably fail
- // if a Content Security Policy forbids using Function, but in that case
- // the proper solution is to fix the accidental strict mode problem. If
- // you've misconfigured your bundler to force strict mode and applied a
- // CSP to forbid Function, and you're not willing to fix either of those
- // problems, please detail your unique predicament in a GitHub issue.
- Function("r", "regeneratorRuntime = r")(runtime);
- }
- });
-
- var _marked = /*#__PURE__*/regeneratorRuntime.mark(numbers);
-
- function number (x) {
- return x === null ? NaN : +x;
- }
- function numbers(values, valueof) {
- var _iterator, _step, value, index, _iterator2, _step2, _value;
-
- return regeneratorRuntime.wrap(function numbers$(_context) {
- while (1) {
- switch (_context.prev = _context.next) {
- case 0:
- if (!(valueof === undefined)) {
- _context.next = 21;
- break;
- }
-
- _iterator = _createForOfIteratorHelper(values);
- _context.prev = 2;
-
- _iterator.s();
-
- case 4:
- if ((_step = _iterator.n()).done) {
- _context.next = 11;
- break;
- }
-
- value = _step.value;
-
- if (!(value != null && (value = +value) >= value)) {
- _context.next = 9;
- break;
- }
-
- _context.next = 9;
- return value;
-
- case 9:
- _context.next = 4;
- break;
-
- case 11:
- _context.next = 16;
- break;
-
- case 13:
- _context.prev = 13;
- _context.t0 = _context["catch"](2);
-
- _iterator.e(_context.t0);
-
- case 16:
- _context.prev = 16;
-
- _iterator.f();
-
- return _context.finish(16);
-
- case 19:
- _context.next = 40;
- break;
-
- case 21:
- index = -1;
- _iterator2 = _createForOfIteratorHelper(values);
- _context.prev = 23;
-
- _iterator2.s();
-
- case 25:
- if ((_step2 = _iterator2.n()).done) {
- _context.next = 32;
- break;
- }
-
- _value = _step2.value;
-
- if (!((_value = valueof(_value, ++index, values)) != null && (_value = +_value) >= _value)) {
- _context.next = 30;
- break;
- }
-
- _context.next = 30;
- return _value;
-
- case 30:
- _context.next = 25;
- break;
-
- case 32:
- _context.next = 37;
- break;
-
- case 34:
- _context.prev = 34;
- _context.t1 = _context["catch"](23);
-
- _iterator2.e(_context.t1);
-
- case 37:
- _context.prev = 37;
-
- _iterator2.f();
-
- return _context.finish(37);
-
- case 40:
- case "end":
- return _context.stop();
- }
- }
- }, _marked, 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;
-
- // `Array.prototype.fill` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.fill
- _export({ target: 'Array', proto: true }, {
- fill: arrayFill
- });
-
- // https://tc39.github.io/ecma262/#sec-array.prototype-@@unscopables
- addToUnscopables('fill');
-
- var INCORRECT_ITERATION$1 = !checkCorrectnessOfIteration(function (iterable) {
- Array.from(iterable);
- });
-
- // `Array.from` method
- // https://tc39.github.io/ecma262/#sec-array.from
- _export({ target: 'Array', stat: true, forced: INCORRECT_ITERATION$1 }, {
- from: arrayFrom
- });
-
- var $some$1 = arrayIteration.some;
-
-
-
- var STRICT_METHOD$4 = arrayMethodIsStrict('some');
- var USES_TO_LENGTH$7 = arrayMethodUsesToLength('some');
-
- // `Array.prototype.some` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.some
- _export({ target: 'Array', proto: true, forced: !STRICT_METHOD$4 || !USES_TO_LENGTH$7 }, {
- some: function some(callbackfn /* , thisArg */) {
- return $some$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- // `Float64Array` constructor
- // https://tc39.github.io/ecma262/#sec-typedarray-objects
- typedArrayConstructor('Float64', function (init) {
- return function Float64Array(data, byteOffset, length) {
- return init(this, data, byteOffset, length);
- };
- });
-
- var exportTypedArrayStaticMethod$1 = arrayBufferViewCore.exportTypedArrayStaticMethod;
-
-
- // `%TypedArray%.from` method
- // https://tc39.github.io/ecma262/#sec-%typedarray%.from
- exportTypedArrayStaticMethod$1('from', typedArrayFrom, typedArrayConstructorsRequireWrappers);
-
- function d3_descending (a, b) {
- return b < a ? -1 : b > a ? 1 : b >= a ? 0 : NaN;
- }
-
- // https://github.com/python/cpython/blob/a74eea238f5baba15797e2e8b570d153bc8690a7/Modules/mathmodule.c#L1423
- var Adder = /*#__PURE__*/function () {
- function Adder() {
- _classCallCheck(this, Adder);
-
- this._partials = new Float64Array(32);
- this._n = 0;
- }
-
- _createClass(Adder, [{
- key: "add",
- value: function add(x) {
- var p = this._partials;
- var i = 0;
-
- for (var j = 0; j < this._n && j < 32; j++) {
- var y = p[j],
- hi = x + y,
- lo = Math.abs(x) < Math.abs(y) ? x - (hi - y) : y - (hi - x);
- if (lo) p[i++] = lo;
- x = hi;
- }
-
- p[i] = x;
- this._n = i + 1;
- return this;
- }
- }, {
- key: "valueOf",
- value: function valueOf() {
- var p = this._partials;
- var n = this._n,
- x,
- y,
- lo,
- hi = 0;
-
- if (n > 0) {
- hi = p[--n];
-
- while (n > 0) {
- x = hi;
- y = p[--n];
- hi = x + y;
- lo = y - (hi - x);
- if (lo) break;
- }
-
- if (n > 0 && (lo < 0 && p[n - 1] < 0 || lo > 0 && p[n - 1] > 0)) {
- y = lo * 2;
- x = hi + y;
- if (y == x - hi) hi = x;
- }
- }
-
- return hi;
- }
- }]);
-
- return Adder;
- }();
-
- // `Map` constructor
- // https://tc39.github.io/ecma262/#sec-map-objects
- var es_map = collection('Map', function (init) {
- return function Map() { return init(this, arguments.length ? arguments[0] : undefined); };
- }, collectionStrong);
-
- var e10 = Math.sqrt(50),
- e5 = Math.sqrt(10),
- e2 = Math.sqrt(2);
- function ticks (start, stop, count) {
- var reverse,
- i = -1,
- n,
- ticks,
- step;
- stop = +stop, start = +start, count = +count;
- if (start === stop && count > 0) return [start];
- if (reverse = stop < start) n = start, start = stop, stop = n;
- 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));
-
- while (++i < n) {
- ticks[i] = (start + i) * step;
- }
- } else {
- step = -step;
- start = Math.ceil(start * step);
- stop = Math.floor(stop * step);
- ticks = new Array(n = Math.ceil(stop - start + 1));
-
- while (++i < n) {
- ticks[i] = (start + i) / step;
- }
- }
-
- if (reverse) ticks.reverse();
- return ticks;
- }
- function tickIncrement(start, stop, count) {
- var step = (stop - start) / Math.max(0, count),
- power = Math.floor(Math.log(step) / Math.LN10),
- error = step / Math.pow(10, power);
- return power >= 0 ? (error >= e10 ? 10 : error >= e5 ? 5 : error >= e2 ? 2 : 1) * Math.pow(10, power) : -Math.pow(10, -power) / (error >= e10 ? 10 : error >= e5 ? 5 : error >= e2 ? 2 : 1);
- }
- function tickStep(start, stop, count) {
- var step0 = Math.abs(stop - start) / Math.max(0, count),
- step1 = Math.pow(10, Math.floor(Math.log(step0) / Math.LN10)),
- error = step0 / step1;
- if (error >= e10) step1 *= 10;else if (error >= e5) step1 *= 5;else if (error >= e2) step1 *= 2;
- return stop < start ? -step1 : step1;
- }
-
- function max$4(values, valueof) {
- var max;
-
- if (valueof === undefined) {
- var _iterator = _createForOfIteratorHelper(values),
- _step;
-
- try {
- for (_iterator.s(); !(_step = _iterator.n()).done;) {
- var value = _step.value;
-
- if (value != null && (max < value || max === undefined && value >= value)) {
- max = value;
- }
- }
- } catch (err) {
- _iterator.e(err);
- } finally {
- _iterator.f();
- }
- } else {
- var index = -1;
-
- var _iterator2 = _createForOfIteratorHelper(values),
- _step2;
-
- try {
- for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
- var _value = _step2.value;
-
- if ((_value = valueof(_value, ++index, values)) != null && (max < _value || max === undefined && _value >= _value)) {
- max = _value;
- }
- }
- } catch (err) {
- _iterator2.e(err);
- } finally {
- _iterator2.f();
- }
- }
-
- return max;
- }
-
- function min$7(values, valueof) {
- var min;
-
- if (valueof === undefined) {
- var _iterator = _createForOfIteratorHelper(values),
- _step;
-
- try {
- for (_iterator.s(); !(_step = _iterator.n()).done;) {
- var value = _step.value;
-
- if (value != null && (min > value || min === undefined && value >= value)) {
- min = value;
- }
- }
- } catch (err) {
- _iterator.e(err);
- } finally {
- _iterator.f();
- }
- } else {
- var index = -1;
-
- var _iterator2 = _createForOfIteratorHelper(values),
- _step2;
-
- try {
- for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
- var _value = _step2.value;
-
- if ((_value = valueof(_value, ++index, values)) != null && (min > _value || min === undefined && _value >= _value)) {
- min = _value;
- }
- }
- } catch (err) {
- _iterator2.e(err);
- } finally {
- _iterator2.f();
- }
- }
-
- return min;
- }
-
- // ISC license, Copyright 2018 Vladimir Agafonkin.
-
- function quickselect(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;
-
- 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));
- quickselect(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);
-
- while (i < j) {
- swap(array, i, j), ++i, --j;
-
- while (compare(array[i], t) < 0) {
- ++i;
- }
-
- while (compare(array[j], t) > 0) {
- --j;
- }
- }
-
- if (compare(array[left], t) === 0) swap(array, left, j);else ++j, swap(array, j, right);
- if (j <= k) left = j + 1;
- if (k <= j) right = j - 1;
- }
-
- return array;
- }
-
- function swap(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);
- 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));
- return value0 + (value1 - value0) * (i - i0);
- }
-
- function d3_median (values, valueof) {
- return quantile(values, 0.5, valueof);
- }
-
- var _marked$1 = /*#__PURE__*/regeneratorRuntime.mark(flatten);
-
- function flatten(arrays) {
- var _iterator, _step, array;
-
- return regeneratorRuntime.wrap(function flatten$(_context) {
- while (1) {
- switch (_context.prev = _context.next) {
- case 0:
- _iterator = _createForOfIteratorHelper(arrays);
- _context.prev = 1;
-
- _iterator.s();
-
- case 3:
- if ((_step = _iterator.n()).done) {
- _context.next = 8;
- break;
- }
-
- array = _step.value;
- return _context.delegateYield(array, "t0", 6);
-
- case 6:
- _context.next = 3;
- break;
-
- case 8:
- _context.next = 13;
- break;
-
- case 10:
- _context.prev = 10;
- _context.t1 = _context["catch"](1);
-
- _iterator.e(_context.t1);
-
- case 13:
- _context.prev = 13;
-
- _iterator.f();
-
- return _context.finish(13);
-
- case 16:
- case "end":
- return _context.stop();
- }
- }
- }, _marked$1, null, [[1, 10, 13, 16]]);
- }
-
- function merge(arrays) {
- return Array.from(flatten(arrays));
- }
-
- function range (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,
- range = new Array(n);
-
- while (++i < n) {
- range[i] = start + i * step;
- }
-
- return range;
- }
-
- var test$2 = [];
- var nativeSort = test$2.sort;
-
- // IE8-
- var FAILS_ON_UNDEFINED = fails(function () {
- test$2.sort(undefined);
- });
- // V8 bug
- var FAILS_ON_NULL = fails(function () {
- test$2.sort(null);
- });
- // Old WebKit
- var STRICT_METHOD$5 = arrayMethodIsStrict('sort');
-
- var FORCED$a = FAILS_ON_UNDEFINED || !FAILS_ON_NULL || !STRICT_METHOD$5;
-
- // `Array.prototype.sort` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.sort
- _export({ target: 'Array', proto: true, forced: FORCED$a }, {
- sort: function sort(comparefn) {
- return comparefn === undefined
- ? nativeSort.call(toObject(this))
- : nativeSort.call(toObject(this), aFunction$1(comparefn));
- }
- });
-
- // `SameValue` abstract operation
- // https://tc39.github.io/ecma262/#sec-samevalue
- var sameValue = Object.is || function is(x, y) {
- // eslint-disable-next-line no-self-compare
- return x === y ? x !== 0 || 1 / x === 1 / y : x != x && y != y;
- };
-
- var $hypot = Math.hypot;
- var abs$1 = Math.abs;
- var sqrt = Math.sqrt;
-
- // Chrome 77 bug
- // https://bugs.chromium.org/p/v8/issues/detail?id=9546
- var BUGGY = !!$hypot && $hypot(Infinity, NaN) !== Infinity;
-
- // `Math.hypot` method
- // https://tc39.github.io/ecma262/#sec-math.hypot
- _export({ target: 'Math', stat: true, forced: BUGGY }, {
- hypot: function hypot(value1, value2) { // eslint-disable-line no-unused-vars
- var sum = 0;
- var i = 0;
- var aLen = arguments.length;
- var larg = 0;
- var arg, div;
- while (i < aLen) {
- arg = abs$1(arguments[i++]);
- if (larg < arg) {
- div = larg / arg;
- sum = sum * div * div + 1;
- larg = arg;
- } else if (arg > 0) {
- div = arg / larg;
- sum += div * div;
- } else sum += arg;
- }
- return larg === Infinity ? Infinity : larg * sqrt(sum);
- }
- });
-
- // `Math.sign` method implementation
- // https://tc39.github.io/ecma262/#sec-math.sign
- var mathSign = Math.sign || function sign(x) {
- // eslint-disable-next-line no-self-compare
- return (x = +x) == 0 || x != x ? x : x < 0 ? -1 : 1;
- };
-
- // `Math.sign` method
- // https://tc39.github.io/ecma262/#sec-math.sign
- _export({ target: 'Math', stat: true }, {
- sign: mathSign
- });
-
- var epsilon = 1e-6;
- var epsilon2 = 1e-12;
- var pi = Math.PI;
- var halfPi = pi / 2;
- var quarterPi = pi / 4;
- var tau = pi * 2;
- var degrees = 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 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 tan = Math.tan;
- function acos(x) {
- return x > 1 ? 0 : x < -1 ? pi : Math.acos(x);
- }
- function asin(x) {
- return x > 1 ? halfPi : x < -1 ? -halfPi : Math.asin(x);
- }
-
- function noop() {}
-
- function streamGeometry(geometry, stream) {
- if (geometry && streamGeometryType.hasOwnProperty(geometry.type)) {
- streamGeometryType[geometry.type](geometry, stream);
- }
- }
-
- var streamObjectType = {
- Feature: function Feature(object, stream) {
- streamGeometry(object.geometry, stream);
- },
- FeatureCollection: function FeatureCollection(object, stream) {
- var features = object.features,
- i = -1,
- n = features.length;
-
- while (++i < n) {
- streamGeometry(features[i].geometry, stream);
- }
- }
- };
- var streamGeometryType = {
- Sphere: function Sphere(object, stream) {
- stream.sphere();
- },
- Point: function Point(object, stream) {
- object = object.coordinates;
- stream.point(object[0], object[1], object[2]);
- },
- MultiPoint: function MultiPoint(object, stream) {
- var coordinates = object.coordinates,
- i = -1,
- n = coordinates.length;
-
- while (++i < n) {
- object = coordinates[i], stream.point(object[0], object[1], object[2]);
- }
- },
- LineString: function LineString(object, stream) {
- streamLine(object.coordinates, stream, 0);
- },
- MultiLineString: function MultiLineString(object, stream) {
- var coordinates = object.coordinates,
- i = -1,
- n = coordinates.length;
-
- while (++i < n) {
- streamLine(coordinates[i], stream, 0);
- }
- },
- Polygon: function Polygon(object, stream) {
- streamPolygon(object.coordinates, stream);
- },
- MultiPolygon: function MultiPolygon(object, stream) {
- var coordinates = object.coordinates,
- i = -1,
- n = coordinates.length;
-
- while (++i < n) {
- streamPolygon(coordinates[i], stream);
- }
- },
- GeometryCollection: function GeometryCollection(object, stream) {
- var geometries = object.geometries,
- i = -1,
- n = geometries.length;
-
- while (++i < n) {
- streamGeometry(geometries[i], stream);
- }
- }
- };
-
- function streamLine(coordinates, stream, closed) {
- var i = -1,
- n = coordinates.length - closed,
- coordinate;
- stream.lineStart();
-
- while (++i < n) {
- coordinate = coordinates[i], stream.point(coordinate[0], coordinate[1], coordinate[2]);
- }
-
- stream.lineEnd();
- }
-
- function streamPolygon(coordinates, stream) {
- var i = -1,
- n = coordinates.length;
- stream.polygonStart();
-
- while (++i < n) {
- streamLine(coordinates[i], stream, 1);
- }
-
- stream.polygonEnd();
- }
-
- function d3_geoStream (object, stream) {
- if (object && streamObjectType.hasOwnProperty(object.type)) {
- streamObjectType[object.type](object, stream);
- } else {
- streamGeometry(object, stream);
- }
- }
-
- var areaRingSum = new Adder(); // hello?
-
- var areaSum = new Adder(),
- lambda00,
- phi00,
- lambda0,
- cosPhi0,
- sinPhi0;
- var areaStream = {
- point: noop,
- lineStart: noop,
- lineEnd: noop,
- polygonStart: function polygonStart() {
- areaRingSum = new Adder();
- areaStream.lineStart = areaRingStart;
- areaStream.lineEnd = areaRingEnd;
- },
- polygonEnd: function polygonEnd() {
- var areaRing = +areaRingSum;
- areaSum.add(areaRing < 0 ? tau + areaRing : areaRing);
- this.lineStart = this.lineEnd = this.point = noop;
- },
- sphere: function sphere() {
- areaSum.add(tau);
- }
- };
-
- function areaRingStart() {
- areaStream.point = areaPointFirst;
- }
-
- function areaRingEnd() {
- areaPoint(lambda00, phi00);
- }
-
- function areaPointFirst(lambda, phi) {
- areaStream.point = areaPoint;
- lambda00 = lambda, phi00 = phi;
- lambda *= radians, phi *= radians;
- lambda0 = lambda, cosPhi0 = cos(phi = phi / 2 + quarterPi), sinPhi0 = sin(phi);
- }
-
- function areaPoint(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,
- sdLambda = dLambda >= 0 ? 1 : -1,
- adLambda = sdLambda * dLambda,
- cosPhi = cos(phi),
- sinPhi = sin(phi),
- k = sinPhi0 * sinPhi,
- u = cosPhi0 * cosPhi + k * cos(adLambda),
- v = k * sdLambda * sin(adLambda);
- areaRingSum.add(atan2(v, u)); // Advance the previous points.
-
- lambda0 = lambda, cosPhi0 = cosPhi, sinPhi0 = sinPhi;
- }
-
- function d3_geoArea (object) {
- areaSum = new Adder();
- d3_geoStream(object, areaStream);
- return areaSum * 2;
- }
-
- function spherical(cartesian) {
- return [atan2(cartesian[1], cartesian[0]), asin(cartesian[2])];
- }
- function cartesian(spherical) {
- var lambda = spherical[0],
- phi = spherical[1],
- cosPhi = cos(phi);
- return [cosPhi * cos(lambda), cosPhi * sin(lambda), sin(phi)];
- }
- function cartesianDot(a, b) {
- return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
- }
- function cartesianCross(a, b) {
- return [a[1] * b[2] - a[2] * b[1], a[2] * b[0] - a[0] * b[2], a[0] * b[1] - a[1] * b[0]];
- } // TODO return a
-
- function cartesianAddInPlace(a, b) {
- a[0] += b[0], a[1] += b[1], a[2] += b[2];
- }
- function cartesianScale(vector, k) {
- return [vector[0] * k, vector[1] * k, vector[2] * k];
- } // TODO return d
-
- function cartesianNormalizeInPlace(d) {
- var l = sqrt$1(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
- p0, // previous 3D point
- deltaSum, ranges, range$1;
- var boundsStream = {
- point: boundsPoint,
- lineStart: boundsLineStart,
- lineEnd: boundsLineEnd,
- polygonStart: function polygonStart() {
- boundsStream.point = boundsRingPoint;
- boundsStream.lineStart = boundsRingStart;
- boundsStream.lineEnd = boundsRingEnd;
- deltaSum = new Adder();
- areaStream.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;
- },
- sphere: function sphere() {
- lambda0$1 = -(lambda1 = 180), phi0 = -(phi1 = 90);
- }
- };
-
- function boundsPoint(lambda, phi) {
- ranges.push(range$1 = [lambda0$1 = lambda, lambda1 = lambda]);
- if (phi < phi0) phi0 = phi;
- if (phi > phi1) phi1 = phi;
- }
-
- function linePoint(lambda, phi) {
- var p = cartesian([lambda * radians, phi * radians]);
-
- if (p0) {
- var normal = cartesianCross(p0, p),
- equatorial = [normal[1], -normal[0], 0],
- inflection = cartesianCross(equatorial, normal);
- cartesianNormalizeInPlace(inflection);
- inflection = spherical(inflection);
- var delta = lambda - lambda2,
- sign = delta > 0 ? 1 : -1,
- lambdai = inflection[0] * degrees * sign,
- phii,
- antimeridian = abs$2(delta) > 180;
-
- if (antimeridian ^ (sign * lambda2 < lambdai && lambdai < sign * lambda)) {
- phii = inflection[1] * degrees;
- if (phii > phi1) phi1 = phii;
- } else if (lambdai = (lambdai + 360) % 360 - 180, antimeridian ^ (sign * lambda2 < lambdai && lambdai < sign * lambda)) {
- phii = -inflection[1] * degrees;
- if (phii < phi0) phi0 = phii;
- } else {
- if (phi < phi0) phi0 = phi;
- if (phi > phi1) phi1 = phi;
- }
-
- if (antimeridian) {
- if (lambda < lambda2) {
- if (angle(lambda0$1, lambda) > angle(lambda0$1, lambda1)) lambda1 = lambda;
- } else {
- if (angle(lambda, lambda1) > angle(lambda0$1, lambda1)) lambda0$1 = lambda;
- }
- } else {
- if (lambda1 >= lambda0$1) {
- if (lambda < lambda0$1) lambda0$1 = lambda;
- if (lambda > lambda1) lambda1 = lambda;
- } else {
- if (lambda > lambda2) {
- if (angle(lambda0$1, lambda) > angle(lambda0$1, lambda1)) lambda1 = lambda;
- } else {
- if (angle(lambda, lambda1) > angle(lambda0$1, lambda1)) lambda0$1 = lambda;
- }
- }
- }
- } else {
- ranges.push(range$1 = [lambda0$1 = lambda, lambda1 = lambda]);
- }
-
- if (phi < phi0) phi0 = phi;
- if (phi > phi1) phi1 = phi;
- p0 = p, lambda2 = lambda;
- }
-
- function boundsLineStart() {
- boundsStream.point = linePoint;
- }
-
- function boundsLineEnd() {
- range$1[0] = lambda0$1, range$1[1] = lambda1;
- boundsStream.point = boundsPoint;
- p0 = null;
- }
-
- function boundsRingPoint(lambda, phi) {
- if (p0) {
- var delta = lambda - lambda2;
- deltaSum.add(abs$2(delta) > 180 ? delta + (delta > 0 ? 360 : -360) : delta);
- } else {
- lambda00$1 = lambda, phi00$1 = phi;
- }
-
- areaStream.point(lambda, phi);
- linePoint(lambda, phi);
- }
-
- function boundsRingStart() {
- areaStream.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;
- p0 = null;
- } // Finds the left-right distance between two longitudes.
- // This is almost the same as (lambda1 - lambda0 + 360°) % 360°, except that we want
- // the distance between ±180° to be 360°.
-
-
- function angle(lambda0, lambda1) {
- return (lambda1 -= lambda0) < 0 ? lambda1 + 360 : lambda1;
- }
-
- function rangeCompare(a, b) {
- return a[0] - b[0];
- }
-
- function rangeContains(range, x) {
- return range[0] <= range[1] ? range[0] <= x && x <= range[1] : x < range[0] || range[1] < x;
- }
-
- function d3_geoBounds (feature) {
- 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.
-
- if (n = ranges.length) {
- ranges.sort(rangeCompare); // Then, merge any ranges that overlap.
-
- for (i = 1, a = ranges[0], merged = [a]; i < n; ++i) {
- b = ranges[i];
-
- if (rangeContains(a, b[0]) || rangeContains(a, b[1])) {
- if (angle(a[0], b[1]) > angle(a[0], a[1])) a[1] = b[1];
- if (angle(b[0], a[1]) > angle(a[0], a[1])) a[0] = b[0];
- } else {
- merged.push(a = b);
- }
- } // Finally, find the largest gap between the merged ranges.
- // The final bounding box will be the inverse of this gap.
-
-
- for (deltaMax = -Infinity, n = merged.length - 1, i = 0, a = merged[n]; i <= n; a = b, ++i) {
- b = merged[i];
- if ((delta = angle(a[1], b[0])) > deltaMax) deltaMax = delta, lambda0$1 = b[0], lambda1 = a[1];
- }
- }
-
- ranges = range$1 = 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]);
- }
-
- if (a.invert && b.invert) compose.invert = function (x, y) {
- return x = b.invert(x, y), x && a.invert(x[0], x[1]);
- };
- return compose;
- }
-
- function rotationIdentity(lambda, phi) {
- return [abs$2(lambda) > pi ? lambda + Math.round(-lambda / tau) * tau : lambda, phi];
- }
-
- rotationIdentity.invert = rotationIdentity;
- function rotateRadians(deltaLambda, deltaPhi, deltaGamma) {
- return (deltaLambda %= tau) ? deltaPhi || deltaGamma ? compose(rotationLambda(deltaLambda), rotationPhiGamma(deltaPhi, deltaGamma)) : rotationLambda(deltaLambda) : deltaPhi || deltaGamma ? rotationPhiGamma(deltaPhi, deltaGamma) : rotationIdentity;
- }
-
- function forwardRotationLambda(deltaLambda) {
- return function (lambda, phi) {
- return lambda += deltaLambda, [lambda > pi ? lambda - tau : lambda < -pi ? lambda + tau : lambda, phi];
- };
- }
-
- function rotationLambda(deltaLambda) {
- var rotation = forwardRotationLambda(deltaLambda);
- rotation.invert = forwardRotationLambda(-deltaLambda);
- return rotation;
- }
-
- function rotationPhiGamma(deltaPhi, deltaGamma) {
- var cosDeltaPhi = cos(deltaPhi),
- sinDeltaPhi = sin(deltaPhi),
- cosDeltaGamma = cos(deltaGamma),
- sinDeltaGamma = sin(deltaGamma);
-
- function rotation(lambda, phi) {
- var cosPhi = cos(phi),
- x = cos(lambda) * cosPhi,
- y = sin(lambda) * cosPhi,
- z = sin(phi),
- k = z * cosDeltaPhi + x * sinDeltaPhi;
- return [atan2(y * cosDeltaGamma - k * sinDeltaGamma, x * cosDeltaPhi - z * sinDeltaPhi), asin(k * cosDeltaGamma + y * sinDeltaGamma)];
- }
-
- rotation.invert = function (lambda, phi) {
- var cosPhi = cos(phi),
- x = cos(lambda) * cosPhi,
- y = sin(lambda) * cosPhi,
- z = sin(phi),
- k = z * cosDeltaGamma - y * sinDeltaGamma;
- return [atan2(y * cosDeltaGamma + z * sinDeltaGamma, x * cosDeltaPhi + k * sinDeltaPhi), asin(k * cosDeltaPhi - x * sinDeltaPhi)];
- };
-
- return rotation;
- }
-
- function rotation (rotate) {
- rotate = rotateRadians(rotate[0] * radians, rotate[1] * radians, rotate.length > 2 ? rotate[2] * radians : 0);
-
- function forward(coordinates) {
- coordinates = rotate(coordinates[0] * radians, coordinates[1] * radians);
- return coordinates[0] *= degrees, coordinates[1] *= degrees, coordinates;
- }
-
- forward.invert = function (coordinates) {
- coordinates = rotate.invert(coordinates[0] * radians, coordinates[1] * radians);
- return coordinates[0] *= degrees, coordinates[1] *= degrees, coordinates;
- };
-
- return forward;
- }
-
- function circleStream(stream, radius, delta, direction, t0, t1) {
- if (!delta) return;
- var cosRadius = cos(radius),
- sinRadius = sin(radius),
- step = direction * delta;
-
- if (t0 == null) {
- t0 = radius + direction * tau;
- t1 = radius - step / 2;
- } else {
- t0 = circleRadius(cosRadius, t0);
- t1 = circleRadius(cosRadius, t1);
- if (direction > 0 ? t0 < t1 : t0 > t1) t0 += direction * tau;
- }
-
- for (var point, t = t0; direction > 0 ? t > t1 : t < t1; t -= step) {
- point = spherical([cosRadius, -sinRadius * cos(t), -sinRadius * sin(t)]);
- stream.point(point[0], point[1]);
- }
- } // Returns the signed angle of a cartesian point relative to [cosRadius, 0, 0].
-
- function circleRadius(cosRadius, point) {
- point = cartesian(point), point[0] -= cosRadius;
- cartesianNormalizeInPlace(point);
- var radius = acos(-point[1]);
- return ((-point[2] < 0 ? -radius : radius) + tau - epsilon) % tau;
- }
-
- function clipBuffer () {
- var lines = [],
- line;
- return {
- point: function point(x, y, m) {
- line.push([x, y, m]);
- },
- lineStart: function lineStart() {
- lines.push(line = []);
- },
- lineEnd: noop,
- rejoin: function rejoin() {
- if (lines.length > 1) lines.push(lines.pop().concat(lines.shift()));
- },
- result: function result() {
- var result = lines;
- lines = [];
- line = null;
- return result;
- }
- };
- }
-
- function pointEqual (a, b) {
- return abs$2(a[0] - b[0]) < epsilon && abs$2(a[1] - b[1]) < epsilon;
- }
-
- function Intersection(point, points, other, entry) {
- this.x = point;
- this.z = points;
- this.o = other; // another intersection
-
- this.e = entry; // is an entry?
-
- this.v = false; // visited
-
- this.n = this.p = null; // next & previous
- } // A generalized polygon clipping algorithm: given a polygon that has been cut
- // into its visible line segments, and rejoins the segments by interpolating
- // along the clip edge.
-
-
- function clipRejoin (segments, compareIntersection, startInside, interpolate, stream) {
- var subject = [],
- clip = [],
- i,
- n;
- segments.forEach(function (segment) {
- if ((n = segment.length - 1) <= 0) return;
- var n,
- p0 = segment[0],
- p1 = segment[n],
- x;
-
- if (pointEqual(p0, p1)) {
- if (!p0[2] && !p1[2]) {
- stream.lineStart();
-
- for (i = 0; i < n; ++i) {
- stream.point((p0 = segment[i])[0], p0[1]);
- }
-
- stream.lineEnd();
- return;
- } // handle degenerate cases by moving the point
-
-
- p1[0] += 2 * epsilon;
- }
-
- subject.push(x = new Intersection(p0, segment, null, true));
- clip.push(x.o = new Intersection(p0, null, x, false));
- subject.push(x = new Intersection(p1, segment, null, false));
- clip.push(x.o = new Intersection(p1, null, x, true));
- });
- if (!subject.length) return;
- clip.sort(compareIntersection);
- link(subject);
- link(clip);
-
- for (i = 0, n = clip.length; i < n; ++i) {
- clip[i].e = startInside = !startInside;
- }
-
- var start = subject[0],
- points,
- point;
-
- while (1) {
- // Find first unvisited intersection.
- var current = start,
- isSubject = true;
-
- while (current.v) {
- if ((current = current.n) === start) return;
- }
-
- points = current.z;
- stream.lineStart();
-
- do {
- current.v = current.o.v = true;
-
- if (current.e) {
- if (isSubject) {
- for (i = 0, n = points.length; i < n; ++i) {
- stream.point((point = points[i])[0], point[1]);
- }
- } else {
- interpolate(current.x, current.n.x, 1, stream);
- }
-
- current = current.n;
- } else {
- if (isSubject) {
- points = current.p.z;
-
- for (i = points.length - 1; i >= 0; --i) {
- stream.point((point = points[i])[0], point[1]);
- }
- } else {
- interpolate(current.x, current.p.x, -1, stream);
- }
-
- current = current.p;
- }
-
- current = current.o;
- points = current.z;
- isSubject = !isSubject;
- } while (!current.v);
-
- stream.lineEnd();
- }
- }
-
- function link(array) {
- if (!(n = array.length)) return;
- var n,
- i = 0,
- a = array[0],
- b;
-
- while (++i < n) {
- a.n = b = array[i];
- b.p = a;
- a = b;
- }
-
- a.n = b = array[0];
- b.p = a;
- }
-
- function longitude(point) {
- if (abs$2(point[0]) <= pi) return point[0];else return sign(point[0]) * ((abs$2(point[0]) + pi) % tau - pi);
- }
-
- function polygonContains (polygon, point) {
- var lambda = longitude(point),
- phi = point[1],
- sinPhi = sin(phi),
- normal = [sin(lambda), -cos(lambda), 0],
- angle = 0,
- winding = 0;
- var sum = new Adder();
- if (sinPhi === 1) phi = halfPi + epsilon;else if (sinPhi === -1) phi = -halfPi - epsilon;
-
- for (var i = 0, n = polygon.length; i < n; ++i) {
- if (!(m = (ring = polygon[i]).length)) continue;
- var ring,
- m,
- point0 = ring[m - 1],
- lambda0 = longitude(point0),
- phi0 = point0[1] / 2 + quarterPi,
- sinPhi0 = sin(phi0),
- cosPhi0 = cos(phi0);
-
- for (var j = 0; j < m; ++j, lambda0 = lambda1, sinPhi0 = sinPhi1, cosPhi0 = cosPhi1, point0 = point1) {
- var point1 = ring[j],
- lambda1 = longitude(point1),
- phi1 = point1[1] / 2 + quarterPi,
- sinPhi1 = sin(phi1),
- cosPhi1 = cos(phi1),
- delta = lambda1 - lambda0,
- sign = delta >= 0 ? 1 : -1,
- absDelta = sign * delta,
- antimeridian = absDelta > pi,
- k = sinPhi0 * sinPhi1;
- sum.add(atan2(k * sign * sin(absDelta), cosPhi0 * cosPhi1 + k * cos(absDelta)));
- angle += antimeridian ? delta + sign * tau : delta; // Are the longitudes either side of the point’s meridian (lambda),
- // and are the latitudes smaller than the parallel (phi)?
-
- if (antimeridian ^ lambda0 >= lambda ^ lambda1 >= lambda) {
- var arc = cartesianCross(cartesian(point0), cartesian(point1));
- cartesianNormalizeInPlace(arc);
- var intersection = cartesianCross(normal, arc);
- cartesianNormalizeInPlace(intersection);
- var phiArc = (antimeridian ^ delta >= 0 ? -1 : 1) * asin(intersection[2]);
-
- if (phi > phiArc || phi === phiArc && (arc[0] || arc[1])) {
- winding += antimeridian ^ delta >= 0 ? 1 : -1;
- }
- }
- }
- } // First, determine whether the South pole is inside or outside:
- //
- // It is inside if:
- // * the polygon winds around it in a clockwise direction.
- // * the polygon does not (cumulatively) wind around it, but has a negative
- // (counter-clockwise) area.
- //
- // Second, count the (signed) number of times a segment crosses a lambda
- // from the point to the South pole. If it is zero, then the point is the
- // same side as the South pole.
-
-
- return (angle < -epsilon || angle < epsilon && sum < -epsilon2) ^ winding & 1;
- }
-
- function clip (pointVisible, clipLine, interpolate, start) {
- return function (sink) {
- var line = clipLine(sink),
- ringBuffer = clipBuffer(),
- ringSink = clipLine(ringBuffer),
- polygonStarted = false,
- polygon,
- segments,
- ring;
- var clip = {
- point: point,
- lineStart: lineStart,
- lineEnd: lineEnd,
- polygonStart: function polygonStart() {
- clip.point = pointRing;
- clip.lineStart = ringStart;
- clip.lineEnd = ringEnd;
- segments = [];
- polygon = [];
- },
- polygonEnd: function polygonEnd() {
- clip.point = point;
- clip.lineStart = lineStart;
- clip.lineEnd = lineEnd;
- segments = merge(segments);
- var startInside = polygonContains(polygon, start);
-
- if (segments.length) {
- if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
- clipRejoin(segments, compareIntersection, startInside, interpolate, sink);
- } else if (startInside) {
- if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
- sink.lineStart();
- interpolate(null, null, 1, sink);
- sink.lineEnd();
- }
-
- if (polygonStarted) sink.polygonEnd(), polygonStarted = false;
- segments = polygon = null;
- },
- sphere: function sphere() {
- sink.polygonStart();
- sink.lineStart();
- interpolate(null, null, 1, sink);
- sink.lineEnd();
- sink.polygonEnd();
- }
- };
-
- function point(lambda, phi) {
- if (pointVisible(lambda, phi)) sink.point(lambda, phi);
- }
-
- function pointLine(lambda, phi) {
- line.point(lambda, phi);
- }
-
- function lineStart() {
- clip.point = pointLine;
- line.lineStart();
- }
-
- function lineEnd() {
- clip.point = point;
- line.lineEnd();
- }
-
- function pointRing(lambda, phi) {
- ring.push([lambda, phi]);
- ringSink.point(lambda, phi);
- }
-
- function ringStart() {
- ringSink.lineStart();
- ring = [];
- }
-
- function ringEnd() {
- pointRing(ring[0][0], ring[0][1]);
- ringSink.lineEnd();
- var clean = ringSink.clean(),
- ringSegments = ringBuffer.result(),
- i,
- n = ringSegments.length,
- m,
- segment,
- point;
- ring.pop();
- polygon.push(ring);
- ring = null;
- if (!n) return; // No intersections.
-
- if (clean & 1) {
- segment = ringSegments[0];
-
- if ((m = segment.length - 1) > 0) {
- if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
- sink.lineStart();
-
- for (i = 0; i < m; ++i) {
- sink.point((point = segment[i])[0], point[1]);
- }
-
- sink.lineEnd();
- }
-
- return;
- } // Rejoin connected segments.
- // TODO reuse ringBuffer.rejoin()?
-
-
- if (n > 1 && clean & 2) ringSegments.push(ringSegments.pop().concat(ringSegments.shift()));
- segments.push(ringSegments.filter(validSegment));
- }
-
- return clip;
- };
- }
-
- function validSegment(segment) {
- return segment.length > 1;
- } // Intersections are sorted along the clip edge. For both antimeridian cutting
- // and circle clipping, the same comparison is used.
-
-
- 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]);
- }
-
- var clipAntimeridian = clip(function () {
- return true;
- }, clipAntimeridianLine, clipAntimeridianInterpolate, [-pi, -halfPi]); // Takes a line and cuts into visible segments. Return values: 0 - there were
- // intersections or the line was empty; 1 - no intersections; 2 - there were
- // intersections, and the first and last segments should be rejoined.
-
- function clipAntimeridianLine(stream) {
- var lambda0 = NaN,
- phi0 = NaN,
- sign0 = NaN,
- _clean; // no intersections
-
-
- return {
- lineStart: function lineStart() {
- stream.lineStart();
- _clean = 1;
- },
- point: function point(lambda1, phi1) {
- var sign1 = lambda1 > 0 ? pi : -pi,
- delta = abs$2(lambda1 - lambda0);
-
- if (abs$2(delta - pi) < epsilon) {
- // line crosses a pole
- stream.point(lambda0, phi0 = (phi0 + phi1) / 2 > 0 ? halfPi : -halfPi);
- stream.point(sign0, phi0);
- stream.lineEnd();
- stream.lineStart();
- stream.point(sign1, phi0);
- stream.point(lambda1, 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(lambda1 - sign1) < epsilon) lambda1 -= sign1 * epsilon;
- phi0 = clipAntimeridianIntersect(lambda0, phi0, lambda1, phi1);
- stream.point(sign0, phi0);
- stream.lineEnd();
- stream.lineStart();
- stream.point(sign1, phi0);
- _clean = 0;
- }
-
- stream.point(lambda0 = lambda1, phi0 = phi1);
- sign0 = sign1;
- },
- lineEnd: function lineEnd() {
- stream.lineEnd();
- lambda0 = phi0 = NaN;
- },
- clean: function clean() {
- return 2 - _clean; // if intersections, rejoin first and last segments
- }
- };
- }
-
- function clipAntimeridianIntersect(lambda0, phi0, lambda1, phi1) {
- 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;
- }
-
- function clipAntimeridianInterpolate(from, to, direction, stream) {
- var phi;
-
- if (from == null) {
- phi = direction * halfPi;
- stream.point(-pi, phi);
- stream.point(0, phi);
- stream.point(pi, phi);
- stream.point(pi, 0);
- stream.point(pi, -phi);
- stream.point(0, -phi);
- stream.point(-pi, -phi);
- stream.point(-pi, 0);
- stream.point(-pi, phi);
- } else if (abs$2(from[0] - to[0]) > epsilon) {
- var lambda = from[0] < to[0] ? pi : -pi;
- phi = direction * lambda / 2;
- stream.point(-lambda, phi);
- stream.point(0, phi);
- stream.point(lambda, phi);
- } else {
- stream.point(to[0], to[1]);
- }
- }
-
- function clipCircle (radius) {
- var cr = cos(radius),
- delta = 6 * radians,
- smallRadius = cr > 0,
- notHemisphere = abs$2(cr) > epsilon; // TODO optimise for this common case
-
- function interpolate(from, to, direction, stream) {
- circleStream(stream, radius, delta, direction, from, to);
- }
-
- function visible(lambda, phi) {
- return cos(lambda) * cos(phi) > cr;
- } // Takes a line and cuts into visible segments. Return values used for polygon
- // clipping: 0 - there were intersections or the line was empty; 1 - no
- // intersections 2 - there were intersections, and the first and last segments
- // should be rejoined.
-
-
- function clipLine(stream) {
- var point0, // previous point
- c0, // code for previous point
- v0, // visibility of previous point
- v00, // visibility of first point
- _clean; // no intersections
-
-
- return {
- lineStart: function lineStart() {
- v00 = v0 = false;
- _clean = 1;
- },
- point: function point(lambda, phi) {
- var point1 = [lambda, phi],
- point2,
- v = visible(lambda, phi),
- c = smallRadius ? v ? 0 : code(lambda, phi) : v ? code(lambda + (lambda < 0 ? pi : -pi), phi) : 0;
- if (!point0 && (v00 = v0 = v)) stream.lineStart();
-
- if (v !== v0) {
- point2 = intersect(point0, point1);
- if (!point2 || pointEqual(point0, point2) || pointEqual(point1, point2)) point1[2] = 1;
- }
-
- if (v !== v0) {
- _clean = 0;
-
- if (v) {
- // outside going in
- stream.lineStart();
- point2 = intersect(point1, point0);
- stream.point(point2[0], point2[1]);
- } else {
- // inside going out
- point2 = intersect(point0, point1);
- stream.point(point2[0], point2[1], 2);
- stream.lineEnd();
- }
-
- point0 = point2;
- } else if (notHemisphere && point0 && smallRadius ^ v) {
- var t; // If the codes for two points are different, or are both zero,
- // and there this segment intersects with the small circle.
-
- if (!(c & c0) && (t = intersect(point1, point0, true))) {
- _clean = 0;
-
- if (smallRadius) {
- stream.lineStart();
- stream.point(t[0][0], t[0][1]);
- stream.point(t[1][0], t[1][1]);
- stream.lineEnd();
- } else {
- stream.point(t[1][0], t[1][1]);
- stream.lineEnd();
- stream.lineStart();
- stream.point(t[0][0], t[0][1], 3);
- }
- }
- }
-
- if (v && (!point0 || !pointEqual(point0, point1))) {
- stream.point(point1[0], point1[1]);
- }
-
- point0 = point1, v0 = v, c0 = c;
- },
- lineEnd: function lineEnd() {
- if (v0) stream.lineEnd();
- point0 = null;
- },
- // Rejoin first and last segments if there were intersections and the first
- // and last points were visible.
- clean: function clean() {
- return _clean | (v00 && v0) << 1;
- }
- };
- } // Intersects the great circle between a and b with the clip circle.
-
-
- function intersect(a, b, two) {
- var pa = cartesian(a),
- pb = cartesian(b); // We have two planes, n1.p = d1 and n2.p = d2.
- // Find intersection line p(t) = c1 n1 + c2 n2 + t (n1 ⨯ n2).
-
- var n1 = [1, 0, 0],
- // normal
- n2 = cartesianCross(pa, pb),
- n2n2 = cartesianDot(n2, n2),
- n1n2 = n2[0],
- // cartesianDot(n1, n2),
- determinant = n2n2 - n1n2 * n1n2; // Two polar points.
-
- if (!determinant) return !two && a;
- var c1 = cr * n2n2 / determinant,
- c2 = -cr * n1n2 / determinant,
- n1xn2 = cartesianCross(n1, n2),
- A = cartesianScale(n1, c1),
- B = cartesianScale(n2, c2);
- cartesianAddInPlace(A, B); // Solve |p(t)|^2 = 1.
-
- var u = n1xn2,
- w = cartesianDot(A, u),
- uu = cartesianDot(u, u),
- t2 = w * w - uu * (cartesianDot(A, A) - 1);
- if (t2 < 0) return;
- var t = sqrt$1(t2),
- q = cartesianScale(u, (-w - t) / uu);
- cartesianAddInPlace(q, A);
- q = spherical(q);
- if (!two) return q; // Two intersection points.
-
- var lambda0 = a[0],
- lambda1 = b[0],
- phi0 = a[1],
- phi1 = b[1],
- z;
- if (lambda1 < lambda0) z = lambda0, lambda0 = lambda1, lambda1 = z;
- var delta = lambda1 - lambda0,
- polar = abs$2(delta - pi) < epsilon,
- meridian = polar || delta < epsilon;
- 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)) {
- var q1 = cartesianScale(u, (-w + t) / uu);
- cartesianAddInPlace(q1, A);
- return [q, spherical(q1)];
- }
- } // Generates a 4-bit vector representing the location of a point relative to
- // the small circle's bounding box.
-
-
- function code(lambda, phi) {
- var r = smallRadius ? radius : pi - radius,
- code = 0;
- if (lambda < -r) code |= 1; // left
- else if (lambda > r) code |= 2; // right
-
- if (phi < -r) code |= 4; // below
- else if (phi > r) code |= 8; // above
-
- return code;
- }
-
- return clip(visible, clipLine, interpolate, smallRadius ? [0, -radius] : [-pi, radius - pi]);
- }
-
- function clipLine (a, b, x0, y0, x1, y1) {
- var ax = a[0],
- ay = a[1],
- bx = b[0],
- by = b[1],
- t0 = 0,
- t1 = 1,
- dx = bx - ax,
- dy = by - ay,
- r;
- r = x0 - ax;
- if (!dx && r > 0) return;
- r /= dx;
-
- if (dx < 0) {
- if (r < t0) return;
- if (r < t1) t1 = r;
- } else if (dx > 0) {
- if (r > t1) return;
- if (r > t0) t0 = r;
- }
-
- r = x1 - ax;
- if (!dx && r < 0) return;
- r /= dx;
-
- if (dx < 0) {
- if (r > t1) return;
- if (r > t0) t0 = r;
- } else if (dx > 0) {
- if (r < t0) return;
- if (r < t1) t1 = r;
- }
-
- r = y0 - ay;
- if (!dy && r > 0) return;
- r /= dy;
-
- if (dy < 0) {
- if (r < t0) return;
- if (r < t1) t1 = r;
- } else if (dy > 0) {
- if (r > t1) return;
- if (r > t0) t0 = r;
- }
-
- r = y1 - ay;
- if (!dy && r < 0) return;
- r /= dy;
-
- if (dy < 0) {
- if (r > t1) return;
- if (r > t0) t0 = r;
- } else if (dy > 0) {
- if (r < t0) return;
- if (r < t1) t1 = r;
- }
-
- if (t0 > 0) a[0] = ax + t0 * dx, a[1] = ay + t0 * dy;
- if (t1 < 1) b[0] = ax + t1 * dx, b[1] = ay + t1 * dy;
- return true;
- }
-
- var clipMax = 1e9,
- clipMin = -clipMax; // TODO Use d3-polygon’s polygonContains here for the ring check?
- // TODO Eliminate duplicate buffering in clipBuffer and polygon.push?
-
- function clipRectangle(x0, y0, x1, y1) {
- function visible(x, y) {
- return x0 <= x && x <= x1 && y0 <= y && y <= y1;
- }
-
- function interpolate(from, to, direction, stream) {
- var a = 0,
- a1 = 0;
-
- if (from == null || (a = corner(from, direction)) !== (a1 = corner(to, direction)) || comparePoint(from, to) < 0 ^ direction > 0) {
- do {
- stream.point(a === 0 || a === 3 ? x0 : x1, a > 1 ? y1 : y0);
- } while ((a = (a + direction + 4) % 4) !== a1);
- } else {
- stream.point(to[0], to[1]);
- }
- }
-
- 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
- }
-
- function compareIntersection(a, b) {
- return comparePoint(a.x, b.x);
- }
-
- function comparePoint(a, b) {
- var ca = corner(a, 1),
- cb = corner(b, 1);
- return ca !== cb ? ca - cb : ca === 0 ? b[1] - a[1] : ca === 1 ? a[0] - b[0] : ca === 2 ? a[1] - b[1] : b[0] - a[0];
- }
-
- return function (stream) {
- var activeStream = stream,
- bufferStream = clipBuffer(),
- segments,
- polygon,
- ring,
- x__,
- y__,
- v__,
- // first point
- x_,
- y_,
- v_,
- // previous point
- first,
- clean;
- var clipStream = {
- point: point,
- lineStart: lineStart,
- lineEnd: lineEnd,
- polygonStart: polygonStart,
- polygonEnd: polygonEnd
- };
-
- function point(x, y) {
- if (visible(x, y)) activeStream.point(x, y);
- }
-
- function polygonInside() {
- var winding = 0;
-
- for (var i = 0, n = polygon.length; i < n; ++i) {
- for (var ring = polygon[i], j = 1, m = ring.length, point = ring[0], a0, a1, b0 = point[0], b1 = point[1]; j < m; ++j) {
- a0 = b0, a1 = b1, point = ring[j], b0 = point[0], b1 = point[1];
-
- if (a1 <= y1) {
- if (b1 > y1 && (b0 - a0) * (y1 - a1) > (b1 - a1) * (x0 - a0)) ++winding;
- } else {
- if (b1 <= y1 && (b0 - a0) * (y1 - a1) < (b1 - a1) * (x0 - a0)) --winding;
- }
- }
- }
-
- return winding;
- } // Buffer geometry within a polygon and then clip it en masse.
-
-
- function polygonStart() {
- activeStream = bufferStream, segments = [], polygon = [], clean = true;
- }
-
- function polygonEnd() {
- var startInside = polygonInside(),
- cleanInside = clean && startInside,
- visible = (segments = merge(segments)).length;
-
- if (cleanInside || visible) {
- stream.polygonStart();
-
- if (cleanInside) {
- stream.lineStart();
- interpolate(null, null, 1, stream);
- stream.lineEnd();
- }
-
- if (visible) {
- clipRejoin(segments, compareIntersection, startInside, interpolate, stream);
- }
-
- stream.polygonEnd();
- }
-
- activeStream = stream, segments = polygon = ring = null;
- }
-
- function lineStart() {
- clipStream.point = linePoint;
- if (polygon) polygon.push(ring = []);
- first = true;
- v_ = false;
- x_ = y_ = NaN;
- } // TODO rather than special-case polygons, simply handle them separately.
- // Ideally, coincident intersection points should be jittered to avoid
- // clipping issues.
-
-
- function lineEnd() {
- if (segments) {
- linePoint(x__, y__);
- if (v__ && v_) bufferStream.rejoin();
- segments.push(bufferStream.result());
- }
-
- clipStream.point = point;
- if (v_) activeStream.lineEnd();
- }
-
- function linePoint(x, y) {
- var v = visible(x, y);
- if (polygon) ring.push([x, y]);
-
- if (first) {
- x__ = x, y__ = y, v__ = v;
- first = false;
-
- if (v) {
- activeStream.lineStart();
- activeStream.point(x, y);
- }
- } else {
- if (v && v_) activeStream.point(x, y);else {
- var a = [x_ = Math.max(clipMin, Math.min(clipMax, x_)), y_ = Math.max(clipMin, Math.min(clipMax, y_))],
- b = [x = Math.max(clipMin, Math.min(clipMax, x)), y = Math.max(clipMin, Math.min(clipMax, y))];
-
- if (clipLine(a, b, x0, y0, x1, y1)) {
- if (!v_) {
- activeStream.lineStart();
- activeStream.point(a[0], a[1]);
- }
-
- activeStream.point(b[0], b[1]);
- if (!v) activeStream.lineEnd();
- clean = false;
- } else if (v) {
- activeStream.lineStart();
- activeStream.point(x, y);
- clean = false;
- }
- }
- }
-
- x_ = x, y_ = y, v_ = v;
- }
-
- return clipStream;
- };
- }
-
- var lengthSum, lambda0$2, sinPhi0$1, cosPhi0$1;
- var lengthStream = {
- sphere: noop,
- point: noop,
- lineStart: lengthLineStart,
- lineEnd: noop,
- polygonStart: noop,
- polygonEnd: noop
- };
-
- function lengthLineStart() {
- lengthStream.point = lengthPointFirst;
- lengthStream.lineEnd = lengthLineEnd;
- }
-
- function lengthLineEnd() {
- lengthStream.point = lengthStream.lineEnd = noop;
- }
-
- function lengthPointFirst(lambda, phi) {
- lambda *= radians, phi *= radians;
- lambda0$2 = lambda, sinPhi0$1 = sin(phi), cosPhi0$1 = cos(phi);
- lengthStream.point = lengthPoint;
- }
-
- function lengthPoint(lambda, phi) {
- lambda *= radians, phi *= radians;
- var sinPhi = sin(phi),
- cosPhi = cos(phi),
- delta = abs$2(lambda - lambda0$2),
- 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;
- }
-
- function d3_geoLength (object) {
- lengthSum = new Adder();
- d3_geoStream(object, lengthStream);
- return +lengthSum;
- }
-
- var identity = (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,
- polygonStart: function polygonStart() {
- areaStream$1.lineStart = areaRingStart$1;
- areaStream$1.lineEnd = areaRingEnd$1;
- },
- polygonEnd: function polygonEnd() {
- areaStream$1.lineStart = areaStream$1.lineEnd = areaStream$1.point = noop;
- areaSum$1.add(abs$2(areaRingSum$1));
- areaRingSum$1 = new Adder();
- },
- result: function result() {
- var area = areaSum$1 / 2;
- areaSum$1 = new Adder();
- return area;
- }
- };
-
- function areaRingStart$1() {
- areaStream$1.point = areaPointFirst$1;
- }
-
- function areaPointFirst$1(x, y) {
- areaStream$1.point = areaPoint$1;
- x00 = x0$1 = x, y00 = y0$1 = y;
- }
-
- function areaPoint$1(x, y) {
- areaRingSum$1.add(y0$1 * x - x0$1 * y);
- x0$1 = x, y0$1 = y;
- }
-
- function areaRingEnd$1() {
- areaPoint$1(x00, y00);
- }
-
- 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,
- result: function result() {
- var bounds = [[x0$2, y0$2], [x1, y1]];
- x1 = y1 = -(y0$2 = x0$2 = Infinity);
- return bounds;
- }
- };
-
- function boundsPoint$1(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,
- x00$1,
- y00$1,
- x0$3,
- y0$3;
- var centroidStream$1 = {
- point: centroidPoint$1,
- lineStart: centroidLineStart$1,
- lineEnd: centroidLineEnd$1,
- polygonStart: function polygonStart() {
- centroidStream$1.lineStart = centroidRingStart$1;
- centroidStream$1.lineEnd = centroidRingEnd$1;
- },
- polygonEnd: function polygonEnd() {
- centroidStream$1.point = centroidPoint$1;
- centroidStream$1.lineStart = centroidLineStart$1;
- centroidStream$1.lineEnd = centroidLineEnd$1;
- },
- 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;
- return centroid;
- }
- };
-
- function centroidPoint$1(x, y) {
- X0$1 += x;
- Y0$1 += y;
- ++Z0$1;
- }
-
- function centroidLineStart$1() {
- centroidStream$1.point = centroidPointFirstLine;
- }
-
- function centroidPointFirstLine(x, y) {
- centroidStream$1.point = centroidPointLine;
- centroidPoint$1(x0$3 = x, y0$3 = 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);
- }
-
- function centroidLineEnd$1() {
- centroidStream$1.point = centroidPoint$1;
- }
-
- function centroidRingStart$1() {
- centroidStream$1.point = centroidPointFirstRing;
- }
-
- function centroidRingEnd$1() {
- 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);
- }
-
- 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);
- }
-
- function PathContext(context) {
- this._context = context;
- }
- PathContext.prototype = {
- _radius: 4.5,
- pointRadius: function pointRadius(_) {
- return this._radius = _, this;
- },
- polygonStart: function polygonStart() {
- this._line = 0;
- },
- polygonEnd: function polygonEnd() {
- this._line = NaN;
- },
- lineStart: function lineStart() {
- this._point = 0;
- },
- lineEnd: function lineEnd() {
- if (this._line === 0) this._context.closePath();
- this._point = NaN;
- },
- point: function point(x, y) {
- switch (this._point) {
- case 0:
- {
- this._context.moveTo(x, y);
-
- this._point = 1;
- break;
- }
-
- case 1:
- {
- this._context.lineTo(x, y);
-
- break;
- }
-
- default:
- {
- this._context.moveTo(x + this._radius, y);
-
- this._context.arc(x, y, this._radius, 0, tau);
-
- break;
- }
- }
- },
- result: noop
- };
-
- var lengthSum$1 = new Adder(),
- lengthRing,
- x00$2,
- y00$2,
- x0$4,
- y0$4;
- var lengthStream$1 = {
- point: noop,
- lineStart: function lineStart() {
- lengthStream$1.point = lengthPointFirst$1;
- },
- lineEnd: function lineEnd() {
- if (lengthRing) lengthPoint$1(x00$2, y00$2);
- lengthStream$1.point = noop;
- },
- polygonStart: function polygonStart() {
- lengthRing = true;
- },
- polygonEnd: function polygonEnd() {
- lengthRing = null;
- },
- result: function result() {
- var length = +lengthSum$1;
- lengthSum$1 = 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 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 PathString() {
- this._string = [];
- }
- PathString.prototype = {
- _radius: 4.5,
- _circle: circle(4.5),
- pointRadius: function pointRadius(_) {
- if ((_ = +_) !== this._radius) this._radius = _, this._circle = null;
- return this;
- },
- polygonStart: function polygonStart() {
- this._line = 0;
- },
- polygonEnd: function polygonEnd() {
- this._line = NaN;
- },
- lineStart: function lineStart() {
- this._point = 0;
- },
- lineEnd: function lineEnd() {
- if (this._line === 0) this._string.push("Z");
- this._point = NaN;
- },
- point: function point(x, y) {
- switch (this._point) {
- case 0:
- {
- this._string.push("M", x, ",", y);
-
- this._point = 1;
- break;
- }
-
- case 1:
- {
- this._string.push("L", x, ",", y);
-
- break;
- }
-
- default:
- {
- if (this._circle == null) this._circle = circle(this._radius);
-
- this._string.push("M", x, ",", y, this._circle);
-
- break;
- }
- }
- },
- result: function result() {
- if (this._string.length) {
- var result = this._string.join("");
-
- this._string = [];
- return result;
- } else {
- return null;
- }
- }
- };
-
- function circle(radius) {
- return "m0," + radius + "a" + radius + "," + radius + " 0 1,1 0," + -2 * radius + "a" + radius + "," + radius + " 0 1,1 0," + 2 * radius + "z";
- }
-
- function d3_geoPath (projection, context) {
- var pointRadius = 4.5,
- projectionStream,
- contextStream;
-
- function path(object) {
- if (object) {
- if (typeof pointRadius === "function") contextStream.pointRadius(+pointRadius.apply(this, arguments));
- d3_geoStream(object, projectionStream(contextStream));
- }
-
- return contextStream.result();
- }
-
- path.area = function (object) {
- d3_geoStream(object, projectionStream(areaStream$1));
- return areaStream$1.result();
- };
-
- path.measure = function (object) {
- d3_geoStream(object, projectionStream(lengthStream$1));
- return lengthStream$1.result();
- };
-
- path.bounds = function (object) {
- d3_geoStream(object, projectionStream(boundsStream$1));
- return boundsStream$1.result();
- };
-
- path.centroid = function (object) {
- d3_geoStream(object, projectionStream(centroidStream$1));
- return centroidStream$1.result();
- };
-
- path.projection = function (_) {
- return arguments.length ? (projectionStream = _ == null ? (projection = null, identity) : (projection = _).stream, path) : projection;
- };
-
- path.context = function (_) {
- if (!arguments.length) return context;
- contextStream = _ == null ? (context = null, new PathString()) : new PathContext(context = _);
- if (typeof pointRadius !== "function") contextStream.pointRadius(pointRadius);
- return path;
- };
-
- path.pointRadius = function (_) {
- if (!arguments.length) return pointRadius;
- pointRadius = typeof _ === "function" ? _ : (contextStream.pointRadius(+_), +_);
- return path;
- };
-
- return path.projection(projection).context(context);
- }
-
- function d3_geoTransform (methods) {
- return {
- stream: transformer(methods)
- };
- }
- function transformer(methods) {
- return function (stream) {
- var s = new TransformStream();
-
- for (var key in methods) {
- s[key] = methods[key];
- }
-
- s.stream = stream;
- return s;
- };
- }
-
- function TransformStream() {}
-
- TransformStream.prototype = {
- constructor: TransformStream,
- point: function point(x, y) {
- this.stream.point(x, y);
- },
- sphere: function sphere() {
- this.stream.sphere();
- },
- lineStart: function lineStart() {
- this.stream.lineStart();
- },
- lineEnd: function lineEnd() {
- this.stream.lineEnd();
- },
- polygonStart: function polygonStart() {
- this.stream.polygonStart();
- },
- polygonEnd: function polygonEnd() {
- this.stream.polygonEnd();
- }
- };
-
- function fit(projection, fitBounds, object) {
- 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());
- if (clip != null) projection.clipExtent(clip);
- return projection;
- }
-
- function fitExtent(projection, extent, object) {
- return fit(projection, function (b) {
- var w = extent[1][0] - extent[0][0],
- h = extent[1][1] - extent[0][1],
- k = Math.min(w / (b[1][0] - b[0][0]), h / (b[1][1] - b[0][1])),
- x = +extent[0][0] + (w - k * (b[1][0] + b[0][0])) / 2,
- y = +extent[0][1] + (h - k * (b[1][1] + b[0][1])) / 2;
- projection.scale(150 * k).translate([x, y]);
- }, object);
- }
- function fitSize(projection, size, object) {
- return fitExtent(projection, [[0, 0], size], object);
- }
- function fitWidth(projection, width, object) {
- return fit(projection, function (b) {
- var w = +width,
- k = w / (b[1][0] - b[0][0]),
- x = (w - k * (b[1][0] + b[0][0])) / 2,
- y = -k * b[0][1];
- projection.scale(150 * k).translate([x, y]);
- }, object);
- }
- function fitHeight(projection, height, object) {
- return fit(projection, function (b) {
- var h = +height,
- k = h / (b[1][1] - b[0][1]),
- x = -k * b[0][0],
- y = (h - k * (b[1][1] + b[0][1])) / 2;
- projection.scale(150 * k).translate([x, y]);
- }, object);
- }
-
- var maxDepth = 16,
- // maximum depth of subdivision
- cosMinDistance = cos(30 * radians); // cos(minimum angular distance)
-
- function resample (project, delta2) {
- return +delta2 ? resample$1(project, delta2) : resampleNone(project);
- }
-
- function resampleNone(project) {
- return transformer({
- point: function point(x, y) {
- x = project(x, y);
- this.stream.point(x[0], x[1]);
- }
- });
- }
-
- function resample$1(project, delta2) {
- function resampleLineTo(x0, y0, lambda0, a0, b0, c0, x1, y1, lambda1, a1, b1, c1, depth, stream) {
- var dx = x1 - x0,
- dy = y1 - y0,
- d2 = dx * dx + dy * dy;
-
- if (d2 > 4 * delta2 && depth--) {
- var a = a0 + a1,
- b = b0 + b1,
- c = c0 + c1,
- m = sqrt$1(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),
- p = project(lambda2, phi2),
- x2 = p[0],
- y2 = p[1],
- dx2 = x2 - x0,
- dy2 = y2 - y0,
- dz = dy * dx2 - dx * dy2;
-
- if (dz * dz / d2 > delta2 // perpendicular projected distance
- || abs$2((dx * dx2 + dy * dy2) / d2 - 0.5) > 0.3 // midpoint close to an end
- || a0 * a1 + b0 * b1 + c0 * c1 < cosMinDistance) {
- // angular distance
- resampleLineTo(x0, y0, lambda0, a0, b0, c0, x2, y2, lambda2, a /= m, b /= m, c, depth, stream);
- stream.point(x2, y2);
- resampleLineTo(x2, y2, lambda2, a, b, c, x1, y1, lambda1, a1, b1, c1, depth, stream);
- }
- }
- }
-
- return function (stream) {
- var lambda00, x00, y00, a00, b00, c00, // first point
- lambda0, x0, y0, a0, b0, c0; // previous point
-
- var resampleStream = {
- point: point,
- lineStart: lineStart,
- lineEnd: lineEnd,
- polygonStart: function polygonStart() {
- stream.polygonStart();
- resampleStream.lineStart = ringStart;
- },
- polygonEnd: function polygonEnd() {
- stream.polygonEnd();
- resampleStream.lineStart = lineStart;
- }
- };
-
- function point(x, y) {
- x = project(x, y);
- stream.point(x[0], x[1]);
- }
-
- function lineStart() {
- x0 = NaN;
- resampleStream.point = linePoint;
- stream.lineStart();
- }
-
- function linePoint(lambda, phi) {
- var c = cartesian([lambda, phi]),
- p = project(lambda, phi);
- resampleLineTo(x0, y0, lambda0, a0, b0, c0, x0 = p[0], y0 = p[1], lambda0 = lambda, a0 = c[0], b0 = c[1], c0 = c[2], maxDepth, stream);
- stream.point(x0, y0);
- }
-
- function lineEnd() {
- resampleStream.point = point;
- stream.lineEnd();
- }
-
- function ringStart() {
- lineStart();
- resampleStream.point = ringPoint;
- resampleStream.lineEnd = ringEnd;
- }
-
- function ringPoint(lambda, phi) {
- linePoint(lambda00 = lambda, phi), x00 = x0, y00 = y0, a00 = a0, b00 = b0, c00 = c0;
- resampleStream.point = linePoint;
- }
-
- function ringEnd() {
- resampleLineTo(x0, y0, lambda0, a0, b0, c0, x00, y00, lambda00, a00, b00, c00, maxDepth, stream);
- resampleStream.lineEnd = lineEnd;
- lineEnd();
- }
-
- return resampleStream;
- };
- }
-
- var transformRadians = transformer({
- point: function point(x, y) {
- this.stream.point(x * radians, y * radians);
- }
- });
-
- function transformRotate(rotate) {
- return transformer({
- point: function point(x, y) {
- var r = rotate(x, y);
- return this.stream.point(r[0], r[1]);
- }
- });
- }
-
- function scaleTranslate(k, dx, dy, sx, sy) {
- function transform(x, y) {
- x *= sx;
- y *= sy;
- return [dx + k * x, dy - k * y];
- }
-
- transform.invert = function (x, y) {
- return [(x - dx) / k * sx, (dy - y) / k * sy];
- };
-
- return transform;
- }
-
- function scaleTranslateRotate(k, dx, dy, sx, sy, alpha) {
- if (!alpha) return scaleTranslate(k, dx, dy, sx, sy);
- var cosAlpha = cos(alpha),
- sinAlpha = sin(alpha),
- a = cosAlpha * k,
- b = sinAlpha * k,
- ai = cosAlpha / k,
- bi = sinAlpha / k,
- ci = (sinAlpha * dy - cosAlpha * dx) / k,
- fi = (sinAlpha * dx + cosAlpha * dy) / k;
-
- function transform(x, y) {
- x *= sx;
- y *= sy;
- return [a * x - b * y + dx, dy - b * x - a * y];
- }
-
- transform.invert = function (x, y) {
- return [sx * (ai * x - bi * y + ci), sy * (fi - bi * x - ai * y)];
- };
-
- return transform;
- }
-
- function projection(project) {
- return projectionMutator(function () {
- return project;
- })();
- }
- function projectionMutator(projectAt) {
- var project,
- k = 150,
- // scale
- x = 480,
- y = 250,
- // translate
- lambda = 0,
- phi = 0,
- // center
- deltaLambda = 0,
- deltaPhi = 0,
- deltaGamma = 0,
- rotate,
- // pre-rotate
- alpha = 0,
- // post-rotate angle
- sx = 1,
- // reflectX
- sy = 1,
- // reflectX
- theta = null,
- preclip = clipAntimeridian,
- // pre-clip angle
- x0 = null,
- y0,
- x1,
- y1,
- postclip = identity,
- // post-clip extent
- delta2 = 0.5,
- // precision
- projectResample,
- projectTransform,
- projectRotateTransform,
- cache,
- cacheStream;
-
- function projection(point) {
- return projectRotateTransform(point[0] * radians, point[1] * radians);
- }
-
- function invert(point) {
- point = projectRotateTransform.invert(point[0], point[1]);
- return point && [point[0] * degrees, point[1] * degrees];
- }
-
- projection.stream = function (stream) {
- return cache && cacheStream === stream ? cache : cache = transformRadians(transformRotate(rotate)(preclip(projectResample(postclip(cacheStream = stream)))));
- };
-
- projection.preclip = function (_) {
- return arguments.length ? (preclip = _, theta = undefined, reset()) : preclip;
- };
-
- projection.postclip = function (_) {
- return arguments.length ? (postclip = _, x0 = y0 = x1 = y1 = null, reset()) : postclip;
- };
-
- projection.clipAngle = function (_) {
- return arguments.length ? (preclip = +_ ? clipCircle(theta = _ * radians) : (theta = null, clipAntimeridian), reset()) : theta * degrees;
- };
-
- 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]];
- };
-
- projection.scale = function (_) {
- return arguments.length ? (k = +_, recenter()) : k;
- };
-
- projection.translate = function (_) {
- return arguments.length ? (x = +_[0], y = +_[1], recenter()) : [x, y];
- };
-
- projection.center = function (_) {
- return arguments.length ? (lambda = _[0] % 360 * radians, phi = _[1] % 360 * radians, recenter()) : [lambda * degrees, phi * degrees];
- };
-
- 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];
- };
-
- projection.angle = function (_) {
- return arguments.length ? (alpha = _ % 360 * radians, recenter()) : alpha * degrees;
- };
-
- projection.reflectX = function (_) {
- return arguments.length ? (sx = _ ? -1 : 1, recenter()) : sx < 0;
- };
-
- projection.reflectY = function (_) {
- return arguments.length ? (sy = _ ? -1 : 1, recenter()) : sy < 0;
- };
-
- projection.precision = function (_) {
- return arguments.length ? (projectResample = resample(projectTransform, delta2 = _ * _), reset()) : sqrt$1(delta2);
- };
-
- projection.fitExtent = function (extent, object) {
- return fitExtent(projection, extent, object);
- };
-
- projection.fitSize = function (size, object) {
- return fitSize(projection, size, object);
- };
-
- projection.fitWidth = function (width, object) {
- return fitWidth(projection, width, object);
- };
-
- projection.fitHeight = function (height, object) {
- return fitHeight(projection, height, object);
- };
-
- function recenter() {
- var center = scaleTranslateRotate(k, 0, 0, sx, sy, alpha).apply(null, project(lambda, phi)),
- transform = scaleTranslateRotate(k, x - center[0], y - center[1], sx, sy, alpha);
- rotate = rotateRadians(deltaLambda, deltaPhi, deltaGamma);
- projectTransform = compose(project, transform);
- projectRotateTransform = compose(rotate, projectTransform);
- projectResample = resample(projectTransform, delta2);
- return reset();
- }
-
- function reset() {
- cache = cacheStream = null;
- return projection;
- }
-
- return function () {
- project = projectAt.apply(this, arguments);
- projection.invert = project.invert && invert;
- return recenter();
- };
- }
-
- function mercatorRaw(lambda, phi) {
- return [lambda, log$1(tan((halfPi + phi) / 2))];
- }
-
- mercatorRaw.invert = function (x, y) {
- return [x, 2 * atan(exp(y)) - halfPi];
- };
-
- function mercator () {
- return mercatorProjection(mercatorRaw).scale(961 / tau);
- }
- function mercatorProjection(project) {
- var m = projection(project),
- center = m.center,
- scale = m.scale,
- translate = m.translate,
- clipExtent = m.clipExtent,
- x0 = null,
- y0,
- x1,
- y1; // clip extent
-
- m.scale = function (_) {
- return arguments.length ? (scale(_), reclip()) : scale();
- };
-
- m.translate = function (_) {
- return arguments.length ? (translate(_), reclip()) : translate();
- };
-
- m.center = function (_) {
- return arguments.length ? (center(_), reclip()) : center();
- };
-
- m.clipExtent = function (_) {
- return arguments.length ? (_ == null ? x0 = y0 = x1 = y1 = null : (x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reclip()) : x0 == null ? null : [[x0, y0], [x1, y1]];
- };
-
- function reclip() {
- var k = pi * scale(),
- t = m(rotation(m.rotate()).invert([0, 0]));
- return clipExtent(x0 == null ? [[t[0] - k, t[1] - k], [t[0] + k, t[1] + k]] : project === mercatorRaw ? [[Math.max(t[0] - k, x0), y0], [Math.min(t[0] + k, x1), y1]] : [[x0, Math.max(t[1] - k, y0)], [x1, Math.min(t[1] + k, y1)]]);
- }
-
- return reclip();
- }
-
- function d3_geoIdentity () {
- var k = 1,
- tx = 0,
- ty = 0,
- sx = 1,
- sy = 1,
- // scale, translate and reflect
- alpha = 0,
- ca,
- sa,
- // angle
- x0 = null,
- y0,
- x1,
- y1,
- // clip extent
- kx = 1,
- ky = 1,
- transform = transformer({
- point: function point(x, y) {
- var p = projection([x, y]);
- this.stream.point(p[0], p[1]);
- }
- }),
- postclip = identity,
- cache,
- cacheStream;
-
- function reset() {
- kx = k * sx;
- ky = k * sy;
- cache = cacheStream = null;
- return projection;
- }
-
- function projection(p) {
- var x = p[0] * kx,
- y = p[1] * ky;
-
- if (alpha) {
- var t = y * ca - x * sa;
- x = x * ca + y * sa;
- y = t;
- }
-
- return [x + tx, y + ty];
- }
-
- projection.invert = function (p) {
- var x = p[0] - tx,
- y = p[1] - ty;
-
- if (alpha) {
- var t = y * ca + x * sa;
- x = x * ca - y * sa;
- y = t;
- }
-
- return [x / kx, y / ky];
- };
-
- projection.stream = function (stream) {
- return cache && cacheStream === stream ? cache : cache = transform(postclip(cacheStream = stream));
- };
-
- projection.postclip = function (_) {
- return arguments.length ? (postclip = _, x0 = y0 = x1 = y1 = null, reset()) : postclip;
- };
-
- 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]];
- };
-
- projection.scale = function (_) {
- return arguments.length ? (k = +_, reset()) : k;
- };
-
- projection.translate = function (_) {
- return arguments.length ? (tx = +_[0], ty = +_[1], reset()) : [tx, ty];
- };
-
- projection.angle = function (_) {
- return arguments.length ? (alpha = _ % 360 * radians, sa = sin(alpha), ca = cos(alpha), reset()) : alpha * degrees;
- };
-
- projection.reflectX = function (_) {
- return arguments.length ? (sx = _ ? -1 : 1, reset()) : sx < 0;
- };
-
- projection.reflectY = function (_) {
- return arguments.length ? (sy = _ ? -1 : 1, reset()) : sy < 0;
- };
-
- projection.fitExtent = function (extent, object) {
- return fitExtent(projection, extent, object);
- };
-
- projection.fitSize = function (size, object) {
- return fitSize(projection, size, object);
- };
-
- projection.fitWidth = function (width, object) {
- return fitWidth(projection, width, object);
- };
-
- projection.fitHeight = function (height, object) {
- return fitHeight(projection, height, object);
- };
-
- return projection;
- }
-
- // constants
- var TAU = 2 * Math.PI;
- var EQUATORIAL_RADIUS = 6356752.314245179;
- var POLAR_RADIUS = 6378137.0;
- function geoLatToMeters(dLat) {
- return dLat * (TAU * POLAR_RADIUS / 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);
- }
- 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)];
- }
- function geoOffsetToMeters(offset, tileSize) {
- tileSize = tileSize || 256;
- return [offset[0] * TAU * EQUATORIAL_RADIUS / tileSize, -offset[1] * TAU * POLAR_RADIUS / tileSize];
- } // Equirectangular approximation of spherical distances on Earth
-
- function geoSphericalDistance(a, b) {
- var x = geoLonToMeters(a[0] - b[0], (a[1] + b[1]) / 2);
- var y = geoLatToMeters(a[1] - b[1]);
- return Math.sqrt(x * x + y * y);
- } // scale to zoom
-
- function geoScaleToZoom(k, tileSize) {
- tileSize = tileSize || 256;
- var log2ts = Math.log(tileSize) * Math.LOG2E;
- return Math.log(k * TAU) / Math.LN2 - log2ts;
- } // zoom to scale
-
- function geoZoomToScale(z, tileSize) {
- tileSize = tileSize || 256;
- return tileSize * Math.pow(2, z) / TAU;
- } // returns info about the node from `nodes` closest to the given `point`
-
- function geoSphericalClosestNode(nodes, point) {
- var minDistance = Infinity,
- distance;
- var indexOfMin;
-
- for (var i in nodes) {
- distance = geoSphericalDistance(nodes[i].loc, point);
-
- if (distance < minDistance) {
- minDistance = distance;
- indexOfMin = i;
- }
- }
-
- if (indexOfMin !== undefined) {
- return {
- index: indexOfMin,
- distance: minDistance,
- node: nodes[indexOfMin]
- };
- } else {
- return null;
- }
- }
-
- function geoExtent(min, max) {
- if (!(this instanceof geoExtent)) {
- return new geoExtent(min, max);
- } else if (min instanceof geoExtent) {
- return min;
- } else if (min && min.length === 2 && min[0].length === 2 && min[1].length === 2) {
- this[0] = min[0];
- this[1] = min[1];
- } else {
- this[0] = min || [Infinity, Infinity];
- this[1] = max || min || [-Infinity, -Infinity];
- }
- }
- geoExtent.prototype = new Array(2);
- Object.assign(geoExtent.prototype, {
- equals: function equals(obj) {
- return this[0][0] === obj[0][0] && this[0][1] === obj[0][1] && this[1][0] === obj[1][0] && this[1][1] === obj[1][1];
- },
- extend: function extend(obj) {
- if (!(obj instanceof geoExtent)) obj = new geoExtent(obj);
- return geoExtent([Math.min(obj[0][0], this[0][0]), Math.min(obj[0][1], this[0][1])], [Math.max(obj[1][0], this[1][0]), Math.max(obj[1][1], this[1][1])]);
- },
- _extend: function _extend(extent) {
- this[0][0] = Math.min(extent[0][0], this[0][0]);
- this[0][1] = Math.min(extent[0][1], this[0][1]);
- this[1][0] = Math.max(extent[1][0], this[1][0]);
- this[1][1] = Math.max(extent[1][1], this[1][1]);
- },
- area: function area() {
- return Math.abs((this[1][0] - this[0][0]) * (this[1][1] - this[0][1]));
- },
- center: function center() {
- return [(this[0][0] + this[1][0]) / 2, (this[0][1] + this[1][1]) / 2];
- },
- rectangle: function rectangle() {
- return [this[0][0], this[0][1], this[1][0], this[1][1]];
- },
- bbox: function bbox() {
- return {
- minX: this[0][0],
- minY: this[0][1],
- maxX: this[1][0],
- maxY: this[1][1]
- };
- },
- polygon: function polygon() {
- return [[this[0][0], this[0][1]], [this[0][0], this[1][1]], [this[1][0], this[1][1]], [this[1][0], this[0][1]], [this[0][0], this[0][1]]];
- },
- contains: function contains(obj) {
- if (!(obj instanceof geoExtent)) obj = new geoExtent(obj);
- return obj[0][0] >= this[0][0] && obj[0][1] >= this[0][1] && obj[1][0] <= this[1][0] && obj[1][1] <= this[1][1];
- },
- intersects: function intersects(obj) {
- if (!(obj instanceof geoExtent)) obj = new geoExtent(obj);
- return obj[0][0] <= this[1][0] && obj[0][1] <= this[1][1] && obj[1][0] >= this[0][0] && obj[1][1] >= this[0][1];
- },
- intersection: function intersection(obj) {
- if (!this.intersects(obj)) return new geoExtent();
- return new geoExtent([Math.max(obj[0][0], this[0][0]), Math.max(obj[0][1], this[0][1])], [Math.min(obj[1][0], this[1][0]), Math.min(obj[1][1], this[1][1])]);
- },
- percentContainedIn: function percentContainedIn(obj) {
- if (!(obj instanceof geoExtent)) obj = new geoExtent(obj);
- var a1 = this.intersection(obj).area();
- var a2 = this.area();
-
- if (a1 === Infinity || a2 === Infinity) {
- return 0;
- } else if (a1 === 0 || a2 === 0) {
- if (obj.contains(this)) {
- return 1;
- }
-
- return 0;
- } else {
- return a1 / a2;
- }
- },
- padByMeters: function padByMeters(meters) {
- var dLat = geoMetersToLat(meters);
- var dLon = geoMetersToLon(meters, this.center()[1]);
- return geoExtent([this[0][0] - dLon, this[0][1] - dLat], [this[1][0] + dLon, this[1][1] + dLat]);
- },
- toParam: function toParam() {
- return this.rectangle().join(',');
- }
- });
-
- var $every$1 = arrayIteration.every;
-
-
-
- var STRICT_METHOD$6 = arrayMethodIsStrict('every');
- var USES_TO_LENGTH$8 = arrayMethodUsesToLength('every');
-
- // `Array.prototype.every` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.every
- _export({ target: 'Array', proto: true, forced: !STRICT_METHOD$6 || !USES_TO_LENGTH$8 }, {
- every: function every(callbackfn /* , thisArg */) {
- return $every$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- var $reduce$1 = arrayReduce.left;
-
-
-
- var STRICT_METHOD$7 = arrayMethodIsStrict('reduce');
- var USES_TO_LENGTH$9 = arrayMethodUsesToLength('reduce', { 1: 0 });
-
- // `Array.prototype.reduce` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.reduce
- _export({ target: 'Array', proto: true, forced: !STRICT_METHOD$7 || !USES_TO_LENGTH$9 }, {
- reduce: function reduce(callbackfn /* , initialValue */) {
- return $reduce$1(this, callbackfn, arguments.length, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- function d3_polygonArea (polygon) {
- var i = -1,
- n = polygon.length,
- a,
- b = polygon[n - 1],
- area = 0;
-
- while (++i < n) {
- a = b;
- b = polygon[i];
- area += a[1] * b[0] - a[0] * b[1];
- }
-
- return area / 2;
- }
-
- function d3_polygonCentroid (polygon) {
- var i = -1,
- n = polygon.length,
- x = 0,
- y = 0,
- a,
- b = polygon[n - 1],
- c,
- k = 0;
-
- while (++i < n) {
- a = b;
- b = polygon[i];
- k += c = a[0] * b[1] - b[0] * a[1];
- x += (a[0] + b[0]) * c;
- y += (a[1] + b[1]) * c;
- }
-
- return k *= 3, [x / k, y / k];
- }
-
- // Returns the 2D cross product of AB and AC vectors, i.e., the z-component of
- // the 3D cross product in a quadrant I Cartesian coordinate system (+x is
- // right, +y is up). Returns a positive value if ABC is counter-clockwise,
- // negative if clockwise, and zero if the points are collinear.
- function cross (a, b, c) {
- return (b[0] - a[0]) * (c[1] - a[1]) - (b[1] - a[1]) * (c[0] - a[0]);
- }
-
- function lexicographicOrder(a, b) {
- return a[0] - b[0] || a[1] - b[1];
- } // Computes the upper convex hull per the monotone chain algorithm.
- // Assumes points.length >= 3, is sorted by x, unique in y.
- // Returns an array of indices into points in left-to-right order.
-
-
- function computeUpperHullIndexes(points) {
- var n = points.length,
- indexes = [0, 1];
- var size = 2,
- i;
-
- for (i = 2; i < n; ++i) {
- while (size > 1 && cross(points[indexes[size - 2]], points[indexes[size - 1]], points[i]) <= 0) {
- --size;
- }
-
- indexes[size++] = i;
- }
-
- return indexes.slice(0, size); // remove popped points
- }
-
- function d3_polygonHull (points) {
- if ((n = points.length) < 3) return null;
- var i,
- n,
- sortedPoints = new Array(n),
- flippedPoints = new Array(n);
-
- for (i = 0; i < n; ++i) {
- sortedPoints[i] = [+points[i][0], +points[i][1], i];
- }
-
- sortedPoints.sort(lexicographicOrder);
-
- for (i = 0; i < n; ++i) {
- flippedPoints[i] = [sortedPoints[i][0], -sortedPoints[i][1]];
- }
-
- var upperIndexes = computeUpperHullIndexes(sortedPoints),
- lowerIndexes = computeUpperHullIndexes(flippedPoints); // Construct the hull polygon, removing possible duplicate endpoints.
-
- var skipLeft = lowerIndexes[0] === upperIndexes[0],
- skipRight = lowerIndexes[lowerIndexes.length - 1] === upperIndexes[upperIndexes.length - 1],
- hull = []; // Add upper hull in right-to-l order.
- // Then add lower hull in left-to-right order.
-
- for (i = upperIndexes.length - 1; i >= 0; --i) {
- hull.push(points[sortedPoints[upperIndexes[i]][2]]);
- }
-
- for (i = +skipLeft; i < lowerIndexes.length - skipRight; ++i) {
- hull.push(points[sortedPoints[lowerIndexes[i]][2]]);
- }
-
- return hull;
- }
-
- // vector equals
- function geoVecEqual(a, b, epsilon) {
- if (epsilon) {
- return Math.abs(a[0] - b[0]) <= epsilon && Math.abs(a[1] - b[1]) <= epsilon;
- } else {
- return a[0] === b[0] && a[1] === b[1];
- }
- } // vector addition
-
- function geoVecAdd(a, b) {
- return [a[0] + b[0], a[1] + b[1]];
- } // vector subtraction
-
- function geoVecSubtract(a, b) {
- return [a[0] - b[0], a[1] - b[1]];
- } // vector scaling
-
- function geoVecScale(a, mag) {
- return [a[0] * mag, a[1] * mag];
- } // vector rounding (was: geoRoundCoordinates)
-
- function geoVecFloor(a) {
- return [Math.floor(a[0]), Math.floor(a[1])];
- } // linear interpolation
-
- function geoVecInterp(a, b, t) {
- return [a[0] + (b[0] - a[0]) * t, a[1] + (b[1] - a[1]) * t];
- } // http://jsperf.com/id-dist-optimization
-
- function geoVecLength(a, b) {
- return Math.sqrt(geoVecLengthSquare(a, b));
- } // length of vector raised to the power two
-
- function geoVecLengthSquare(a, b) {
- b = b || [0, 0];
- var x = a[0] - b[0];
- var y = a[1] - b[1];
- return x * x + y * y;
- } // get a unit vector
-
- function geoVecNormalize(a) {
- var length = Math.sqrt(a[0] * a[0] + a[1] * a[1]);
-
- if (length !== 0) {
- return geoVecScale(a, 1 / length);
- }
-
- return [0, 0];
- } // Return the counterclockwise angle in the range (-pi, pi)
- // between the positive X axis and the line intersecting a and b.
-
- function geoVecAngle(a, b) {
- return Math.atan2(b[1] - a[1], b[0] - a[0]);
- } // dot product
-
- function geoVecDot(a, b, origin) {
- origin = origin || [0, 0];
- var p = geoVecSubtract(a, origin);
- var q = geoVecSubtract(b, origin);
- return p[0] * q[0] + p[1] * q[1];
- } // normalized dot product
-
- function geoVecNormalizedDot(a, b, origin) {
- origin = origin || [0, 0];
- var p = geoVecNormalize(geoVecSubtract(a, origin));
- var q = geoVecNormalize(geoVecSubtract(b, origin));
- return geoVecDot(p, q);
- } // 2D cross product of OA and OB vectors, returns magnitude of Z vector
- // Returns a positive value, if OAB makes a counter-clockwise turn,
- // negative for clockwise turn, and zero if the points are collinear.
-
- function geoVecCross(a, b, origin) {
- origin = origin || [0, 0];
- var p = geoVecSubtract(a, origin);
- var q = geoVecSubtract(b, origin);
- return p[0] * q[1] - p[1] * q[0];
- } // find closest orthogonal projection of point onto points array
-
- function geoVecProject(a, points) {
- var min = Infinity;
- var idx;
- var target;
-
- for (var i = 0; i < points.length - 1; i++) {
- var o = points[i];
- var s = geoVecSubtract(points[i + 1], o);
- var v = geoVecSubtract(a, o);
- var proj = geoVecDot(v, s) / geoVecDot(s, s);
- var p;
-
- if (proj < 0) {
- p = o;
- } else if (proj > 1) {
- p = points[i + 1];
- } else {
- p = [o[0] + proj * s[0], o[1] + proj * s[1]];
- }
-
- var dist = geoVecLength(p, a);
-
- if (dist < min) {
- min = dist;
- idx = i + 1;
- target = p;
- }
- }
-
- if (idx !== undefined) {
- return {
- index: idx,
- distance: min,
- target: target
- };
- } else {
- return null;
- }
- }
-
- // between the positive X axis and the line intersecting a and b.
-
- function geoAngle(a, b, projection) {
- return geoVecAngle(projection(a.loc), projection(b.loc));
- }
- function geoEdgeEqual(a, b) {
- return a[0] === b[0] && a[1] === b[1] || a[0] === b[1] && a[1] === b[0];
- } // Rotate all points counterclockwise around a pivot point by given angle
-
- function geoRotate(points, angle, around) {
- return points.map(function (point) {
- var radial = geoVecSubtract(point, around);
- return [radial[0] * Math.cos(angle) - radial[1] * Math.sin(angle) + around[0], radial[0] * Math.sin(angle) + radial[1] * Math.cos(angle) + around[1]];
- });
- } // Choose the edge with the minimal distance from `point` to its orthogonal
- // projection onto that edge, if such a projection exists, or the distance to
- // the closest vertex on that edge. Returns an object with the `index` of the
- // chosen edge, the chosen `loc` on that edge, and the `distance` to to it.
-
- function geoChooseEdge(nodes, point, projection, activeID) {
- var dist = geoVecLength;
- var points = nodes.map(function (n) {
- return projection(n.loc);
- });
- var ids = nodes.map(function (n) {
- return n.id;
- });
- var min = Infinity;
- var idx;
- var loc;
-
- for (var i = 0; i < points.length - 1; i++) {
- if (ids[i] === activeID || ids[i + 1] === activeID) continue;
- var o = points[i];
- var s = geoVecSubtract(points[i + 1], o);
- var v = geoVecSubtract(point, o);
- var proj = geoVecDot(v, s) / geoVecDot(s, s);
- var p;
-
- if (proj < 0) {
- p = o;
- } else if (proj > 1) {
- p = points[i + 1];
- } else {
- p = [o[0] + proj * s[0], o[1] + proj * s[1]];
- }
-
- var d = dist(p, point);
-
- if (d < min) {
- min = d;
- idx = i + 1;
- loc = projection.invert(p);
- }
- }
-
- if (idx !== undefined) {
- return {
- index: idx,
- distance: min,
- loc: loc
- };
- } else {
- return null;
- }
- } // Test active (dragged or drawing) segments against inactive segments
- // This is used to test e.g. multipolygon rings that cross
- // `activeNodes` is the ring containing the activeID being dragged.
- // `inactiveNodes` is the other ring to test against
-
- function geoHasLineIntersections(activeNodes, inactiveNodes, activeID) {
- var actives = [];
- var inactives = [];
- var j, k, n1, n2, segment; // gather active segments (only segments in activeNodes that contain the activeID)
-
- for (j = 0; j < activeNodes.length - 1; j++) {
- n1 = activeNodes[j];
- n2 = activeNodes[j + 1];
- segment = [n1.loc, n2.loc];
-
- if (n1.id === activeID || n2.id === activeID) {
- actives.push(segment);
- }
- } // gather inactive segments
-
-
- for (j = 0; j < inactiveNodes.length - 1; j++) {
- n1 = inactiveNodes[j];
- n2 = inactiveNodes[j + 1];
- segment = [n1.loc, n2.loc];
- inactives.push(segment);
- } // test
-
-
- for (j = 0; j < actives.length; j++) {
- for (k = 0; k < inactives.length; k++) {
- var p = actives[j];
- var q = inactives[k];
- var hit = geoLineIntersection(p, q);
-
- if (hit) {
- return true;
- }
- }
- }
-
- return false;
- } // Test active (dragged or drawing) segments against inactive segments
- // This is used to test whether a way intersects with itself.
-
- function geoHasSelfIntersections(nodes, activeID) {
- var actives = [];
- var inactives = [];
- var j, k; // group active and passive segments along the nodes
-
- for (j = 0; j < nodes.length - 1; j++) {
- var n1 = nodes[j];
- var n2 = nodes[j + 1];
- var segment = [n1.loc, n2.loc];
-
- if (n1.id === activeID || n2.id === activeID) {
- actives.push(segment);
- } else {
- inactives.push(segment);
- }
- } // test
-
-
- for (j = 0; j < actives.length; j++) {
- for (k = 0; k < inactives.length; k++) {
- var p = actives[j];
- var q = inactives[k]; // skip if segments share an endpoint
-
- if (geoVecEqual(p[1], q[0]) || geoVecEqual(p[0], q[1]) || geoVecEqual(p[0], q[0]) || geoVecEqual(p[1], q[1])) {
- continue;
- }
-
- var hit = geoLineIntersection(p, q);
-
- if (hit) {
- var epsilon = 1e-8; // skip if the hit is at the segment's endpoint
-
- if (geoVecEqual(p[1], hit, epsilon) || geoVecEqual(p[0], hit, epsilon) || geoVecEqual(q[1], hit, epsilon) || geoVecEqual(q[0], hit, epsilon)) {
- continue;
- } else {
- return true;
- }
- }
- }
- }
-
- return false;
- } // Return the intersection point of 2 line segments.
- // From https://github.com/pgkelley4/line-segments-intersect
- // This uses the vector cross product approach described below:
- // http://stackoverflow.com/a/565282/786339
-
- function geoLineIntersection(a, b) {
- var p = [a[0][0], a[0][1]];
- var p2 = [a[1][0], a[1][1]];
- var q = [b[0][0], b[0][1]];
- var q2 = [b[1][0], b[1][1]];
- var r = geoVecSubtract(p2, p);
- var s = geoVecSubtract(q2, q);
- var uNumerator = geoVecCross(geoVecSubtract(q, p), r);
- var denominator = geoVecCross(r, s);
-
- if (uNumerator && denominator) {
- var u = uNumerator / denominator;
- var t = geoVecCross(geoVecSubtract(q, p), s) / denominator;
-
- if (t >= 0 && t <= 1 && u >= 0 && u <= 1) {
- return geoVecInterp(p, p2, t);
- }
- }
-
- return null;
- }
- function geoPathIntersections(path1, path2) {
- var intersections = [];
-
- for (var i = 0; i < path1.length - 1; i++) {
- for (var j = 0; j < path2.length - 1; j++) {
- var a = [path1[i], path1[i + 1]];
- var b = [path2[j], path2[j + 1]];
- var hit = geoLineIntersection(a, b);
-
- if (hit) {
- intersections.push(hit);
- }
- }
- }
-
- return intersections;
- }
- function geoPathHasIntersections(path1, path2) {
- for (var i = 0; i < path1.length - 1; i++) {
- for (var j = 0; j < path2.length - 1; j++) {
- var a = [path1[i], path1[i + 1]];
- var b = [path2[j], path2[j + 1]];
- var hit = geoLineIntersection(a, b);
-
- if (hit) {
- return true;
- }
- }
- }
-
- return false;
- } // Return whether point is contained in polygon.
- //
- // `point` should be a 2-item array of coordinates.
- // `polygon` should be an array of 2-item arrays of coordinates.
- //
- // From https://github.com/substack/point-in-polygon.
- // ray-casting algorithm based on
- // http://www.ecse.rpi.edu/Homepages/wrf/Research/Short_Notes/pnpoly.html
- //
-
- function geoPointInPolygon(point, polygon) {
- var x = point[0];
- var y = point[1];
- var inside = false;
-
- for (var i = 0, j = polygon.length - 1; i < polygon.length; j = i++) {
- var xi = polygon[i][0];
- var yi = polygon[i][1];
- var xj = polygon[j][0];
- var yj = polygon[j][1];
- var intersect = yi > y !== yj > y && x < (xj - xi) * (y - yi) / (yj - yi) + xi;
- if (intersect) inside = !inside;
- }
-
- return inside;
- }
- function geoPolygonContainsPolygon(outer, inner) {
- return inner.every(function (point) {
- return geoPointInPolygon(point, outer);
- });
- }
- function geoPolygonIntersectsPolygon(outer, inner, checkSegments) {
- function testPoints(outer, inner) {
- return inner.some(function (point) {
- return geoPointInPolygon(point, outer);
- });
- }
-
- return testPoints(outer, inner) || !!checkSegments && geoPathHasIntersections(outer, inner);
- } // http://gis.stackexchange.com/questions/22895/finding-minimum-area-rectangle-for-given-points
- // http://gis.stackexchange.com/questions/3739/generalisation-strategies-for-building-outlines/3756#3756
-
- function geoGetSmallestSurroundingRectangle(points) {
- var hull = d3_polygonHull(points);
- var centroid = d3_polygonCentroid(hull);
- var minArea = Infinity;
- var ssrExtent = [];
- var ssrAngle = 0;
- var c1 = hull[0];
-
- for (var i = 0; i <= hull.length - 1; i++) {
- var c2 = i === hull.length - 1 ? hull[0] : hull[i + 1];
- var angle = Math.atan2(c2[1] - c1[1], c2[0] - c1[0]);
- var poly = geoRotate(hull, -angle, centroid);
- var extent = poly.reduce(function (extent, point) {
- return extent.extend(geoExtent(point));
- }, geoExtent());
- var area = extent.area();
-
- if (area < minArea) {
- minArea = area;
- ssrExtent = extent;
- ssrAngle = angle;
- }
-
- c1 = c2;
- }
-
- return {
- poly: geoRotate(ssrExtent.polygon(), ssrAngle, centroid),
- angle: ssrAngle
- };
- }
- function geoPathLength(path) {
- var length = 0;
-
- for (var i = 0; i < path.length - 1; i++) {
- length += geoVecLength(path[i], path[i + 1]);
- }
-
- return length;
- } // If the given point is at the edge of the padded viewport,
- // return a vector that will nudge the viewport in that direction
-
- function geoViewportEdge(point, dimensions) {
- var pad = [80, 20, 50, 20]; // top, right, bottom, left
-
- 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 (x || y) {
- return [x, y];
- } else {
- return null;
- }
- }
-
- var noop$1 = {
- value: function value() {}
- };
-
- function dispatch() {
- 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(_);
- }
-
- function Dispatch$1(_) {
- this._ = _;
- }
-
- function parseTypenames(typenames, types) {
- return typenames.trim().split(/^|\s+/).map(function (t) {
- var name = "",
- i = t.indexOf(".");
- if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
- if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
- return {
- type: t,
- name: name
- };
- });
- }
-
- Dispatch$1.prototype = dispatch.prototype = {
- constructor: Dispatch$1,
- on: function on(typename, callback) {
- var _ = this._,
- T = parseTypenames(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;
- }
-
- return;
- } // If a type was specified, set the callback for the given type and name.
- // Otherwise, if a null callback was specified, remove callbacks of the given name.
-
-
- 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);
- }
- }
-
- return this;
- },
- copy: function copy() {
- var copy = {},
- _ = this._;
-
- for (var t in _) {
- copy[t] = _[t].slice();
- }
-
- return new Dispatch$1(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) {
- args[i] = arguments[i + 2];
- }
- if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
-
- for (t = this._[type], i = 0, n = t.length; i < n; ++i) {
- t[i].value.apply(that, args);
- }
- },
- apply: function apply(type, that, args) {
- if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
-
- for (var t = this._[type], i = 0, n = t.length; i < n; ++i) {
- t[i].value.apply(that, args);
- }
- }
- };
-
- function get$3(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) {
- 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));
- break;
- }
- }
-
- if (callback != null) type.push({
- name: name,
- value: callback
- });
- return type;
- }
-
- var xhtml = "http://www.w3.org/1999/xhtml";
- var namespaces = {
- svg: "http://www.w3.org/2000/svg",
- xhtml: xhtml,
- xlink: "http://www.w3.org/1999/xlink",
- xml: "http://www.w3.org/XML/1998/namespace",
- xmlns: "http://www.w3.org/2000/xmlns/"
- };
-
- function namespace (name) {
- var prefix = name += "",
- i = prefix.indexOf(":");
- if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
- return namespaces.hasOwnProperty(prefix) ? {
- space: namespaces[prefix],
- local: name
- } : name; // eslint-disable-line no-prototype-builtins
- }
-
- function creatorInherit(name) {
- return function () {
- var document = this.ownerDocument,
- uri = this.namespaceURI;
- return uri === xhtml && document.documentElement.namespaceURI === xhtml ? document.createElement(name) : document.createElementNS(uri, name);
- };
- }
-
- function creatorFixed(fullname) {
- return function () {
- return this.ownerDocument.createElementNS(fullname.space, fullname.local);
- };
- }
-
- function creator (name) {
- var fullname = namespace(name);
- return (fullname.local ? creatorFixed : creatorInherit)(fullname);
- }
-
- function none() {}
-
- function selector (selector) {
- return selector == null ? none : function () {
- return this.querySelector(selector);
- };
- }
-
- function selection_select (select) {
- if (typeof select !== "function") select = selector(select);
-
- for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
- for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
- if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
- if ("__data__" in node) subnode.__data__ = node.__data__;
- subgroup[i] = subnode;
- }
- }
- }
-
- return new Selection(subgroups, this._parents);
- }
-
- function array (x) {
- return _typeof(x) === "object" && "length" in x ? x // Array, TypedArray, NodeList, array-like
- : Array.from(x); // Map, Set, iterable, string, or anything else
- }
-
- function empty() {
- return [];
- }
-
- function selectorAll (selector) {
- return selector == null ? empty : function () {
- return this.querySelectorAll(selector);
- };
- }
-
- function arrayAll(select) {
- return function () {
- var group = select.apply(this, arguments);
- return group == null ? [] : array(group);
- };
- }
-
- function selection_selectAll (select) {
- if (typeof select === "function") select = arrayAll(select);else select = selectorAll(select);
-
- 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]) {
- subgroups.push(select.call(node, node.__data__, i, group));
- parents.push(node);
- }
- }
- }
-
- return new Selection(subgroups, parents);
- }
-
- var $find$1 = arrayIteration.find;
-
-
-
- var FIND = 'find';
- var SKIPS_HOLES = true;
-
- var USES_TO_LENGTH$a = arrayMethodUsesToLength(FIND);
-
- // Shouldn't skip holes
- if (FIND in []) Array(1)[FIND](function () { SKIPS_HOLES = false; });
-
- // `Array.prototype.find` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.find
- _export({ target: 'Array', proto: true, forced: SKIPS_HOLES || !USES_TO_LENGTH$a }, {
- find: function find(callbackfn /* , that = undefined */) {
- return $find$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- // https://tc39.github.io/ecma262/#sec-array.prototype-@@unscopables
- addToUnscopables(FIND);
-
- function matcher (selector) {
- return function () {
- return this.matches(selector);
- };
- }
- function childMatcher(selector) {
- return function (node) {
- return node.matches(selector);
- };
- }
-
- var find$1 = Array.prototype.find;
-
- function childFind(match) {
- return function () {
- return find$1.call(this.children, match);
- };
- }
-
- function childFirst() {
- return this.firstElementChild;
- }
-
- function selection_selectChild (match) {
- return this.select(match == null ? childFirst : childFind(typeof match === "function" ? match : childMatcher(match)));
- }
-
- var filter = Array.prototype.filter;
-
- function children() {
- return this.children;
- }
-
- function childrenFilter(match) {
- return function () {
- return filter.call(this.children, match);
- };
- }
-
- function selection_selectChildren (match) {
- return this.selectAll(match == null ? children : childrenFilter(typeof match === "function" ? match : childMatcher(match)));
- }
-
- function selection_filter (match) {
- if (typeof match !== "function") match = matcher(match);
-
- for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
- for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
- if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
- subgroup.push(node);
- }
- }
- }
-
- return new Selection(subgroups, this._parents);
- }
-
- function sparse (update) {
- return new Array(update.length);
- }
-
- function selection_enter () {
- return new Selection(this._enter || this._groups.map(sparse), this._parents);
- }
- function EnterNode(parent, datum) {
- this.ownerDocument = parent.ownerDocument;
- this.namespaceURI = parent.namespaceURI;
- this._next = null;
- this._parent = parent;
- this.__data__ = datum;
- }
- EnterNode.prototype = {
- constructor: EnterNode,
- appendChild: function appendChild(child) {
- return this._parent.insertBefore(child, this._next);
- },
- insertBefore: function insertBefore(child, next) {
- return this._parent.insertBefore(child, next);
- },
- querySelector: function querySelector(selector) {
- return this._parent.querySelector(selector);
- },
- querySelectorAll: function querySelectorAll(selector) {
- return this._parent.querySelectorAll(selector);
- }
- };
-
- function constant (x) {
- return function () {
- return x;
- };
- }
-
- function bindIndex(parent, group, enter, update, exit, data) {
- var i = 0,
- node,
- groupLength = group.length,
- dataLength = data.length; // Put any non-null nodes that fit into update.
- // Put any null nodes into enter.
- // Put any remaining data into enter.
-
- for (; i < dataLength; ++i) {
- if (node = group[i]) {
- node.__data__ = data[i];
- update[i] = node;
- } else {
- enter[i] = new EnterNode(parent, data[i]);
- }
- } // Put any non-null nodes that don’t fit into exit.
-
-
- for (; i < groupLength; ++i) {
- if (node = group[i]) {
- exit[i] = node;
- }
- }
- }
-
- function bindKey(parent, group, enter, update, exit, data, key) {
- var i,
- node,
- nodeByKeyValue = new Map(),
- groupLength = group.length,
- dataLength = data.length,
- keyValues = new Array(groupLength),
- keyValue; // Compute the key for each node.
- // If multiple nodes have the same key, the duplicates are added to exit.
-
- for (i = 0; i < groupLength; ++i) {
- if (node = group[i]) {
- keyValues[i] = keyValue = key.call(node, node.__data__, i, group) + "";
-
- if (nodeByKeyValue.has(keyValue)) {
- exit[i] = node;
- } else {
- nodeByKeyValue.set(keyValue, node);
- }
- }
- } // Compute the key for each datum.
- // If there a node associated with this key, join and add it to update.
- // If there is not (or the key is a duplicate), add it to enter.
-
-
- for (i = 0; i < dataLength; ++i) {
- keyValue = key.call(parent, data[i], i, data) + "";
-
- if (node = nodeByKeyValue.get(keyValue)) {
- update[i] = node;
- node.__data__ = data[i];
- nodeByKeyValue["delete"](keyValue);
- } else {
- enter[i] = new EnterNode(parent, data[i]);
- }
- } // Add any remaining nodes that were not bound to data to exit.
-
-
- for (i = 0; i < groupLength; ++i) {
- if ((node = group[i]) && nodeByKeyValue.get(keyValues[i]) === node) {
- exit[i] = node;
- }
- }
- }
-
- function datum(node) {
- return node.__data__;
- }
-
- function selection_data (value, key) {
- if (!arguments.length) return Array.from(this, datum);
- var bind = key ? bindKey : bindIndex,
- parents = this._parents,
- groups = this._groups;
- if (typeof value !== "function") value = constant(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],
- group = groups[j],
- groupLength = group.length,
- data = array(value.call(parent, parent && parent.__data__, j, parents)),
- dataLength = data.length,
- enterGroup = enter[j] = new Array(dataLength),
- updateGroup = update[j] = new Array(dataLength),
- exitGroup = exit[j] = new Array(groupLength);
- bind(parent, group, enterGroup, updateGroup, exitGroup, data, key); // Now connect the enter nodes to their following update node, such that
- // appendChild can insert the materialized enter node before this node,
- // rather than at the end of the parent node.
-
- for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
- if (previous = enterGroup[i0]) {
- if (i0 >= i1) i1 = i0 + 1;
-
- while (!(next = updateGroup[i1]) && ++i1 < dataLength) {
- }
-
- previous._next = next || null;
- }
- }
- }
-
- update = new Selection(update, parents);
- update._enter = enter;
- update._exit = exit;
- return update;
- }
-
- function selection_exit () {
- return new Selection(this._exit || this._groups.map(sparse), this._parents);
- }
-
- function selection_join (onenter, onupdate, onexit) {
- var enter = this.enter(),
- update = this,
- exit = this.exit();
- enter = typeof onenter === "function" ? onenter(enter) : enter.append(onenter + "");
- if (onupdate != null) update = onupdate(update);
- if (onexit == null) exit.remove();else onexit(exit);
- return enter && update ? enter.merge(update).order() : update;
- }
-
- function selection_merge (selection) {
- if (!(selection instanceof Selection)) 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) {
- if (node = group0[i] || group1[i]) {
- merge[i] = node;
- }
- }
- }
-
- for (; j < m0; ++j) {
- merges[j] = groups0[j];
- }
-
- return new Selection(merges, this._parents);
- }
-
- function selection_order () {
- for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
- for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
- if (node = group[i]) {
- if (next && node.compareDocumentPosition(next) ^ 4) next.parentNode.insertBefore(node, next);
- next = node;
- }
- }
- }
-
- return this;
- }
-
- function selection_sort (compare) {
- if (!compare) compare = ascending;
-
- function compareNode(a, b) {
- return a && b ? compare(a.__data__, b.__data__) : !a - !b;
- }
-
- for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
- for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
- if (node = group[i]) {
- sortgroup[i] = node;
- }
- }
-
- sortgroup.sort(compareNode);
- }
-
- return new Selection(sortgroups, this._parents).order();
- }
-
- function ascending(a, b) {
- return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
- }
-
- function selection_call () {
- var callback = arguments[0];
- arguments[0] = this;
- callback.apply(null, arguments);
- return this;
- }
-
- function selection_nodes () {
- return Array.from(this);
- }
-
- function selection_node () {
- for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
- for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
- var node = group[i];
- if (node) return node;
- }
- }
-
- return null;
- }
-
- function selection_size () {
- var size = 0;
-
- var _iterator = _createForOfIteratorHelper(this),
- _step;
-
- try {
- for (_iterator.s(); !(_step = _iterator.n()).done;) {
- var node = _step.value;
- ++size;
- } // eslint-disable-line no-unused-vars
-
- } catch (err) {
- _iterator.e(err);
- } finally {
- _iterator.f();
- }
-
- return size;
- }
-
- function selection_empty () {
- return !this.node();
- }
-
- function selection_each (callback) {
- for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
- for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
- if (node = group[i]) callback.call(node, node.__data__, i, group);
- }
- }
-
- return this;
- }
-
- function attrRemove(name) {
- return function () {
- this.removeAttribute(name);
- };
- }
-
- function attrRemoveNS(fullname) {
- return function () {
- this.removeAttributeNS(fullname.space, fullname.local);
- };
- }
-
- function attrConstant(name, value) {
- return function () {
- this.setAttribute(name, value);
- };
- }
-
- function attrConstantNS(fullname, value) {
- return function () {
- this.setAttributeNS(fullname.space, fullname.local, value);
- };
- }
-
- function attrFunction(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) {
- 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);
- };
- }
-
- function selection_attr (name, value) {
- var fullname = namespace(name);
-
- if (arguments.length < 2) {
- var node = this.node();
- 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));
- }
-
- function defaultView (node) {
- return node.ownerDocument && node.ownerDocument.defaultView || // node is a Node
- node.document && node // node is a Window
- || node.defaultView; // node is a Document
- }
-
- function styleRemove(name) {
- return function () {
- this.style.removeProperty(name);
- };
- }
-
- function styleConstant(name, value, priority) {
- return function () {
- this.style.setProperty(name, value, priority);
- };
- }
-
- function styleFunction(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);
- }
- function styleValue(node, name) {
- return node.style.getPropertyValue(name) || defaultView(node).getComputedStyle(node, null).getPropertyValue(name);
- }
-
- function propertyRemove(name) {
- return function () {
- delete this[name];
- };
- }
-
- function propertyConstant(name, value) {
- return function () {
- this[name] = value;
- };
- }
-
- function propertyFunction(name, value) {
- return function () {
- var v = value.apply(this, arguments);
- if (v == null) delete this[name];else this[name] = v;
- };
- }
-
- function selection_property (name, value) {
- return arguments.length > 1 ? this.each((value == null ? propertyRemove : typeof value === "function" ? propertyFunction : propertyConstant)(name, value)) : this.node()[name];
- }
-
- function classArray(string) {
- return string.trim().split(/^|\s+/);
- }
-
- function classList(node) {
- return node.classList || new ClassList(node);
- }
-
- function ClassList(node) {
- this._node = node;
- this._names = classArray(node.getAttribute("class") || "");
- }
-
- ClassList.prototype = {
- add: function add(name) {
- var i = this._names.indexOf(name);
-
- if (i < 0) {
- this._names.push(name);
-
- this._node.setAttribute("class", this._names.join(" "));
- }
- },
- remove: function remove(name) {
- var i = this._names.indexOf(name);
-
- if (i >= 0) {
- this._names.splice(i, 1);
-
- this._node.setAttribute("class", this._names.join(" "));
- }
- },
- contains: function contains(name) {
- return this._names.indexOf(name) >= 0;
- }
- };
-
- function classedAdd(node, names) {
- var list = classList(node),
- i = -1,
- n = names.length;
-
- while (++i < n) {
- list.add(names[i]);
- }
- }
-
- function classedRemove(node, names) {
- var list = classList(node),
- i = -1,
- n = names.length;
-
- while (++i < n) {
- list.remove(names[i]);
- }
- }
-
- function classedTrue(names) {
- return function () {
- classedAdd(this, names);
- };
- }
-
- function classedFalse(names) {
- return function () {
- classedRemove(this, names);
- };
- }
-
- function classedFunction(names, value) {
- return function () {
- (value.apply(this, arguments) ? classedAdd : classedRemove)(this, names);
- };
- }
-
- function selection_classed (name, value) {
- var names = classArray(name + "");
-
- if (arguments.length < 2) {
- var list = classList(this.node()),
- i = -1,
- n = names.length;
-
- while (++i < n) {
- if (!list.contains(names[i])) return false;
- }
-
- return true;
- }
-
- return this.each((typeof value === "function" ? classedFunction : value ? classedTrue : classedFalse)(names, value));
- }
-
- function textRemove() {
- this.textContent = "";
- }
-
- function textConstant(value) {
- return function () {
- this.textContent = value;
- };
- }
-
- function textFunction(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;
- }
-
- function htmlRemove() {
- this.innerHTML = "";
- }
-
- function htmlConstant(value) {
- return function () {
- this.innerHTML = value;
- };
- }
-
- function htmlFunction(value) {
- return function () {
- var v = value.apply(this, arguments);
- this.innerHTML = v == null ? "" : v;
- };
- }
-
- function selection_html (value) {
- return arguments.length ? this.each(value == null ? htmlRemove : (typeof value === "function" ? htmlFunction : htmlConstant)(value)) : this.node().innerHTML;
- }
-
- function raise() {
- if (this.nextSibling) this.parentNode.appendChild(this);
- }
-
- function selection_raise () {
- return this.each(raise);
- }
-
- function lower() {
- if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
- }
-
- function selection_lower () {
- return this.each(lower);
- }
-
- function selection_append (name) {
- var create = typeof name === "function" ? name : creator(name);
- return this.select(function () {
- return this.appendChild(create.apply(this, arguments));
- });
- }
-
- function constantNull() {
- return null;
- }
-
- function selection_insert (name, before) {
- var create = typeof name === "function" ? name : creator(name),
- select = before == null ? constantNull : typeof before === "function" ? before : selector(before);
- return this.select(function () {
- return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
- });
- }
-
- function remove() {
- var parent = this.parentNode;
- if (parent) parent.removeChild(this);
- }
-
- function selection_remove () {
- return this.each(remove);
- }
-
- function selection_cloneShallow() {
- var clone = this.cloneNode(false),
- parent = this.parentNode;
- return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
- }
-
- function selection_cloneDeep() {
- var clone = this.cloneNode(true),
- parent = this.parentNode;
- return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
- }
-
- function selection_clone (deep) {
- return this.select(deep ? selection_cloneDeep : selection_cloneShallow);
- }
-
- function selection_datum (value) {
- return arguments.length ? this.property("__data__", value) : this.node().__data__;
- }
-
- function contextListener(listener) {
- return function (event) {
- listener.call(this, event, this.__data__);
- };
- }
-
- function parseTypenames$1(typenames) {
- return typenames.trim().split(/^|\s+/).map(function (t) {
- var name = "",
- i = t.indexOf(".");
- if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
- return {
- type: t,
- name: name
- };
- });
- }
-
- function onRemove(typename) {
- return function () {
- var on = this.__on;
- if (!on) return;
-
- for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
- if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
- this.removeEventListener(o.type, o.listener, o.options);
- } else {
- on[++i] = o;
- }
- }
-
- if (++i) on.length = i;else delete this.__on;
- };
- }
-
- function onAdd(typename, value, options) {
- return function () {
- var on = this.__on,
- o,
- listener = contextListener(value);
- if (on) for (var j = 0, m = on.length; j < m; ++j) {
- if ((o = on[j]).type === typename.type && o.name === typename.name) {
- this.removeEventListener(o.type, o.listener, o.options);
- this.addEventListener(o.type, o.listener = listener, o.options = options);
- o.value = value;
- return;
- }
- }
- this.addEventListener(typename.type, listener, options);
- o = {
- type: typename.type,
- name: typename.name,
- value: value,
- listener: listener,
- options: options
- };
- if (!on) this.__on = [o];else on.push(o);
- };
- }
-
- function selection_on (typename, value, options) {
- var typenames = parseTypenames$1(typename + ""),
- i,
- n = typenames.length,
- t;
-
- if (arguments.length < 2) {
- var on = this.node().__on;
-
- if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
- for (i = 0, o = on[j]; i < n; ++i) {
- if ((t = typenames[i]).type === o.type && t.name === o.name) {
- return o.value;
- }
- }
- }
- return;
- }
-
- on = value ? onAdd : onRemove;
-
- for (i = 0; i < n; ++i) {
- this.each(on(typenames[i], value, options));
- }
-
- return this;
- }
-
- function dispatchEvent$1(node, type, params) {
- var window = defaultView(node),
- event = window.CustomEvent;
-
- if (typeof event === "function") {
- event = new event(type, params);
- } else {
- event = window.document.createEvent("Event");
- if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;else event.initEvent(type, false, false);
- }
-
- node.dispatchEvent(event);
- }
-
- function dispatchConstant(type, params) {
- return function () {
- return dispatchEvent$1(this, type, params);
- };
- }
-
- function dispatchFunction(type, params) {
- return function () {
- return dispatchEvent$1(this, type, params.apply(this, arguments));
- };
- }
-
- function selection_dispatch (type, params) {
- return this.each((typeof params === "function" ? dispatchFunction : dispatchConstant)(type, params));
- }
-
- var _marked$2 = /*#__PURE__*/regeneratorRuntime.mark(_callee);
-
- function _callee() {
- var groups, j, m, group, i, n, node;
- return regeneratorRuntime.wrap(function _callee$(_context) {
- while (1) {
- switch (_context.prev = _context.next) {
- case 0:
- groups = this._groups, j = 0, m = groups.length;
-
- case 1:
- if (!(j < m)) {
- _context.next = 13;
- break;
- }
-
- group = groups[j], i = 0, n = group.length;
-
- case 3:
- if (!(i < n)) {
- _context.next = 10;
- break;
- }
-
- if (!(node = group[i])) {
- _context.next = 7;
- break;
- }
-
- _context.next = 7;
- return node;
-
- case 7:
- ++i;
- _context.next = 3;
- break;
-
- case 10:
- ++j;
- _context.next = 1;
- break;
-
- case 13:
- case "end":
- return _context.stop();
- }
- }
- }, _marked$2, this);
- }
-
- var root = [null];
- function Selection(groups, parents) {
- this._groups = groups;
- this._parents = parents;
- }
-
- function selection() {
- return new Selection([[document.documentElement]], root);
- }
-
- function selection_selection() {
- return this;
- }
-
- Selection.prototype = selection.prototype = _defineProperty({
- constructor: Selection,
- select: selection_select,
- selectAll: selection_selectAll,
- selectChild: selection_selectChild,
- selectChildren: selection_selectChildren,
- filter: selection_filter,
- data: selection_data,
- enter: selection_enter,
- exit: selection_exit,
- join: selection_join,
- merge: selection_merge,
- selection: selection_selection,
- order: selection_order,
- sort: selection_sort,
- call: selection_call,
- nodes: selection_nodes,
- node: selection_node,
- size: selection_size,
- empty: selection_empty,
- each: selection_each,
- attr: selection_attr,
- style: selection_style,
- property: selection_property,
- classed: selection_classed,
- text: selection_text,
- html: selection_html,
- raise: selection_raise,
- lower: selection_lower,
- append: selection_append,
- insert: selection_insert,
- remove: selection_remove,
- clone: selection_clone,
- datum: selection_datum,
- on: selection_on,
- dispatch: selection_dispatch
- }, Symbol.iterator, _callee);
-
- function select (selector) {
- return typeof selector === "string" ? new Selection([[document.querySelector(selector)]], [document.documentElement]) : new Selection([[selector]], root);
- }
-
- function sourceEvent (event) {
- var sourceEvent;
-
- while (sourceEvent = event.sourceEvent) {
- event = sourceEvent;
- }
-
- return event;
- }
-
- function pointer (event, node) {
- event = sourceEvent(event);
- if (node === undefined) node = event.currentTarget;
-
- if (node) {
- var svg = node.ownerSVGElement || node;
-
- if (svg.createSVGPoint) {
- var point = svg.createSVGPoint();
- point.x = event.clientX, point.y = event.clientY;
- point = point.matrixTransform(node.getScreenCTM().inverse());
- return [point.x, point.y];
- }
-
- if (node.getBoundingClientRect) {
- var rect = node.getBoundingClientRect();
- return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
- }
- }
-
- return [event.pageX, event.pageY];
- }
-
- function selectAll (selector) {
- return typeof selector === "string" ? new Selection([document.querySelectorAll(selector)], [document.documentElement]) : new Selection([selector == null ? [] : array(selector)], root);
- }
-
- function nopropagation(event) {
- event.stopImmediatePropagation();
- }
- function noevent (event) {
- event.preventDefault();
- event.stopImmediatePropagation();
- }
-
- function dragDisable (view) {
- var root = view.document.documentElement,
- selection = select(view).on("dragstart.drag", noevent, true);
-
- if ("onselectstart" in root) {
- selection.on("selectstart.drag", noevent, true);
- } else {
- root.__noselect = root.style.MozUserSelect;
- root.style.MozUserSelect = "none";
- }
- }
- function yesdrag(view, noclick) {
- var root = view.document.documentElement,
- selection = select(view).on("dragstart.drag", null);
-
- if (noclick) {
- selection.on("click.drag", noevent, true);
- setTimeout(function () {
- selection.on("click.drag", null);
- }, 0);
- }
-
- if ("onselectstart" in root) {
- selection.on("selectstart.drag", null);
- } else {
- root.style.MozUserSelect = root.__noselect;
- delete root.__noselect;
- }
- }
-
- var constant$1 = (function (x) {
- return function () {
- return x;
- };
- });
-
- // `Object.defineProperties` method
- // https://tc39.github.io/ecma262/#sec-object.defineproperties
- _export({ target: 'Object', stat: true, forced: !descriptors, sham: !descriptors }, {
- defineProperties: objectDefineProperties
- });
-
- function DragEvent(type, _ref) {
- var sourceEvent = _ref.sourceEvent,
- subject = _ref.subject,
- target = _ref.target,
- identifier = _ref.identifier,
- active = _ref.active,
- x = _ref.x,
- y = _ref.y,
- dx = _ref.dx,
- dy = _ref.dy,
- dispatch = _ref.dispatch;
- Object.defineProperties(this, {
- type: {
- value: type,
- enumerable: true,
- configurable: true
- },
- sourceEvent: {
- value: sourceEvent,
- enumerable: true,
- configurable: true
- },
- subject: {
- value: subject,
- enumerable: true,
- configurable: true
- },
- target: {
- value: target,
- enumerable: true,
- configurable: true
- },
- identifier: {
- value: identifier,
- enumerable: true,
- configurable: true
- },
- active: {
- value: active,
- enumerable: true,
- configurable: true
- },
- x: {
- value: x,
- enumerable: true,
- configurable: true
- },
- y: {
- value: y,
- enumerable: true,
- configurable: true
- },
- dx: {
- value: dx,
- enumerable: true,
- configurable: true
- },
- dy: {
- value: dy,
- enumerable: true,
- configurable: true
- },
- _: {
- value: dispatch
- }
- });
- }
-
- DragEvent.prototype.on = function () {
- var value = this._.on.apply(this._, arguments);
-
- return value === this._ ? this : value;
- };
-
- function defaultFilter(event) {
- return !event.ctrlKey && !event.button;
- }
-
- function defaultContainer() {
- return this.parentNode;
- }
-
- function defaultSubject(event, d) {
- return d == null ? {
- x: event.x,
- y: event.y
- } : d;
- }
-
- function defaultTouchable() {
- return navigator.maxTouchPoints || "ontouchstart" in this;
- }
-
- function d3_drag () {
- var filter = defaultFilter,
- container = defaultContainer,
- subject = defaultSubject,
- touchable = defaultTouchable,
- gestures = {},
- listeners = dispatch("start", "drag", "end"),
- active = 0,
- mousedownx,
- mousedowny,
- mousemoving,
- touchending,
- clickDistance2 = 0;
-
- function drag(selection) {
- selection.on("mousedown.drag", mousedowned).filter(touchable).on("touchstart.drag", touchstarted).on("touchmove.drag", touchmoved).on("touchend.drag touchcancel.drag", touchended).style("touch-action", "none").style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
- }
-
- function mousedowned(event, d) {
- if (touchending || !filter.call(this, event, d)) return;
- var gesture = beforestart(this, container.call(this, event, d), event, d, "mouse");
- if (!gesture) return;
- select(event.view).on("mousemove.drag", mousemoved, true).on("mouseup.drag", mouseupped, true);
- dragDisable(event.view);
- nopropagation(event);
- mousemoving = false;
- mousedownx = event.clientX;
- mousedowny = event.clientY;
- gesture("start", event);
- }
-
- function mousemoved(event) {
- noevent(event);
-
- if (!mousemoving) {
- var dx = event.clientX - mousedownx,
- dy = event.clientY - mousedowny;
- mousemoving = dx * dx + dy * dy > clickDistance2;
- }
-
- gestures.mouse("drag", event);
- }
-
- function mouseupped(event) {
- select(event.view).on("mousemove.drag mouseup.drag", null);
- yesdrag(event.view, mousemoving);
- noevent(event);
- gestures.mouse("end", event);
- }
-
- function touchstarted(event, d) {
- if (!filter.call(this, event, d)) return;
- var touches = event.changedTouches,
- c = container.call(this, event, d),
- n = touches.length,
- i,
- gesture;
-
- for (i = 0; i < n; ++i) {
- if (gesture = beforestart(this, c, event, d, touches[i].identifier, touches[i])) {
- nopropagation(event);
- gesture("start", event, touches[i]);
- }
- }
- }
-
- function touchmoved(event) {
- var touches = event.changedTouches,
- n = touches.length,
- i,
- gesture;
-
- for (i = 0; i < n; ++i) {
- if (gesture = gestures[touches[i].identifier]) {
- noevent(event);
- gesture("drag", event, touches[i]);
- }
- }
- }
-
- function touchended(event) {
- var touches = event.changedTouches,
- n = touches.length,
- i,
- gesture;
- if (touchending) clearTimeout(touchending);
- touchending = setTimeout(function () {
- touchending = null;
- }, 500); // Ghost clicks are delayed!
-
- for (i = 0; i < n; ++i) {
- if (gesture = gestures[touches[i].identifier]) {
- nopropagation(event);
- gesture("end", event, touches[i]);
- }
- }
- }
-
- function beforestart(that, container, event, d, identifier, touch) {
- var dispatch = listeners.copy(),
- p = pointer(touch || event, container),
- dx,
- dy,
- s;
- if ((s = subject.call(that, new DragEvent("beforestart", {
- sourceEvent: event,
- target: drag,
- identifier: identifier,
- active: active,
- x: p[0],
- y: p[1],
- dx: 0,
- dy: 0,
- dispatch: dispatch
- }), d)) == null) return;
- dx = s.x - p[0] || 0;
- dy = s.y - p[1] || 0;
- return function gesture(type, event, touch) {
- var p0 = p,
- n;
-
- switch (type) {
- case "start":
- gestures[identifier] = gesture, n = active++;
- break;
-
- case "end":
- delete gestures[identifier], --active;
- // nobreak
-
- case "drag":
- p = pointer(touch || event, container), n = active;
- break;
- }
-
- dispatch.call(type, that, new DragEvent(type, {
- sourceEvent: event,
- subject: s,
- target: drag,
- identifier: identifier,
- active: n,
- x: p[0] + dx,
- y: p[1] + dy,
- dx: p[0] - p0[0],
- dy: p[1] - p0[1],
- dispatch: dispatch
- }), d);
- };
- }
-
- drag.filter = function (_) {
- return arguments.length ? (filter = typeof _ === "function" ? _ : constant$1(!!_), drag) : filter;
- };
-
- drag.container = function (_) {
- return arguments.length ? (container = typeof _ === "function" ? _ : constant$1(_), drag) : container;
- };
-
- drag.subject = function (_) {
- return arguments.length ? (subject = typeof _ === "function" ? _ : constant$1(_), drag) : subject;
- };
-
- drag.touchable = function (_) {
- return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$1(!!_), drag) : touchable;
- };
-
- drag.on = function () {
- var value = listeners.on.apply(listeners, arguments);
- return value === listeners ? drag : value;
- };
-
- drag.clickDistance = function (_) {
- return arguments.length ? (clickDistance2 = (_ = +_) * _, drag) : Math.sqrt(clickDistance2);
- };
-
- return drag;
- }
-
- var defineProperty$9 = objectDefineProperty.f;
- var getOwnPropertyNames$1 = objectGetOwnPropertyNames.f;
-
-
-
-
-
- var setInternalState$8 = internalState.set;
-
-
-
- var MATCH$1 = wellKnownSymbol('match');
- var NativeRegExp = global_1.RegExp;
- var RegExpPrototype$1 = NativeRegExp.prototype;
- 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 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';
- })));
-
- // `RegExp` constructor
- // https://tc39.github.io/ecma262/#sec-regexp-constructor
- if (FORCED$b) {
- var RegExpWrapper = function RegExp(pattern, flags) {
- var thisIsRegExp = this instanceof RegExpWrapper;
- var patternIsRegExp = isRegexp(pattern);
- var flagsAreUndefined = flags === undefined;
- var sticky;
-
- if (!thisIsRegExp && patternIsRegExp && pattern.constructor === RegExpWrapper && flagsAreUndefined) {
- return pattern;
- }
-
- if (CORRECT_NEW) {
- if (patternIsRegExp && !flagsAreUndefined) pattern = pattern.source;
- } else if (pattern instanceof RegExpWrapper) {
- if (flagsAreUndefined) flags = regexpFlags.call(pattern);
- pattern = pattern.source;
- }
-
- if (UNSUPPORTED_Y$2) {
- 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 (UNSUPPORTED_Y$2 && sticky) setInternalState$8(result, { sticky: sticky });
-
- return result;
- };
- var proxy = function (key) {
- key in RegExpWrapper || defineProperty$9(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);
- }
-
- // https://tc39.github.io/ecma262/#sec-get-regexp-@@species
- setSpecies('RegExp');
-
- function define (constructor, factory, prototype) {
- constructor.prototype = factory.prototype = prototype;
- prototype.constructor = constructor;
- }
- function extend(parent, definition) {
- var prototype = Object.create(parent.prototype);
-
- for (var key in definition) {
- prototype[key] = definition[key];
- }
-
- return prototype;
- }
-
- function Color() {}
- var _darker = 0.7;
-
- var _brighter = 1 / _darker;
- var reI = "\\s*([+-]?\\d+)\\s*",
- reN = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\s*",
- reP = "\\s*([+-]?\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)%\\s*",
- reHex = /^#([0-9a-f]{3,8})$/,
- reRgbInteger = new RegExp("^rgb\\(" + [reI, reI, reI] + "\\)$"),
- reRgbPercent = new RegExp("^rgb\\(" + [reP, reP, reP] + "\\)$"),
- reRgbaInteger = new RegExp("^rgba\\(" + [reI, reI, reI, reN] + "\\)$"),
- reRgbaPercent = new RegExp("^rgba\\(" + [reP, reP, reP, reN] + "\\)$"),
- reHslPercent = new RegExp("^hsl\\(" + [reN, reP, reP] + "\\)$"),
- reHslaPercent = new RegExp("^hsla\\(" + [reN, reP, reP, reN] + "\\)$");
- var named = {
- aliceblue: 0xf0f8ff,
- antiquewhite: 0xfaebd7,
- aqua: 0x00ffff,
- aquamarine: 0x7fffd4,
- azure: 0xf0ffff,
- beige: 0xf5f5dc,
- bisque: 0xffe4c4,
- black: 0x000000,
- blanchedalmond: 0xffebcd,
- blue: 0x0000ff,
- blueviolet: 0x8a2be2,
- brown: 0xa52a2a,
- burlywood: 0xdeb887,
- cadetblue: 0x5f9ea0,
- chartreuse: 0x7fff00,
- chocolate: 0xd2691e,
- coral: 0xff7f50,
- cornflowerblue: 0x6495ed,
- cornsilk: 0xfff8dc,
- crimson: 0xdc143c,
- cyan: 0x00ffff,
- darkblue: 0x00008b,
- darkcyan: 0x008b8b,
- darkgoldenrod: 0xb8860b,
- darkgray: 0xa9a9a9,
- darkgreen: 0x006400,
- darkgrey: 0xa9a9a9,
- darkkhaki: 0xbdb76b,
- darkmagenta: 0x8b008b,
- darkolivegreen: 0x556b2f,
- darkorange: 0xff8c00,
- darkorchid: 0x9932cc,
- darkred: 0x8b0000,
- darksalmon: 0xe9967a,
- darkseagreen: 0x8fbc8f,
- darkslateblue: 0x483d8b,
- darkslategray: 0x2f4f4f,
- darkslategrey: 0x2f4f4f,
- darkturquoise: 0x00ced1,
- darkviolet: 0x9400d3,
- deeppink: 0xff1493,
- deepskyblue: 0x00bfff,
- dimgray: 0x696969,
- dimgrey: 0x696969,
- dodgerblue: 0x1e90ff,
- firebrick: 0xb22222,
- floralwhite: 0xfffaf0,
- forestgreen: 0x228b22,
- fuchsia: 0xff00ff,
- gainsboro: 0xdcdcdc,
- ghostwhite: 0xf8f8ff,
- gold: 0xffd700,
- goldenrod: 0xdaa520,
- gray: 0x808080,
- green: 0x008000,
- greenyellow: 0xadff2f,
- grey: 0x808080,
- honeydew: 0xf0fff0,
- hotpink: 0xff69b4,
- indianred: 0xcd5c5c,
- indigo: 0x4b0082,
- ivory: 0xfffff0,
- khaki: 0xf0e68c,
- lavender: 0xe6e6fa,
- lavenderblush: 0xfff0f5,
- lawngreen: 0x7cfc00,
- lemonchiffon: 0xfffacd,
- lightblue: 0xadd8e6,
- lightcoral: 0xf08080,
- lightcyan: 0xe0ffff,
- lightgoldenrodyellow: 0xfafad2,
- lightgray: 0xd3d3d3,
- lightgreen: 0x90ee90,
- lightgrey: 0xd3d3d3,
- lightpink: 0xffb6c1,
- lightsalmon: 0xffa07a,
- lightseagreen: 0x20b2aa,
- lightskyblue: 0x87cefa,
- lightslategray: 0x778899,
- lightslategrey: 0x778899,
- lightsteelblue: 0xb0c4de,
- lightyellow: 0xffffe0,
- lime: 0x00ff00,
- limegreen: 0x32cd32,
- linen: 0xfaf0e6,
- magenta: 0xff00ff,
- maroon: 0x800000,
- mediumaquamarine: 0x66cdaa,
- mediumblue: 0x0000cd,
- mediumorchid: 0xba55d3,
- mediumpurple: 0x9370db,
- mediumseagreen: 0x3cb371,
- mediumslateblue: 0x7b68ee,
- mediumspringgreen: 0x00fa9a,
- mediumturquoise: 0x48d1cc,
- mediumvioletred: 0xc71585,
- midnightblue: 0x191970,
- mintcream: 0xf5fffa,
- mistyrose: 0xffe4e1,
- moccasin: 0xffe4b5,
- navajowhite: 0xffdead,
- navy: 0x000080,
- oldlace: 0xfdf5e6,
- olive: 0x808000,
- olivedrab: 0x6b8e23,
- orange: 0xffa500,
- orangered: 0xff4500,
- orchid: 0xda70d6,
- palegoldenrod: 0xeee8aa,
- palegreen: 0x98fb98,
- paleturquoise: 0xafeeee,
- palevioletred: 0xdb7093,
- papayawhip: 0xffefd5,
- peachpuff: 0xffdab9,
- peru: 0xcd853f,
- pink: 0xffc0cb,
- plum: 0xdda0dd,
- powderblue: 0xb0e0e6,
- purple: 0x800080,
- rebeccapurple: 0x663399,
- red: 0xff0000,
- rosybrown: 0xbc8f8f,
- royalblue: 0x4169e1,
- saddlebrown: 0x8b4513,
- salmon: 0xfa8072,
- sandybrown: 0xf4a460,
- seagreen: 0x2e8b57,
- seashell: 0xfff5ee,
- sienna: 0xa0522d,
- silver: 0xc0c0c0,
- skyblue: 0x87ceeb,
- slateblue: 0x6a5acd,
- slategray: 0x708090,
- slategrey: 0x708090,
- snow: 0xfffafa,
- springgreen: 0x00ff7f,
- steelblue: 0x4682b4,
- tan: 0xd2b48c,
- teal: 0x008080,
- thistle: 0xd8bfd8,
- tomato: 0xff6347,
- turquoise: 0x40e0d0,
- violet: 0xee82ee,
- wheat: 0xf5deb3,
- white: 0xffffff,
- whitesmoke: 0xf5f5f5,
- yellow: 0xffff00,
- yellowgreen: 0x9acd32
- };
- define(Color, color, {
- copy: function copy(channels) {
- return Object.assign(new this.constructor(), this, channels);
- },
- displayable: function displayable() {
- return this.rgb().displayable();
- },
- hex: color_formatHex,
- // Deprecated! Use color.formatHex.
- formatHex: color_formatHex,
- formatHsl: color_formatHsl,
- formatRgb: color_formatRgb,
- toString: color_formatRgb
- });
-
- function color_formatHex() {
- return this.rgb().formatHex();
- }
-
- function color_formatHsl() {
- return hslConvert(this).formatHsl();
- }
-
- function color_formatRgb() {
- return this.rgb().formatRgb();
- }
-
- function color(format) {
- var m, l;
- format = (format + "").trim().toLowerCase();
- return (m = reHex.exec(format)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn(m) // #ff0000
- : l === 3 ? new Rgb(m >> 8 & 0xf | m >> 4 & 0xf0, m >> 4 & 0xf | m & 0xf0, (m & 0xf) << 4 | m & 0xf, 1) // #f00
- : l === 8 ? rgba(m >> 24 & 0xff, m >> 16 & 0xff, m >> 8 & 0xff, (m & 0xff) / 0xff) // #ff000000
- : l === 4 ? rgba(m >> 12 & 0xf | m >> 8 & 0xf0, m >> 8 & 0xf | m >> 4 & 0xf0, m >> 4 & 0xf | m & 0xf0, ((m & 0xf) << 4 | m & 0xf) / 0xff) // #f000
- : null // invalid hex
- ) : (m = reRgbInteger.exec(format)) ? new Rgb(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
- : (m = reRgbPercent.exec(format)) ? new Rgb(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
- : (m = reRgbaInteger.exec(format)) ? rgba(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
- : (m = reRgbaPercent.exec(format)) ? rgba(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
- : (m = reHslPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
- : (m = reHslaPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
- : named.hasOwnProperty(format) ? rgbn(named[format]) // eslint-disable-line no-prototype-builtins
- : format === "transparent" ? new Rgb(NaN, NaN, NaN, 0) : null;
- }
-
- function rgbn(n) {
- return new Rgb(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
- }
-
- function rgba(r, g, b, a) {
- if (a <= 0) r = g = b = NaN;
- return new Rgb(r, g, b, a);
- }
-
- function rgbConvert(o) {
- if (!(o instanceof Color)) o = color(o);
- if (!o) return new Rgb();
- o = o.rgb();
- return new Rgb(o.r, o.g, o.b, o.opacity);
- }
- function rgb(r, g, b, opacity) {
- return arguments.length === 1 ? rgbConvert(r) : new Rgb(r, g, b, opacity == null ? 1 : opacity);
- }
- function Rgb(r, g, b, opacity) {
- this.r = +r;
- this.g = +g;
- this.b = +b;
- this.opacity = +opacity;
- }
- define(Rgb, rgb, extend(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);
- },
- darker: function darker(k) {
- k = k == null ? _darker : Math.pow(_darker, k);
- return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
- },
- rgb: function rgb() {
- return this;
- },
- displayable: function displayable() {
- return -0.5 <= this.r && this.r < 255.5 && -0.5 <= this.g && this.g < 255.5 && -0.5 <= this.b && this.b < 255.5 && 0 <= this.opacity && this.opacity <= 1;
- },
- hex: rgb_formatHex,
- // Deprecated! Use color.formatHex.
- formatHex: rgb_formatHex,
- formatRgb: rgb_formatRgb,
- toString: rgb_formatRgb
- }));
-
- function rgb_formatHex() {
- return "#" + hex$2(this.r) + hex$2(this.g) + hex$2(this.b);
- }
-
- function rgb_formatRgb() {
- var a = this.opacity;
- a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
- 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) {
- value = Math.max(0, Math.min(255, Math.round(value) || 0));
- return (value < 16 ? "0" : "") + value.toString(16);
- }
-
- function hsla(h, s, l, a) {
- if (a <= 0) h = s = l = NaN;else if (l <= 0 || l >= 1) h = s = NaN;else if (s <= 0) h = NaN;
- return new Hsl(h, s, l, a);
- }
-
- function hslConvert(o) {
- if (o instanceof Hsl) return new Hsl(o.h, o.s, o.l, o.opacity);
- if (!(o instanceof Color)) o = color(o);
- if (!o) return new Hsl();
- if (o instanceof Hsl) return o;
- o = o.rgb();
- var r = o.r / 255,
- g = o.g / 255,
- b = o.b / 255,
- min = Math.min(r, g, b),
- max = Math.max(r, g, b),
- h = NaN,
- s = max - min,
- l = (max + min) / 2;
-
- if (s) {
- if (r === max) h = (g - b) / s + (g < b) * 6;else if (g === max) h = (b - r) / s + 2;else h = (r - g) / s + 4;
- s /= l < 0.5 ? max + min : 2 - max - min;
- h *= 60;
- } else {
- s = l > 0 && l < 1 ? 0 : h;
- }
-
- return new Hsl(h, s, l, o.opacity);
- }
- function hsl(h, s, l, opacity) {
- return arguments.length === 1 ? hslConvert(h) : new Hsl(h, s, l, opacity == null ? 1 : opacity);
- }
-
- function Hsl(h, s, l, opacity) {
- this.h = +h;
- this.s = +s;
- this.l = +l;
- this.opacity = +opacity;
- }
-
- define(Hsl, hsl, extend(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);
- },
- darker: function darker(k) {
- k = k == null ? _darker : Math.pow(_darker, k);
- return new Hsl(this.h, this.s, this.l * k, this.opacity);
- },
- rgb: function rgb() {
- var h = this.h % 360 + (this.h < 0) * 360,
- s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
- l = this.l,
- m2 = l + (l < 0.5 ? l : 1 - l) * s,
- m1 = 2 * l - m2;
- return new Rgb(hsl2rgb(h >= 240 ? h - 240 : h + 120, m1, m2), hsl2rgb(h, m1, m2), hsl2rgb(h < 120 ? h + 240 : h - 120, m1, m2), this.opacity);
- },
- displayable: function displayable() {
- return (0 <= this.s && this.s <= 1 || isNaN(this.s)) && 0 <= this.l && this.l <= 1 && 0 <= this.opacity && this.opacity <= 1;
- },
- formatHsl: function formatHsl() {
- var a = this.opacity;
- a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
- return (a === 1 ? "hsl(" : "hsla(") + (this.h || 0) + ", " + (this.s || 0) * 100 + "%, " + (this.l || 0) * 100 + "%" + (a === 1 ? ")" : ", " + a + ")");
- }
- }));
- /* From FvD 13.37, CSS Color Module Level 3 */
-
- function hsl2rgb(h, m1, m2) {
- 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) {
- return function () {
- return x;
- };
- });
-
- function linear(a, d) {
- return function (t) {
- return a + t * d;
- };
- }
-
- function exponential(a, b, y) {
- return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function (t) {
- return Math.pow(a + t * b, y);
- };
- }
- function gamma(y) {
- return (y = +y) === 1 ? nogamma : function (a, b) {
- return b - a ? exponential(a, b, y) : constant$2(isNaN(a) ? b : a);
- };
- }
- function nogamma(a, b) {
- var d = b - a;
- return d ? linear(a, d) : constant$2(isNaN(a) ? b : a);
- }
-
- var d3_interpolateRgb = (function rgbGamma(y) {
- var color = gamma(y);
-
- function rgb$1(start, end) {
- var r = color((start = rgb(start)).r, (end = rgb(end)).r),
- g = color(start.g, end.g),
- b = color(start.b, end.b),
- opacity = nogamma(start.opacity, end.opacity);
- return function (t) {
- start.r = r(t);
- start.g = g(t);
- start.b = b(t);
- start.opacity = opacity(t);
- return start + "";
- };
- }
-
- rgb$1.gamma = rgbGamma;
- return rgb$1;
- })(1);
-
- function numberArray (a, b) {
- if (!b) b = [];
- var n = a ? Math.min(b.length, a.length) : 0,
- c = b.slice(),
- i;
- return function (t) {
- for (i = 0; i < n; ++i) {
- c[i] = a[i] * (1 - t) + b[i] * t;
- }
-
- return c;
- };
- }
- function isNumberArray(x) {
- return ArrayBuffer.isView(x) && !(x instanceof DataView);
- }
-
- function genericArray(a, b) {
- var nb = b ? b.length : 0,
- na = a ? Math.min(nb, a.length) : 0,
- x = new Array(na),
- c = new Array(nb),
- i;
-
- for (i = 0; i < na; ++i) {
- x[i] = interpolate(a[i], b[i]);
- }
-
- for (; i < nb; ++i) {
- c[i] = b[i];
- }
-
- return function (t) {
- for (i = 0; i < na; ++i) {
- c[i] = x[i](t);
- }
-
- return c;
- };
- }
-
- function date (a, b) {
- var d = new Date();
- return a = +a, b = +b, function (t) {
- return d.setTime(a * (1 - t) + b * t), d;
- };
- }
-
- function d3_interpolateNumber (a, b) {
- return a = +a, b = +b, function (t) {
- return a * (1 - t) + b * t;
- };
- }
-
- function object (a, b) {
- var i = {},
- c = {},
- k;
- if (a === null || _typeof(a) !== "object") a = {};
- if (b === null || _typeof(b) !== "object") b = {};
-
- for (k in b) {
- if (k in a) {
- i[k] = interpolate(a[k], b[k]);
- } else {
- c[k] = b[k];
- }
- }
-
- return function (t) {
- for (k in i) {
- c[k] = i[k](t);
- }
-
- return c;
- };
- }
-
- var reA = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g,
- reB = new RegExp(reA.source, "g");
-
- function zero(b) {
- return function () {
- return b;
- };
- }
-
- function one(b) {
- return function (t) {
- return b(t) + "";
- };
- }
-
- function interpolateString (a, b) {
- var bi = reA.lastIndex = reB.lastIndex = 0,
- // scan index for next number in b
- am,
- // current match in a
- bm,
- // current match in b
- bs,
- // string preceding current number in b, if any
- i = -1,
- // index in s
- s = [],
- // string constants and placeholders
- q = []; // number interpolators
- // Coerce inputs to strings.
-
- a = a + "", b = b + ""; // Interpolate pairs of numbers in a & b.
-
- while ((am = reA.exec(a)) && (bm = reB.exec(b))) {
- if ((bs = bm.index) > bi) {
- // a string precedes the next number in b
- bs = b.slice(bi, bs);
- if (s[i]) s[i] += bs; // coalesce with previous string
- else s[++i] = bs;
- }
-
- if ((am = am[0]) === (bm = bm[0])) {
- // numbers in a & b match
- if (s[i]) s[i] += bm; // coalesce with previous string
- else s[++i] = bm;
- } else {
- // interpolate non-matching numbers
- s[++i] = null;
- q.push({
- i: i,
- x: d3_interpolateNumber(am, bm)
- });
- }
-
- bi = reB.lastIndex;
- } // Add remains of b.
-
-
- if (bi < b.length) {
- bs = b.slice(bi);
- if (s[i]) s[i] += bs; // coalesce with previous string
- else s[++i] = bs;
- } // Special optimization for only a single match.
- // Otherwise, interpolate each of the numbers and rejoin the string.
-
-
- return s.length < 2 ? q[0] ? one(q[0].x) : zero(b) : (b = q.length, function (t) {
- for (var i = 0, o; i < b; ++i) {
- s[(o = q[i]).i] = o.x(t);
- }
-
- return s.join("");
- });
- }
-
- function interpolate (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);
- }
-
- function interpolateRound (a, b) {
- return a = +a, b = +b, function (t) {
- return Math.round(a * (1 - t) + b * t);
- };
- }
-
- var degrees$1 = 180 / Math.PI;
- var identity$1 = {
- translateX: 0,
- translateY: 0,
- rotate: 0,
- skewX: 0,
- scaleX: 1,
- scaleY: 1
- };
- function decompose (a, b, c, d, e, f) {
- var scaleX, scaleY, skewX;
- if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;
- if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;
- if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;
- if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
- return {
- translateX: e,
- translateY: f,
- rotate: Math.atan2(b, a) * degrees$1,
- skewX: Math.atan(skewX) * degrees$1,
- scaleX: scaleX,
- scaleY: scaleY
- };
- }
-
- var svgNode;
- /* eslint-disable no-undef */
-
- 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);
- }
- function parseSvg(value) {
- if (value == null) return identity$1;
- 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;
- value = value.matrix;
- return decompose(value.a, value.b, value.c, value.d, value.e, value.f);
- }
-
- function interpolateTransform(parse, pxComma, pxParen, degParen) {
- function pop(s) {
- return s.length ? s.pop() + " " : "";
- }
-
- function translate(xa, ya, xb, yb, s, q) {
- if (xa !== xb || ya !== yb) {
- var i = s.push("translate(", null, pxComma, null, pxParen);
- q.push({
- i: i - 4,
- x: d3_interpolateNumber(xa, xb)
- }, {
- i: i - 2,
- x: d3_interpolateNumber(ya, yb)
- });
- } else if (xb || yb) {
- s.push("translate(" + xb + pxComma + yb + pxParen);
- }
- }
-
- function rotate(a, b, s, q) {
- if (a !== b) {
- if (a - b > 180) b += 360;else if (b - a > 180) a += 360; // shortest path
-
- q.push({
- i: s.push(pop(s) + "rotate(", null, degParen) - 2,
- x: d3_interpolateNumber(a, b)
- });
- } else if (b) {
- s.push(pop(s) + "rotate(" + b + degParen);
- }
- }
-
- function skewX(a, b, s, q) {
- if (a !== b) {
- q.push({
- i: s.push(pop(s) + "skewX(", null, degParen) - 2,
- x: d3_interpolateNumber(a, b)
- });
- } else if (b) {
- s.push(pop(s) + "skewX(" + b + degParen);
- }
- }
-
- function scale(xa, ya, xb, yb, s, q) {
- if (xa !== xb || ya !== yb) {
- var i = s.push(pop(s) + "scale(", null, ",", null, ")");
- q.push({
- i: i - 4,
- x: d3_interpolateNumber(xa, xb)
- }, {
- i: i - 2,
- x: d3_interpolateNumber(ya, yb)
- });
- } else if (xb !== 1 || yb !== 1) {
- s.push(pop(s) + "scale(" + xb + "," + yb + ")");
- }
- }
-
- return function (a, b) {
- var s = [],
- // string constants and placeholders
- q = []; // number interpolators
-
- a = parse(a), b = parse(b);
- translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
- rotate(a.rotate, b.rotate, s, q);
- skewX(a.skewX, b.skewX, s, q);
- scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
- a = b = null; // gc
-
- return function (t) {
- var i = -1,
- n = q.length,
- o;
-
- while (++i < n) {
- s[(o = q[i]).i] = o.x(t);
- }
-
- return s.join("");
- };
- };
- }
-
- var interpolateTransformCss = interpolateTransform(parseCss, "px, ", "px)", "deg)");
- var interpolateTransformSvg = interpolateTransform(parseSvg, ", ", ")", ")");
-
- var epsilon2$1 = 1e-12;
-
- function cosh(x) {
- return ((x = Math.exp(x)) + 1 / x) / 2;
- }
-
- function sinh(x) {
- return ((x = Math.exp(x)) - 1 / x) / 2;
- }
-
- function tanh(x) {
- return ((x = Math.exp(2 * x)) - 1) / (x + 1);
- }
-
- var interpolateZoom = (function zoomRho(rho, rho2, rho4) {
- // p0 = [ux0, uy0, w0]
- // p1 = [ux1, uy1, w1]
- function zoom(p0, p1) {
- var ux0 = p0[0],
- uy0 = p0[1],
- w0 = p0[2],
- ux1 = p1[0],
- uy1 = p1[1],
- w1 = p1[2],
- dx = ux1 - ux0,
- dy = uy1 - uy0,
- d2 = dx * dx + dy * dy,
- i,
- S; // Special case for u0 ≅ u1.
-
- if (d2 < epsilon2$1) {
- S = Math.log(w1 / w0) / rho;
-
- i = function i(t) {
- return [ux0 + t * dx, uy0 + t * dy, w0 * Math.exp(rho * t * S)];
- };
- } // General case.
- else {
- var d1 = Math.sqrt(d2),
- b0 = (w1 * w1 - w0 * w0 + rho4 * d2) / (2 * w0 * rho2 * d1),
- b1 = (w1 * w1 - w0 * w0 - rho4 * d2) / (2 * w1 * rho2 * d1),
- r0 = Math.log(Math.sqrt(b0 * b0 + 1) - b0),
- r1 = Math.log(Math.sqrt(b1 * b1 + 1) - b1);
- S = (r1 - r0) / rho;
-
- i = function i(t) {
- var s = t * S,
- coshr0 = cosh(r0),
- u = w0 / (rho2 * d1) * (coshr0 * tanh(rho * s + r0) - sinh(r0));
- return [ux0 + u * dx, uy0 + u * dy, w0 * coshr0 / cosh(rho * s + r0)];
- };
- }
-
- i.duration = S * 1000 * rho / Math.SQRT2;
- return i;
- }
-
- zoom.rho = function (_) {
- var _1 = Math.max(1e-3, +_),
- _2 = _1 * _1,
- _4 = _2 * _2;
-
- return zoomRho(_1, _2, _4);
- };
-
- return zoom;
- })(Math.SQRT2, 2, 4);
-
- function d3_quantize (interpolator, n) {
- var samples = new Array(n);
-
- for (var i = 0; i < n; ++i) {
- samples[i] = interpolator(i / (n - 1));
- }
-
- return samples;
- }
-
- // `Function.prototype.bind` method
- // https://tc39.github.io/ecma262/#sec-function.prototype.bind
- _export({ target: 'Function', proto: true }, {
- bind: functionBind
- });
-
- var frame = 0,
- // is an animation frame pending?
- timeout = 0,
- // is a timeout pending?
- interval = 0,
- // are any timers active?
- pokeDelay = 1000,
- // how frequently we check for clock skew
- taskHead,
- taskTail,
- clockLast = 0,
- clockNow = 0,
- clockSkew = 0,
- clock = (typeof performance === "undefined" ? "undefined" : _typeof(performance)) === "object" && performance.now ? performance : Date,
- setFrame = (typeof window === "undefined" ? "undefined" : _typeof(window)) === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function (f) {
- setTimeout(f, 17);
- };
- function now() {
- return clockNow || (setFrame(clearNow), clockNow = clock.now() + clockSkew);
- }
-
- function clearNow() {
- clockNow = 0;
- }
-
- function Timer() {
- this._call = this._time = this._next = null;
- }
- Timer.prototype = timer.prototype = {
- 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);
-
- if (!this._next && taskTail !== this) {
- if (taskTail) taskTail._next = this;else taskHead = this;
- taskTail = this;
- }
-
- this._call = callback;
- this._time = time;
- sleep();
- },
- stop: function stop() {
- if (this._call) {
- this._call = null;
- this._time = Infinity;
- sleep();
- }
- }
- };
- function timer(callback, delay, time) {
- var t = new Timer();
- t.restart(callback, delay, time);
- return t;
- }
- function timerFlush() {
- now(); // Get the current time, if not already set.
-
- ++frame; // Pretend we’ve set an alarm, if we haven’t already.
-
- var t = taskHead,
- e;
-
- while (t) {
- if ((e = clockNow - t._time) >= 0) t._call.call(null, e);
- t = t._next;
- }
-
- --frame;
- }
-
- function wake() {
- clockNow = (clockLast = clock.now()) + clockSkew;
- frame = timeout = 0;
-
- try {
- timerFlush();
- } finally {
- frame = 0;
- nap();
- clockNow = 0;
- }
- }
-
- function poke() {
- var now = clock.now(),
- delay = now - clockLast;
- if (delay > pokeDelay) clockSkew -= delay, clockLast = now;
- }
-
- function nap() {
- var t0,
- t1 = taskHead,
- t2,
- time = Infinity;
-
- while (t1) {
- if (t1._call) {
- if (time > t1._time) time = t1._time;
- t0 = t1, t1 = t1._next;
- } else {
- t2 = t1._next, t1._next = null;
- t1 = t0 ? t0._next = t2 : taskHead = t2;
- }
- }
-
- taskTail = t0;
- sleep(time);
- }
-
- function sleep(time) {
- if (frame) return; // Soonest alarm already set, or will be.
-
- if (timeout) timeout = clearTimeout(timeout);
- var delay = time - clockNow; // Strictly less than if we recomputed clockNow.
-
- if (delay > 24) {
- if (time < Infinity) timeout = setTimeout(wake, time - clock.now() - clockSkew);
- if (interval) interval = clearInterval(interval);
- } else {
- if (!interval) clockLast = clock.now(), interval = setInterval(poke, pokeDelay);
- frame = 1, setFrame(wake);
- }
- }
-
- function d3_timeout (callback, delay, time) {
- var t = new Timer();
- delay = delay == null ? 0 : +delay;
- t.restart(function (elapsed) {
- t.stop();
- callback(elapsed + delay);
- }, delay, time);
- return t;
- }
-
- var emptyOn = dispatch("start", "end", "cancel", "interrupt");
- var emptyTween = [];
- var CREATED = 0;
- var SCHEDULED = 1;
- var STARTING = 2;
- var STARTED = 3;
- var RUNNING = 4;
- var ENDING = 5;
- var ENDED = 6;
- 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, {
- name: name,
- index: index,
- // For context during callback.
- group: group,
- // For context during callback.
- on: emptyOn,
- tween: emptyTween,
- time: timing.time,
- delay: timing.delay,
- duration: timing.duration,
- ease: timing.ease,
- timer: null,
- state: CREATED
- });
- }
- function init(node, id) {
- var schedule = get$4(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);
- if (schedule.state > STARTED) throw new Error("too late; already running");
- return schedule;
- }
- function get$4(node, id) {
- var schedule = node.__transition;
- if (!schedule || !(schedule = schedule[id])) throw new Error("transition not found");
- return schedule;
- }
-
- function create(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!
-
- schedules[id] = self;
- self.timer = timer(schedule, 0, self.time);
-
- function schedule(elapsed) {
- self.state = SCHEDULED;
- self.timer.restart(start, self.delay, self.time); // If the elapsed delay is less than our first sleep, start immediately.
-
- if (self.delay <= elapsed) start(elapsed - self.delay);
- }
-
- function start(elapsed) {
- var i, j, n, o; // If the state is not SCHEDULED, then we previously errored on start.
-
- if (self.state !== SCHEDULED) return stop();
-
- for (i in schedules) {
- o = schedules[i];
- if (o.name !== self.name) continue; // While this element already has a starting transition during this frame,
- // defer starting an interrupting transition until that transition has a
- // chance to tick (and possibly end); see d3/d3-transition#54!
-
- if (o.state === STARTED) return d3_timeout(start); // Interrupt the active transition, if any.
-
- if (o.state === RUNNING) {
- o.state = ENDED;
- o.timer.stop();
- o.on.call("interrupt", node, node.__data__, o.index, o.group);
- delete schedules[i];
- } // Cancel any pre-empted transitions.
- else if (+i < id) {
- o.state = ENDED;
- o.timer.stop();
- o.on.call("cancel", node, node.__data__, o.index, o.group);
- delete schedules[i];
- }
- } // Defer the first tick to end of the current frame; see d3/d3#1576.
- // Note the transition may be canceled after start and before the first tick!
- // Note this must be scheduled before the start event; see d3/d3-transition#16!
- // Assuming this is successful, subsequent callbacks go straight to tick.
-
-
- d3_timeout(function () {
- if (self.state === STARTED) {
- self.state = RUNNING;
- self.timer.restart(tick, self.delay, self.time);
- tick(elapsed);
- }
- }); // Dispatch the start event.
- // Note this must be done before the tween are initialized.
-
- self.state = STARTING;
- self.on.call("start", node, node.__data__, self.index, self.group);
- if (self.state !== STARTING) return; // interrupted
-
- self.state = STARTED; // Initialize the tween, deleting null tween.
-
- tween = new Array(n = self.tween.length);
-
- for (i = 0, j = -1; i < n; ++i) {
- if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {
- tween[++j] = o;
- }
- }
-
- tween.length = j + 1;
- }
-
- function tick(elapsed) {
- var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.timer.restart(stop), self.state = ENDING, 1),
- i = -1,
- n = tween.length;
-
- while (++i < n) {
- tween[i].call(node, t);
- } // Dispatch the end event.
-
-
- if (self.state === ENDING) {
- self.on.call("end", node, node.__data__, self.index, self.group);
- stop();
- }
- }
-
- function stop() {
- self.state = ENDED;
- self.timer.stop();
- delete schedules[id];
-
- for (var i in schedules) {
- return;
- } // eslint-disable-line no-unused-vars
-
-
- delete node.__transition;
- }
- }
-
- function interrupt (node, name) {
- var schedules = node.__transition,
- schedule,
- active,
- empty = true,
- i;
- if (!schedules) return;
- name = name == null ? null : name + "";
-
- for (i in schedules) {
- if ((schedule = schedules[i]).name !== name) {
- empty = false;
- continue;
- }
-
- active = schedule.state > STARTING && schedule.state < ENDING;
- schedule.state = ENDED;
- schedule.timer.stop();
- schedule.on.call(active ? "interrupt" : "cancel", node, node.__data__, schedule.index, schedule.group);
- delete schedules[i];
- }
-
- if (empty) delete node.__transition;
- }
-
- function selection_interrupt (name) {
- return this.each(function () {
- interrupt(this, name);
- });
- }
-
- function tweenRemove(id, name) {
- var tween0, tween1;
- return function () {
- var schedule = set$4(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.
-
- if (tween !== tween0) {
- tween1 = tween0 = tween;
-
- for (var i = 0, n = tween1.length; i < n; ++i) {
- if (tween1[i].name === name) {
- tween1 = tween1.slice();
- tween1.splice(i, 1);
- break;
- }
- }
- }
-
- schedule.tween = tween1;
- };
- }
-
- function tweenFunction(id, name, value) {
- var tween0, tween1;
- if (typeof value !== "function") throw new Error();
- return function () {
- var schedule = set$4(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.
-
- if (tween !== tween0) {
- tween1 = (tween0 = tween).slice();
-
- for (var t = {
- name: name,
- value: value
- }, i = 0, n = tween1.length; i < n; ++i) {
- if (tween1[i].name === name) {
- tween1[i] = t;
- break;
- }
- }
-
- if (i === n) tween1.push(t);
- }
-
- schedule.tween = tween1;
- };
- }
-
- function transition_tween (name, value) {
- var id = this._id;
- name += "";
-
- if (arguments.length < 2) {
- var tween = get$4(this.node(), id).tween;
-
- for (var i = 0, n = tween.length, t; i < n; ++i) {
- if ((t = tween[i]).name === name) {
- return t.value;
- }
- }
-
- return null;
- }
-
- return this.each((value == null ? tweenRemove : tweenFunction)(id, name, value));
- }
- function tweenValue(transition, name, value) {
- var id = transition._id;
- transition.each(function () {
- var schedule = set$4(this, id);
- (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
- });
- return function (node) {
- return get$4(node, id).value[name];
- };
- }
-
- function interpolate$1 (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) {
- return function () {
- this.removeAttribute(name);
- };
- }
-
- function attrRemoveNS$1(fullname) {
- return function () {
- this.removeAttributeNS(fullname.space, fullname.local);
- };
- }
-
- function attrConstant$1(name, interpolate, value1) {
- var string00,
- string1 = value1 + "",
- interpolate0;
- return function () {
- var string0 = this.getAttribute(name);
- return string0 === string1 ? null : string0 === string00 ? interpolate0 : interpolate0 = interpolate(string00 = string0, value1);
- };
- }
-
- function attrConstantNS$1(fullname, interpolate, value1) {
- var string00,
- string1 = value1 + "",
- interpolate0;
- return function () {
- var string0 = this.getAttributeNS(fullname.space, fullname.local);
- return string0 === string1 ? null : string0 === string00 ? interpolate0 : interpolate0 = interpolate(string00 = string0, value1);
- };
- }
-
- function attrFunction$1(name, interpolate, value) {
- var string00, string10, interpolate0;
- return function () {
- var string0,
- value1 = value(this),
- string1;
- if (value1 == null) return void this.removeAttribute(name);
- string0 = this.getAttribute(name);
- string1 = value1 + "";
- return string0 === string1 ? null : string0 === string00 && string1 === string10 ? interpolate0 : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
- };
- }
-
- function attrFunctionNS$1(fullname, interpolate, value) {
- var string00, string10, interpolate0;
- return function () {
- var string0,
- value1 = value(this),
- string1;
- if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);
- string0 = this.getAttributeNS(fullname.space, fullname.local);
- string1 = value1 + "";
- return string0 === string1 ? null : string0 === string00 && string1 === string10 ? interpolate0 : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
- };
- }
-
- 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));
- }
-
- function attrInterpolate(name, i) {
- return function (t) {
- this.setAttribute(name, i.call(this, t));
- };
- }
-
- function attrInterpolateNS(fullname, i) {
- return function (t) {
- this.setAttributeNS(fullname.space, fullname.local, i.call(this, t));
- };
- }
-
- function attrTweenNS(fullname, value) {
- var t0, i0;
-
- function tween() {
- var i = value.apply(this, arguments);
- if (i !== i0) t0 = (i0 = i) && attrInterpolateNS(fullname, i);
- return t0;
- }
-
- tween._value = value;
- return tween;
- }
-
- function attrTween(name, value) {
- var t0, i0;
-
- function tween() {
- var i = value.apply(this, arguments);
- if (i !== i0) t0 = (i0 = i) && attrInterpolate(name, i);
- return t0;
- }
-
- tween._value = value;
- return tween;
- }
-
- function transition_attrTween (name, value) {
- var key = "attr." + name;
- if (arguments.length < 2) return (key = this.tween(key)) && key._value;
- if (value == null) return this.tween(key, null);
- if (typeof value !== "function") throw new Error();
- var fullname = namespace(name);
- return this.tween(key, (fullname.local ? attrTweenNS : attrTween)(fullname, value));
- }
-
- function delayFunction(id, value) {
- return function () {
- init(this, id).delay = +value.apply(this, arguments);
- };
- }
-
- function delayConstant(id, value) {
- return value = +value, function () {
- init(this, id).delay = value;
- };
- }
-
- 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;
- }
-
- function durationFunction(id, value) {
- return function () {
- set$4(this, id).duration = +value.apply(this, arguments);
- };
- }
-
- function durationConstant(id, value) {
- return value = +value, function () {
- set$4(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;
- }
-
- function easeConstant(id, value) {
- if (typeof value !== "function") throw new Error();
- return function () {
- set$4(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;
- }
-
- 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;
- };
- }
-
- function transition_easeVarying (value) {
- if (typeof value !== "function") throw new Error();
- return this.each(easeVarying(this._id, value));
- }
-
- function transition_filter (match) {
- if (typeof match !== "function") match = matcher(match);
-
- for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
- for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
- if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
- subgroup.push(node);
- }
- }
- }
-
- return new Transition(subgroups, this._parents, this._name, this._id);
- }
-
- function transition_merge (transition) {
- if (transition._id !== this._id) throw new Error();
-
- for (var groups0 = this._groups, groups1 = transition._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) {
- if (node = group0[i] || group1[i]) {
- merge[i] = node;
- }
- }
- }
-
- for (; j < m0; ++j) {
- merges[j] = groups0[j];
- }
-
- return new Transition(merges, this._parents, this._name, this._id);
- }
-
- function start(name) {
- return (name + "").trim().split(/^|\s+/).every(function (t) {
- var i = t.indexOf(".");
- if (i >= 0) t = t.slice(0, i);
- return !t || t === "start";
- });
- }
-
- function onFunction(id, name, listener) {
- var on0,
- on1,
- sit = start(name) ? init : set$4;
- return function () {
- var schedule = sit(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.
-
- if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);
- schedule.on = on1;
- };
- }
-
- 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));
- }
-
- function removeFunction(id) {
- return function () {
- var parent = this.parentNode;
-
- for (var i in this.__transition) {
- if (+i !== id) return;
- }
-
- if (parent) parent.removeChild(this);
- };
- }
-
- function transition_remove () {
- return this.on("end.remove", removeFunction(this._id));
- }
-
- function transition_select (select) {
- var name = this._name,
- id = this._id;
- if (typeof select !== "function") select = selector(select);
-
- for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
- for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
- 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));
- }
- }
- }
-
- return new Transition(subgroups, this._parents, name, id);
- }
-
- function transition_selectAll (select) {
- var name = this._name,
- id = this._id;
- if (typeof select !== "function") select = selectorAll(select);
-
- 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) {
- if (child = children[k]) {
- schedule(child, name, id, k, children, inherit);
- }
- }
-
- subgroups.push(children);
- parents.push(node);
- }
- }
- }
-
- return new Transition(subgroups, parents, name, id);
- }
-
- var Selection$1 = selection.prototype.constructor;
- function transition_selection () {
- return new Selection$1(this._groups, this._parents);
- }
-
- function styleNull(name, interpolate) {
- var string00, string10, interpolate0;
- return function () {
- var string0 = styleValue(this, name),
- string1 = (this.style.removeProperty(name), styleValue(this, name));
- return string0 === string1 ? null : string0 === string00 && string1 === string10 ? interpolate0 : interpolate0 = interpolate(string00 = string0, string10 = string1);
- };
- }
-
- function styleRemove$1(name) {
- return function () {
- this.style.removeProperty(name);
- };
- }
-
- function styleConstant$1(name, interpolate, value1) {
- var string00,
- string1 = value1 + "",
- interpolate0;
- return function () {
- var string0 = styleValue(this, name);
- return string0 === string1 ? null : string0 === string00 ? interpolate0 : interpolate0 = interpolate(string00 = string0, value1);
- };
- }
-
- function styleFunction$1(name, interpolate, value) {
- var string00, string10, interpolate0;
- return function () {
- var string0 = styleValue(this, name),
- value1 = value(this),
- string1 = value1 + "";
- if (value1 == null) string1 = value1 = (this.style.removeProperty(name), styleValue(this, name));
- return string0 === string1 ? null : string0 === string00 && string1 === string10 ? interpolate0 : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
- };
- }
-
- function styleMaybeRemove(id, name) {
- var on0,
- on1,
- listener0,
- key = "style." + name,
- event = "end." + key,
- remove;
- return function () {
- var schedule = set$4(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,
- // just assign the updated shared dispatch and we’re done!
- // Otherwise, copy-on-write.
-
- if (on !== on0 || listener0 !== listener) (on1 = (on0 = on).copy()).on(event, listener0 = listener);
- schedule.on = on1;
- };
- }
-
- 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);
- }
-
- function styleInterpolate(name, i, priority) {
- return function (t) {
- this.style.setProperty(name, i.call(this, t), priority);
- };
- }
-
- function styleTween(name, value, priority) {
- var t, i0;
-
- function tween() {
- var i = value.apply(this, arguments);
- if (i !== i0) t = (i0 = i) && styleInterpolate(name, i, priority);
- return t;
- }
-
- tween._value = value;
- return tween;
- }
-
- function transition_styleTween (name, value, priority) {
- var key = "style." + (name += "");
- if (arguments.length < 2) return (key = this.tween(key)) && key._value;
- if (value == null) return this.tween(key, null);
- if (typeof value !== "function") throw new Error();
- return this.tween(key, styleTween(name, value, priority == null ? "" : priority));
- }
-
- function textConstant$1(value) {
- return function () {
- this.textContent = value;
- };
- }
-
- function textFunction$1(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 + ""));
- }
-
- function textInterpolate(i) {
- return function (t) {
- this.textContent = i.call(this, t);
- };
- }
-
- function textTween(value) {
- var t0, i0;
-
- function tween() {
- var i = value.apply(this, arguments);
- if (i !== i0) t0 = (i0 = i) && textInterpolate(i);
- return t0;
- }
-
- tween._value = value;
- return tween;
- }
-
- function transition_textTween (value) {
- var key = "text";
- if (arguments.length < 1) return (key = this.tween(key)) && key._value;
- if (value == null) return this.tween(key, null);
- if (typeof value !== "function") throw new Error();
- return this.tween(key, textTween(value));
- }
-
- function transition_transition () {
- var name = this._name,
- id0 = this._id,
- id1 = newId();
-
- 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);
- schedule(node, name, id1, i, group, {
- time: inherit.time + inherit.delay + inherit.duration,
- delay: 0,
- duration: inherit.duration,
- ease: inherit.ease
- });
- }
- }
- }
-
- return new Transition(groups, this._parents, name, id1);
- }
-
- function transition_end () {
- var on0,
- on1,
- that = this,
- id = that._id,
- size = that.size();
- return new Promise(function (resolve, reject) {
- var cancel = {
- value: reject
- },
- end = {
- value: function value() {
- if (--size === 0) resolve();
- }
- };
- that.each(function () {
- var schedule = set$4(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.
-
- if (on !== on0) {
- on1 = (on0 = on).copy();
-
- on1._.cancel.push(cancel);
-
- on1._.interrupt.push(cancel);
-
- on1._.end.push(end);
- }
-
- schedule.on = on1;
- }); // The selection was empty, resolve end immediately
-
- if (size === 0) resolve();
- });
- }
-
- var id$1 = 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;
- }
- var selection_prototype = selection.prototype;
- Transition.prototype = transition.prototype = _defineProperty({
- constructor: Transition,
- select: transition_select,
- selectAll: transition_selectAll,
- filter: transition_filter,
- merge: transition_merge,
- selection: transition_selection,
- transition: transition_transition,
- call: selection_prototype.call,
- nodes: selection_prototype.nodes,
- node: selection_prototype.node,
- size: selection_prototype.size,
- empty: selection_prototype.empty,
- each: selection_prototype.each,
- on: transition_on,
- attr: transition_attr,
- attrTween: transition_attrTween,
- style: transition_style,
- styleTween: transition_styleTween,
- text: transition_text,
- textTween: transition_textTween,
- remove: transition_remove,
- tween: transition_tween,
- delay: transition_delay,
- duration: transition_duration,
- ease: transition_ease,
- easeVarying: transition_easeVarying,
- end: transition_end
- }, Symbol.iterator, selection_prototype[Symbol.iterator]);
-
- var linear$1 = function linear(t) {
- return +t;
- };
-
- function cubicInOut(t) {
- return ((t *= 2) <= 1 ? t * t * t : (t -= 2) * t * t + 2) / 2;
- }
-
- var defaultTiming = {
- time: null,
- // Set on use.
- delay: 0,
- duration: 250,
- ease: cubicInOut
- };
-
- function inherit(node, id) {
- var timing;
-
- while (!(timing = node.__transition) || !(timing = timing[id])) {
- if (!(node = node.parentNode)) {
- throw new Error("transition ".concat(id, " not found"));
- }
- }
-
- return timing;
- }
-
- function selection_transition (name) {
- var id, timing;
-
- if (name instanceof Transition) {
- id = name._id, name = name._name;
- } else {
- id = newId(), (timing = defaultTiming).time = now(), name = name == null ? null : name + "";
- }
-
- 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]) {
- schedule(node, name, id, i, group, timing || inherit(node, id));
- }
- }
- }
-
- return new Transition(groups, this._parents, name, id);
- }
-
- selection.prototype.interrupt = selection_interrupt;
- selection.prototype.transition = selection_transition;
-
- var constant$3 = (function (x) {
- return function () {
- return x;
- };
- });
-
- function ZoomEvent(type, _ref) {
- var sourceEvent = _ref.sourceEvent,
- target = _ref.target,
- transform = _ref.transform,
- dispatch = _ref.dispatch;
- Object.defineProperties(this, {
- type: {
- value: type,
- enumerable: true,
- configurable: true
- },
- sourceEvent: {
- value: sourceEvent,
- enumerable: true,
- configurable: true
- },
- target: {
- value: target,
- enumerable: true,
- configurable: true
- },
- transform: {
- value: transform,
- enumerable: true,
- configurable: true
- },
- _: {
- value: dispatch
- }
- });
- }
-
- function Transform(k, x, y) {
- this.k = k;
- this.x = x;
- this.y = y;
- }
- Transform.prototype = {
- constructor: Transform,
- scale: function scale(k) {
- return k === 1 ? this : new Transform(this.k * k, this.x, this.y);
- },
- translate: function translate(x, y) {
- return x === 0 & y === 0 ? this : new Transform(this.k, this.x + this.k * x, this.y + this.k * y);
- },
- apply: function apply(point) {
- return [point[0] * this.k + this.x, point[1] * this.k + this.y];
- },
- applyX: function applyX(x) {
- return x * this.k + this.x;
- },
- applyY: function applyY(y) {
- return y * this.k + this.y;
- },
- invert: function invert(location) {
- return [(location[0] - this.x) / this.k, (location[1] - this.y) / this.k];
- },
- invertX: function invertX(x) {
- return (x - this.x) / this.k;
- },
- invertY: function invertY(y) {
- return (y - this.y) / this.k;
- },
- rescaleX: function rescaleX(x) {
- return x.copy().domain(x.range().map(this.invertX, this).map(x.invert, x));
- },
- rescaleY: function rescaleY(y) {
- return y.copy().domain(y.range().map(this.invertY, this).map(y.invert, y));
- },
- toString: function toString() {
- return "translate(" + this.x + "," + this.y + ") scale(" + this.k + ")";
- }
- };
- var identity$2 = new Transform(1, 0, 0);
-
- function nopropagation$1(event) {
- event.stopImmediatePropagation();
- }
- function noevent$1 (event) {
- event.preventDefault();
- event.stopImmediatePropagation();
- }
-
- // except for pinch-to-zoom, which is sent as a wheel+ctrlKey event
-
- function defaultFilter$1(event) {
- return (!event.ctrlKey || event.type === 'wheel') && !event.button;
- }
-
- function defaultExtent() {
- var e = this;
-
- if (e instanceof SVGElement) {
- e = e.ownerSVGElement || e;
-
- if (e.hasAttribute("viewBox")) {
- e = e.viewBox.baseVal;
- return [[e.x, e.y], [e.x + e.width, e.y + e.height]];
- }
-
- return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];
- }
-
- return [[0, 0], [e.clientWidth, e.clientHeight]];
- }
-
- function defaultTransform() {
- return this.__zoom || identity$2;
- }
-
- function defaultWheelDelta(event) {
- return -event.deltaY * (event.deltaMode === 1 ? 0.05 : event.deltaMode ? 1 : 0.002) * (event.ctrlKey ? 10 : 1);
- }
-
- function defaultTouchable$1() {
- return navigator.maxTouchPoints || "ontouchstart" in this;
- }
-
- 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));
- }
-
- function d3_zoom () {
- var filter = defaultFilter$1,
- extent = defaultExtent,
- constrain = defaultConstrain,
- wheelDelta = defaultWheelDelta,
- touchable = defaultTouchable$1,
- scaleExtent = [0, Infinity],
- translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]],
- duration = 250,
- interpolate = interpolateZoom,
- listeners = dispatch("start", "zoom", "end"),
- touchstarting,
- touchfirst,
- touchending,
- touchDelay = 500,
- wheelDelay = 150,
- clickDistance2 = 0,
- tapDistance = 10;
-
- function zoom(selection) {
- selection.property("__zoom", defaultTransform).on("wheel.zoom", wheeled).on("mousedown.zoom", mousedowned).on("dblclick.zoom", dblclicked).filter(touchable).on("touchstart.zoom", touchstarted).on("touchmove.zoom", touchmoved).on("touchend.zoom touchcancel.zoom", touchended).style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
- }
-
- zoom.transform = function (collection, transform, point, event) {
- var selection = collection.selection ? collection.selection() : collection;
- selection.property("__zoom", defaultTransform);
-
- if (collection !== selection) {
- schedule(collection, transform, point, event);
- } else {
- selection.interrupt().each(function () {
- gesture(this, arguments).event(event).start().zoom(null, typeof transform === "function" ? transform.apply(this, arguments) : transform).end();
- });
- }
- };
-
- zoom.scaleBy = function (selection, k, p, event) {
- zoom.scaleTo(selection, function () {
- var k0 = this.__zoom.k,
- k1 = typeof k === "function" ? k.apply(this, arguments) : k;
- return k0 * k1;
- }, p, event);
- };
-
- zoom.scaleTo = function (selection, k, p, event) {
- zoom.transform(selection, function () {
- var e = extent.apply(this, arguments),
- t0 = this.__zoom,
- p0 = p == null ? 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, event);
- };
-
- zoom.translateBy = function (selection, x, y, event) {
- zoom.transform(selection, function () {
- return constrain(this.__zoom.translate(typeof x === "function" ? x.apply(this, arguments) : x, typeof y === "function" ? y.apply(this, arguments) : y), extent.apply(this, arguments), translateExtent);
- }, null, event);
- };
-
- zoom.translateTo = function (selection, x, y, p, event) {
- zoom.transform(selection, function () {
- var e = extent.apply(this, arguments),
- t = this.__zoom,
- p0 = p == null ? 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, event);
- };
-
- 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);
- }
-
- function centroid(extent) {
- return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];
- }
-
- function schedule(transition, transform, point, event) {
- transition.on("start.zoom", function () {
- gesture(this, arguments).event(event).start();
- }).on("interrupt.zoom end.zoom", function () {
- gesture(this, arguments).event(event).end();
- }).tween("zoom", function () {
- var that = this,
- args = arguments,
- g = gesture(that, args).event(event),
- e = extent.apply(that, args),
- p = point == null ? 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 = that.__zoom,
- 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, t);
- };
- });
- }
-
- function gesture(that, args, clean) {
- return !clean && that.__zooming || new Gesture(that, args);
- }
-
- function Gesture(that, args) {
- this.that = that;
- this.args = args;
- this.active = 0;
- this.sourceEvent = null;
- this.extent = extent.apply(that, args);
- this.taps = 0;
- }
-
- Gesture.prototype = {
- event: function event(_event) {
- if (_event) this.sourceEvent = _event;
- return this;
- },
- start: function start() {
- if (++this.active === 1) {
- this.that.__zooming = this;
- this.emit("start");
- }
-
- return this;
- },
- zoom: function zoom(key, transform) {
- if (this.mouse && key !== "mouse") this.mouse[1] = transform.invert(this.mouse[0]);
- if (this.touch0 && key !== "touch") this.touch0[1] = transform.invert(this.touch0[0]);
- if (this.touch1 && key !== "touch") this.touch1[1] = transform.invert(this.touch1[0]);
- this.that.__zoom = transform;
- this.emit("zoom");
- return this;
- },
- end: function end() {
- if (--this.active === 0) {
- delete this.that.__zooming;
- this.emit("end");
- }
-
- return this;
- },
- emit: function emit(type) {
- var d = select(this.that).datum();
- listeners.call(type, this.that, new ZoomEvent(type, {
- sourceEvent: this.sourceEvent,
- target: zoom,
- type: type,
- transform: this.that.__zoom,
- dispatch: listeners
- }), d);
- }
- };
-
- function wheeled(event) {
- for (var _len = arguments.length, args = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {
- args[_key - 1] = arguments[_key];
- }
-
- if (!filter.apply(this, arguments)) return;
- var g = gesture(this, args).event(event),
- t = this.__zoom,
- k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t.k * Math.pow(2, wheelDelta.apply(this, arguments)))),
- p = pointer(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);
- }
-
- clearTimeout(g.wheel);
- } // If this wheel event won’t trigger a transform change, ignore it.
- else if (t.k === k) return; // Otherwise, capture the mouse point and location at the start.
- else {
- g.mouse = [p, t.invert(p)];
- interrupt(this);
- g.start();
- }
-
- noevent$1(event);
- g.wheel = setTimeout(wheelidled, wheelDelay);
- g.zoom("mouse", constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));
-
- function wheelidled() {
- g.wheel = null;
- g.end();
- }
- }
-
- function mousedowned(event) {
- for (var _len2 = arguments.length, args = new Array(_len2 > 1 ? _len2 - 1 : 0), _key2 = 1; _key2 < _len2; _key2++) {
- args[_key2 - 1] = arguments[_key2];
- }
-
- if (touchending || !filter.apply(this, arguments)) return;
- var g = gesture(this, args, true).event(event),
- v = select(event.view).on("mousemove.zoom", mousemoved, true).on("mouseup.zoom", mouseupped, true),
- p = pointer(event, currentTarget),
- currentTarget = event.currentTarget,
- x0 = event.clientX,
- y0 = event.clientY;
- dragDisable(event.view);
- nopropagation$1(event);
- g.mouse = [p, this.__zoom.invert(p)];
- interrupt(this);
- g.start();
-
- function mousemoved(event) {
- noevent$1(event);
-
- if (!g.moved) {
- var dx = event.clientX - x0,
- dy = event.clientY - y0;
- g.moved = dx * dx + dy * dy > clickDistance2;
- }
-
- g.event(event).zoom("mouse", constrain(translate(g.that.__zoom, g.mouse[0] = pointer(event, currentTarget), g.mouse[1]), g.extent, translateExtent));
- }
-
- function mouseupped(event) {
- v.on("mousemove.zoom mouseup.zoom", null);
- yesdrag(event.view, g.moved);
- noevent$1(event);
- g.event(event).end();
- }
- }
-
- function dblclicked(event) {
- for (var _len3 = arguments.length, args = new Array(_len3 > 1 ? _len3 - 1 : 0), _key3 = 1; _key3 < _len3; _key3++) {
- args[_key3 - 1] = arguments[_key3];
- }
-
- if (!filter.apply(this, arguments)) return;
- var t0 = this.__zoom,
- p0 = pointer(event.changedTouches ? event.changedTouches[0] : event, this),
- 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);
- if (duration > 0) select(this).transition().duration(duration).call(schedule, t1, p0, event);else select(this).call(zoom.transform, t1, p0, event);
- }
-
- function touchstarted(event) {
- for (var _len4 = arguments.length, args = new Array(_len4 > 1 ? _len4 - 1 : 0), _key4 = 1; _key4 < _len4; _key4++) {
- args[_key4 - 1] = arguments[_key4];
- }
-
- if (!filter.apply(this, arguments)) return;
- var touches = event.touches,
- n = touches.length,
- g = gesture(this, args, event.changedTouches.length === n).event(event),
- started,
- i,
- t,
- p;
- nopropagation$1(event);
-
- for (i = 0; i < n; ++i) {
- t = touches[i], p = pointer(t, this);
- p = [p, this.__zoom.invert(p), t.identifier];
- if (!g.touch0) g.touch0 = p, started = true, g.taps = 1 + !!touchstarting;else if (!g.touch1 && g.touch0[2] !== p[2]) g.touch1 = p, g.taps = 0;
- }
-
- if (touchstarting) touchstarting = clearTimeout(touchstarting);
-
- if (started) {
- if (g.taps < 2) touchfirst = p[0], touchstarting = setTimeout(function () {
- touchstarting = null;
- }, touchDelay);
- interrupt(this);
- g.start();
- }
- }
-
- function touchmoved(event) {
- if (!this.__zooming) return;
-
- for (var _len5 = arguments.length, args = new Array(_len5 > 1 ? _len5 - 1 : 0), _key5 = 1; _key5 < _len5; _key5++) {
- args[_key5 - 1] = arguments[_key5];
- }
-
- var g = gesture(this, args).event(event),
- touches = event.changedTouches,
- n = touches.length,
- i,
- t,
- p,
- l;
- noevent$1(event);
-
- for (i = 0; i < n; ++i) {
- t = touches[i], p = pointer(t, this);
- if (g.touch0 && g.touch0[2] === t.identifier) g.touch0[0] = p;else if (g.touch1 && g.touch1[2] === t.identifier) g.touch1[0] = p;
- }
-
- t = g.that.__zoom;
-
- if (g.touch1) {
- var p0 = g.touch0[0],
- l0 = g.touch0[1],
- p1 = g.touch1[0],
- l1 = g.touch1[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.touch0) p = g.touch0[0], l = g.touch0[1];else return;
-
- g.zoom("touch", constrain(translate(t, p, l), g.extent, translateExtent));
- }
-
- function touchended(event) {
- for (var _len6 = arguments.length, args = new Array(_len6 > 1 ? _len6 - 1 : 0), _key6 = 1; _key6 < _len6; _key6++) {
- args[_key6 - 1] = arguments[_key6];
- }
-
- if (!this.__zooming) return;
- var g = gesture(this, args).event(event),
- touches = event.changedTouches,
- n = touches.length,
- i,
- t;
- nopropagation$1(event);
- if (touchending) clearTimeout(touchending);
- touchending = setTimeout(function () {
- touchending = null;
- }, touchDelay);
-
- for (i = 0; i < n; ++i) {
- t = touches[i];
- if (g.touch0 && g.touch0[2] === t.identifier) delete g.touch0;else if (g.touch1 && g.touch1[2] === t.identifier) delete g.touch1;
- }
-
- if (g.touch1 && !g.touch0) g.touch0 = g.touch1, delete g.touch1;
- if (g.touch0) g.touch0[1] = this.__zoom.invert(g.touch0[0]);else {
- g.end(); // If this was a dbltap, reroute to the (optional) dblclick.zoom handler.
-
- if (g.taps === 2) {
- t = pointer(t, this);
-
- if (Math.hypot(touchfirst[0] - t[0], touchfirst[1] - t[1]) < tapDistance) {
- var p = select(this).on("dblclick.zoom");
- if (p) p.apply(this, arguments);
- }
- }
- }
- }
-
- zoom.wheelDelta = function (_) {
- return arguments.length ? (wheelDelta = typeof _ === "function" ? _ : constant$3(+_), zoom) : wheelDelta;
- };
-
- zoom.filter = function (_) {
- return arguments.length ? (filter = typeof _ === "function" ? _ : constant$3(!!_), zoom) : filter;
- };
-
- zoom.touchable = function (_) {
- return arguments.length ? (touchable = typeof _ === "function" ? _ : constant$3(!!_), zoom) : touchable;
- };
-
- zoom.extent = function (_) {
- return arguments.length ? (extent = typeof _ === "function" ? _ : constant$3([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
- };
-
- zoom.scaleExtent = function (_) {
- return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];
- };
-
- 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]]];
- };
-
- zoom.constrain = function (_) {
- return arguments.length ? (constrain = _, zoom) : constrain;
- };
-
- zoom.duration = function (_) {
- return arguments.length ? (duration = +_, zoom) : duration;
- };
-
- zoom.interpolate = function (_) {
- return arguments.length ? (interpolate = _, zoom) : interpolate;
- };
-
- zoom.on = function () {
- var value = listeners.on.apply(listeners, arguments);
- return value === listeners ? zoom : value;
- };
-
- zoom.clickDistance = function (_) {
- return arguments.length ? (clickDistance2 = (_ = +_) * _, zoom) : Math.sqrt(clickDistance2);
- };
-
- zoom.tapDistance = function (_) {
- return arguments.length ? (tapDistance = +_, zoom) : tapDistance;
- };
-
- return zoom;
- }
-
- /*
- Bypasses features of D3's default projection stream pipeline that are unnecessary:
- * Antimeridian clipping
- * Spherical rotation
- * Resampling
- */
-
- function geoRawMercator() {
- var project = mercatorRaw;
- var k = 512 / Math.PI; // scale
-
- var x = 0;
- var y = 0; // translate
-
- var clipExtent = [[0, 0], [0, 0]];
-
- function projection(point) {
- point = project(point[0] * Math.PI / 180, point[1] * Math.PI / 180);
- return [point[0] * k + x, y - point[1] * k];
- }
-
- projection.invert = function (point) {
- point = project.invert((point[0] - x) / k, (y - point[1]) / k);
- return point && [point[0] * 180 / Math.PI, point[1] * 180 / Math.PI];
- };
-
- projection.scale = function (_) {
- if (!arguments.length) return k;
- k = +_;
- return projection;
- };
-
- projection.translate = function (_) {
- if (!arguments.length) return [x, y];
- x = +_[0];
- y = +_[1];
- return projection;
- };
-
- projection.clipExtent = function (_) {
- if (!arguments.length) return clipExtent;
- clipExtent = _;
- return projection;
- };
-
- projection.transform = function (obj) {
- if (!arguments.length) return identity$2.translate(x, y).scale(k);
- x = +obj.x;
- y = +obj.y;
- k = +obj.k;
- return projection;
- };
-
- projection.stream = d3_geoTransform({
- point: function point(x, y) {
- var vec = projection([x, y]);
- this.stream.point(vec[0], vec[1]);
- }
- }).stream;
- return projection;
- }
-
- function geoOrthoNormalizedDotProduct(a, b, origin) {
- if (geoVecEqual(origin, a) || geoVecEqual(origin, b)) {
- return 1; // coincident points, treat as straight and try to remove
- }
-
- return geoVecNormalizedDot(a, b, origin);
- }
-
- function geoOrthoFilterDotProduct(dotp, epsilon, lowerThreshold, upperThreshold, allowStraightAngles) {
- var val = Math.abs(dotp);
-
- if (val < epsilon) {
- return 0; // already orthogonal
- } else if (allowStraightAngles && Math.abs(val - 1) < epsilon) {
- return 0; // straight angle, which is okay in this case
- } else if (val < lowerThreshold || val > upperThreshold) {
- return dotp; // can be adjusted
- } else {
- return null; // ignore vertex
- }
- }
-
- function geoOrthoCalcScore(points, isClosed, epsilon, threshold) {
- var score = 0;
- var first = isClosed ? 0 : 1;
- var last = isClosed ? points.length : points.length - 1;
- var coords = points.map(function (p) {
- return p.coord;
- });
- var lowerThreshold = Math.cos((90 - threshold) * Math.PI / 180);
- var upperThreshold = Math.cos(threshold * Math.PI / 180);
-
- for (var i = first; i < last; i++) {
- var a = coords[(i - 1 + coords.length) % coords.length];
- var origin = coords[i];
- var b = coords[(i + 1) % coords.length];
- var dotp = geoOrthoFilterDotProduct(geoOrthoNormalizedDotProduct(a, b, origin), epsilon, lowerThreshold, upperThreshold);
- if (dotp === null) continue; // ignore vertex
-
- score = score + 2.0 * Math.min(Math.abs(dotp - 1.0), Math.min(Math.abs(dotp), Math.abs(dotp + 1)));
- }
-
- return score;
- } // returns the maximum angle less than `lessThan` between the actual corner and a 0° or 90° corner
-
- function geoOrthoMaxOffsetAngle(coords, isClosed, lessThan) {
- var max = -Infinity;
- var first = isClosed ? 0 : 1;
- var last = isClosed ? coords.length : coords.length - 1;
-
- for (var i = first; i < last; i++) {
- var a = coords[(i - 1 + coords.length) % coords.length];
- var origin = coords[i];
- var b = coords[(i + 1) % coords.length];
- var normalizedDotP = geoOrthoNormalizedDotProduct(a, b, origin);
- var angle = Math.acos(Math.abs(normalizedDotP)) * 180 / Math.PI;
- if (angle > 45) angle = 90 - angle;
- if (angle >= lessThan) continue;
- if (angle > max) max = angle;
- }
-
- if (max === -Infinity) return null;
- return max;
- } // similar to geoOrthoCalcScore, but returns quickly if there is something to do
-
- function geoOrthoCanOrthogonalize(coords, isClosed, epsilon, threshold, allowStraightAngles) {
- var score = null;
- var first = isClosed ? 0 : 1;
- var last = isClosed ? coords.length : coords.length - 1;
- var lowerThreshold = Math.cos((90 - threshold) * Math.PI / 180);
- var upperThreshold = Math.cos(threshold * Math.PI / 180);
-
- for (var i = first; i < last; i++) {
- var a = coords[(i - 1 + coords.length) % coords.length];
- var origin = coords[i];
- var b = coords[(i + 1) % coords.length];
- var dotp = geoOrthoFilterDotProduct(geoOrthoNormalizedDotProduct(a, b, origin), epsilon, lowerThreshold, upperThreshold, allowStraightAngles);
- if (dotp === null) continue; // ignore vertex
-
- if (Math.abs(dotp) > 0) return 1; // something to do
-
- score = 0; // already square
- }
-
- return score;
- }
-
- var onFreeze = internalMetadata.onFreeze;
-
- var nativeFreeze = Object.freeze;
- var FAILS_ON_PRIMITIVES$4 = fails(function () { nativeFreeze(1); });
-
- // `Object.freeze` method
- // https://tc39.github.io/ecma262/#sec-object.freeze
- _export({ target: 'Object', stat: true, forced: FAILS_ON_PRIMITIVES$4, sham: !freezing }, {
- freeze: function freeze(it) {
- return nativeFreeze && isObject(it) ? nativeFreeze(onFreeze(it)) : it;
- }
- });
-
- // Returns true if a and b have the same elements at the same indices.
- function utilArrayIdentical(a, b) {
- // an array is always identical to itself
- if (a === b) return true;
- var i = a.length;
- if (i !== b.length) return false;
-
- while (i--) {
- if (a[i] !== b[i]) return false;
- }
-
- return true;
- } // http://2ality.com/2015/01/es6-set-operations.html
- // Difference (a \ b): create a set that contains those elements of set a that are not in set b.
- // This operation is also sometimes called minus (-).
- // var a = [1,2,3];
- // var b = [4,3,2];
- // utilArrayDifference(a, b)
- // [1]
- // utilArrayDifference(b, a)
- // [4]
-
- function utilArrayDifference(a, b) {
- var other = new Set(b);
- return Array.from(new Set(a)).filter(function (v) {
- return !other.has(v);
- });
- } // Intersection (a ∩ b): create a set that contains those elements of set a that are also in set b.
- // var a = [1,2,3];
- // var b = [4,3,2];
- // utilArrayIntersection(a, b)
- // [2,3]
-
- function utilArrayIntersection(a, b) {
- var other = new Set(b);
- return Array.from(new Set(a)).filter(function (v) {
- return other.has(v);
- });
- } // Union (a ∪ b): create a set that contains the elements of both set a and set b.
- // var a = [1,2,3];
- // var b = [4,3,2];
- // utilArrayUnion(a, b)
- // [1,2,3,4]
-
- function utilArrayUnion(a, b) {
- var result = new Set(a);
- b.forEach(function (v) {
- result.add(v);
- });
- return Array.from(result);
- } // Returns an Array with all the duplicates removed
- // var a = [1,1,2,3,3];
- // utilArrayUniq(a)
- // [1,2,3]
-
- function utilArrayUniq(a) {
- return Array.from(new Set(a));
- } // Splits array into chunks of given chunk size
- // var a = [1,2,3,4,5,6,7];
- // utilArrayChunk(a, 3);
- // [[1,2,3],[4,5,6],[7]];
-
- function utilArrayChunk(a, chunkSize) {
- if (!chunkSize || chunkSize < 0) return [a.slice()];
- var result = new Array(Math.ceil(a.length / chunkSize));
- return Array.from(result, function (item, i) {
- return a.slice(i * chunkSize, i * chunkSize + chunkSize);
- });
- } // Flattens two level array into a single level
- // var a = [[1,2,3],[4,5,6],[7]];
- // utilArrayFlatten(a);
- // [1,2,3,4,5,6,7];
-
- function utilArrayFlatten(a) {
- return a.reduce(function (acc, val) {
- return acc.concat(val);
- }, []);
- } // Groups the items of the Array according to the given key
- // `key` can be passed as a property or as a key function
- //
- // var pets = [
- // { type: 'Dog', name: 'Spot' },
- // { type: 'Cat', name: 'Tiger' },
- // { type: 'Dog', name: 'Rover' },
- // { type: 'Cat', name: 'Leo' }
- // ];
- //
- // utilArrayGroupBy(pets, 'type')
- // {
- // 'Dog': [{type: 'Dog', name: 'Spot'}, {type: 'Dog', name: 'Rover'}],
- // 'Cat': [{type: 'Cat', name: 'Tiger'}, {type: 'Cat', name: 'Leo'}]
- // }
- //
- // utilArrayGroupBy(pets, function(item) { return item.name.length; })
- // {
- // 3: [{type: 'Cat', name: 'Leo'}],
- // 4: [{type: 'Dog', name: 'Spot'}],
- // 5: [{type: 'Cat', name: 'Tiger'}, {type: 'Dog', name: 'Rover'}]
- // }
-
- function utilArrayGroupBy(a, key) {
- return a.reduce(function (acc, item) {
- var group = typeof key === 'function' ? key(item) : item[key];
- (acc[group] = acc[group] || []).push(item);
- return acc;
- }, {});
- } // Returns an Array with all the duplicates removed
- // where uniqueness determined by the given key
- // `key` can be passed as a property or as a key function
- //
- // var pets = [
- // { type: 'Dog', name: 'Spot' },
- // { type: 'Cat', name: 'Tiger' },
- // { type: 'Dog', name: 'Rover' },
- // { type: 'Cat', name: 'Leo' }
- // ];
- //
- // utilArrayUniqBy(pets, 'type')
- // [
- // { type: 'Dog', name: 'Spot' },
- // { type: 'Cat', name: 'Tiger' }
- // ]
- //
- // utilArrayUniqBy(pets, function(item) { return item.name.length; })
- // [
- // { type: 'Dog', name: 'Spot' },
- // { type: 'Cat', name: 'Tiger' },
- // { type: 'Cat', name: 'Leo' }
- // }
-
- function utilArrayUniqBy(a, key) {
- var seen = new Set();
- return a.reduce(function (acc, item) {
- var val = typeof key === 'function' ? key(item) : item[key];
-
- if (val && !seen.has(val)) {
- seen.add(val);
- acc.push(item);
- }
-
- return acc;
- }, []);
- }
-
- // @@match logic
- fixRegexpWellKnownSymbolLogic('match', 1, function (MATCH, nativeMatch, maybeCallNative) {
- return [
- // `String.prototype.match` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.match
- function match(regexp) {
- var O = requireObjectCoercible(this);
- var matcher = regexp == undefined ? undefined : regexp[MATCH];
- return matcher !== undefined ? matcher.call(regexp, O) : new RegExp(regexp)[MATCH](String(O));
- },
- // `RegExp.prototype[@@match]` method
- // https://tc39.github.io/ecma262/#sec-regexp.prototype-@@match
- function (regexp) {
- var res = maybeCallNative(nativeMatch, regexp, this);
- if (res.done) return res.value;
-
- var rx = anObject(regexp);
- var S = String(this);
-
- if (!rx.global) return regexpExecAbstract(rx, S);
-
- var fullUnicode = rx.unicode;
- rx.lastIndex = 0;
- var A = [];
- var n = 0;
- var result;
- while ((result = regexpExecAbstract(rx, S)) !== null) {
- var matchStr = String(result[0]);
- A[n] = matchStr;
- if (matchStr === '') rx.lastIndex = advanceStringIndex(S, toLength(rx.lastIndex), fullUnicode);
- n++;
- }
- return n === 0 ? null : A;
- }
- ];
- });
-
- var remove$1 = removeDiacritics;
- var replacementList = [{
- base: ' ',
- chars: "\xA0"
- }, {
- base: '0',
- chars: "\u07C0"
- }, {
- base: 'A',
- chars: "\u24B6\uFF21\xC0\xC1\xC2\u1EA6\u1EA4\u1EAA\u1EA8\xC3\u0100\u0102\u1EB0\u1EAE\u1EB4\u1EB2\u0226\u01E0\xC4\u01DE\u1EA2\xC5\u01FA\u01CD\u0200\u0202\u1EA0\u1EAC\u1EB6\u1E00\u0104\u023A\u2C6F"
- }, {
- base: 'AA',
- chars: "\uA732"
- }, {
- base: 'AE',
- chars: "\xC6\u01FC\u01E2"
- }, {
- base: 'AO',
- chars: "\uA734"
- }, {
- base: 'AU',
- chars: "\uA736"
- }, {
- base: 'AV',
- chars: "\uA738\uA73A"
- }, {
- base: 'AY',
- chars: "\uA73C"
- }, {
- base: 'B',
- chars: "\u24B7\uFF22\u1E02\u1E04\u1E06\u0243\u0181"
- }, {
- base: 'C',
- chars: "\u24B8\uFF23\uA73E\u1E08\u0106C\u0108\u010A\u010C\xC7\u0187\u023B"
- }, {
- base: 'D',
- chars: "\u24B9\uFF24\u1E0A\u010E\u1E0C\u1E10\u1E12\u1E0E\u0110\u018A\u0189\u1D05\uA779"
- }, {
- base: 'Dh',
- chars: "\xD0"
- }, {
- base: 'DZ',
- chars: "\u01F1\u01C4"
- }, {
- base: 'Dz',
- chars: "\u01F2\u01C5"
- }, {
- base: 'E',
- chars: "\u025B\u24BA\uFF25\xC8\xC9\xCA\u1EC0\u1EBE\u1EC4\u1EC2\u1EBC\u0112\u1E14\u1E16\u0114\u0116\xCB\u1EBA\u011A\u0204\u0206\u1EB8\u1EC6\u0228\u1E1C\u0118\u1E18\u1E1A\u0190\u018E\u1D07"
- }, {
- base: 'F',
- chars: "\uA77C\u24BB\uFF26\u1E1E\u0191\uA77B"
- }, {
- base: 'G',
- chars: "\u24BC\uFF27\u01F4\u011C\u1E20\u011E\u0120\u01E6\u0122\u01E4\u0193\uA7A0\uA77D\uA77E\u0262"
- }, {
- base: 'H',
- chars: "\u24BD\uFF28\u0124\u1E22\u1E26\u021E\u1E24\u1E28\u1E2A\u0126\u2C67\u2C75\uA78D"
- }, {
- base: 'I',
- chars: "\u24BE\uFF29\xCC\xCD\xCE\u0128\u012A\u012C\u0130\xCF\u1E2E\u1EC8\u01CF\u0208\u020A\u1ECA\u012E\u1E2C\u0197"
- }, {
- base: 'J',
- chars: "\u24BF\uFF2A\u0134\u0248\u0237"
- }, {
- base: 'K',
- chars: "\u24C0\uFF2B\u1E30\u01E8\u1E32\u0136\u1E34\u0198\u2C69\uA740\uA742\uA744\uA7A2"
- }, {
- base: 'L',
- chars: "\u24C1\uFF2C\u013F\u0139\u013D\u1E36\u1E38\u013B\u1E3C\u1E3A\u0141\u023D\u2C62\u2C60\uA748\uA746\uA780"
- }, {
- base: 'LJ',
- chars: "\u01C7"
- }, {
- base: 'Lj',
- chars: "\u01C8"
- }, {
- base: 'M',
- chars: "\u24C2\uFF2D\u1E3E\u1E40\u1E42\u2C6E\u019C\u03FB"
- }, {
- base: 'N',
- chars: "\uA7A4\u0220\u24C3\uFF2E\u01F8\u0143\xD1\u1E44\u0147\u1E46\u0145\u1E4A\u1E48\u019D\uA790\u1D0E"
- }, {
- base: 'NJ',
- chars: "\u01CA"
- }, {
- base: 'Nj',
- chars: "\u01CB"
- }, {
- base: 'O',
- chars: "\u24C4\uFF2F\xD2\xD3\xD4\u1ED2\u1ED0\u1ED6\u1ED4\xD5\u1E4C\u022C\u1E4E\u014C\u1E50\u1E52\u014E\u022E\u0230\xD6\u022A\u1ECE\u0150\u01D1\u020C\u020E\u01A0\u1EDC\u1EDA\u1EE0\u1EDE\u1EE2\u1ECC\u1ED8\u01EA\u01EC\xD8\u01FE\u0186\u019F\uA74A\uA74C"
- }, {
- base: 'OE',
- chars: "\u0152"
- }, {
- base: 'OI',
- chars: "\u01A2"
- }, {
- base: 'OO',
- chars: "\uA74E"
- }, {
- base: 'OU',
- chars: "\u0222"
- }, {
- base: 'P',
- chars: "\u24C5\uFF30\u1E54\u1E56\u01A4\u2C63\uA750\uA752\uA754"
- }, {
- base: 'Q',
- chars: "\u24C6\uFF31\uA756\uA758\u024A"
- }, {
- base: 'R',
- chars: "\u24C7\uFF32\u0154\u1E58\u0158\u0210\u0212\u1E5A\u1E5C\u0156\u1E5E\u024C\u2C64\uA75A\uA7A6\uA782"
- }, {
- base: 'S',
- chars: "\u24C8\uFF33\u1E9E\u015A\u1E64\u015C\u1E60\u0160\u1E66\u1E62\u1E68\u0218\u015E\u2C7E\uA7A8\uA784"
- }, {
- base: 'T',
- chars: "\u24C9\uFF34\u1E6A\u0164\u1E6C\u021A\u0162\u1E70\u1E6E\u0166\u01AC\u01AE\u023E\uA786"
- }, {
- base: 'Th',
- chars: "\xDE"
- }, {
- base: 'TZ',
- chars: "\uA728"
- }, {
- base: 'U',
- chars: "\u24CA\uFF35\xD9\xDA\xDB\u0168\u1E78\u016A\u1E7A\u016C\xDC\u01DB\u01D7\u01D5\u01D9\u1EE6\u016E\u0170\u01D3\u0214\u0216\u01AF\u1EEA\u1EE8\u1EEE\u1EEC\u1EF0\u1EE4\u1E72\u0172\u1E76\u1E74\u0244"
- }, {
- base: 'V',
- chars: "\u24CB\uFF36\u1E7C\u1E7E\u01B2\uA75E\u0245"
- }, {
- base: 'VY',
- chars: "\uA760"
- }, {
- base: 'W',
- chars: "\u24CC\uFF37\u1E80\u1E82\u0174\u1E86\u1E84\u1E88\u2C72"
- }, {
- base: 'X',
- chars: "\u24CD\uFF38\u1E8A\u1E8C"
- }, {
- base: 'Y',
- chars: "\u24CE\uFF39\u1EF2\xDD\u0176\u1EF8\u0232\u1E8E\u0178\u1EF6\u1EF4\u01B3\u024E\u1EFE"
- }, {
- base: 'Z',
- chars: "\u24CF\uFF3A\u0179\u1E90\u017B\u017D\u1E92\u1E94\u01B5\u0224\u2C7F\u2C6B\uA762"
- }, {
- base: 'a',
- chars: "\u24D0\uFF41\u1E9A\xE0\xE1\xE2\u1EA7\u1EA5\u1EAB\u1EA9\xE3\u0101\u0103\u1EB1\u1EAF\u1EB5\u1EB3\u0227\u01E1\xE4\u01DF\u1EA3\xE5\u01FB\u01CE\u0201\u0203\u1EA1\u1EAD\u1EB7\u1E01\u0105\u2C65\u0250\u0251"
- }, {
- base: 'aa',
- chars: "\uA733"
- }, {
- base: 'ae',
- chars: "\xE6\u01FD\u01E3"
- }, {
- base: 'ao',
- chars: "\uA735"
- }, {
- base: 'au',
- chars: "\uA737"
- }, {
- base: 'av',
- chars: "\uA739\uA73B"
- }, {
- base: 'ay',
- chars: "\uA73D"
- }, {
- base: 'b',
- chars: "\u24D1\uFF42\u1E03\u1E05\u1E07\u0180\u0183\u0253\u0182"
- }, {
- base: 'c',
- chars: "\uFF43\u24D2\u0107\u0109\u010B\u010D\xE7\u1E09\u0188\u023C\uA73F\u2184"
- }, {
- base: 'd',
- chars: "\u24D3\uFF44\u1E0B\u010F\u1E0D\u1E11\u1E13\u1E0F\u0111\u018C\u0256\u0257\u018B\u13E7\u0501\uA7AA"
- }, {
- base: 'dh',
- chars: "\xF0"
- }, {
- base: 'dz',
- chars: "\u01F3\u01C6"
- }, {
- base: 'e',
- chars: "\u24D4\uFF45\xE8\xE9\xEA\u1EC1\u1EBF\u1EC5\u1EC3\u1EBD\u0113\u1E15\u1E17\u0115\u0117\xEB\u1EBB\u011B\u0205\u0207\u1EB9\u1EC7\u0229\u1E1D\u0119\u1E19\u1E1B\u0247\u01DD"
- }, {
- base: 'f',
- chars: "\u24D5\uFF46\u1E1F\u0192"
- }, {
- base: 'ff',
- chars: "\uFB00"
- }, {
- base: 'fi',
- chars: "\uFB01"
- }, {
- base: 'fl',
- chars: "\uFB02"
- }, {
- base: 'ffi',
- chars: "\uFB03"
- }, {
- base: 'ffl',
- chars: "\uFB04"
- }, {
- base: 'g',
- chars: "\u24D6\uFF47\u01F5\u011D\u1E21\u011F\u0121\u01E7\u0123\u01E5\u0260\uA7A1\uA77F\u1D79"
- }, {
- base: 'h',
- chars: "\u24D7\uFF48\u0125\u1E23\u1E27\u021F\u1E25\u1E29\u1E2B\u1E96\u0127\u2C68\u2C76\u0265"
- }, {
- base: 'hv',
- chars: "\u0195"
- }, {
- base: 'i',
- chars: "\u24D8\uFF49\xEC\xED\xEE\u0129\u012B\u012D\xEF\u1E2F\u1EC9\u01D0\u0209\u020B\u1ECB\u012F\u1E2D\u0268\u0131"
- }, {
- base: 'j',
- chars: "\u24D9\uFF4A\u0135\u01F0\u0249"
- }, {
- base: 'k',
- chars: "\u24DA\uFF4B\u1E31\u01E9\u1E33\u0137\u1E35\u0199\u2C6A\uA741\uA743\uA745\uA7A3"
- }, {
- base: 'l',
- chars: "\u24DB\uFF4C\u0140\u013A\u013E\u1E37\u1E39\u013C\u1E3D\u1E3B\u017F\u0142\u019A\u026B\u2C61\uA749\uA781\uA747\u026D"
- }, {
- base: 'lj',
- chars: "\u01C9"
- }, {
- base: 'm',
- chars: "\u24DC\uFF4D\u1E3F\u1E41\u1E43\u0271\u026F"
- }, {
- base: 'n',
- chars: "\u24DD\uFF4E\u01F9\u0144\xF1\u1E45\u0148\u1E47\u0146\u1E4B\u1E49\u019E\u0272\u0149\uA791\uA7A5\u043B\u0509"
- }, {
- base: 'nj',
- chars: "\u01CC"
- }, {
- base: 'o',
- chars: "\u24DE\uFF4F\xF2\xF3\xF4\u1ED3\u1ED1\u1ED7\u1ED5\xF5\u1E4D\u022D\u1E4F\u014D\u1E51\u1E53\u014F\u022F\u0231\xF6\u022B\u1ECF\u0151\u01D2\u020D\u020F\u01A1\u1EDD\u1EDB\u1EE1\u1EDF\u1EE3\u1ECD\u1ED9\u01EB\u01ED\xF8\u01FF\uA74B\uA74D\u0275\u0254\u1D11"
- }, {
- base: 'oe',
- chars: "\u0153"
- }, {
- base: 'oi',
- chars: "\u01A3"
- }, {
- base: 'oo',
- chars: "\uA74F"
- }, {
- base: 'ou',
- chars: "\u0223"
- }, {
- base: 'p',
- chars: "\u24DF\uFF50\u1E55\u1E57\u01A5\u1D7D\uA751\uA753\uA755\u03C1"
- }, {
- base: 'q',
- chars: "\u24E0\uFF51\u024B\uA757\uA759"
- }, {
- base: 'r',
- chars: "\u24E1\uFF52\u0155\u1E59\u0159\u0211\u0213\u1E5B\u1E5D\u0157\u1E5F\u024D\u027D\uA75B\uA7A7\uA783"
- }, {
- base: 's',
- chars: "\u24E2\uFF53\u015B\u1E65\u015D\u1E61\u0161\u1E67\u1E63\u1E69\u0219\u015F\u023F\uA7A9\uA785\u1E9B\u0282"
- }, {
- base: 'ss',
- chars: "\xDF"
- }, {
- base: 't',
- chars: "\u24E3\uFF54\u1E6B\u1E97\u0165\u1E6D\u021B\u0163\u1E71\u1E6F\u0167\u01AD\u0288\u2C66\uA787"
- }, {
- base: 'th',
- chars: "\xFE"
- }, {
- base: 'tz',
- chars: "\uA729"
- }, {
- base: 'u',
- chars: "\u24E4\uFF55\xF9\xFA\xFB\u0169\u1E79\u016B\u1E7B\u016D\xFC\u01DC\u01D8\u01D6\u01DA\u1EE7\u016F\u0171\u01D4\u0215\u0217\u01B0\u1EEB\u1EE9\u1EEF\u1EED\u1EF1\u1EE5\u1E73\u0173\u1E77\u1E75\u0289"
- }, {
- base: 'v',
- chars: "\u24E5\uFF56\u1E7D\u1E7F\u028B\uA75F\u028C"
- }, {
- base: 'vy',
- chars: "\uA761"
- }, {
- base: 'w',
- chars: "\u24E6\uFF57\u1E81\u1E83\u0175\u1E87\u1E85\u1E98\u1E89\u2C73"
- }, {
- base: 'x',
- chars: "\u24E7\uFF58\u1E8B\u1E8D"
- }, {
- base: 'y',
- chars: "\u24E8\uFF59\u1EF3\xFD\u0177\u1EF9\u0233\u1E8F\xFF\u1EF7\u1E99\u1EF5\u01B4\u024F\u1EFF"
- }, {
- base: 'z',
- chars: "\u24E9\uFF5A\u017A\u1E91\u017C\u017E\u1E93\u1E95\u01B6\u0225\u0240\u2C6C\uA763"
- }];
- var diacriticsMap = {};
-
- for (var i = 0; i < replacementList.length; i += 1) {
- var chars = replacementList[i].chars;
-
- for (var j$1 = 0; j$1 < chars.length; j$1 += 1) {
- diacriticsMap[chars[j$1]] = replacementList[i].base;
- }
- }
-
- function removeDiacritics(str) {
- return str.replace(/[^\u0000-\u007e]/g, function (c) {
- return diacriticsMap[c] || c;
- });
- }
-
- var replacementList_1 = replacementList;
- var diacriticsMap_1 = diacriticsMap;
- var diacritics = {
- remove: remove$1,
- 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
- ];
-
- 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);
-
- for (var i = 0; i < arabicBlocks.length; i++) {
- var block = arabicBlocks[i];
-
- 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');
- }
-
- var code = _char2.charCodeAt(0);
-
- return code >= 0x660 && code <= 0x66C || code >= 0x6F0 && code <= 0x6F9;
- }
-
- exports.isMath = isMath;
- });
-
- var unicodeArabic = createCommonjsModule(function (module, exports) {
-
- 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"
- },
- "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"
- },
- "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"
- }
- },
- "rohingya yeh": {
- "normal": ["\u08AC"]
- },
- "low alef": {
- "normal": ["\u08AD"]
- },
- "straight waw": {
- "normal": ["\u08B1"]
- },
- "african feh": {
- "normal": ["\u08BB"]
- },
- "african qaf": {
- "normal": ["\u08BC"]
- },
- "african noon": {
- "normal": ["\u08BD"]
- }
- };
- exports["default"] = arabicReference;
- });
-
- 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) {
-
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- var letterList = Object.keys(unicodeArabic["default"]);
- exports.letterList = letterList;
- var ligatureList = Object.keys(unicodeLigatures["default"]);
- exports.ligatureList = ligatureList;
- var ligatureWordList = Object.keys(unicodeLigatures["default"].words);
- exports.ligatureWordList = ligatureWordList;
- var lams = "\u0644\u06B5\u06B6\u06B7\u06B8";
- exports.lams = lams;
- var alefs = "\u0627\u0622\u0623\u0625\u0671\u0672\u0673\u0675\u0773\u0774";
- exports.alefs = alefs; // for (var l = 1; l < lams.length; l++) {
- // console.log('-');
- // for (var a = 0; a < alefs.length; a++) {
- // console.log(a + ': ' + lams[l] + alefs[a]);
- // }
- // }
-
- var tashkeel = "\u0605\u0640\u0670\u0674\u06DF\u06E7\u06E8";
- exports.tashkeel = tashkeel;
-
- function addToTashkeel(start, finish) {
- for (var i = start; i <= finish; i++) {
- exports.tashkeel = tashkeel += String.fromCharCode(i);
- }
- }
-
- addToTashkeel(0x0610, 0x061A);
- addToTashkeel(0x064B, 0x065F);
- addToTashkeel(0x06D6, 0x06DC);
- addToTashkeel(0x06E0, 0x06E4);
- addToTashkeel(0x06EA, 0x06ED);
- addToTashkeel(0x08D3, 0x08E1);
- addToTashkeel(0x08E3, 0x08FF);
- addToTashkeel(0xFE70, 0xFE7F);
- var lineBreakers = "\u0627\u0629\u0648\u06C0\u06CF\u06FD\u06FE\u076B\u076C\u0771\u0773\u0774\u0778\u0779\u08E2\u08B1\u08B2\u08B9";
- exports.lineBreakers = lineBreakers;
-
- function addToLineBreakers(start, finish) {
- for (var i = start; i <= finish; i++) {
- exports.lineBreakers = lineBreakers += String.fromCharCode(i);
- }
- }
-
- addToLineBreakers(0x0600, 0x061F); // it's OK to include tashkeel in this range as it is ignored
-
- addToLineBreakers(0x0621, 0x0625);
- addToLineBreakers(0x062F, 0x0632);
- addToLineBreakers(0x0660, 0x066D); // numerals, math
-
- addToLineBreakers(0x0671, 0x0677);
- addToLineBreakers(0x0688, 0x0699);
- addToLineBreakers(0x06C3, 0x06CB);
- addToLineBreakers(0x06D2, 0x06F9);
- addToLineBreakers(0x0759, 0x075B);
- addToLineBreakers(0x08AA, 0x08AE);
- addToLineBreakers(0xFB50, 0xFDFD); // presentation forms look like they could connect, but never do
- // Presentation Forms A includes diacritics but they are meant to stand alone
-
- addToLineBreakers(0xFE80, 0xFEFC); // presentation forms look like they could connect, but never do
- // numerals, math
-
- addToLineBreakers(0x10E60, 0x10E7F);
- addToLineBreakers(0x1EC70, 0x1ECBF);
- 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);
- }
- } else {
- letters.push(letter);
- }
-
- if (reference.tashkeel.indexOf(letter) === -1) {
- lastLetter = letter;
- }
- });
- return letters;
- }
-
- exports.GlyphSplitter = GlyphSplitter;
- });
-
- 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 {
- letters.push(letter);
- }
-
- if (reference.tashkeel.indexOf(letter) === -1) {
- // don't allow tashkeel to hide line break
- lastLetter = letter;
- }
- });
- return letters;
- }
-
- exports.BaselineSplitter = BaselineSplitter;
- });
-
- 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;
- }
-
- 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 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);
-
- 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'];
- }
-
- 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');
-
- 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'];
- }
-
- 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');
-
- return;
- }
- }
- } // try ligatures
-
-
- for (var v2 = 0; v2 < reference.ligatureList.length; v2++) {
- var normalForm = reference.ligatureList[v2];
-
- 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;
- }
- }
- }
- } // try words ligatures
-
-
- for (var v3 = 0; v3 < reference.ligatureWordList.length; v3++) {
- var _normalForm = reference.ligatureWordList[v3];
-
- if (unicodeLigatures["default"].words[_normalForm] === letter) {
- returnable += _normalForm;
- return;
- }
- }
-
- returnable += letter; // console.log('kept the letter')
- });
- return returnable;
- }
-
- exports.Normal = Normal;
- });
-
- 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');
- }
-
- 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 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);
-
- 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 WordShaper_1 = createCommonjsModule(function (module, exports) {
-
- 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;
- }
-
- if (reference.tashkeel.indexOf(word[nxw]) === -1) {
- nextLetter = 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'];
-
- while (word[w] !== nextLetter) {
- w++;
- }
-
- state = 'initial';
- } else {
- output += CharShaper_1.CharShaper(word[w], state);
- state = 'medial';
- }
- }
-
- return output;
- }
-
- exports.WordShaper = WordShaper;
- });
-
- var ParentLetter_1 = createCommonjsModule(function (module, exports) {
-
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
-
- 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 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);
-
- 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;
- }
- }
- } else if (localVersion === letter || _typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
- // match
- return letterForms;
- }
- }
-
- return null;
- }
- }
-
- exports.ParentLetter = ParentLetter;
-
- 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 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);
-
- 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;
- }
- }
- } else if (localVersion === letter || _typeof(localVersion) === 'object' && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
- // match
- return letterForms;
- }
- }
-
- return null;
- }
- }
-
- exports.GrandparentLetter = GrandparentLetter;
- });
-
- 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;
- });
-
- var rtlRegex = /[\u0590-\u05FF\u0600-\u06FF\u0750-\u07BF\u08A0–\u08BF]/;
- function fixRTLTextForSvg(inputText) {
- var ret = '',
- rtlBuffer = [];
- var arabicRegex = /[\u0600-\u06FF]/g;
- var arabicDiacritics = /[\u0610-\u061A\u064B-\u065F\u0670\u06D6-\u06ED]/g;
- var arabicMath = /[\u0660-\u066C\u06F0-\u06F9]+/g;
- var thaanaVowel = /[\u07A6-\u07B0]/;
- var hebrewSign = /[\u0591-\u05bd\u05bf\u05c1-\u05c5\u05c7]/; // Arabic word shaping
-
- if (arabicRegex.test(inputText)) {
- inputText = lib.WordShaper(inputText);
- }
-
- for (var n = 0; n < inputText.length; n++) {
- var c = inputText[n];
-
- if (arabicMath.test(c)) {
- // Arabic numbers go LTR
- ret += rtlBuffer.reverse().join('');
- rtlBuffer = [c];
- } else {
- if (rtlBuffer.length && arabicMath.test(rtlBuffer[rtlBuffer.length - 1])) {
- ret += rtlBuffer.reverse().join('');
- rtlBuffer = [];
- }
-
- if ((thaanaVowel.test(c) || hebrewSign.test(c) || arabicDiacritics.test(c)) && rtlBuffer.length) {
- rtlBuffer[rtlBuffer.length - 1] += c;
- } else if (rtlRegex.test(c) // include Arabic presentation forms
- || c.charCodeAt(0) >= 64336 && c.charCodeAt(0) <= 65023 || c.charCodeAt(0) >= 65136 && c.charCodeAt(0) <= 65279) {
- rtlBuffer.push(c);
- } else if (c === ' ' && rtlBuffer.length) {
- // whitespace within RTL text
- rtlBuffer = [rtlBuffer.reverse().join('') + ' '];
- } else {
- // non-RTL character
- ret += rtlBuffer.reverse().join('') + c;
- rtlBuffer = [];
- }
- }
- }
-
- ret += rtlBuffer.reverse().join('');
- return ret;
- }
-
- var propertyIsEnumerable = objectPropertyIsEnumerable.f;
-
- // `Object.{ entries, values }` methods implementation
- var createMethod$5 = function (TO_ENTRIES) {
- return function (it) {
- var O = toIndexedObject(it);
- var keys = objectKeys(O);
- var length = keys.length;
- var i = 0;
- var result = [];
- var key;
- while (length > i) {
- key = keys[i++];
- if (!descriptors || propertyIsEnumerable.call(O, key)) {
- result.push(TO_ENTRIES ? [key, O[key]] : O[key]);
- }
- }
- return result;
- };
- };
-
- var objectToArray = {
- // `Object.entries` method
- // https://tc39.github.io/ecma262/#sec-object.entries
- entries: createMethod$5(true),
- // `Object.values` method
- // https://tc39.github.io/ecma262/#sec-object.values
- values: createMethod$5(false)
- };
-
- var $values = objectToArray.values;
-
- // `Object.values` method
- // https://tc39.github.io/ecma262/#sec-object.values
- _export({ target: 'Object', stat: true }, {
- values: function values(O) {
- return $values(O);
- }
- });
-
- // https://github.com/openstreetmap/iD/issues/772
- // http://mathiasbynens.be/notes/localstorage-pattern#comment-9
- var _storage;
-
- try {
- _storage = localStorage;
- } catch (e) {} // eslint-disable-line no-empty
-
-
- _storage = _storage || function () {
- var s = {};
- return {
- getItem: function getItem(k) {
- return s[k];
- },
- setItem: function setItem(k, v) {
- return s[k] = v;
- },
- removeItem: function removeItem(k) {
- return delete s[k];
- }
- };
- }(); //
- // corePreferences is an interface for persisting basic key-value strings
- // within and between iD sessions on the same site.
- //
-
-
- function corePreferences(k, v) {
- try {
- if (arguments.length === 1) return _storage.getItem(k);else if (v === null) _storage.removeItem(k);else _storage.setItem(k, v);
- } catch (e) {
- /* eslint-disable no-console */
- if (typeof console !== 'undefined') {
- console.error('localStorage quota exceeded');
- }
- /* eslint-enable no-console */
-
- }
- }
-
- function responseText(response) {
- if (!response.ok) throw new Error(response.status + " " + response.statusText);
- return response.text();
- }
-
- function d3_text (input, init) {
- return fetch(input, init).then(responseText);
- }
-
- 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 d3_json (input, init) {
- return fetch(input, init).then(responseJson);
- }
-
- function parser(type) {
- return function (input, init) {
- return d3_text(input, init).then(function (text) {
- return new DOMParser().parseFromString(text, type);
- });
- };
- }
-
- var d3_xml = parser("application/xml");
- var svg = parser("image/svg+xml");
-
- var _mainFileFetcher = coreFileFetcher(); // singleton
- // coreFileFetcher asynchronously fetches data from JSON files
- //
-
- function coreFileFetcher() {
- var _this = {};
- var _inflight = {};
- var _fileMap = {
- 'address_formats': 'data/address_formats.min.json',
- 'deprecated': 'data/deprecated.min.json',
- 'discarded': 'data/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',
- 'phone_formats': 'data/phone_formats.min.json',
- 'qa_data': 'data/qa_data.min.json',
- 'shortcuts': 'data/shortcuts.min.json',
- 'territory_languages': 'data/territory_languages.min.json',
- 'wmf_sitematrix': 'https://cdn.jsdelivr.net/npm/wmf-sitematrix@0.1/wikipedia.min.json'
- };
- var _cachedData = {}; // expose the cache; useful for tests
-
- _this.cache = function () {
- return _cachedData;
- }; // Returns a Promise to fetch data
- // (resolved with the data if we have it already)
-
-
- _this.get = function (which) {
- if (_cachedData[which]) {
- return Promise.resolve(_cachedData[which]);
- }
-
- var file = _fileMap[which];
-
- var url = file && _this.asset(file);
-
- if (!url) {
- return Promise.reject("Unknown data file for \"".concat(which, "\""));
- }
-
- var prom = _inflight[url];
-
- if (!prom) {
- _inflight[url] = prom = d3_json(url).then(function (result) {
- delete _inflight[url];
-
- if (!result) {
- throw new Error("No data loaded for \"".concat(which, "\""));
- }
-
- _cachedData[which] = result;
- return result;
- })["catch"](function (err) {
- delete _inflight[url];
- throw err;
- });
- }
-
- return prom;
- }; // Accessor for the file map
-
-
- _this.fileMap = function (val) {
- if (!arguments.length) return _fileMap;
- _fileMap = val;
- return _this;
- };
-
- var _assetPath = '';
-
- _this.assetPath = function (val) {
- if (!arguments.length) return _assetPath;
- _assetPath = val;
- return _this;
- };
-
- var _assetMap = {};
-
- _this.assetMap = function (val) {
- if (!arguments.length) return _assetMap;
- _assetMap = val;
- return _this;
- };
-
- _this.asset = function (val) {
- if (/^http(s)?:\/\//i.test(val)) return val;
- var filename = _assetPath + val;
- return _assetMap[filename] || filename;
- };
-
- return _this;
- }
-
- var $findIndex$1 = arrayIteration.findIndex;
-
-
-
- var FIND_INDEX = 'findIndex';
- var SKIPS_HOLES$1 = true;
-
- var USES_TO_LENGTH$b = arrayMethodUsesToLength(FIND_INDEX);
-
- // Shouldn't skip holes
- if (FIND_INDEX in []) Array(1)[FIND_INDEX](function () { SKIPS_HOLES$1 = false; });
-
- // `Array.prototype.findIndex` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.findindex
- _export({ target: 'Array', proto: true, forced: SKIPS_HOLES$1 || !USES_TO_LENGTH$b }, {
- findIndex: function findIndex(callbackfn /* , that = undefined */) {
- return $findIndex$1(this, callbackfn, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- // https://tc39.github.io/ecma262/#sec-array.prototype-@@unscopables
- addToUnscopables(FIND_INDEX);
-
- var $includes$1 = arrayIncludes.includes;
-
-
-
- var USES_TO_LENGTH$c = arrayMethodUsesToLength('indexOf', { ACCESSORS: true, 1: 0 });
-
- // `Array.prototype.includes` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.includes
- _export({ target: 'Array', proto: true, forced: !USES_TO_LENGTH$c }, {
- includes: function includes(el /* , fromIndex = 0 */) {
- return $includes$1(this, el, arguments.length > 1 ? arguments[1] : undefined);
- }
- });
-
- // https://tc39.github.io/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 (e) {
- try {
- regexp[MATCH$2] = false;
- return '/./'[METHOD_NAME](regexp);
- } catch (f) { /* empty */ }
- } return false;
- };
-
- // `String.prototype.includes` method
- // https://tc39.github.io/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;
- }
-
- var getOwnPropertyNames$2 = objectGetOwnPropertyNames.f;
- var getOwnPropertyDescriptor$3 = objectGetOwnPropertyDescriptor.f;
- var defineProperty$a = objectDefineProperty.f;
- var trim$2 = stringTrim.trim;
-
- var NUMBER = 'Number';
- var NativeNumber = global_1[NUMBER];
- var NumberPrototype = NativeNumber.prototype;
-
- // Opera ~12 has broken Object#toString
- var BROKEN_CLASSOF = classofRaw(objectCreate(NumberPrototype)) == NUMBER;
-
- // `ToNumber` abstract operation
- // https://tc39.github.io/ecma262/#sec-tonumber
- var toNumber = function (argument) {
- var it = toPrimitive(argument, false);
- var first, third, radix, maxCode, digits, length, index, code;
- if (typeof it == 'string' && it.length > 2) {
- it = trim$2(it);
- first = it.charCodeAt(0);
- if (first === 43 || first === 45) {
- third = it.charCodeAt(2);
- if (third === 88 || third === 120) return NaN; // Number('+0x1') should be NaN, old V8 fix
- } else if (first === 48) {
- switch (it.charCodeAt(1)) {
- case 66: case 98: radix = 2; maxCode = 49; break; // fast equal of /^0b[01]+$/i
- case 79: case 111: radix = 8; maxCode = 55; break; // fast equal of /^0o[0-7]+$/i
- default: return +it;
- }
- digits = it.slice(2);
- length = digits.length;
- for (index = 0; index < length; index++) {
- code = digits.charCodeAt(index);
- // parseInt parses a string to a first unavailable symbol
- // but ToNumber should return NaN if a string contains unavailable symbols
- if (code < 48 || code > maxCode) return NaN;
- } return parseInt(digits, radix);
- }
- } return +it;
- };
-
- // `Number` constructor
- // https://tc39.github.io/ecma262/#sec-number-constructor
- if (isForced_1(NUMBER, !NativeNumber(' 0o1') || !NativeNumber('0b1') || NativeNumber('+0x1'))) {
- var NumberWrapper = function Number(value) {
- var it = arguments.length < 1 ? 0 : value;
- var dummy = this;
- 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);
- };
- for (var keys$3 = descriptors ? getOwnPropertyNames$2(NativeNumber) : (
- // ES3:
- 'MAX_VALUE,MIN_VALUE,NaN,NEGATIVE_INFINITY,POSITIVE_INFINITY,' +
- // ES2015 (in case, if modules with ES2015 Number statics required before):
- 'EPSILON,isFinite,isInteger,isNaN,isSafeInteger,MAX_SAFE_INTEGER,' +
- 'MIN_SAFE_INTEGER,parseFloat,parseInt,isInteger'
- ).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$a(NumberWrapper, key$1, getOwnPropertyDescriptor$3(NativeNumber, key$1));
- }
- }
- NumberWrapper.prototype = NumberPrototype;
- NumberPrototype.constructor = NumberWrapper;
- redefine(global_1, NUMBER, NumberWrapper);
- }
-
- // `Number.MAX_SAFE_INTEGER` constant
- // https://tc39.github.io/ecma262/#sec-number.max_safe_integer
- _export({ target: 'Number', stat: true }, {
- MAX_SAFE_INTEGER: 0x1FFFFFFFFFFFFF
- });
-
- 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;
- }
-
- 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;
- }
-
- 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);
- }
- }
-
- return arg;
- } // It's an array; check it is a valid representation of a byte
-
-
- if (Array.isArray(arg)) {
- if (!checkInts(arg)) {
- throw new Error('Array contains invalid value: ' + arg);
- }
-
- return new Uint8Array(arg);
- } // Something else, but behaves like an array (maybe a Buffer? Arguments?)
-
-
- if (checkInt(arg.length) && checkInts(arg)) {
- return new Uint8Array(arg);
- }
-
- throw new Error('unsupported array-like object');
- }
-
- function createArray(length) {
- return new Uint8Array(length);
- }
-
- 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);
- }
- }
-
- targetArray.set(sourceArray, targetStart);
- }
-
- var convertUtf8 = function () {
- function toBytes(text) {
- var result = [],
- i = 0;
- text = encodeURI(text);
-
- while (i < text.length) {
- var c = text.charCodeAt(i++); // if it is a % sign, encode the following 2 bytes as a hex value
-
- if (c === 37) {
- result.push(parseInt(text.substr(i, 2), 16));
- i += 2; // otherwise, just the actual byte
- } else {
- result.push(c);
- }
- }
-
- return coerceArray(result);
- }
-
- function fromBytes(bytes) {
- var result = [],
- i = 0;
-
- 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;
- }
- }
-
- return result.join('');
- }
-
- return {
- toBytes: toBytes,
- fromBytes: fromBytes
- };
- }();
-
- var convertHex = function () {
- function toBytes(text) {
- var result = [];
-
- for (var i = 0; i < text.length; i += 2) {
- result.push(parseInt(text.substr(i, 2), 16));
- }
-
- return result;
- } // http://ixti.net/development/javascript/2011/11/11/base64-encodedecode-of-utf8-in-browser-with-js.html
-
-
- var Hex = '0123456789abcdef';
-
- function fromBytes(bytes) {
- var result = [];
-
- for (var i = 0; i < bytes.length; i++) {
- var v = bytes[i];
- result.push(Hex[(v & 0xf0) >> 4] + Hex[v & 0x0f]);
- }
-
- return result.join('');
- }
-
- return {
- toBytes: toBytes,
- fromBytes: fromBytes
- };
- }(); // Number of rounds by keysize
-
-
- var numberOfRounds = {
- 16: 10,
- 24: 12,
- 32: 14
- }; // Round constant words
-
- 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)
-
- 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 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 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
-
- 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];
-
- 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]);
- }
-
- return result;
- }
-
- var AES = function AES(key) {
- if (!(this instanceof AES)) {
- throw Error('AES must be instanitated with `new`');
- }
-
- Object.defineProperty(this, 'key', {
- value: coerceArray(key, true)
- });
-
- this._prepare();
- };
-
- AES.prototype._prepare = function () {
- var rounds = numberOfRounds[this.key.length];
-
- if (rounds == null) {
- throw new Error('invalid key size (must be 16, 24 or 32 bytes)');
- } // encryption round keys
-
-
- this._Ke = []; // decryption round keys
-
- this._Kd = [];
-
- for (var i = 0; i <= rounds; i++) {
- this._Ke.push([0, 0, 0, 0]);
-
- this._Kd.push([0, 0, 0, 0]);
- }
-
- 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
-
- 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 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)
-
- 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)
-
- } else {
- for (var i = 1; i < KC / 2; i++) {
- tk[i] ^= tk[i - 1];
- }
-
- 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;
-
- for (var i = KC / 2 + 1; i < KC; i++) {
- tk[i] ^= tk[i - 1];
- }
- } // copy values into round key arrays
-
-
- var i = 0,
- r,
- c;
-
- 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];
- }
- }
- };
-
- 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)
-
- var t = convertToInt32(plaintext);
-
- for (var i = 0; i < 4; i++) {
- t[i] ^= this._Ke[0][i];
- } // apply round transforms
-
-
- 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];
- }
-
- t = a.slice();
- } // the last round is special
-
-
- var result = createArray(16),
- tt;
-
- 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;
- }
-
- return result;
- };
-
- AES.prototype.decrypt = function (ciphertext) {
- if (ciphertext.length != 16) {
- throw new Error('invalid ciphertext size (must be 16 bytes)');
- }
-
- var rounds = this._Kd.length - 1;
- var a = [0, 0, 0, 0]; // convert plaintext to (ints ^ key)
-
- var t = convertToInt32(ciphertext);
-
- for (var i = 0; i < 4; i++) {
- t[i] ^= this._Kd[0][i];
- } // apply round transforms
-
-
- 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];
- }
-
- 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)
- */
-
-
- var ModeOfOperationECB = function ModeOfOperationECB(key) {
- if (!(this instanceof ModeOfOperationECB)) {
- throw Error('AES must be instanitated with `new`');
- }
-
- this.description = "Electronic Code Block";
- this.name = "ecb";
- this._aes = new AES(key);
- };
-
- ModeOfOperationECB.prototype.encrypt = function (plaintext) {
- plaintext = coerceArray(plaintext);
-
- if (plaintext.length % 16 !== 0) {
- throw new Error('invalid plaintext size (must be multiple of 16 bytes)');
- }
-
- 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 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)');
- }
-
- 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);
- }
-
- return plaintext;
- };
- /**
- * Mode Of Operation - Cipher Block Chaining (CBC)
- */
-
-
- var ModeOfOperationCBC = function ModeOfOperationCBC(key, iv) {
- if (!(this instanceof ModeOfOperationCBC)) {
- throw Error('AES must be instanitated with `new`');
- }
-
- 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);
- };
-
- ModeOfOperationCBC.prototype.encrypt = function (plaintext) {
- plaintext = coerceArray(plaintext);
-
- if (plaintext.length % 16 !== 0) {
- throw new Error('invalid plaintext size (must be multiple of 16 bytes)');
- }
-
- 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);
-
- for (var j = 0; j < 16; j++) {
- block[j] ^= this._lastCipherblock[j];
- }
-
- this._lastCipherblock = this._aes.encrypt(block);
- copyArray(this._lastCipherblock, ciphertext, i);
- }
-
- return ciphertext;
- };
-
- ModeOfOperationCBC.prototype.decrypt = function (ciphertext) {
- ciphertext = coerceArray(ciphertext);
-
- if (ciphertext.length % 16 !== 0) {
- throw new Error('invalid ciphertext size (must be multiple of 16 bytes)');
- }
-
- 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);
- }
-
- 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`');
- }
-
- this.description = "Cipher Feedback";
- this.name = "cfb";
-
- if (!iv) {
- iv = createArray(16);
- } else if (iv.length != 16) {
- throw new Error('invalid initialation vector size (must be 16 size)');
- }
-
- if (!segmentSize) {
- segmentSize = 1;
- }
-
- 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)');
- }
-
- var encrypted = coerceArray(plaintext, true);
- var xorSegment;
-
- for (var i = 0; i < encrypted.length; i += this.segmentSize) {
- xorSegment = this._aes.encrypt(this._shiftRegister);
-
- for (var j = 0; j < this.segmentSize; j++) {
- encrypted[i + j] ^= xorSegment[j];
- } // Shift the register
-
-
- copyArray(this._shiftRegister, this._shiftRegister, 0, this.segmentSize);
- copyArray(encrypted, this._shiftRegister, 16 - this.segmentSize, i, i + this.segmentSize);
- }
-
- return encrypted;
- };
-
- 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;
- }
-
- 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);
- } 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;
- }
-
- 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];
-
- if (padder > 16) {
- throw new Error('PKCS#7 padding byte out of range');
- }
-
- 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');
- }
- }
-
- var result = createArray(length);
- copyArray(data, result, 0, 0, length);
- return result;
- } ///////////////////////
- // Exporting
- // The block cipher
-
-
- 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
- }
- })();
- });
-
- // 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);
- }
- }
-
- return out;
-
- function cleanValue(k, v) {
- function keepSpaces(k) {
- return /_hours|_times|:conditional$/.test(k);
- }
-
- function skip(k) {
- return /^(description|note|fixme)$/.test(k);
- }
-
- 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
- }
-
- return cleaned;
- }
- }
-
- // 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;
- }
-
- function valueConstant() {
- if (this.value !== value) {
- this.value = value;
- }
- }
-
- function valueFunction() {
- var x = value.apply(this, arguments);
-
- if (x === null || x === undefined) {
- delete this.value;
- } else if (this.value !== x) {
- this.value = x;
- }
- }
-
- 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 = {};
-
- function testBindings(d3_event, isCapturing) {
- var didMatch = false;
- var bindings = Object.keys(_keybindings).map(function (id) {
- return _keybindings[id];
- });
- 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;
- }
- }
-
- if (didMatch) return; // then unshifted keybindings
-
- for (i = 0; i < bindings.length; i++) {
- binding = bindings[i];
- if (binding.event.modifiers.shiftKey) continue; // shift
-
- if (!!binding.capture !== isCapturing) continue;
-
- if (matches(d3_event, binding, false)) {
- binding.callback(d3_event);
- break;
- }
- }
-
- function matches(d3_event, binding, testShift) {
- var event = d3_event;
- var isMatch = false;
- var tryKeyCode = true; // Prefer a match on `KeyboardEvent.key`
-
- if (event.key !== undefined) {
- tryKeyCode = event.key.charCodeAt(0) > 255; // outside ISO-Latin-1
-
- isMatch = 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?)
-
-
- if (!isMatch && tryKeyCode) {
- isMatch = event.keyCode === binding.event.keyCode;
- }
-
- if (!isMatch) return false; // test modifier keys
-
- 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;
- }
-
- if (event.metaKey !== binding.event.modifiers.metaKey) return false;
- if (testShift && event.shiftKey !== binding.event.modifiers.shiftKey) return false;
- return true;
- }
- }
-
- function capture(d3_event) {
- testBindings(d3_event, true);
- }
-
- function bubble(d3_event) {
- var tagName = select(d3_event.target).node().tagName;
-
- if (tagName === 'INPUT' || tagName === 'SELECT' || tagName === 'TEXTAREA') {
- return;
- }
-
- testBindings(d3_event, 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()
-
-
- 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;
- }, {});
- }
-
- // 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]);
- }
-
- 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';
- }
-
- 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));
- }
-
- 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;
- }
-
- 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;
- }
- /**
- * 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);
- }
-
- 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';
- }
- /* 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';
- }
- /**
- * 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));
- });
- }
- }
-
- return presetCollection(utilArrayUniq(results));
- };
-
- return _this;
- }
-
- // `presetCategory` builds a `presetCollection` of member presets,
- // decorated with some extra methods for searching and matching geometry
- //
-
- function presetCategory(categoryID, category, all) {
- var _this = Object.assign({}, category); // shallow copy
-
-
- _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];
-
- if (acc.indexOf(geometry) === -1) {
- acc.push(geometry);
- }
- }
-
- return acc;
- }, []);
-
- _this.matchGeometry = function (geom) {
- return _this.geometry.indexOf(geom) >= 0;
- };
-
- _this.matchAllGeometry = function (geometries) {
- return _this.members.collection.some(function (preset) {
- return preset.matchAllGeometry(geometries);
- });
- };
-
- _this.matchScore = function () {
- return -1;
- };
-
- _this.name = function () {
- return _t("presets.categories.".concat(categoryID, ".name"), {
- 'default': categoryID
- });
- };
-
- _this.nameLabel = function () {
- return _t.html("presets.categories.".concat(categoryID, ".name"), {
- 'default': categoryID
- });
- };
-
- _this.terms = function () {
- return [];
- };
-
- return _this;
- }
-
- // `presetField` decorates a given `field` Object
- // with some extra methods for searching and matching geometry
- //
-
- function presetField(fieldID, field) {
- var _this = Object.assign({}, field); // shallow copy
-
-
- _this.id = fieldID; // for use in classes, element ids, css selectors
-
- _this.safeid = utilSafeClassName(fieldID);
-
- _this.matchGeometry = function (geom) {
- return !_this.geometry || _this.geometry.indexOf(geom) !== -1;
- };
-
- _this.matchAllGeometry = function (geometries) {
- return !_this.geometry || geometries.every(function (geom) {
- return _this.geometry.indexOf(geom) !== -1;
- });
- };
-
- _this.t = function (scope, options) {
- return _t("presets.fields.".concat(fieldID, ".").concat(scope), options);
- };
-
- _this.t.html = function (scope, options) {
- return _t.html("presets.fields.".concat(fieldID, ".").concat(scope), options);
- };
-
- _this.title = function () {
- return _this.overrideLabel || _this.t('label', {
- 'default': fieldID
- });
- };
-
- _this.label = function () {
- return _this.overrideLabel || _this.t.html('label', {
- 'default': fieldID
- });
- };
-
- var _placeholder = _this.placeholder;
-
- _this.placeholder = function () {
- return _this.t('placeholder', {
- 'default': _placeholder
- });
- };
-
- _this.originalTerms = (_this.terms || []).join();
-
- _this.terms = function () {
- return _this.t('terms', {
- 'default': _this.originalTerms
- }).toLowerCase().trim().split(/\s*,+\s*/);
- };
-
- _this.increment = _this.type === 'number' ? _this.increment || 1 : undefined;
- return _this;
- }
-
- // `Array.prototype.lastIndexOf` method
- // https://tc39.github.io/ecma262/#sec-array.prototype.lastindexof
- _export({ target: 'Array', proto: true, forced: arrayLastIndexOf !== [].lastIndexOf }, {
- lastIndexOf: arrayLastIndexOf
- });
-
- // `presetPreset` decorates a given `preset` Object
- // with some extra methods for searching and matching geometry
- //
-
- function presetPreset(presetID, preset, addable, allFields, allPresets) {
- allFields = allFields || {};
- allPresets = allPresets || {};
-
- var _this = Object.assign({}, preset); // shallow copy
-
-
- var _addable = addable || false;
-
- var _resolvedFields; // cache
-
-
- var _resolvedMoreFields; // cache
-
-
- _this.id = presetID;
- _this.safeid = utilSafeClassName(presetID); // for use in css classes, selectors, element ids
-
- _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 || [];
-
- _this.fields = function () {
- return _resolvedFields || (_resolvedFields = resolve('fields'));
- };
-
- _this.moreFields = function () {
- return _resolvedMoreFields || (_resolvedMoreFields = resolve('moreFields'));
- };
-
- _this.resetFields = function () {
- return _resolvedFields = _resolvedMoreFields = null;
- };
-
- _this.tags = _this.tags || {};
- _this.addTags = _this.addTags || _this.tags;
- _this.removeTags = _this.removeTags || _this.addTags;
- _this.geometry = _this.geometry || [];
-
- _this.matchGeometry = function (geom) {
- return _this.geometry.indexOf(geom) >= 0;
- };
-
- _this.matchAllGeometry = function (geoms) {
- return geoms.every(_this.matchGeometry);
- };
-
- _this.matchScore = function (entityTags) {
- var tags = _this.tags;
- var seen = {};
- var score = 0; // match on tags
-
- for (var k in tags) {
- seen[k] = true;
-
- 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
-
-
- var addTags = _this.addTags;
-
- for (var _k in addTags) {
- if (!seen[_k] && entityTags[_k] === addTags[_k]) {
- score += _this.originalScore;
- }
- }
-
- return score;
- };
-
- _this.t = function (scope, options) {
- var textID = "presets.presets.".concat(presetID, ".").concat(scope);
- return _t(textID, options);
- };
-
- _this.t.html = function (scope, options) {
- var textID = "presets.presets.".concat(presetID, ".").concat(scope);
- return _t.html(textID, options);
- };
-
- _this.name = function () {
- return _this.t('name', {
- 'default': _this.originalName
- });
- };
-
- _this.nameLabel = function () {
- return _this.t.html('name', {
- 'default': _this.originalName
- });
- };
-
- _this.subtitle = function () {
- if (_this.suggestion) {
- var path = presetID.split('/');
- path.pop(); // remove brand name
-
- return _t('presets.presets.' + path.join('/') + '.name');
- }
-
- return null;
- };
-
- _this.subtitleLabel = function () {
- if (_this.suggestion) {
- var path = presetID.split('/');
- path.pop(); // remove brand name
-
- return _t.html('presets.presets.' + path.join('/') + '.name');
- }
-
- return null;
- };
-
- _this.terms = function () {
- return _this.t('terms', {
- 'default': _this.originalTerms
- }).toLowerCase().trim().split(/\s*,+\s*/);
- };
-
- _this.isFallback = function () {
- var tagCount = Object.keys(_this.tags).length;
- return tagCount === 0 || tagCount === 1 && _this.tags.hasOwnProperty('area');
- };
-
- _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['brand:wikidata'] || _this.tags['operator:wikidata'];
-
- if (qid) {
- return {
- qid: qid
- };
- } // Lookup documentation on OSM Wikibase...
-
-
- var key = _this.originalReference.key || Object.keys(utilObjectOmit(_this.tags, 'name'))[0];
- var value = _this.originalReference.value || _this.tags[key];
-
- if (value === '*') {
- return {
- key: key
- };
- } else {
- return {
- key: key,
- value: value
- };
- }
- };
-
- _this.unsetTags = function (tags, geometry, skipFieldDefaults) {
- tags = utilObjectOmit(tags, Object.keys(_this.removeTags));
-
- if (geometry && !skipFieldDefaults) {
- _this.fields().forEach(function (field) {
- if (field.matchGeometry(geometry) && field.key && field["default"] === tags[field.key]) {
- delete tags[field.key];
- }
- });
- }
-
- delete tags.area;
- return tags;
- };
-
- _this.setTags = function (tags, geometry, skipFieldDefaults) {
- var addTags = _this.addTags;
- tags = Object.assign({}, tags); // shallow copy
-
- 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 (!addTags.hasOwnProperty('area')) {
- delete tags.area;
-
- if (geometry === 'area') {
- var needsAreaTag = true;
-
- if (_this.geometry.indexOf('line') === -1) {
- for (var _k2 in addTags) {
- if (_k2 in osmAreaKeys) {
- needsAreaTag = false;
- break;
- }
- }
- }
-
- if (needsAreaTag) {
- tags.area = 'yes';
- }
- }
- }
-
- if (geometry && !skipFieldDefaults) {
- _this.fields().forEach(function (field) {
- if (field.matchGeometry(geometry) && field.key && !tags[field.key] && field["default"]) {
- tags[field.key] = field["default"];
- }
- });
- }
-
- return tags;
- }; // For a preset without fields, use the fields of the parent preset.
- // Replace {preset} placeholders with the fields of the specified presets.
-
-
- function resolve(which) {
- var fieldIDs = which === 'fields' ? _this.originalFields : _this.originalMoreFields;
- var resolved = [];
- fieldIDs.forEach(function (fieldID) {
- var match = fieldID.match(/\{(.*)\}/);
-
- 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
-
- if (!resolved.length) {
- var endIndex = _this.id.lastIndexOf('/');
-
- var parentID = endIndex && _this.id.substring(0, endIndex);
-
- if (parentID) {
- resolved = inheritFields(parentID, which);
- }
- }
-
- return utilArrayUniq(resolved); // returns an array of fields to inherit from the given presetID, if found
-
- function inheritFields(presetID, which) {
- var parent = allPresets[presetID];
- if (!parent) return [];
-
- 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=*`
-
-
- 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 _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
-
- var _recents;
-
- var _favorites; // Index of presets by (geometry, tag key).
-
-
- var _geometryIndex = {
- point: {},
- vertex: {},
- line: {},
- area: {},
- relation: {}
- };
-
- var _loadPromise;
-
- _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());
- });
- };
-
- _this.merge = function (d) {
- // Merge Fields
- if (d.fields) {
- Object.keys(d.fields).forEach(function (fieldID) {
- var f = d.fields[fieldID];
-
- if (f) {
- // add or replace
- _fields[fieldID] = presetField(fieldID, f);
- } else {
- // remove
- delete _fields[fieldID];
- }
- });
- } // Merge Presets
-
-
- if (d.presets) {
- Object.keys(d.presets).forEach(function (presetID) {
- var p = d.presets[presetID];
-
- if (p) {
- // add or replace
- var isAddable = !_addablePresetIDs || _addablePresetIDs.has(presetID);
-
- _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
-
-
- _this.collection = Object.values(_presets).concat(Object.values(_categories)); // Merge Categories
-
- if (d.categories) {
- Object.keys(d.categories).forEach(function (categoryID) {
- var c = d.categories[categoryID];
-
- if (c) {
- // add or replace
- _categories[categoryID] = presetCategory(categoryID, c, _this);
- } else {
- // remove
- delete _categories[categoryID];
- }
- });
- } // Rebuild _this.collection after loading categories
-
-
- _this.collection = Object.values(_presets).concat(Object.values(_categories)); // Merge Defaults
-
- if (d.defaults) {
- Object.keys(d.defaults).forEach(function (geometry) {
- var def = d.defaults[geometry];
-
- 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
-
-
- _universal = Object.values(_fields).filter(function (field) {
- return field.universal;
- }); // Reset all the preset fields - they'll need to be resolved again
-
- Object.values(_presets).forEach(function (preset) {
- return preset.resetFields();
- }); // Rebuild geometry index
-
- _geometryIndex = {
- point: {},
- vertex: {},
- line: {},
- area: {},
- relation: {}
- };
-
- _this.collection.forEach(function (preset) {
- (preset.geometry || []).forEach(function (geometry) {
- var g = _geometryIndex[geometry];
-
- for (var key in preset.tags) {
- (g[key] = g[key] || []).push(preset);
- }
- });
- });
-
- 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';
- }
-
- return _this.matchTags(entity.tags, geometry);
- });
- };
-
- _this.matchTags = function (tags, geometry) {
- var geometryMatches = _geometryIndex[geometry];
- var address;
- var best = -1;
- var match;
-
- 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 keyMatches = geometryMatches[k];
- if (!keyMatches) continue;
-
- for (var i = 0; i < keyMatches.length; i++) {
- var score = keyMatches[i].matchScore(tags);
-
- if (score > best) {
- best = score;
- match = keyMatches[i];
- }
- }
- }
-
- if (address && (!match || match.isFallback())) {
- match = address;
- }
-
- return match || _this.fallback(geometry);
- };
-
- _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;
- });
- }; // 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.
-
-
- _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
-
- var presets = _this.collection.filter(function (p) {
- return !p.suggestion && !p.replacement;
- }); // keeplist
-
-
- presets.forEach(function (p) {
- var keys = p.tags && Object.keys(p.tags);
- var key = keys && keys.length && keys[0]; // pick the first tag
-
- if (!key) return;
- if (ignore.indexOf(key) !== -1) return;
-
- if (p.geometry.indexOf('area') !== -1) {
- // probably an area..
- areaKeys[key] = areaKeys[key] || {};
- }
- }); // discardlist
-
- presets.forEach(function (p) {
- var key;
-
- 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;
- };
-
- _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
-
- var keys = d.tags && Object.keys(d.tags);
- var key = keys && keys.length && keys[0]; // pick the first tag
-
- if (!key) return pointTags; // if this can be a point
-
- if (d.geometry.indexOf('point') !== -1) {
- pointTags[key] = pointTags[key] || {};
- pointTags[key][d.tags[key]] = true;
- }
-
- 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 keys = d.tags && Object.keys(d.tags);
- var key = keys && keys.length && keys[0]; // pick the first tag
-
- if (!key) return vertexTags; // if this can be a vertex
-
- if (d.geometry.indexOf('vertex') !== -1) {
- vertexTags[key] = vertexTags[key] || {};
- vertexTags[key][d.tags[key]] = true;
- }
-
- return vertexTags;
- }, {});
- };
-
- _this.field = function (id) {
- return _fields[id];
- };
-
- _this.universal = function () {
- return _universal;
- };
-
- _this.defaults = function (geometry, n, startWithRecents) {
- var recents = [];
-
- if (startWithRecents) {
- recents = _this.recent().matchGeometry(geometry).collection.slice(0, 4);
- }
-
- var defaults;
-
- if (_addablePresetIDs) {
- defaults = Array.from(_addablePresetIDs).map(function (id) {
- var preset = _this.item(id);
-
- if (preset && preset.matchGeometry(geometry)) return preset;
- return null;
- }).filter(Boolean);
- } else {
- defaults = _defaults[geometry].collection.concat(_this.fallback(geometry));
- }
-
- return presetCollection(utilArrayUniq(recents.concat(defaults)).slice(0, n - 1));
- }; // pass a Set of addable preset ids
-
-
- _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 _this;
- };
-
- _this.recent = function () {
- return presetCollection(utilArrayUniq(_this.getRecents().map(function (d) {
- return d.preset;
- })));
- };
-
- function RibbonItem(preset, source) {
- var item = {};
- item.preset = preset;
- item.source = source;
-
- item.isFavorite = function () {
- return item.source === 'favorite';
- };
-
- item.isRecent = function () {
- return item.source === 'recent';
- };
-
- item.matches = function (preset) {
- return item.preset.id === preset.id;
- };
-
- item.minified = function () {
- return {
- pID: item.preset.id
- };
- };
-
- return item;
- }
-
- function ribbonItemForMinified(d, source) {
- if (d && d.pID) {
- var preset = _this.item(d.pID);
-
- if (!preset) return null;
- return RibbonItem(preset, source);
- }
-
- return null;
- }
-
- _this.getGenericRibbonItems = function () {
- return ['point', 'line', 'area'].map(function (id) {
- return RibbonItem(_this.item(id), 'generic');
- });
- };
-
- _this.getAddable = function () {
- if (!_addablePresetIDs) return [];
- return _addablePresetIDs.map(function (id) {
- var preset = _this.item(id);
-
- 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');
- }
-
- _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;
- }, []);
- }
-
- return _recents;
- };
-
- _this.addRecent = function (preset, besidePreset, after) {
- var recents = _this.getRecents();
-
- 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);
- };
-
- _this.removeRecent = function (preset) {
- var item = _this.recentMatching(preset);
-
- if (item) {
- var items = _this.getRecents();
-
- items.splice(items.indexOf(item), 1);
- setRecents(items);
- }
- };
-
- _this.recentMatching = function (preset) {
- var items = _this.getRecents();
-
- for (var i in items) {
- if (items[i].matches(preset)) {
- return items[i];
- }
- }
-
- return 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;
- };
-
- _this.moveRecent = function (item, beforeItem) {
- var recents = _this.getRecents();
-
- var fromIndex = recents.indexOf(item);
- var toIndex = recents.indexOf(beforeItem);
-
- var items = _this.moveItem(recents, fromIndex, toIndex);
-
- if (items) setRecents(items);
- };
-
- _this.setMostRecent = function (preset) {
- if (preset.searchable === false) return;
-
- var items = _this.getRecents();
-
- var item = _this.recentMatching(preset);
-
- if (item) {
- items.splice(items.indexOf(item), 1);
- } else {
- item = RibbonItem(preset, 'recent');
- } // remove the last recent (first in, first out)
-
-
- while (items.length >= MAXRECENTS) {
- items.pop();
- } // prepend array
-
-
- items.unshift(item);
- setRecents(items);
- };
-
- function setFavorites(items) {
- _favorites = items;
- var minifiedItems = items.map(function (d) {
- return d.minified();
- });
- corePreferences('preset_favorites', JSON.stringify(minifiedItems)); // call update
-
- dispatch$1.call('favoritePreset');
- }
-
- _this.addFavorite = function (preset, besidePreset, after) {
- var favorites = _this.getFavorites();
-
- var beforeItem = _this.favoriteMatching(besidePreset);
-
- var toIndex = favorites.indexOf(beforeItem);
- if (after) toIndex += 1;
- var newItem = RibbonItem(preset, 'favorite');
- favorites.splice(toIndex, 0, newItem);
- setFavorites(favorites);
- };
-
- _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
-
-
- favs.push(RibbonItem(preset, 'favorite'));
- }
-
- setFavorites(favs);
- };
-
- _this.removeFavorite = function (preset) {
- var item = _this.favoriteMatching(preset);
-
- if (item) {
- var items = _this.getFavorites();
-
- items.splice(items.indexOf(item), 1);
- setFavorites(items);
- }
- };
-
- _this.getFavorites = function () {
- if (!_favorites) {
- // fetch from local storage
- var rawFavorites = JSON.parse(corePreferences('preset_favorites'));
-
- if (!rawFavorites) {
- rawFavorites = [];
- corePreferences('preset_favorites', JSON.stringify(rawFavorites));
- }
-
- _favorites = rawFavorites.reduce(function (output, d) {
- var item = ribbonItemForMinified(d, 'favorite');
- if (item && item.preset.addable()) output.push(item);
- return output;
- }, []);
- }
-
- return _favorites;
- };
-
- _this.favoriteMatching = function (preset) {
- var favs = _this.getFavorites();
-
- for (var index in favs) {
- if (favs[index].matches(preset)) {
- return favs[index];
- }
- }
-
- return null;
- };
-
- 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;
-
- for (var i = 0; i < array.length; i++) {
- val = array[i];
- entity = typeof val === 'string' ? graph.hasEntity(val) : val;
-
- 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];
-
- if ((oldVal || oldVal === '') && (newVal === undefined || newVal !== oldVal)) {
- tagDiff.push({
- type: '-',
- key: k,
- oldVal: oldVal,
- newVal: newVal,
- display: '- ' + k + '=' + oldVal
- });
- }
-
- 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));
-
- 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 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 utilEntityAndDeepMemberIDs(ids, graph) {
- var seen = new Set();
- ids.forEach(collectDeepDescendants);
- return Array.from(seen);
-
- 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 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));
-
- function collectDeepDescendants(id) {
- if (seen.has(id)) return;
- seen.add(id);
-
- if (!idsSet.has(id)) {
- returners.add(id);
- }
-
- 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 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
-
- function utilGetAllNodes(ids, graph) {
- var seen = new Set();
- var nodes = new Set();
- ids.forEach(collectNodes);
- return Array.from(nodes);
-
- function collectNodes(id) {
- if (seen.has(id)) return;
- seen.add(id);
- var entity = graph.hasEntity(id);
- if (!entity) return;
-
- 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 || '';
- var network = entity.tags.cycle_network || entity.tags.network;
-
- if (!name && entity.tags.ref) {
- name = entity.tags.ref;
-
- if (network) {
- name = network + ' ' + name;
- }
- }
-
- return name;
- }
- function utilDisplayNameForPath(entity) {
- var name = utilDisplayName(entity);
- var isFirefox = utilDetect().browser.toLowerCase().indexOf('firefox') > -1;
-
- if (!isFirefox && name && rtlRegex.test(name)) {
- name = fixRTLTextForSvg(name);
- }
-
- 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);
-
- 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);
-
- if (preset && preset.name()) {
- // use the preset name if there is a match
- return preset.name();
- } // fallback to the display type (node/way/relation)
-
-
- 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 ]
- // }
-
- 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
-
- 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`
-
- 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);
- }
- }
- }
-
- var tagHash = key + '=' + value;
- if (!tagCounts[tagHash]) tagCounts[tagHash] = 0;
- tagCounts[tagHash] += 1;
- });
- });
-
- for (var key in tags) {
- if (!Array.isArray(tags[key])) continue; // sort values by frequency then alphabetically
-
- tags[key] = tags[key].sort(function (val1, val2) {
- var key = key; // capture
-
- var count2 = tagCounts[key + '=' + val2];
- var count1 = tagCounts[key + '=' + val1];
-
- if (count2 !== count1) {
- return count2 - count1;
- }
-
- if (val2 && val1) {
- return val1.localeCompare(val2);
- }
-
- return val1 ? 1 : -1;
- });
- }
-
- return tags;
- }
- function utilStringQs(str) {
- var i = 0; // advance past any leading '?' or '#' characters
-
- while (i < str.length && (str[i] === '?' || str[i] === '#')) {
- i++;
- }
-
- str = str.slice(i);
- return str.split('&').reduce(function (obj, pair) {
- var parts = pair.split('=');
-
- if (parts.length === 2) {
- obj[parts[0]] = null === parts[1] ? '' : decodeURIComponent(parts[1]);
- }
-
- 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);
- }
-
- 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;
- }
- }
-
- return false;
- }
- 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();
- }
-
- while (++i < n) {
- if (prefixes[i] + property in s) {
- return '-' + prefixes[i].toLowerCase() + property.replace(/([A-Z])/g, '-$1').toLowerCase();
- }
- }
-
- return false;
- }
- 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;
-
- for (i = 0; i <= b.length; i++) {
- matrix[i] = [i];
- }
-
- for (j = 0; j <= a.length; j++) {
- matrix[0][j] = j;
- }
-
- 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 matrix[b.length][a.length];
- } // a d3.mouse-alike which
- // 1. Only works on HTML elements, not SVG
- // 2. Does not cause style recalculation
-
- 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]
-
- function utilWrap(index, length) {
- if (index < 0) {
- index += Math.ceil(-index / length) * length;
- }
-
- 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
- */
-
- 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;
-
- if (str.length === 0) {
- return hash;
- }
-
- for (var i = 0; i < str.length; i++) {
- var _char = str.charCodeAt(i);
-
- hash = (hash << 5) - hash + _char;
- hash = hash & hash; // Convert to 32bit integer
- }
-
- return hash;
- } // Returns version of `str` with all runs of special characters replaced by `_`;
- // suitable for HTML ids, classes, selectors, etc.
-
- 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.
-
- 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.
-
- function utilUnicodeCharsTruncated(str, limit) {
- return Array.from(str).slice(0, limit).join('');
- }
-
- function osmEntity(attrs) {
- // For prototypal inheritance.
- if (this instanceof osmEntity) return; // Create the appropriate subtype.
-
- 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).
-
-
- return new osmEntity().initialize(arguments);
- }
-
- osmEntity.id = function (type) {
- return osmEntity.id.fromOSM(type, osmEntity.id.next[type]--);
- };
-
- osmEntity.id.next = {
- changeset: -1,
- node: -1,
- way: -1,
- relation: -1
- };
-
- osmEntity.id.fromOSM = function (type, id) {
- return type[0] + id;
- };
-
- osmEntity.id.toOSM = function (id) {
- return id.slice(1);
- };
-
- 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().
-
-
- osmEntity.key = function (entity) {
- return entity.id + 'v' + (entity.v || 0);
- };
-
- var _deprecatedTagValuesByKey;
-
- osmEntity.deprecatedTagValuesByKey = function (dataDeprecated) {
- if (!_deprecatedTagValuesByKey) {
- _deprecatedTagValuesByKey = {};
- dataDeprecated.forEach(function (d) {
- var oldKeys = Object.keys(d.old);
-
- 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);
- }
- }
- }
- });
- }
-
- return _deprecatedTagValuesByKey;
- };
-
- osmEntity.prototype = {
- tags: {},
- initialize: function initialize(sources) {
- for (var i = 0; i < sources.length; ++i) {
- var source = sources[i];
-
- 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 (!this.id && this.type) {
- this.id = osmEntity.id(this.type);
- }
-
- if (!this.hasOwnProperty('visible')) {
- this.visible = true;
- }
-
- 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;
-
- for (var k in tags) {
- var t1 = merged[k];
- var t2 = tags[k];
-
- 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
-
- 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
-
- if (hasExistingValues) return;
- }
-
- 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;
- }
- }
- }
-
- return false;
- });
-
- if (matchesDeprecatedTags) {
- deprecated.push(d);
- }
- });
- return deprecated;
- }
- };
-
- 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
-
- 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
- };
- }
-
- 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;
-
- default:
- count = isOneWay ? 1 : 2;
- break;
- }
-
- 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 {
- forward: forward,
- backward: backward,
- bothways: bothways
- };
- }
-
- 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 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 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 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 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;
- });
- }
-
- 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 i = 0; i < this.nodes.length; i++) {
- var node = resolver.hasEntity(this.nodes[i]);
-
- if (node) {
- extent._extend(node.extent());
- }
- }
-
- 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..
-
-
- 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]];
-
- if (key === 'highway') {
- var laneCount = this.tags.lanes && parseInt(this.tags.lanes, 10);
- if (!laneCount) laneCount = this.isOneWay() ? 1 : 2;
- return width * laneCount;
- }
-
- return width;
- }
- }
-
- return null;
- },
- isOneWay: function isOneWay() {
- // explicit oneway tag..
- var values = {
- 'yes': true,
- '1': true,
- '-1': true,
- 'reversible': true,
- 'alternating': true,
- 'no': false,
- '0': false
- };
-
- if (values[this.tags.oneway] !== undefined) {
- return values[this.tags.oneway];
- } // implied oneway tag..
-
-
- 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 (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;
- }
-
- 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;
-
- 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;
- }
-
- prev = curr;
- }
-
- 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;
- }
- }
-
- 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])]]);
- }
-
- return graph["transient"](this, 'segments', function () {
- var segments = [];
-
- 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
- });
- }
-
- 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;
- }
-
- 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;
-
- if (index === undefined) {
- index = max;
- }
-
- 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..
-
-
- if (isClosed) {
- var connector = this.first(); // leading connectors..
-
- var i = 1;
-
- while (i < nodes.length && nodes.length > 2 && nodes[i] === connector) {
- nodes.splice(i, 1);
- if (index > i) index--;
- } // trailing connectors..
-
-
- i = nodes.length - 1;
-
- while (i > 0 && nodes.length > 1 && nodes[i] === connector) {
- nodes.splice(i, 1);
- if (index > i) index--;
- i = nodes.length - 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 (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 (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 (isClosed) {
- var connector = this.first(); // leading connectors..
-
- var i = 1;
-
- while (i < nodes.length && nodes.length > 2 && nodes[i] === connector) {
- nodes.splice(i, 1);
- if (index > i) index--;
- } // trailing connectors..
-
-
- i = nodes.length - 1;
-
- while (i > 0 && nodes.length > 1 && nodes[i] === connector) {
- nodes.splice(i, 1);
- if (index === i) index = 0; // update leading connector instead
-
- i = nodes.length - 1;
- }
- }
-
- nodes.splice(index, 1, id);
- nodes = nodes.filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
-
- if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
- nodes.push(nodes[0]);
- }
-
- 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;
- }
- }
-
- nodes = nodes.filter(noRepeatNodes); // If the way was closed before, append a connector node to keep it closed..
-
- if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
- nodes.push(nodes[0]);
- }
-
- 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..
-
- if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
- nodes.push(nodes[0]);
- }
-
- 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)
- }
- };
-
- 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 (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);
- }
-
- 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);
- }
-
- return isNaN(area) ? 0 : area;
- });
- }
- }); // Filter function to eliminate consecutive duplicates.
-
- function noRepeatNodes(node, i, arr) {
- return i === 0 || node !== arr[i - 1];
- }
-
- //
- // 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;
- }
-
- var outerMember;
-
- for (var memberIndex in entity.members) {
- var member = entity.members[memberIndex];
-
- 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 (Object.keys(outerMember.tags).filter(osmIsInterestingTag).length === 0) {
- return false;
- }
- }
- }
-
- 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;
- 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;
-
- 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 (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;
-
- for (var i = 0; i < members.length; i++) {
- member = members[i];
-
- if (!member.role || member.role === 'outer') {
- if (outerMember) return false; // Not a simple multipolygon
-
- outerMember = member;
- }
- }
-
- 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.
- //
-
- 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
-
-
- 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)
-
- var i;
- var joinAsMembers = true;
-
- for (i = 0; i < toJoin.length; i++) {
- if (toJoin[i] instanceof osmWay) {
- joinAsMembers = false;
- break;
- }
- }
-
- var sequences = [];
- sequences.actions = [];
-
- while (toJoin.length) {
- // start a new sequence
- var item = toJoin.shift();
- var currWays = [item];
- var currNodes = resolve(item).slice(); // add to it
-
- 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 (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 (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
-
- nodes = nodes.slice(1);
- break;
- } else if (nodes[nodes.length - 1] === end) {
- fn = currNodes.push; // join to end
-
- nodes = nodes.slice(0, -1).reverse();
- item = reverse(item);
- break;
- } else if (nodes[nodes.length - 1] === start) {
- fn = currNodes.unshift; // join to beginning
-
- nodes = nodes.slice(0, -1);
- break;
- } else if (nodes[0] === start) {
- fn = currNodes.unshift; // join to beginning
-
- nodes = nodes.slice(1).reverse();
- item = reverse(item);
- break;
- } else {
- fn = nodes = null;
- }
- }
-
- if (!nodes) {
- // couldn't find a joinable way/member
- break;
- }
-
- fn.apply(currWays, [item]);
- fn.apply(currNodes, nodes);
- toJoin.splice(i, 1);
- }
-
- currWays.nodes = currNodes;
- sequences.push(currWays);
- }
-
- return sequences;
- }
-
- 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 isPTv2 = /stop|platform/.test(member.role);
-
- 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;
- }
-
- graph = graph.replace(relation.addMember(member, memberIndex));
- }
-
- return graph;
- }; // Add a way member into the relation "wherever it makes sense".
- // In this situation we were not supplied a memberIndex.
-
- function addWayMember(relation, graph) {
- var groups, tempWay, item, i, j, k; // remove PTv2 stops and platforms before doing anything.
-
- var PTv2members = [];
- var members = [];
-
- for (i = 0; i < relation.members.length; i++) {
- var m = relation.members[i];
-
- if (/stop|platform/.test(m.role)) {
- PTv2members.push(m);
- } else {
- members.push(m);
- }
- }
-
- relation = relation.update({
- members: members
- });
-
- 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);
- }
-
- 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
-
- for (j = 0; j < members.length; j++) {
- if (members[j].index === startIndex) {
- break;
- }
- } // k = each member in segment
-
-
- 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
-
-
- if (k > 0) {
- if (j + k >= members.length || item.index !== members[j + k].index) {
- moveMember(members, item.index, j + k);
- }
- }
-
- nodes.splice(0, way.nodes.length - 1);
- }
- }
-
- if (tempWay) {
- graph = graph.remove(tempWay);
- } // Final pass: skip dead items, split pairs, remove index properties
-
-
- var wayMembers = [];
-
- for (i = 0; i < members.length; i++) {
- item = members[i];
- if (item.index === -1) continue;
-
- 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
-
-
- 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 moveMember(arr, findIndex, toIndex) {
- var i;
-
- for (i = 0; i < arr.length; i++) {
- if (arr[i].index === findIndex) {
- break;
- }
- }
-
- var item = Object.assign({}, arr[i]); // shallow copy
-
- arr[i].index = -1; // mark as dead
-
- 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
-
-
- function withIndex(arr) {
- var result = new Array(arr.length);
-
- for (var i = 0; i < arr.length; i++) {
- result[i] = Object.assign({}, arr[i]); // shallow copy
-
- result[i].index = i;
- }
-
- return result;
- }
- }
- }
-
- 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.
-
- return;
- }
- }
- });
- 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));
- };
- }
-
- function actionChangeMember(relationId, member, memberIndex) {
- return function (graph) {
- return graph.replace(graph.entity(relationId).updateMember(member, memberIndex));
- };
- }
-
- 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
- }));
- };
- }
-
- function actionChangeTags(entityId, tags) {
- return function (graph) {
- var entity = graph.entity(entityId);
- return graph.replace(entity.update({
- tags: tags
- }));
- };
- }
-
- 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?
-
- 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
-
- var re = /:direction$/i;
- var keys = Object.keys(this.tags);
-
- for (i = 0; i < keys.length; i++) {
- if (re.test(keys[i])) {
- val = this.tags[keys[i]].toLowerCase();
- break;
- }
- }
- }
-
- 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 (v !== '' && !isNaN(+v)) {
- results.push(+v);
- return;
- } // string direction - inspect parent ways
-
-
- 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;
-
- 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
- }
-
- 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);
-
- 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 (way.isClosed()) {
- nodes.pop();
- } // ignore connecting node if closed
- // return true if vertex appears multiple times (way is self intersecting)
-
-
- return nodes.indexOf(this.id) !== nodes.lastIndexOf(this.id);
- }
-
- 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
- };
- }
- });
-
- function actionCircularize(wayId, projection, maxAngle) {
- maxAngle = (maxAngle || 20) * Math.PI / 180;
-
- 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;
- });
-
- 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 (!keyNodes.length) {
- keyNodes = [nodes[0]];
- keyPoints = [points[0]];
- }
-
- 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 (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;
-
- if (indexRange < 0) {
- indexRange += nodes.length;
- } // position this key node
-
-
- 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
-
- 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 (totalAngle * sign > 0) {
- totalAngle = -sign * (2 * Math.PI - Math.abs(totalAngle));
- }
-
- do {
- numberNewPoints++;
- eachAngle = totalAngle / (indexRange + numberNewPoints);
- } while (Math.abs(eachAngle) > maxAngle); // move existing nodes
-
-
- 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
-
-
- 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 min = Infinity;
-
- for (var nodeId in nearNodes) {
- var nearAngle = nearNodes[nodeId];
- var dist = Math.abs(nearAngle - angle);
-
- if (dist < min) {
- min = dist;
- origNode = origNodes[nodeId];
- }
- }
-
- 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..
-
-
- if (indexRange === 1 && inBetweenNodes.length) {
- var startIndex1 = way.nodes.lastIndexOf(startNode.id);
- var endIndex1 = way.nodes.lastIndexOf(endNode.id);
- var wayDirection1 = endIndex1 - startIndex1;
-
- if (wayDirection1 < -1) {
- wayDirection1 = 1;
- }
-
- var parentWays = graph.parentWays(keyNodes[i]);
-
- for (j = 0; j < parentWays.length; j++) {
- var sharedWay = parentWays[j];
- if (sharedWay === way) continue;
-
- 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;
-
- if (wayDirection2 < -1) {
- wayDirection2 = 1;
- }
-
- if (wayDirection1 !== wayDirection2) {
- inBetweenNodes.reverse();
- insertAt = startIndex2;
- }
-
- for (k = 0; k < inBetweenNodes.length; k++) {
- sharedWay = sharedWay.addNode(inBetweenNodes[k], insertAt + k);
- }
-
- graph = graph.replace(sharedWay);
- }
- }
- }
- } // update the way to have all the new nodes
-
-
- ids = nodes.map(function (n) {
- return n.id;
- });
- ids.push(ids[0]);
- way = way.update({
- nodes: ids
- });
- graph = graph.replace(way);
- return graph;
- };
-
- 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();
- }
-
- 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;
-
- if (indexRange < 0) {
- indexRange += nodes.length;
- } // move interior nodes to the surface of the convex hull..
-
-
- 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;
- };
-
- action.disabled = function (graph) {
- if (!graph.entity(wayId).isClosed()) {
- return 'not_closed';
- } //disable when already circular
-
-
- 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;
-
- 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
-
- 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 (diff > 0.05 * radius) {
- return false;
- }
- } //check if central angles are smaller than maxAngle
-
-
- 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;
- }
-
- if (angle > Math.PI) {
- angle = 2 * Math.PI - angle;
- }
-
- if (angle > maxAngle + epsilonAngle) {
- return false;
- }
- }
-
- return 'already_circular';
- };
-
- action.transitionable = true;
- return action;
- }
-
- 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 (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
-
- return !node.hasInterestingTags();
- }
-
- var action = function action(graph) {
- var way = graph.entity(wayID);
- graph.parentRelations(way).forEach(function (parent) {
- parent = parent.removeMembersWithID(wayID);
- graph = graph.replace(parent);
-
- 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 (canDeleteNode(node, graph)) {
- graph = graph.remove(node);
- }
- });
- return graph.remove(way);
- };
-
- return action;
- }
-
- function actionDeleteMultiple(ids) {
- var actions = {
- way: actionDeleteWay,
- node: actionDeleteNode,
- relation: actionDeleteRelation
- };
-
- 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;
- };
-
- return action;
- }
-
- function actionDeleteRelation(relationID, allowUntaggedMembers) {
- function canDeleteEntity(entity, graph) {
- return !graph.parentWays(entity).length && !graph.parentRelations(entity).length && !entity.hasInterestingTags() && !allowUntaggedMembers;
- }
-
- var action = function action(graph) {
- var relation = graph.entity(relationID);
- graph.parentRelations(relation).forEach(function (parent) {
- parent = parent.removeMembersWithID(relationID);
- graph = graph.replace(parent);
-
- 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);
-
- if (canDeleteEntity(entity, graph)) {
- graph = actionDeleteMultiple([memberID])(graph);
- }
- });
- return graph.remove(relation);
- };
-
- return action;
- }
-
- 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 (parent.isDegenerate()) {
- graph = actionDeleteWay(parent.id)(graph);
- }
- });
- graph.parentRelations(node).forEach(function (parent) {
- parent = parent.removeMembersWithID(nodeId);
- graph = graph.replace(parent);
-
- if (parent.isDegenerate()) {
- graph = actionDeleteRelation(parent.id)(graph);
- }
- });
- return graph.remove(node);
- };
-
- return action;
- }
-
- //
- // 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
- //
-
- 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.
-
-
- for (i = 0; i < nodeIDs.length; i++) {
- node = graph.entity(nodeIDs[i]);
- if (node.id === survivor.id) continue;
- parents = graph.parentWays(node);
-
- for (j = 0; j < parents.length; j++) {
- graph = graph.replace(parents[j].replaceNode(node.id, survivor.id));
- }
-
- parents = graph.parentRelations(node);
-
- for (j = 0; j < parents.length; j++) {
- graph = graph.replace(parents[j].replaceMember(node, survivor));
- }
-
- survivor = survivor.mergeTags(node.tags);
- graph = actionDeleteNode(node.id)(graph);
- }
-
- graph = graph.replace(survivor); // find and delete any degenerate ways created by connecting adjacent vertices
-
- parents = graph.parentWays(survivor);
-
- for (i = 0; i < parents.length; i++) {
- if (parents[i].isDegenerate()) {
- graph = actionDeleteWay(parents[i].id)(graph);
- }
- }
-
- return graph;
- };
-
- 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 (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
-
-
- for (i = 0; i < nodeIDs.length; i++) {
- node = graph.entity(nodeIDs[i]);
- relations = graph.parentRelations(node);
-
- 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
-
- if (relation.hasFromViaTo()) {
- restrictionIDs.push(relation.id);
- }
-
- if (seen[relation.id] !== undefined && seen[relation.id] !== role) {
- return 'relation';
- } else {
- seen[relation.id] = role;
- }
- }
- } // gather restrictions for parent ways
-
-
- for (i = 0; i < nodeIDs.length; i++) {
- node = graph.entity(nodeIDs[i]);
- var parents = graph.parentWays(node);
-
- for (j = 0; j < parents.length; j++) {
- var parent = parents[j];
- relations = graph.parentRelations(parent);
-
- for (k = 0; k < relations.length; k++) {
- relation = relations[k];
-
- if (relation.hasFromViaTo()) {
- restrictionIDs.push(relation.id);
- }
- }
- }
- } // test restrictions
-
-
- restrictionIDs = utilArrayUniq(restrictionIDs);
-
- 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 nodes = {
- from: [],
- via: [],
- to: [],
- keyfrom: [],
- keyto: []
- };
-
- for (j = 0; j < relation.members.length; j++) {
- collectNodes(relation.members[j], nodes);
- }
-
- 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;
-
- for (j = 0; j < nodeIDs.length; j++) {
- var n = nodeIDs[j];
-
- if (nodes.from.indexOf(n) !== -1) {
- connectFrom = true;
- }
-
- if (nodes.via.indexOf(n) !== -1) {
- connectVia = true;
- }
-
- if (nodes.to.indexOf(n) !== -1) {
- connectTo = true;
- }
-
- if (nodes.keyfrom.indexOf(n) !== -1) {
- connectKeyFrom = true;
- }
-
- if (nodes.keyto.indexOf(n) !== -1) {
- connectKeyTo = true;
- }
- }
-
- if (connectFrom && connectTo && !isUturn) {
- return 'restriction';
- }
-
- if (connectFrom && connectVia) {
- return 'restriction';
- }
-
- 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 (connectKeyFrom || connectKeyTo) {
- if (nodeIDs.length !== 2) {
- return 'restriction';
- }
-
- var n0 = null;
- var n1 = null;
-
- for (j = 0; j < memberWays.length; j++) {
- way = memberWays[j];
-
- if (way.contains(nodeIDs[0])) {
- n0 = nodeIDs[0];
- }
-
- if (way.contains(nodeIDs[1])) {
- n1 = nodeIDs[1];
- }
- }
-
- if (n0 && n1) {
- // both nodes are part of the restriction
- var ok = false;
-
- for (j = 0; j < memberWays.length; j++) {
- way = memberWays[j];
-
- 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)
-
-
- for (j = 0; j < memberWays.length; j++) {
- way = memberWays[j].update({}); // make copy
-
- 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 (way.isDegenerate()) {
- return 'restriction';
- }
- }
- }
-
- return false; // if a key node appears multiple times (indexOf !== lastIndexOf) it's a FROM-VIA or TO-VIA junction
-
- function hasDuplicates(n, i, arr) {
- return arr.indexOf(n) !== arr.lastIndexOf(n);
- }
-
- function keyNodeFilter(froms, tos) {
- return function (n) {
- return froms.indexOf(n) === -1 && tos.indexOf(n) === -1;
- };
- }
-
- function collectNodes(member, collection) {
- var entity = graph.hasEntity(member.id);
- if (!entity) return;
- var role = member.role || '';
-
- if (!collection[role]) {
- collection[role] = [];
- }
-
- if (member.type === 'node') {
- collection[role].push(member.id);
-
- 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());
- }
-
- if (role === 'to' || role === 'via') {
- collection.keyto.push(entity.first());
- collection.keyto.push(entity.last());
- }
- }
- }
- };
-
- return action;
- }
-
- function actionCopyEntities(ids, fromGraph) {
- var _copies = {};
-
- var action = function action(graph) {
- ids.forEach(function (id) {
- fromGraph.entity(id).copy(fromGraph, _copies);
- });
-
- for (var id in _copies) {
- graph = graph.replace(_copies[id]);
- }
-
- return graph;
- };
-
- action.copies = function () {
- return _copies;
- };
-
- return action;
- }
-
- 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;
-
- function checkTags(entity) {
- var keys = Object.keys(entity.tags);
- var didDiscard = false;
- var tags = {};
-
- for (var i = 0; i < keys.length; i++) {
- var k = keys[i];
-
- if (discardTags[k] || !entity.tags[k]) {
- didDiscard = true;
- } else {
- tags[k] = entity.tags[k];
- }
- }
-
- if (didDiscard) {
- graph = graph.replace(entity.update({
- tags: tags
- }));
- }
- }
- };
- }
-
- //
- // 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
- //
-
- function actionDisconnect(nodeId, newNodeId) {
- var wayIds;
-
- 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 (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 {
- // replace shared node with multiple new nodes..
- graph = graph.replace(way.updateNode(newNode.id, connection.index));
- }
- });
- 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];
-
- if (wayIds && wayIds.indexOf(way.id) === -1) {
- keeping = true;
- continue;
- }
-
- 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];
-
- 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
- });
- }
- }
- }
- }
-
- return keeping ? candidates : candidates.slice(1);
- };
-
- 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';
- };
-
- action.limitWays = function (val) {
- if (!arguments.length) return wayIds;
- wayIds = val;
- return action;
- };
-
- return action;
- }
-
- var geojsonRewind = rewind;
-
- function rewind(gj, outer) {
- var type = gj && gj.type,
- i;
-
- 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;
- }
-
- 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);
- }
- }
-
- function rewindRing(ring, dir) {
- var area = 0;
-
- 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]);
- }
-
- if (area >= 0 !== !!dir) ring.reverse();
- }
-
- function actionExtract(entityID) {
- var extractedNodeID;
-
- var action = function action(graph) {
- var entity = graph.entity(entityID);
-
- if (entity.type === 'node') {
- return extractFromNode(entity, graph);
- }
-
- return extractFromWayOrRelation(entity, graph);
- };
-
- function extractFromNode(node, graph) {
- extractedNodeID = node.id; // Create a new node to replace the one we will detach
-
- var replacement = osmNode({
- loc: node.loc
- });
- graph = graph.replace(replacement); // Process each way in turn, updating the graph as we go
-
- graph = graph.parentWays(node).reduce(function (accGraph, parentWay) {
- return accGraph.replace(parentWay.replaceNode(entityID, replacement.id));
- }, graph); // Process any relations too
-
- return graph.parentRelations(node).reduce(function (accGraph, parentRel) {
- return accGraph.replace(parentRel.replaceMember(node, replacement));
- }, graph);
- }
-
- 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));
-
- if (!extractedLoc || !isFinite(extractedLoc[0]) || !isFinite(extractedLoc[1])) {
- extractedLoc = entity.extent(graph).center();
- }
-
- 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 = {};
-
- for (var key in entityTags) {
- if (entity.type === 'relation' && key === 'type') {
- continue;
- }
-
- 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 (!isBuilding && !isIndoorArea && fromGeometry === 'area') {
- // ensure that areas keep area geometry
- entityTags.area = 'yes';
- }
-
- var replacement = osmNode({
- loc: extractedLoc,
- tags: pointTags
- });
- graph = graph.replace(replacement);
- extractedNodeID = replacement.id;
- return graph.replace(entity.update({
- tags: entityTags
- }));
- }
-
- action.getExtractedNodeID = function () {
- return extractedNodeID;
- };
-
- return action;
- }
-
- //
- // 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
- //
-
- 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
-
- 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.
-
- for (var i = 0; i < ways.length; i++) {
- if (!ways[i].isNew()) {
- survivorID = ways[i].id;
- break;
- }
- }
-
- 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.
-
- 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 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;
- }
-
- 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 (survivor.geometry(graph) !== 'area') {
- // ensure the feature persists as an area
- tags.area = 'yes';
- }
-
- delete tags.type; // remove type=multipolygon
-
- survivor = survivor.update({
- tags: tags
- });
- graph = graph.replace(survivor);
- }
-
- checkForSimpleMultipolygon();
- return graph;
- }; // Returns the number of nodes the resultant way is expected to have
-
-
- action.resultingWayNodesLength = function (graph) {
- return ids.reduce(function (count, id) {
- return count + graph.entity(id).nodes.length;
- }, 0) - ids.length - 1;
- };
-
- action.disabled = function (graph) {
- var geometries = groupEntitiesByGeometry(graph);
-
- if (ids.length < 2 || ids.length !== geometries.line.length) {
- return 'not_eligible';
- }
-
- var joined = osmJoinWays(ids.map(graph.entity, graph), graph);
-
- if (joined.length > 1) {
- return 'not_adjacent';
- } // Loop through all combinations of path-pairs
- // to check potential intersections between all pairs
-
-
- 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 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';
- }
- }
- }
-
- 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;
- }
- }
- });
-
- if (relation) {
- return 'restriction';
- }
-
- if (conflicting) {
- return 'conflicting_tags';
- }
- };
-
- return action;
- }
-
- 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 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;
-
- for (var i = 0; i < nodes.length; i++) {
- var node = nodes[i];
-
- 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
-
-
- graph = graph.replace(point.update({
- tags: {},
- loc: node.loc
- }));
- target = target.replaceNode(node.id, point.id);
- graph = graph.replace(target);
- removeNode = node;
- break;
- }
-
- graph = graph.remove(removeNode);
- });
-
- if (target.tags.area === 'yes') {
- var tags = Object.assign({}, target.tags); // shallow copy
-
- delete tags.area;
-
- if (osmTagSuggestingArea(tags)) {
- // remove the `area` tag if area geometry is now implied - #3851
- target = target.update({
- tags: tags
- });
- graph = graph.replace(target);
- }
- }
-
- return graph;
- };
-
- action.disabled = function (graph) {
- var geometries = groupEntitiesByGeometry(graph);
-
- if (geometries.point.length === 0 || geometries.area.length + geometries.line.length !== 1 || geometries.relation.length !== 0) {
- return 'not_eligible';
- }
- };
-
- return action;
- }
-
- //
- // 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;
-
- for (var i = 0; i < nodeIDs.length; i++) {
- var node = graph.entity(nodeIDs[i]);
-
- if (node.hasInterestingTags()) {
- interestingLoc = ++interestingCount === 1 ? node.loc : null;
- }
-
- sum = geoVecAdd(sum, node.loc);
- }
-
- return interestingLoc || geoVecScale(sum, 1 / nodeIDs.length);
- }
-
- var action = function action(graph) {
- if (nodeIDs.length < 2) return graph;
- var toLoc = loc;
-
- if (!toLoc) {
- toLoc = chooseLoc(graph);
- }
-
- for (var i = 0; i < nodeIDs.length; i++) {
- var node = graph.entity(nodeIDs[i]);
-
- if (node.loc !== toLoc) {
- graph = graph.replace(node.move(toLoc));
- }
- }
-
- return actionConnect(nodeIDs)(graph);
- };
-
- action.disabled = function (graph) {
- if (nodeIDs.length < 2) return 'not_eligible';
-
- 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);
- };
-
- return action;
- }
-
- 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;
-
- 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]);
- }
-
- var ordered = {};
- order.forEach(function (o) {
- if (groups[o]) ordered[o] = groups[o];
- });
- return ordered;
- } // sort relations in a changeset by dependencies
-
-
- 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 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 i = 0; i < relations.length; i++) {
- var relation = relations[i]; // skip relation if already sorted
-
- if (!sorted[relation['@id']]) {
- processing.push(relation);
- }
-
- while (processing.length > 0) {
- var next = processing[0],
- deps = next.member.map(resolve).filter(Boolean).filter(isNew);
-
- if (deps.length === 0) {
- sorted[next['@id']] = next;
- processing.shift();
- } else {
- processing = deps.concat(processing);
- }
- }
- }
-
- 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 {};
- }
- });
-
- function osmNote() {
- if (!(this instanceof osmNote)) {
- return new osmNote().initialize(arguments);
- } else if (arguments.length) {
- this.initialize(arguments);
- }
- }
-
- osmNote.id = function () {
- return osmNote.id.next--;
- };
-
- 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 prop in source) {
- if (Object.prototype.hasOwnProperty.call(source, prop)) {
- if (source[prop] === undefined) {
- delete this[prop];
- } else {
- this[prop] = source[prop];
- }
- }
- }
- }
-
- if (!this.id) {
- this.id = osmNote.id().toString();
- }
-
- 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);
-
- 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;
- };
-
- 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();
-
- 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));
- }
- }
-
- 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);
-
- for (var i = 0; i < this.members.length; i++) {
- result[i] = Object.assign({}, this.members[i], {
- index: i
- });
- }
-
- 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 = [];
-
- for (var i = 0; i < this.members.length; i++) {
- if (this.members[i].role === role) {
- result.push(Object.assign({}, this.members[i], {
- index: 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 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 = [];
-
- for (var i = 0; i < this.members.length; i++) {
- var member = this.members[i];
-
- 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
- });
- }
- }
-
- 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 (changeset_id) {
- r.relation['@changeset'] = changeset_id;
- }
-
- 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;
- }
- }
-
- 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 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]);
- }
-
- return sequence.nodes.map(function (node) {
- return node.loc;
- });
- };
-
- 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];
- });
-
- function findOuter(inner) {
- var o, outer;
-
- for (o = 0; o < outers.length; o++) {
- outer = outers[o];
- if (geoPolygonContainsPolygon(outer, inner)) return o;
- }
-
- for (o = 0; o < outers.length; o++) {
- outer = outers[o];
- if (geoPolygonIntersectsPolygon(outer, inner, false)) return o;
- }
- }
-
- for (var i = 0; i < inners.length; i++) {
- var inner = inners[i];
-
- if (d3_geoArea({
- type: 'Polygon',
- coordinates: [inner]
- }) < 2 * Math.PI) {
- inner = inner.reverse();
- }
-
- var o = findOuter(inners[i]);
-
- if (o !== undefined) {
- result[o].push(inners[i]);
- } else {
- result.push([inners[i]]); // Invalid geometry
- }
- }
-
- return result;
- }
- });
-
- var QAItem = /*#__PURE__*/function () {
- function QAItem(loc, service, itemType, id, props) {
- _classCallCheck(this, QAItem);
-
- // 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
-
- 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);
- }
- }
-
- _createClass(QAItem, [{
- key: "update",
- value: function update(props) {
- var _this = this;
-
- // 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
-
- }], [{
- key: "id",
- value: function id() {
- return this.nextId--;
- }
- }]);
-
- return QAItem;
- }();
- QAItem.nextId = -1;
-
- //
- // 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
- //
-
- 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
-
-
- var _keepHistoryOn = 'longest'; // 'longest', 'first'
- // The IDs of the ways actually created by running this action
-
- var _createdWayIDs = [];
-
- 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 splitArea(nodes, idxA, graph) {
- var lengths = new Array(nodes.length);
- var length;
- var i;
- var best = 0;
- var idxB;
-
- 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;
- }
-
- length = 0;
-
- for (i = wrap(idxA - 1); i !== idxA; i = wrap(i - 1)) {
- length += dist(graph, nodes[i], nodes[wrap(i + 1)]);
-
- if (length < lengths[i]) {
- lengths[i] = length;
- }
- } // determine best opposite node to split
-
-
- for (i = 0; i < nodes.length; i++) {
- var cost = lengths[i] / dist(graph, nodes[idxA], nodes[i]);
-
- if (cost > best) {
- idxB = i;
- best = cost;
- }
- }
-
- return idxB;
- }
-
- function totalLengthBetweenNodes(graph, nodes) {
- var totalLength = 0;
-
- for (var i = 0; i < nodes.length - 1; i++) {
- totalLength += dist(graph, nodes[i], nodes[i + 1]);
- }
-
- return totalLength;
- }
-
- function split(graph, nodeId, wayA, newWayId) {
- var wayB = osmWay({
- id: newWayId,
- tags: wayA.tags
- }); // `wayB` is the NEW way
-
- var origNodes = wayA.nodes.slice();
- var nodesA;
- var nodesB;
- var isArea = wayA.isArea();
- var isOuter = osmIsOldMultipolygonOuterMember(wayA, graph);
-
- if (wayA.isClosed()) {
- var nodes = wayA.nodes.slice(0, -1);
- var idxA = nodes.indexOf(nodeId);
- var idxB = splitArea(nodes, idxA, graph);
-
- 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);
- }
-
- 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
- });
- }
-
- if (wayA.tags.step_count) {
- // divide up the the step count proportionally between the two ways
- var stepCount = parseFloat(wayA.tags.step_count);
-
- 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
- });
- }
- }
-
- 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
-
- 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
-
- if (f.id === wayA.id || t.id === wayA.id) {
- var keepB = false;
-
- 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);
-
- if (wayVia && utilArrayIntersection(wayB.nodes, wayVia.nodes).length) {
- keepB = true;
- break;
- }
- }
- }
- }
-
- if (keepB) {
- relation = relation.replaceMember(wayA, wayB);
- graph = graph.replace(relation);
- } // 2. split a VIA
-
- } 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: {}
- }));
- }
-
- 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);
- }
- });
-
- 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: {}
- }));
- }
-
- _createdWayIDs.push(wayB.id);
-
- return graph;
- }
-
- var action = function action(graph) {
- _createdWayIDs = [];
- var newWayIndex = 0;
-
- 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;
- }
- }
-
- return graph;
- };
-
- action.getCreatedWayIDs = function () {
- return _createdWayIDs;
- };
-
- action.waysForNode = function (nodeId, graph) {
- var node = graph.entity(nodeId);
- var splittableParents = graph.parentWays(node).filter(isSplittable);
-
- 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';
- });
- }
- }
-
- return splittableParents;
-
- 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 (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;
- }
-
- return false;
- }
- };
-
- action.ways = function (graph) {
- return utilArrayUniq([].concat.apply([], nodeIds.map(function (nodeId) {
- return action.waysForNode(nodeId, graph);
- })));
- };
-
- action.disabled = function (graph) {
- for (var i = 0; i < nodeIds.length; i++) {
- var nodeId = nodeIds[i];
- var candidates = action.waysForNode(nodeId, graph);
-
- if (candidates.length === 0 || _wayIDs && _wayIDs.length !== candidates.length) {
- return 'not_eligible';
- }
- }
- };
-
- 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];
- },
- 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 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 (transients[key] !== undefined) {
- return transients[key];
- }
-
- transients[key] = fn.call(entity);
- return transients[key];
- },
- parentWays: function parentWays(entity) {
- var parents = this._parentWays[entity.id];
- var result = [];
-
- 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 = [];
-
- if (parents) {
- parents.forEach(function (id) {
- result.push(this.entity(id));
- }, this);
- }
-
- 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 = [];
-
- 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;
-
- 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
-
- base.entities[entity.id] = entity;
-
- this._updateCalculated(undefined, entity, base.parentWays, base.parentRels); // Restore provisionally-deleted nodes that are discovered to have an extant parent
-
-
- if (entity.type === 'way') {
- for (j = 0; j < entity.nodes.length; j++) {
- id = entity.nodes[j];
-
- for (k = 1; k < stack.length; k++) {
- var ents = stack[k].entities;
-
- if (ents.hasOwnProperty(id) && ents[id] === undefined) {
- delete ents[id];
- }
- }
- }
- }
- }
-
- 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;
- }
-
- 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);
- }
-
- 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;
- }) : [];
-
- if (oldentity && entity) {
- removed = utilArrayDifference(oldentityMemberIDs, entityMemberIDs);
- added = utilArrayDifference(entityMemberIDs, oldentityMemberIDs);
- } else if (oldentity) {
- removed = oldentityMemberIDs;
- added = [];
- } else if (entity) {
- removed = [];
- added = entityMemberIDs;
- }
-
- 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);
- }
-
- 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);
-
- this.entities[entity.id] = entity;
- });
- },
- remove: function remove(entity) {
- return this.update(function () {
- this._updateCalculated(entity, undefined);
-
- 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);
-
- delete this.entities[id];
- });
- },
- update: function update() {
- var graph = this.frozen ? coreGraph(this, true) : this;
-
- for (var i = 0; i < arguments.length; i++) {
- arguments[i].call(graph, graph);
- }
-
- 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);
-
- for (var i in entities) {
- this.entities[i] = entities[i];
-
- this._updateCalculated(base.entities[i], this.entities[i]);
- }
-
- return this;
- }
- };
-
- 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
-
- var vgraph = coreGraph(); // virtual graph
-
- var i, j, k;
-
- 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];
- }
-
- 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
-
- var actions = []; // STEP 1: walk the graph outwards from starting vertex to search
- // for more key vertices and ways to include in the intersection..
-
- while (checkVertices.length) {
- vertex = checkVertices.pop(); // check this vertex for parent ways that are roads
-
- checkWays = graph.parentWays(vertex);
- var hasWays = false;
-
- 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
-
- nodes = utilArrayUniq(graph.childNodes(way));
-
- 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
-
- if (geoSphericalDistance(node.loc, startNode.loc) > maxDistance) continue; // too far from start
- // a key vertex will have parents that are also roads
-
- var hasParents = false;
- parents = graph.parentWays(node);
-
- 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
-
- if (!isRoad(parent)) continue; // not a road
-
- hasParents = true;
- break;
- }
-
- if (hasParents) {
- checkVertices.push(node);
- }
- }
- }
-
- if (hasWays) {
- vertices.push(vertex);
- }
- }
-
- 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
-
- 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));
- }
- }
- });
- }); // STEP 3: Force all oneways to be drawn in the forward direction
-
- 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 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
-
- osmEntity.id.next.way = origCount; // STEP 5: Update arrays to point to vgraph entities
-
- 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?
-
-
- var __first = vertexIds.indexOf(way.first()) !== -1;
-
- var __last = vertexIds.indexOf(way.last()) !== -1; // what roles is this way eligible for?
-
-
- var __via = __first && __last;
-
- var __from = __first && !__oneWay || __last;
-
- var __to = __first || __last && !__oneWay;
-
- 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
-
- var keepGoing;
- var removeWayIds = [];
- var removeVertexIds = [];
-
- do {
- keepGoing = false;
- checkVertices = vertexIds.slice();
-
- for (i = 0; i < checkVertices.length; i++) {
- var vertexId = checkVertices[i];
- vertex = vgraph.hasEntity(vertexId);
-
- if (!vertex) {
- if (vertexIds.indexOf(vertexId) !== -1) {
- vertexIds.splice(vertexIds.indexOf(vertexId), 1); // stop checking this one
- }
-
- removeVertexIds.push(vertexId);
- continue;
- }
-
- parents = vgraph.parentWays(vertex);
-
- if (parents.length < 3) {
- if (vertexIds.indexOf(vertexId) !== -1) {
- vertexIds.splice(vertexIds.indexOf(vertexId), 1); // stop checking this one
- }
- }
-
- 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;
-
- if (aIsLeaf && !bIsLeaf) {
- leaf = a;
- survivor = b;
- } else if (!aIsLeaf && bIsLeaf) {
- leaf = b;
- survivor = a;
- }
-
- if (leaf && survivor) {
- survivor = withMetadata(survivor, vertexIds); // update survivor way
-
- vgraph = vgraph.replace(survivor).remove(leaf); // update graph
-
- removeWayIds.push(leaf.id);
- keepGoing = true;
- }
- }
-
- 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;
- }
-
- 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 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
-
- 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
-
- var i, j;
-
- 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
-
- if (way.__oneWay && way.nodes[0] !== entity.id) continue; // if we have seen it before (allowing for an initial u-turn), skip
-
- if (currPath.indexOf(way.id) !== -1 && currPath.length >= 3) continue; // Check all "current" restrictions (where we've already walked the `FROM`)
-
- var restrict = null;
-
- 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?
-
- var matchesFrom = f.id === fromWayId;
- var matchesViaTo = false;
- var isAlongOnlyPath = false;
-
- 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...)
- }
-
- var restrictionVias = [];
-
- for (k = 0; k < v.length; k++) {
- if (v[k].type === 'way') {
- restrictionVias.push(v[k].id);
- }
- }
-
- 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;
- }
- }
- }
-
- 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 (restrict && restrict.direct) break;
- }
-
- 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)
-
- 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);
-
- if (turn) {
- if (matchedRestriction) {
- turn.restrictionID = matchedRestriction.id;
- turn.no = matchedRestriction.no;
- turn.only = matchedRestriction.only;
- turn.direct = matchedRestriction.direct;
- }
-
- turns.push(osmTurn(turn));
- }
-
- 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?
-
- var n1 = vgraph.entity(entity.first());
- var n2 = vgraph.entity(entity.last());
- var dist = geoSphericalDistance(n1.loc, n2.loc);
- var nextNodes = [];
-
- if (currPath.length > 1) {
- if (dist > maxDistance) return; // the next node is too far
-
- 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
- }
-
- 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
- }
-
- 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
-
- 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);
-
- if (viaWay.first() === nextNode.id || viaWay.last() === nextNode.id) {
- isOnlyVia = true;
- break;
- }
- }
- }
-
- return isOnlyVia;
- });
- step(nextNode, currPath, currRestrictions.concat(fromRestrictions), false);
- });
- }
- } // assumes path is alternating way-node-way of odd length
-
-
- 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];
-
- 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 (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
- };
-
- function adjacentNode(wayId, affixId) {
- var nodes = vgraph.entity(wayId).nodes;
- return affixId === nodes[0] ? nodes[1] : nodes[nodes.length - 2];
- }
- }
- };
-
- 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;
- }
-
- 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)
-
- if (angle < 158) return 'no_right_turn';
- if (angle > 202) return 'no_left_turn';
- return 'no_straight_on';
- }
-
- 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);
- }
-
- 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.
-
- 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 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
-
- var members = [];
- var outer = true;
-
- while (polygons.length) {
- extractUncontained(polygons);
- polygons = polygons.filter(isContained);
- contained = contained.filter(isContained).map(filterContained);
- }
-
- function isContained(d, i) {
- return contained[i].some(function (val) {
- return val;
- });
- }
-
- function filterContained(d) {
- return d.filter(isContained);
- }
-
- 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
-
-
- 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';
- }
-
- 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'])
- }));
- };
-
- action.disabled = function (graph) {
- var entities = groupEntities(graph);
-
- if (entities.other.length > 0 || entities.closedWay.length + entities.multipolygon.length < 2) {
- return 'not_eligible';
- }
-
- 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;
- });
-
- 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 action;
- }
-
- var UNSUPPORTED_Y$3 = regexpStickyHelpers.UNSUPPORTED_Y;
-
- // `RegExp.prototype.flags` getter
- // https://tc39.github.io/ecma262/#sec-get-regexp.prototype.flags
- if (descriptors && (/./g.flags != 'g' || UNSUPPORTED_Y$3)) {
- objectDefineProperty.f(RegExp.prototype, 'flags', {
- configurable: true,
- get: regexpFlags
- });
- }
-
- 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;
-
- if (Array.isArray(a)) {
- length = a.length;
- if (length != b.length) return false;
-
- for (i = length; i-- !== 0;) {
- if (!equal(a[i], b[i])) return false;
- }
-
- return true;
- }
-
- 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;
-
- for (i = length; i-- !== 0;) {
- if (!Object.prototype.hasOwnProperty.call(b, keys[i])) return false;
- }
-
- for (i = length; i-- !== 0;) {
- var key = keys[i];
- if (!equal(a[key], b[key])) return false;
- }
-
- return true;
- } // true if both NaN, false otherwise
-
-
- 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
- //
- // Expects two arrays, finds longest common sequence
-
- function LCS(buffer1, buffer2) {
- var equivalenceClasses = {};
-
- for (var j = 0; j < buffer2.length; j++) {
- var item = buffer2[j];
-
- if (equivalenceClasses[item]) {
- equivalenceClasses[item].push(j);
- } else {
- equivalenceClasses[item] = [j];
- }
- }
-
- var NULLRESULT = {
- buffer1index: -1,
- buffer2index: -1,
- chain: null
- };
- 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];
-
- 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;
- }
- }
-
- if (s < candidates.length) {
- var newCandidate = {
- buffer1index: i,
- buffer2index: _j,
- chain: candidates[s]
- };
-
- 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
- }
- }
- }
-
- 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 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.
-
-
- function diffIndices(buffer1, buffer2) {
- var lcs = LCS(buffer1, buffer2);
- var result = [];
- var tail1 = buffer1.length;
- var tail2 = buffer2.length;
-
- 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)
- });
- }
- }
-
- 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 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 = [];
-
- 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])
-
- });
- }
-
- 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;
- }
- }
-
- 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
-
- regionEnd = Math.max(regionEnd, nextHunkStart + nextHunk.oLength);
- regionHunks.push(hunks.shift());
- }
-
- 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);
- }
-
- 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
- });
- }
-
- okBuffer = [];
- }
-
- 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;
- }
-
- return true;
- }
-
- regions.forEach(function (region) {
- if (region.stable) {
- var _okBuffer;
-
- (_okBuffer = okBuffer).push.apply(_okBuffer, _toConsumableArray(region.bufferContent));
- } else {
- if (options.excludeFalseConflicts && isFalseConflict(region.aContent, region.bContent)) {
- var _okBuffer2;
-
- (_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
- }
- });
- }
- }
- });
- flushOk();
- return results;
- }
-
- function actionMergeRemoteChanges(id, localGraph, remoteGraph, discardTags, formatUser) {
- discardTags = discardTags || {};
- var _option = 'safe'; // 'safe', 'force_local', 'force_remote'
-
- var _conflicts = [];
-
- function user(d) {
- return typeof formatUser === 'function' ? formatUser(d) : d;
- }
-
- 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 (_option === 'force_local' || pointEqual(target.loc, remote.loc)) {
- return target;
- }
-
- if (_option === 'force_remote') {
- return target.update({
- loc: remote.loc
- });
- }
-
- _conflicts.push(_t('merge_remote_changes.conflict.location', {
- user: user(remote.user)
- }));
-
- return target;
- }
-
- function mergeNodes(base, remote, target) {
- if (_option === 'force_local' || fastDeepEqual(target.nodes, remote.nodes)) {
- return target;
- }
-
- if (_option === 'force_remote') {
- return target.update({
- nodes: remote.nodes
- });
- }
-
- 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
- });
-
- for (var i = 0; i < hunks.length; i++) {
- var hunk = hunks[i];
-
- 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 (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;
- }
- }
- }
-
- 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;
- }
-
- var ccount = _conflicts.length;
-
- for (var i = 0; i < children.length; i++) {
- var id = children[i];
- var node = graph.hasEntity(id); // remove unused childNodes..
-
- if (targetWay.nodes.indexOf(id) === -1) {
- if (node && !isUsed(node, targetWay)) {
- updates.removeIds.push(id);
- }
-
- continue;
- } // restore used childNodes..
-
-
- 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
- });
- }
-
- updates.replacements.push(target);
- } else if (_option === 'safe' && local && remote && local.version !== remote.version) {
- target = osmEntity(local, {
- version: remote.version
- });
-
- if (remote.visible) {
- target = mergeLocation(remote, target);
- } else {
- _conflicts.push(_t('merge_remote_changes.conflict.deleted', {
- user: user(remote.user)
- }));
- }
-
- if (_conflicts.length !== ccount) break;
- updates.replacements.push(target);
- }
- }
-
- return targetWay;
- }
-
- function updateChildren(updates, graph) {
- for (var i = 0; i < updates.replacements.length; i++) {
- graph = graph.replace(updates.replacements[i]);
- }
-
- if (updates.removeIds.length) {
- graph = actionDeleteMultiple(updates.removeIds)(graph);
- }
-
- return graph;
- }
-
- function mergeMembers(remote, target) {
- if (_option === 'force_local' || fastDeepEqual(target.members, remote.members)) {
- return target;
- }
-
- if (_option === 'force_remote') {
- return target.update({
- members: remote.members
- });
- }
-
- _conflicts.push(_t('merge_remote_changes.conflict.memberlist', {
- user: user(remote.user)
- }));
-
- return target;
- }
-
- function mergeTags(base, remote, target) {
- if (_option === 'force_local' || fastDeepEqual(target.tags, remote.tags)) {
- return target;
- }
-
- if (_option === 'force_remote') {
- return target.update({
- tags: remote.tags
- });
- }
-
- 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
-
- var changed = false;
-
- for (var i = 0; i < keys.length; i++) {
- var k = keys[i];
-
- 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];
- }
-
- changed = true;
- }
- }
- }
-
- 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 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 (!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);
- }
-
- return graph.replace(target);
- } else {
- _conflicts.push(_t('merge_remote_changes.conflict.deleted', {
- user: user(remote.user)
- }));
-
- return graph; // do nothing
- }
- } // merge
-
-
- 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);
- }
-
- target = mergeTags(base, remote, target);
-
- if (!_conflicts.length) {
- graph = updateChildren(updates, graph).replace(target);
- }
-
- return graph;
- };
-
- action.withOption = function (opt) {
- _option = opt;
- return action;
- };
-
- action.conflicts = function () {
- return _conflicts;
- };
-
- return action;
- }
-
- // https://github.com/openstreetmap/potlatch2/blob/master/net/systemeD/halcyon/connection/actions/MoveNodeAction.as
-
- function actionMove(moveIDs, tryDelta, projection, cache) {
- var _delta = tryDelta;
-
- 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 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;
-
- 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);
- }
-
- 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));
-
- 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;
-
- for (var k = 0; k < parents.length; k++) {
- var way = parents[k];
-
- if (!cache.moving[way.id]) {
- unmoved = way;
- break;
- }
- }
-
- if (!unmoved) continue; // exclude ways that are overly connected..
-
- 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)
- });
- }
- }
- }
-
- if (!cache) {
- cache = {};
- }
-
- 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
- //
- //
-
-
- 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;
- }
-
- var prev = graph.hasEntity(way.nodes[prevIndex]);
- var next = graph.hasEntity(way.nodes[nextIndex]); // Don't add orig vertex at endpoint..
-
- if (!prev || !next) return graph;
- var key = wayId + '_' + nodeId;
- var orig = cache.replacedVertex[key];
-
- if (!orig) {
- orig = osmNode();
- cache.replacedVertex[key] = orig;
- cache.startLoc[orig.id] = cache.startLoc[nodeId];
- }
-
- var start, end;
-
- if (delta) {
- start = projection(cache.startLoc[nodeId]);
- end = projection.invert(geoVecAdd(start, delta));
- } else {
- end = cache.startLoc[nodeId];
- }
-
- 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?
-
- 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.
-
-
- function removeDuplicateVertices(wayId, graph) {
- var way = graph.entity(wayId);
- var epsilon = 1e-6;
- var prev, curr;
-
- 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 (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;
- }
-
- 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
- //
-
-
- 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 (!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;
- }
-
- graph = graph.replace(vertex.move(loc)); // if zorro happened, reorder nodes..
-
- if (!isEP1 && edge1.index !== way1.nodes.indexOf(vertex.id)) {
- way1 = way1.removeNode(vertex.id).addNode(vertex.id, edge1.index);
- graph = graph.replace(way1);
- }
-
- 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;
- }
-
- 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..
-
-
- function limitDelta(graph) {
- function moveNode(loc) {
- return geoVecAdd(projection(loc), _delta);
- }
-
- 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 (!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 action = function action(graph) {
- if (_delta[0] === 0 && _delta[1] === 0) return graph;
- setupCache(graph);
-
- if (cache.intersections.length) {
- limitDelta(graph);
- }
-
- 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);
- }
-
- return graph;
- };
-
- action.delta = function () {
- return _delta;
- };
-
- return action;
- }
-
- function actionMoveMember(relationId, fromIndex, toIndex) {
- return function (graph) {
- return graph.replace(graph.entity(relationId).moveMember(fromIndex, toIndex));
- };
- }
-
- 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;
- }
-
- function actionNoop() {
- return function (graph) {
- return graph;
- };
- }
-
- 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)
-
- var lowerThreshold = Math.cos((90 - threshold) * Math.PI / 180);
- var upperThreshold = Math.cos(threshold * Math.PI / 180);
-
- 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
-
- if (way.tags.nonsquare) {
- var tags = Object.assign({}, way.tags); // since we're squaring, remove indication that this is physically unsquare
-
- delete tags.nonsquare;
- way = way.update({
- tags: tags
- });
- }
-
- graph = graph.replace(way);
- var isClosed = way.isClosed();
- var nodes = graph.childNodes(way).slice(); // shallow copy
-
- if (isClosed) nodes.pop();
-
- 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
-
-
- var nodeCount = {};
- var points = [];
- var corner = {
- i: 0,
- dotp: 1
- };
- var node, point, loc, score, motions, i, j;
-
- 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 (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;
- }
- }
-
- 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
-
- for (i = 0; i < points.length; i++) {
- point = points[i];
- var dotp = 0;
-
- 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));
- }
-
- if (dotp > upperThreshold) {
- straights.push(point);
- } else {
- simplified.push(point);
- }
- } // Orthogonalize the simplified shape
-
-
- var bestPoints = clonePoints(simplified);
- var originalPoints = clonePoints(simplified);
- score = Infinity;
-
- for (i = 0; i < 1000; i++) {
- motions = simplified.map(calcMotion);
-
- for (j = 0; j < motions.length; j++) {
- simplified[j].coord = geoVecAdd(simplified[j].coord, motions[j]);
- }
-
- var newScore = geoOrthoCalcScore(simplified, isClosed, epsilon, threshold);
-
- if (newScore < score) {
- bestPoints = clonePoints(simplified);
- score = newScore;
- }
-
- if (score < epsilon) {
- break;
- }
- }
-
- var bestCoords = bestPoints.map(function (p) {
- return p.coord;
- });
- if (isClosed) bestCoords.push(bestCoords[0]); // move the nodes that should move
-
- for (i = 0; i < bestPoints.length; i++) {
- point = bestPoints[i];
-
- 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
-
-
- 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 (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)));
- }
- }
- }
- }
-
- return graph;
-
- function clonePoints(array) {
- return array.map(function (p) {
- return {
- id: p.id,
- coord: [p.coord[0], p.coord[1]]
- };
- });
- }
-
- 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 (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
-
-
- function nodeSubset(nodes, vertexID, isClosed) {
- var first = isClosed ? 0 : 1;
- var last = isClosed ? nodes.length : nodes.length - 1;
-
- 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 [];
- }
-
- 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';
- }
-
- var coords = nodes.map(function (n) {
- return projection(n.loc);
- });
- var score = geoOrthoCanOrthogonalize(coords, isClosed, epsilon, threshold, allowStraightAngles);
-
- if (score === null) {
- return 'not_squarish';
- } else if (score === 0) {
- return 'square_enough';
- } else {
- return false;
- }
- };
-
- action.transitionable = true;
- return action;
- }
-
- //
- // `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.
- //
-
- 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 (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'
- });
- });
- }
-
- members.push({
- id: toWay.id,
- type: 'way',
- role: 'to'
- });
- return graph.replace(osmRelation({
- id: restrictionID,
- tags: {
- type: 'restriction',
- restriction: restrictionType
- },
- members: members
- }));
- };
- }
-
- function actionRevert(id) {
- var action = function action(graph) {
- var entity = graph.hasEntity(id),
- base = graph.base().entities[id];
-
- 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 (parent.isDegenerate()) {
- graph = actionDeleteWay(parent.id)(graph);
- }
- });
- }
-
- graph.parentRelations(entity).forEach(function (parent) {
- parent = parent.removeMembersWithID(id);
- graph = graph.replace(parent);
-
- if (parent.isDegenerate()) {
- graph = actionDeleteRelation(parent.id)(graph);
- }
- });
- }
-
- return graph.revert(id);
- };
-
- return action;
- }
-
- 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)));
- });
- });
- };
-
- return action;
- }
-
- 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)));
- });
- });
- };
- }
-
- /* 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
-
-
- 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
-
- 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];
- }
-
- return [p2, q2];
- }
-
- 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)));
- }
-
- 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;
-
- 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 (!isNaN(dist) && dist > maxDistance) {
- maxDistance = dist;
- }
- }
-
- if (maxDistance < 0.0001) {
- return 'straight_enough';
- }
- };
-
- action.transitionable = true;
- return action;
- }
-
- /*
- * Based on https://github.com/openstreetmap/potlatch2/net/systemeD/potlatch2/tools/Straighten.as
- */
-
- 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
-
-
- 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"]
-
-
- 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
-
-
- while (remainingWays.length) {
- nextWay = getNextWay(currNode, remainingWays);
- remainingWays = utilArrayDifference(remainingWays, [nextWay]);
-
- if (nextWay[0] !== currNode) {
- nextWay.reverse();
- }
-
- nodes = nodes.concat(nextWay);
- currNode = nodes[nodes.length - 1];
- } // If user selected 2 nodes to straighten between, then slice nodes array to those nodes
-
-
- 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);
- }
-
- return nodes.map(function (n) {
- return graph.entity(n);
- });
- }
-
- function shouldKeepNode(node, graph) {
- return graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags();
- }
-
- 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 (i = 1; i < points.length - 1; i++) {
- var node = nodes[i];
- var point = points[i];
-
- 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);
- }
- }
- }
-
- for (i = 0; i < toDelete.length; i++) {
- graph = actionDeleteNode(toDelete[i].id)(graph);
- }
-
- return graph;
- };
-
- 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;
-
- if (threshold === 0) {
- return 'too_bendy';
- }
-
- var maxDistance = 0;
-
- 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 (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);
- });
-
- if (maxDistance < 0.0001 && // Allow straightening even if already straight in order to remove extraneous nodes
- keepingAllNodes) {
- return 'straight_enough';
- }
- };
-
- action.transitionable = true;
- return action;
- }
-
- //
- // `turn` must be an `osmTurn` object with a `restrictionID` property.
- // see osm/intersection.js, pathToTurn()
- //
-
- function actionUnrestrictTurn(turn) {
- return function (graph) {
- return actionDeleteRelation(turn.restrictionID)(graph);
- };
- }
-
- /* Reflect the given area around its axis of symmetry */
-
- function actionReflect(reflectIds, projection) {
- var _useLongAxis = true;
-
- 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.
-
- 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 (_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
-
-
- 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 (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);
- }
-
- return graph;
- };
-
- action.useLongAxis = function (val) {
- if (!arguments.length) return _useLongAxis;
- _useLongAxis = val;
- return action;
- };
-
- action.transitionable = true;
- return action;
- }
-
- function actionUpgradeTags(entityId, oldTags, replaceTags) {
- return function (graph) {
- var entity = graph.entity(entityId);
- var tags = Object.assign({}, entity.tags); // shallow copy
-
- var transferValue;
- var semiIndex;
-
- for (var oldTagKey in oldTags) {
- if (!(oldTagKey in tags)) continue; // wildcard match
-
- 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 (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;
- }
-
- vals.splice(oldIndex, 1);
- tags[oldTagKey] = vals.join(';');
- }
- }
- }
-
- if (replaceTags) {
- for (var replaceKey in replaceTags) {
- var replaceValue = replaceTags[replaceKey];
-
- 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;
- }
- }
- }
- }
-
- return graph.replace(entity.update({
- tags: tags
- }));
- };
- }
-
- function behaviorEdit(context) {
- function behavior() {
- context.map().minzoom(context.minEditableZoom());
- }
-
- 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 behaviorHover(context) {
- var dispatch$1 = dispatch('hover');
-
- var _selection = select(null);
-
- var _newNodeId = null;
- var _initialNodeID = null;
-
- var _altDisables;
-
- var _ignoreVertex;
-
- 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$1.call('hover', this, null);
- }
- }
-
- 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);
- }
- }
-
- function behavior(selection) {
- _selection = selection;
- _targets = [];
-
- if (_initialNodeID) {
- _newNodeId = _initialNodeID;
- _initialNodeID = null;
- } else {
- _newNodeId = null;
- }
-
- _selection.on(_pointerPrefix + 'over.hover', pointerover).on(_pointerPrefix + 'out.hover', pointerout) // treat pointerdown as pointerover for touch devices
- .on(_pointerPrefix + 'down.hover', pointerover);
-
- select(window).on(_pointerPrefix + 'up.hover pointercancel.hover', pointerout, true).on('keydown.hover', keydown).on('keyup.hover', keyup);
-
- function eventTarget(d3_event) {
- var datum = d3_event.target && d3_event.target.__data__;
- if (_typeof(datum) !== 'object') return null;
-
- if (!(datum instanceof osmEntity) && datum.properties && datum.properties.entity instanceof osmEntity) {
- return datum.properties.entity;
- }
-
- return datum;
- }
-
- 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);
-
- updateHover(d3_event, _targets);
- }
- }
-
- function pointerout(d3_event) {
- var target = eventTarget(d3_event);
-
- var index = _targets.indexOf(target);
-
- if (index !== -1) {
- _targets.splice(index);
-
- updateHover(d3_event, _targets);
- }
- }
-
- function allowsVertex(d) {
- return d.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(d, context.graph());
- }
-
- function modeAllowsHover(target) {
- var mode = context.mode();
-
- if (mode.id === 'add-point') {
- return mode.preset.matchGeometry('vertex') || target.type !== 'way' && target.geometry(context.graph()) !== 'vertex';
- }
-
- return true;
- }
-
- function updateHover(d3_event, targets) {
- _selection.selectAll('.hover').classed('hover', false);
-
- _selection.selectAll('.hover-suppressed').classed('hover-suppressed', false);
-
- 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;
- }
-
- 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 = '';
-
- for (var i in targets) {
- var datum = targets[i]; // What are we hovering over?
-
- 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 (datum.type === 'relation') {
- for (var j in datum.members) {
- selector += ', .' + datum.members[j].id;
- }
- }
- }
- }
-
- var suppressed = _altDisables && d3_event && d3_event.altKey;
-
- if (selector.trim().length) {
- // remove the first comma
- selector = selector.slice(1);
-
- _selection.selectAll(selector).classed(suppressed ? 'hover-suppressed' : 'hover', true);
- }
-
- dispatch$1.call('hover', this, !suppressed && targets);
- }
- }
-
- 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);
- };
-
- behavior.altDisables = function (val) {
- if (!arguments.length) return _altDisables;
- _altDisables = val;
- return behavior;
- };
-
- behavior.ignoreVertex = function (val) {
- if (!arguments.length) return _ignoreVertex;
- _ignoreVertex = val;
- return behavior;
- };
-
- behavior.initialNodeID = function (nodeId) {
- _initialNodeID = nodeId;
- return behavior;
- };
-
- return utilRebind(behavior, dispatch$1, 'on');
- }
-
- 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');
-
- var _hover = behaviorHover(context).altDisables(true).ignoreVertex(true).on('hover', context.ui().sidebar.hover);
-
- var _edit = behaviorEdit(context);
-
- var _closeTolerance = 4;
- var _tolerance = 12;
- var _mouseLeave = false;
- var _lastMouse = null;
-
- var _lastPointerUpEvent;
-
- var _downPointer; // use pointer events on supported platforms; fallback to mouse events
-
-
- var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse'; // related code
- // - `mode/drag_node.js` `datum()`
-
-
- 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 (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)
-
-
- var d = element.__data__;
- return d && d.properties && d.properties.target ? d : {};
- }
-
- 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 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 (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 pointermove(d3_event) {
- if (_downPointer && _downPointer.id === (d3_event.pointerId || 'mouse') && !_downPointer.isCancelled) {
- var p2 = _downPointer.pointerLocGetter(d3_event);
-
- var dist = geoVecLength(_downPointer.downLoc, p2);
-
- if (dist >= _closeTolerance) {
- _downPointer.isCancelled = true;
- dispatch$1.call('downcancel', this);
- }
- }
-
- 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));
- }
-
- function pointercancel(d3_event) {
- if (_downPointer && _downPointer.id === (d3_event.pointerId || 'mouse')) {
- if (!_downPointer.isCancelled) {
- dispatch$1.call('downcancel', this);
- }
-
- _downPointer = null;
- }
- }
-
- function mouseenter() {
- _mouseLeave = false;
- }
-
- function mouseleave() {
- _mouseLeave = true;
- }
-
- 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 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 (choice) {
- var edge = [target.nodes[choice.index - 1], target.nodes[choice.index]];
- dispatch$1.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$1.call('click', this, locLatLng, d);
- }
- } // treat a spacebar press like a click
-
-
- function space(d3_event) {
- d3_event.preventDefault();
- d3_event.stopPropagation();
- var currSpace = context.map().mouse();
-
- if (_disableSpace && _lastSpace) {
- var dist = geoVecLength(_lastSpace, currSpace);
-
- if (dist > _tolerance) {
- _disableSpace = false;
- }
- }
-
- if (_disableSpace || _mouseLeave || !_lastMouse) return; // user must move mouse or release space bar to allow another click
-
- _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
-
- 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');
- }
-
- function del(d3_event) {
- d3_event.preventDefault();
- dispatch$1.call('cancel');
- }
-
- function ret(d3_event) {
- d3_event.preventDefault();
- dispatch$1.call('finish');
- }
-
- 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;
- }
-
- 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
-
- select(document).call(keybinding.unbind);
- };
-
- behavior.hover = function () {
- return _hover;
- };
-
- return utilRebind(behavior, dispatch$1, 'on');
- }
-
- function initRange(domain, range) {
- switch (arguments.length) {
- case 0:
- break;
-
- case 1:
- this.range(domain);
- break;
-
- default:
- this.range(range).domain(domain);
- break;
- }
-
- return this;
- }
-
- function constants(x) {
- return function () {
- return x;
- };
- }
-
- function number$1(x) {
- return +x;
- }
-
- var unit = [0, 1];
- function identity$3(x) {
- return x;
- }
-
- function normalize$1(a, b) {
- return (b -= a = +a) ? function (x) {
- return (x - a) / b;
- } : constants(isNaN(b) ? NaN : 0.5);
- }
-
- 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].
-
-
- 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));
- };
- }
-
- 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();
- }
-
- while (++i < j) {
- d[i] = normalize$1(domain[i], domain[i + 1]);
- r[i] = interpolate(range[i], range[i + 1]);
- }
-
- 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());
- }
- function transformer$1() {
- var domain = unit,
- range = unit,
- interpolate$1 = interpolate,
- transform,
- untransform,
- unknown,
- clamp = identity$3,
- piecewise,
- output,
- input;
-
- 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 scale(x) {
- return isNaN(x = +x) ? unknown : (output || (output = piecewise(domain.map(transform), range, interpolate$1)))(transform(clamp(x)));
- }
-
- 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();
- };
-
- scale.range = function (_) {
- return arguments.length ? (range = Array.from(_), rescale()) : range.slice();
- };
-
- scale.rangeRound = function (_) {
- return range = Array.from(_), interpolate$1 = interpolateRound, rescale();
- };
-
- scale.clamp = function (_) {
- return arguments.length ? (clamp = _ ? true : identity$3, rescale()) : clamp !== identity$3;
- };
-
- scale.interpolate = function (_) {
- return arguments.length ? (interpolate$1 = _, rescale()) : interpolate$1;
- };
-
- scale.unknown = function (_) {
- return arguments.length ? (unknown = _, scale) : unknown;
- };
-
- return function (t, u) {
- transform = t, untransform = u;
- return rescale();
- };
- }
- function continuous() {
- return transformer$1()(identity$3, identity$3);
- }
-
- 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
-
- 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 [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;
- }
-
- function formatGroup (grouping, thousands) {
- return function (value, width) {
- var i = value.length,
- t = [],
- j = 0,
- g = grouping[0],
- length = 0;
-
- 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);
- };
- }
-
- function formatNumerals (numerals) {
- return function (value) {
- return value.replace(/[0-9]/g, function (i) {
- return numerals[+i];
- });
- };
- }
-
- // [[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
-
- 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;
-
- case "0":
- if (i0 === 0) i0 = i;
- i1 = i;
- break;
-
- default:
- if (!+s[i]) break out;
- if (i0 > 0) i0 = 0;
- break;
- }
- }
-
- return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;
- }
-
- // `thisNumberValue` abstract operation
- // https://tc39.github.io/ecma262/#sec-thisnumbervalue
- var thisNumberValue = function (value) {
- if (typeof value != 'number' && classofRaw(value) != 'Number') {
- throw TypeError('Incorrect invocation');
- }
- return +value;
- };
-
- // `String.prototype.repeat` method implementation
- // https://tc39.github.io/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;
- };
-
- 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);
- };
-
- 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;
- };
-
- 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({});
- });
-
- // `Number.prototype.toFixed` method
- // https://tc39.github.io/ecma262/#sec-number.prototype.tofixed
- _export({ target: 'Number', proto: true, forced: FORCED$c }, {
- // eslint-disable-next-line max-statements
- 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;
-
- var multiply = function (n, c) {
- var index = -1;
- var c2 = c;
- while (++index < 6) {
- c2 += n * data[index];
- data[index] = c2 % 1e7;
- c2 = floor$6(c2 / 1e7);
- }
- };
-
- var divide = function (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 () {
- 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 (fractDigits < 0 || fractDigits > 20) throw RangeError('Incorrect fraction digits');
- // eslint-disable-next-line no-self-compare
- 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(0, z);
- j = fractDigits;
- while (j >= 7) {
- multiply(1e7, 0);
- j -= 7;
- }
- multiply(pow$2(10, j, 1), 0);
- j = e - 1;
- while (j >= 23) {
- divide(1 << 23);
- j -= 23;
- }
- divide(1 << j);
- multiply(1, 1);
- divide(2);
- result = dataToString();
- } else {
- multiply(0, z);
- multiply(1 << -e, 0);
- result = dataToString() + 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.github.io/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);
- }
- };
-
- 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".
-
- 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.
-
- if (zero || fill === "0" && align === "=") zero = true, fill = "0", align = "="; // Compute the prefix and suffix.
- // For SI-prefix, the suffix is lazily computed.
-
- 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 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;
-
- 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 valueNegative = value < 0 || 1 / value < 0; // Perform the initial formatting.
-
- value = isNaN(value) ? nan : formatType(Math.abs(value), precision); // Trim insignificant zeros.
-
- 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 (valueNegative && +value === 0 && sign !== "+") valueNegative = false; // Compute the prefix and suffix.
-
- 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 (maybeSuffix) {
- i = -1, n = value.length;
-
- 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.
-
-
- if (comma && !zero) value = group(value, Infinity); // Compute the padding.
-
- 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 (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = ""; // Reconstruct the final output based on the desired alignment.
-
- switch (align) {
- case "<":
- value = valuePrefix + value + valueSuffix + padding;
- break;
-
- case "=":
- value = valuePrefix + padding + value + valueSuffix;
- break;
-
- case "^":
- value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length);
- break;
-
- default:
- value = padding + valuePrefix + value + valueSuffix;
- break;
- }
-
- return numerals(value);
- }
-
- format.toString = function () {
- return specifier + "";
- };
-
- return format;
- }
-
- 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 {
- format: newFormat,
- formatPrefix: formatPrefix
- };
- }
-
- 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 precisionFixed (step) {
- return Math.max(0, -exponent(Math.abs(step)));
- }
-
- 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 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);
-
- 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);
- }
-
- 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;
- }
- }
-
- return format(specifier);
- }
-
- 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);
- };
-
- scale.tickFormat = function (count, specifier) {
- var d = domain();
- return tickFormat(d[0], d[d.length - 1], count == null ? 10 : count, specifier);
- };
-
- 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;
- }
-
- while (maxIter-- > 0) {
- step = tickIncrement(start, stop, count);
-
- 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;
- }
-
- 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.github.io/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;
-
- function scale(x) {
- return x <= x ? range[bisectRight(domain, x, 0, n)] : unknown;
- }
-
- function rescale() {
- var i = -1;
- domain = new Array(n);
-
- while (++i < n) {
- domain[i] = ((i + 1) * x1 - (i - n) * x0) / (n + 1);
- }
-
- 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
-
-
-
-
- var ceil$1 = Math.ceil;
-
- // `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;
- };
- };
-
- var stringPad = {
- // `String.prototype.padStart` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.padstart
- start: createMethod$6(false),
- // `String.prototype.padEnd` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.padend
- end: createMethod$6(true)
- };
-
- var padStart = stringPad.start;
-
- var abs$3 = Math.abs;
- var DatePrototype$1 = Date.prototype;
- var getTime$1 = DatePrototype$1.getTime;
- var nativeDateToISOString = DatePrototype$1.toISOString;
-
- // `Date.prototype.toISOString` method implementation
- // https://tc39.github.io/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;
-
- // `Date.prototype.toISOString` method
- // https://tc39.github.io/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 behaviorBreathe() {
- var duration = 800;
- var steps = 4;
- var selector = '.selected.shadow, .selected .shadow';
-
- var _selected = select(null);
-
- var _classed = '';
- var _params = {};
- var _done = false;
-
- var _timer;
-
- 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 || '');
- };
- }
-
- 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');
- });
- }
-
- 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..
-
-
- 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;
- });
- }
-
- 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);
-
- _selected = select(null);
- return;
- }
-
- if (!fastDeepEqual(currSelected.data(), _selected.data()) || currClassed !== _classed) {
- _selected.call(reset);
-
- _classed = currClassed;
- _selected = currSelected.call(calcAnimationParams);
- }
-
- var didCallNextRun = false;
-
- _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 (!select(this).classed('selected')) {
- reset(select(this));
- }
- });
- }
-
- function behavior(surface) {
- _done = false;
- _timer = timer(function () {
- // wait for elements to actually become selected
- if (surface.selectAll(selector).empty()) {
- return false;
- }
-
- surface.call(run, 'from');
-
- _timer.stop();
-
- return true;
- }, 20);
- }
-
- behavior.restartIfNeeded = function (surface) {
- if (_selected.empty()) {
- surface.call(run, 'from');
-
- if (_timer) {
- _timer.stop();
- }
- }
- };
-
- behavior.off = function () {
- _done = true;
-
- if (_timer) {
- _timer.stop();
- }
-
- _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();
- }
- }
-
- function behavior() {
- if (_operation && _operation.available()) {
- context.keybinding().on(_operation.keys, keypress);
- }
-
- return behavior;
- }
-
- behavior.off = function () {
- context.keybinding().off(_operation.keys);
- };
-
- behavior.which = function (_) {
- if (!arguments.length) return _operation;
- _operation = _;
- return behavior;
- };
-
- return behavior;
- }
-
- function operationCircularize(context, selectedIDs) {
- var _extent;
-
- var _actions = selectedIDs.map(getAction).filter(Boolean);
-
- var _amount = _actions.length === 1 ? 'single' : 'multiple';
-
- var _coords = utilGetAllNodes(selectedIDs, context.graph()).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 (!_extent) {
- _extent = entity.extent(context.graph());
- } else {
- _extent = _extent.extend(entity.extent(context.graph()));
- }
-
- return actionCircularize(entityID, context.projection);
- }
-
- var operation = function operation() {
- if (!_actions.length) return;
-
- var combinedAction = function combinedAction(graph, t) {
- _actions.forEach(function (action) {
- if (!action.disabled(graph)) {
- graph = action(graph, t);
- }
- });
-
- 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
-
-
- operation.disabled = function () {
- if (!_actions.length) return '';
-
- var actionDisableds = _actions.map(function (action) {
- return action.disabled(context.graph());
- }).filter(Boolean);
-
- if (actionDisableds.length === _actions.length) {
- // none of the features can be circularized
- if (new Set(actionDisableds).size > 1) {
- return 'multiple_blockers';
- }
-
- 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 false;
-
- function someMissing() {
- if (context.inIntro()) return false;
- var osm = context.connection();
-
- if (osm) {
- var missing = _coords.filter(function (loc) {
- return !osm.isDataLoaded(loc);
- });
-
- if (missing.length) {
- missing.forEach(function (loc) {
- context.loadTileAtLoc(loc);
- });
- return true;
- }
- }
-
- return false;
- }
- };
-
- 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
- });
- };
-
- operation.id = 'circularize';
- operation.keys = [_t('operations.circularize.key')];
- operation.title = _t('operations.circularize.title');
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
- }
-
- // For example, ⌘Z -> Ctrl+Z
-
- var uiCmd = function uiCmd(code) {
- var detected = utilDetect();
-
- if (detected.os === 'mac') {
- return code;
- }
-
- if (detected.os === 'win') {
- if (code === '⌘⇧Z') return 'Ctrl+Y';
- }
-
- var result = '',
- replacements = {
- '⌘': 'Ctrl',
- '⇧': 'Shift',
- '⌥': 'Alt',
- '⌫': 'Backspace',
- '⌦': 'Delete'
- };
-
- 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 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;
- };
-
- 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());
-
- var operation = function operation() {
- var nextSelectedID;
- var nextSelectedLoc;
-
- 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;
- }
- }
-
- context.perform(action, operation.annotation());
- context.validator().validate();
-
- if (nextSelectedID && nextSelectedLoc) {
- if (context.hasEntity(nextSelectedID)) {
- context.enter(modeSelect(context, [nextSelectedID]).follow(true));
- } else {
- context.map().centerEase(nextSelectedLoc);
- context.enter(modeBrowse(context));
- }
- } else {
- context.enter(modeBrowse(context));
- }
- };
-
- operation.available = function () {
- return true;
- };
-
- operation.disabled = function () {
- if (extent.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';
- }
-
- return false;
-
- function someMissing() {
- if (context.inIntro()) return false;
- var osm = context.connection();
-
- if (osm) {
- var missing = coords.filter(function (loc) {
- return !osm.isDataLoaded(loc);
- });
-
- if (missing.length) {
- missing.forEach(function (loc) {
- context.loadTileAtLoc(loc);
- });
- return true;
- }
- }
-
- return false;
- }
-
- function hasWikidataTag(id) {
- var entity = context.entity(id);
- return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
- }
-
- function incompleteRelation(id) {
- var entity = context.entity(id);
- return entity.type === 'relation' && !entity.isComplete(context.graph());
- }
-
- function protectedMember(id) {
- var entity = context.entity(id);
- if (entity.type !== 'way') return false;
- var parents = context.graph().parentRelations(entity);
-
- for (var i = 0; i < parents.length; i++) {
- var parent = parents[i];
- var type = parent.tags.type;
- var role = parent.memberById(id).role || 'outer';
-
- if (type === 'route' || type === 'boundary' || type === 'multipolygon' && role === 'outer') {
- return true;
- }
- }
-
- return false;
- }
- };
-
- operation.tooltip = function () {
- var disable = operation.disabled();
- return disable ? _t('operations.delete.' + disable + '.' + multi) : _t('operations.delete.description.' + multi);
- };
-
- operation.annotation = function () {
- return selectedIDs.length === 1 ? _t('operations.delete.annotation.' + context.graph().geometry(selectedIDs[0])) : _t('operations.delete.annotation.feature', {
- n: selectedIDs.length
- });
- };
-
- operation.id = 'delete';
- operation.keys = [uiCmd('⌘⌫'), uiCmd('⌘⌦'), uiCmd('⌦')];
- operation.title = _t('operations.delete.title');
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
- }
-
- function operationOrthogonalize(context, selectedIDs) {
- var _extent;
-
- var _type;
-
- var _actions = selectedIDs.map(chooseAction).filter(Boolean);
-
- var _amount = _actions.length === 1 ? 'single' : 'multiple';
-
- var _coords = utilGetAllNodes(selectedIDs, context.graph()).map(function (n) {
- return n.loc;
- });
-
- function chooseAction(entityID) {
- var entity = context.entity(entityID);
- var geometry = entity.geometry(context.graph());
-
- if (!_extent) {
- _extent = entity.extent(context.graph());
- } else {
- _extent = _extent.extend(entity.extent(context.graph()));
- } // square a line/area
-
-
- 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 (parents.length === 1) {
- var way = parents[0];
-
- if (way.nodes.indexOf(entityID) !== -1) {
- return actionOrthogonalize(way.id, context.projection, entityID);
- }
- }
- }
-
- return null;
- }
-
- var operation = function operation() {
- if (!_actions.length) return;
-
- var combinedAction = function combinedAction(graph, t) {
- _actions.forEach(function (action) {
- if (!action.disabled(graph)) {
- graph = action(graph, t);
- }
- });
-
- 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
-
-
- operation.disabled = function () {
- if (!_actions.length) return '';
-
- var actionDisableds = _actions.map(function (action) {
- return action.disabled(context.graph());
- }).filter(Boolean);
-
- if (actionDisableds.length === _actions.length) {
- // none of the features can be squared
- if (new Set(actionDisableds).size > 1) {
- return 'multiple_blockers';
- }
-
- 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 (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;
- }
- };
-
- 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
- });
- };
-
- operation.id = 'orthogonalize';
- operation.keys = [_t('operations.orthogonalize.key')];
- operation.title = _t('operations.orthogonalize.title');
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
- }
-
- 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());
-
- 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
-
-
- 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 someMissing() {
- if (context.inIntro()) return false;
- var osm = context.connection();
-
- if (osm) {
- var missing = coords.filter(function (loc) {
- return !osm.isDataLoaded(loc);
- });
-
- if (missing.length) {
- missing.forEach(function (loc) {
- context.loadTileAtLoc(loc);
- });
- return true;
- }
- }
-
- return false;
- }
-
- function 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.reflect.' + disable + '.' + multi) : _t('operations.reflect.description.' + axis + '.' + multi);
- };
-
- operation.annotation = function () {
- return _t('operations.reflect.annotation.' + axis + '.feature', {
- n: selectedIDs.length
- });
- };
-
- 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 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());
-
- var operation = function operation() {
- context.enter(modeMove(context, selectedIDs));
- };
-
- operation.available = function () {
- return selectedIDs.length > 1 || context.entity(selectedIDs[0]).type !== 'node';
- };
-
- 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 someMissing() {
- if (context.inIntro()) return false;
- var osm = context.connection();
-
- if (osm) {
- var missing = coords.filter(function (loc) {
- return !osm.isDataLoaded(loc);
- });
-
- if (missing.length) {
- missing.forEach(function (loc) {
- context.loadTileAtLoc(loc);
- });
- return true;
- }
- }
-
- return false;
- }
-
- function 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
- });
- };
-
- 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
- });
-
- var _prevGraph;
-
- var _prevAngle;
-
- var _prevTransform;
-
- var _pivot;
-
- function doRotate() {
- var fn;
-
- if (context.graph() !== _prevGraph) {
- fn = context.perform;
- } else {
- fn = context.replace;
- } // projection changed, recalculate _pivot
-
-
- var projection = context.projection;
- var currTransform = projection.transform();
-
- 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;
- }
-
- 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));
- }
- }
-
- 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));
- }
-
- 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);
- };
-
- 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([]);
- };
-
- mode.selectedIDs = function () {
- if (!arguments.length) return entityIDs; // no assign
-
- return mode;
- };
-
- return mode;
- }
-
- 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 operation = function operation() {
- context.enter(modeRotate(context, selectedIDs));
- };
-
- operation.available = function () {
- return nodes.length >= 2;
- };
-
- 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 someMissing() {
- if (context.inIntro()) return false;
- var osm = context.connection();
-
- if (osm) {
- var missing = coords.filter(function (loc) {
- return !osm.isDataLoaded(loc);
- });
-
- if (missing.length) {
- missing.forEach(function (loc) {
- context.loadTileAtLoc(loc);
- });
- return true;
- }
- }
-
- return false;
- }
-
- function 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);
- };
-
- 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;
- }
-
- 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 _prevGraph;
-
- var _cache;
-
- var _origin;
-
- var _nudgeInterval;
-
- function doMove(nudge) {
- nudge = nudge || [0, 0];
- var fn;
-
- if (_prevGraph !== context.graph()) {
- _cache = {};
- _origin = context.map().mouseCoordinates();
- fn = context.perform;
- } else {
- fn = context.overwrite;
- }
-
- 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();
- }
-
- function startNudge(nudge) {
- if (_nudgeInterval) window.clearInterval(_nudgeInterval);
- _nudgeInterval = window.setInterval(function () {
- context.map().pan(nudge);
- doMove(nudge);
- }, 50);
- }
-
- function stopNudge() {
- if (_nudgeInterval) {
- window.clearInterval(_nudgeInterval);
- _nudgeInterval = null;
- }
- }
-
- function move() {
- doMove();
- var nudge = geoViewportEdge(context.map().mouse(), context.map().dimensions());
-
- if (nudge) {
- startNudge(nudge);
- } else {
- stopNudge();
- }
- }
-
- function finish(d3_event) {
- d3_event.stopPropagation();
- context.replace(actionNoop(), annotation);
- context.enter(modeSelect(context, entityIDs));
- stopNudge();
- }
-
- function cancel() {
- if (baseGraph) {
- while (context.graph() !== baseGraph) {
- context.pop();
- }
-
- context.enter(modeBrowse(context));
- } else {
- context.pop();
- context.enter(modeSelect(context, entityIDs));
- }
-
- stopNudge();
- }
-
- function undone() {
- context.enter(modeBrowse(context));
- }
-
- 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);
- };
-
- 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([]);
- };
-
- mode.selectedIDs = function () {
- if (!arguments.length) return entityIDs; // no assign
-
- return mode;
- };
-
- return mode;
- }
-
- 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];
-
- extent._extend(oldEntity.extent(oldGraph)); // Exclude child nodes from newIDs if their parent way was also copied.
-
-
- var parents = context.graph().parentWays(newEntity);
- var parentCopied = parents.some(function (parent) {
- return originals.has(parent.id);
- });
-
- if (!parentCopied) {
- newIDs.push(newEntity.id);
- }
- } // Put pasted objects where mouse pointer is..
-
-
- 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));
- }
-
- function behavior() {
- context.keybinding().on(uiCmd('⌘V'), doPaste);
- return behavior;
- }
-
- behavior.off = function () {
- context.keybinding().off(uiCmd('⌘V'));
- };
-
- return behavior;
- }
-
- // `String.prototype.repeat` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.repeat
- _export({ target: 'String', proto: true }, {
- repeat: stringRepeat
- });
-
- /*
- `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 behaviorDrag() {
- var dispatch$1 = dispatch('start', 'move', 'end'); // see also behaviorSelect
-
- var _tolerancePx = 1; // keep this low to facilitate pixel-perfect micromapping
-
- var _penTolerancePx = 4; // styluses can be touchy so require greater movement - #1981
-
- var _origin = null;
- var _selector = '';
-
- var _targetNode;
-
- var _targetEntity;
-
- var _surface;
-
- var _pointerId; // use pointer events on supported platforms; fallback to mouse events
-
-
- var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
-
- var d3_event_userSelectProperty = utilPrefixCSSProperty('UserSelect');
-
- 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];
- }
-
- d3_event.stopPropagation();
-
- function pointermove(d3_event) {
- if (_pointerId !== (d3_event.pointerId || 'mouse')) return;
- var p = pointerLocGetter(d3_event);
-
- 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 (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 pointerup(d3_event) {
- if (_pointerId !== (d3_event.pointerId || 'mouse')) return;
- _pointerId = null;
-
- 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();
- }
- }
-
- 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;
-
- 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);
- }
- }
- };
- }
-
- selection.on(_pointerPrefix + 'down.drag' + _selector, delegate);
- }
-
- behavior.off = function (selection) {
- selection.on(_pointerPrefix + 'down.drag' + _selector, null);
- };
-
- behavior.selector = function (_) {
- if (!arguments.length) return _selector;
- _selector = _;
- return behavior;
- };
-
- 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;
- };
-
- behavior.targetNode = function (_) {
- if (!arguments.length) return _targetNode;
- _targetNode = _;
- return behavior;
- };
-
- behavior.targetEntity = function (_) {
- if (!arguments.length) return _targetEntity;
- _targetEntity = _;
- return behavior;
- };
-
- behavior.surface = function (_) {
- if (!arguments.length) return _surface;
- _surface = _;
- return behavior;
- };
-
- return utilRebind(behavior, dispatch$1, 'on');
- }
-
- 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);
-
- var _nudgeInterval;
-
- var _restoreSelectedIDs = [];
- var _wasMidpoint = false;
- var _isCancelled = false;
-
- var _activeEntity;
-
- var _startLoc;
-
- var _lastLoc;
-
- 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);
- }
-
- function stopNudge() {
- if (_nudgeInterval) {
- window.clearInterval(_nudgeInterval);
- _nudgeInterval = null;
- }
- }
-
- 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());
-
- 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
-
- 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');
- }
-
- return _t('operations.connect.annotation.from_vertex.to_sibling_vertex');
- }
- }
-
- return _t('operations.connect.annotation.from_' + nodeGeometry + '.to_' + targetGeometry);
- }
-
- function shouldSnapToNode(target) {
- if (!_activeEntity) return false;
- return _activeEntity.geometry(context.graph()) !== 'vertex' || target.geometry(context.graph()) === 'vertex' || _mainPresetIndex.allowsVertex(target, context.graph());
- }
-
- function origin(entity) {
- return context.projection(entity.loc);
- }
-
- function keydown(d3_event) {
- if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
- if (context.surface().classed('nope')) {
- context.surface().classed('nope-suppressed', true);
- }
-
- 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);
- }
-
- 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 (_isCancelled) {
- if (hasHidden) {
- context.ui().flash.duration(4000).iconName('#iD-icon-no').label(_t('modes.drag_node.connected_to_hidden'))();
- }
-
- return drag.cancel();
- }
-
- 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());
- }
-
- _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()`
-
-
- 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 : {};
- }
- }
-
- 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 (!_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 (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);
-
- if (edge) {
- loc = edge.loc;
- }
- }
- }
-
- context.replace(actionMoveNode(entity.id, loc)); // Below here: validations
-
- var isInvalid = false; // Check if this connection to `target` could cause relations to break..
-
- if (target) {
- isInvalid = hasRelationConflict(entity, target, edge, context.graph());
- } // Check if this drag causes the geometry to break..
-
-
- if (!isInvalid) {
- isInvalid = hasInvalidGeometry(entity, context.graph());
- }
-
- 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('')();
- }
- }
-
- 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);
- }
-
- _lastLoc = loc;
- } // Uses `actionConnect.disabled()` to know whether this connection is ok..
-
-
- function hasRelationConflict(entity, target, edge, graph) {
- var testGraph = graph.update(); // copy
- // if snapping to way - add midpoint there and consider that the target..
-
- 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?
-
-
- var ids = [entity.id, target.id];
- return actionConnect(ids).disabled(testGraph);
- }
-
- function hasInvalidGeometry(entity, graph) {
- var parents = graph.parentWays(entity);
- var i, j, k;
-
- 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
-
- var relations = graph.parentRelations(parent);
-
- 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 (k = 0; k < rings.length; k++) {
- nodes = rings[k].nodes;
-
- if (nodes.find(function (n) {
- return n.id === entity.id;
- })) {
- activeIndex = k;
-
- if (geoHasSelfIntersections(nodes, entity.id)) {
- return 'multipolygonMember';
- }
- }
-
- rings[k].coords = nodes.map(function (n) {
- return n.loc;
- });
- } // test active ring for intersections with other rings in the multipolygon
-
-
- for (k = 0; k < rings.length; k++) {
- if (k === activeIndex) continue; // make sure active ring doesn't cross passive rings
-
- 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.
-
-
- if (activeIndex === null) {
- nodes = parent.nodes.map(function (nodeID) {
- return graph.entity(nodeID);
- });
-
- if (nodes.length && geoHasSelfIntersections(nodes, entity.id)) {
- return parent.geometry(graph);
- }
- }
- }
-
- 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();
- }
- }
-
- 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));
- }
-
- if (wasPoint) {
- context.enter(modeSelect(context, [entity.id]));
- } else {
- var reselection = _restoreSelectedIDs.filter(function (id) {
- return context.graph().hasEntity(id);
- });
-
- if (reselection.length) {
- context.enter(modeSelect(context, reselection));
- } else {
- context.enter(modeBrowse(context));
- }
- }
- }
-
- function _actionBounceBack(nodeID, toLoc) {
- var moveNode = actionMoveNode(nodeID, toLoc);
-
- 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;
- }
-
- function cancel() {
- drag.cancel();
- context.enter(modeBrowse(context));
- }
-
- 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);
-
- 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();
- };
-
- mode.selectedIDs = function () {
- if (!arguments.length) return _activeEntity ? [_activeEntity.id] : []; // no assign
-
- return mode;
- };
-
- mode.activeID = function () {
- if (!arguments.length) return _activeEntity && _activeEntity.id; // no assign
-
- return mode;
- };
-
- mode.restoreSelectedIDs = function (_) {
- if (!arguments.length) return _restoreSelectedIDs;
- _restoreSelectedIDs = _;
- return mode;
- };
-
- mode.behavior = drag;
- return mode;
- }
-
- // 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.github.io/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
- );
- }
- });
-
- // patch native Promise.prototype for native async functions
- if ( typeof nativePromiseConstructor == 'function' && !nativePromiseConstructor.prototype['finally']) {
- redefine(nativePromiseConstructor.prototype, 'finally', getBuiltIn('Promise').prototype['finally']);
- }
-
- // @@search logic
- fixRegexpWellKnownSymbolLogic('search', 1, function (SEARCH, nativeSearch, maybeCallNative) {
- return [
- // `String.prototype.search` method
- // https://tc39.github.io/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.github.io/ecma262/#sec-regexp.prototype-@@search
- function (regexp) {
- var res = maybeCallNative(nativeSearch, regexp, this);
- if (res.done) return res.value;
-
- var rx = anObject(regexp);
- var S = String(this);
-
- 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;
- }
- ];
- });
-
- function quickselect$1(arr, k, left, right, compare) {
- quickselectStep(arr, k, left || 0, right || arr.length - 1, compare || defaultCompare);
- }
-
- 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$1(arr, left, k);
- if (compare(arr[right], t) > 0) swap$1(arr, left, right);
-
- while (i < j) {
- swap$1(arr, i, j);
- i++;
- j--;
-
- while (compare(arr[i], t) < 0) {
- i++;
- }
-
- while (compare(arr[j], t) > 0) {
- j--;
- }
- }
-
- if (compare(arr[left], t) === 0) swap$1(arr, left, j);else {
- j++;
- swap$1(arr, j, right);
- }
- if (j <= k) left = j + 1;
- if (k <= j) right = j - 1;
- }
- }
-
- function swap$1(arr, i, j) {
- var tmp = arr[i];
- arr[i] = arr[j];
- arr[j] = tmp;
- }
-
- function defaultCompare(a, b) {
- return a < b ? -1 : a > b ? 1 : 0;
- }
-
- var RBush = /*#__PURE__*/function () {
- function RBush() {
- var maxEntries = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 9;
-
- _classCallCheck(this, RBush);
-
- // 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();
- }
-
- _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;
-
- if (intersects(bbox, childBBox)) {
- if (node.leaf) result.push(child);else if (contains(bbox, childBBox)) this._all(child, result);else nodesToSearch.push(child);
- }
- }
-
- node = nodesToSearch.pop();
- }
-
- 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;
-
- if (intersects(bbox, childBBox)) {
- if (node.leaf || contains(bbox, childBBox)) return true;
- nodesToSearch.push(child);
- }
- }
-
- node = nodesToSearch.pop();
- }
-
- return false;
- }
- }, {
- key: "load",
- value: function load(data) {
- if (!(data && data.length)) return this;
-
- 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
-
-
- 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
-
-
- 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
-
- 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);
-
- if (index !== -1) {
- // item found, remove the item and condense tree upwards
- node.children.splice(index, 1);
- path.push(node);
-
- this._condense(path);
-
- return this;
- }
- }
-
- 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 = [];
-
- while (node) {
- if (node.leaf) result.push.apply(result, _toConsumableArray(node.children));else nodesToSearch.push.apply(nodesToSearch, _toConsumableArray(node.children));
- node = nodesToSearch.pop();
- }
-
- 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;
- }
-
- 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
-
- M = Math.ceil(N / Math.pow(M, height - 1));
- }
-
- node = createNode([]);
- node.leaf = false;
- node.height = height; // split the items into M mostly square tiles
-
- var N2 = Math.ceil(N / M);
- var N1 = N2 * Math.ceil(Math.sqrt(M));
- multiSelect(items, left, right, N1, this.compareMinX);
-
- for (var i = left; i <= right; i += N1) {
- var right2 = Math.min(i + N1 - 1, right);
- multiSelect(items, i, right2, N2, this.compareMinY);
-
- 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));
- }
- }
-
- 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;
-
- 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 (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;
- }
- }
- }
-
- node = targetNode || node.children[0];
- }
-
- 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
-
-
- node.children.push(item);
- extend$1(node, bbox); // split on node overflow; propagate upwards if necessary
-
- while (level >= 0) {
- if (insertPath[level].children.length > this._maxEntries) {
- this._split(insertPath, level);
-
- level--;
- } else break;
- } // adjust bboxes along the insertion path
-
-
- this._adjustParentBBoxes(bbox, insertPath, level);
- } // split overflowed node into two
-
- }, {
- key: "_split",
- value: function _split(insertPath, level) {
- var node = insertPath[level];
- var M = node.children.length;
- var m = this._minEntries;
-
- this._chooseSplitAxis(node, m, M);
-
- var splitIndex = this._chooseSplitIndex(node, m, M);
-
- 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
-
- 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 index || M - m;
- } // sorts node children by the best axis for split
-
- }, {
- 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);
-
- 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
-
-
- 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);
-
- 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);
- }
-
- 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 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;
- }
-
- return -1;
- } // calculate node's bbox from bboxes of its children
-
-
- function calcBBox(node, toBBox) {
- distBBox(node, 0, node.children.length, toBBox, node);
- } // min bounding rectangle of node children from k to p-1
-
-
- function distBBox(node, k, p, toBBox, destNode) {
- if (!destNode) destNode = createNode(null);
- destNode.minX = Infinity;
- destNode.minY = Infinity;
- destNode.maxX = -Infinity;
- destNode.maxY = -Infinity;
-
- for (var i = k; i < p; i++) {
- var child = node.children[i];
- extend$1(destNode, node.leaf ? toBBox(child) : child);
- }
-
- return destNode;
- }
-
- 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 compareNodeMinX(a, b) {
- return a.minX - b.minX;
- }
-
- function compareNodeMinY(a, b) {
- return a.minY - b.minY;
- }
-
- function bboxArea(a) {
- return (a.maxX - a.minX) * (a.maxY - a.minY);
- }
-
- function bboxMargin(a) {
- return a.maxX - a.minX + (a.maxY - a.minY);
- }
-
- 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 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);
- }
-
- function contains(a, b) {
- return a.minX <= b.minX && a.minY <= b.minY && b.maxX <= a.maxX && b.maxY <= a.maxY;
- }
-
- function intersects(a, b) {
- return b.minX <= a.maxX && b.minY <= a.maxY && b.maxX >= a.minX && b.maxY >= a.minY;
- }
-
- 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
-
-
- function multiSelect(arr, left, right, n, compare) {
- var stack = [left, right];
-
- 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);
- }
- }
-
- 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
-
- var _cache;
-
- 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];
-
- function abortRequest(controller) {
- if (controller) {
- controller.abort();
- }
- }
-
- function abortUnwantedRequests(cache, tiles) {
- Object.keys(cache.inflightTile).forEach(function (k) {
- var wanted = tiles.find(function (tile) {
- return k === tile.id;
- });
-
- if (!wanted) {
- abortRequest(cache.inflightTile[k]);
- delete cache.inflightTile[k];
- }
- });
- }
-
- 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
-
-
- function updateRtree(item, replace) {
- _cache.rtree.remove(item, function (a, b) {
- return a.data.id === b.data.id;
- });
-
- if (replace) {
- _cache.rtree.insert(item);
- }
- }
-
- function tokenReplacements(d) {
- if (!(d instanceof QAItem)) return;
- var htmlRegex = new RegExp(/<\/[a-z][\s\S]*>/);
- var replacements = {};
- var issueTemplate = _krData.errorTypes[d.whichType];
-
- if (!issueTemplate) {
- /* eslint-disable no-console */
- console.log('No Template: ', d.whichType);
- console.log(' ', d.description);
- /* eslint-enable no-console */
-
- return;
- } // some descriptions are just fixed text
-
-
- 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);
-
- if (!errorMatch) {
- /* eslint-disable no-console */
- console.log('Unmatched: ', d.whichType);
- console.log(' ', d.description);
- console.log(' ', errorRegex);
- /* eslint-enable no-console */
-
- return;
- }
-
- 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 (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]);
- }
- }
-
- replacements['var' + i] = capture;
- }
-
- return replacements;
- }
-
- 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]);
- }
-
- switch (idType) {
- // link a string like "this node"
- case 'this':
- capture = linkErrorObject(capture);
- break;
-
- case 'url':
- capture = linkURL(capture);
- break;
- // link an entity ID
-
- case 'n':
- case 'w':
- case 'r':
- capture = linkEntity(idType + capture);
- break;
- // some errors have more complex ID lists/variance
-
- case '20':
- capture = parse20(capture);
- break;
-
- case '211':
- capture = parse211(capture);
- break;
-
- case '231':
- capture = parse231(capture);
- break;
-
- case '294':
- capture = parse294(capture);
- break;
-
- case '370':
- capture = parse370(capture);
- break;
- }
-
- return capture;
-
- function linkErrorObject(d) {
- return "<a class=\"error_object_link\">".concat(d, "</a>");
- }
-
- function linkEntity(d) {
- return "<a class=\"error_entity_link\">".concat(d, "</a>");
- }
-
- 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 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 parse231(capture) {
- var newList = []; // unfortunately 'layer' can itself contain commas, so we split on '),'
-
- 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 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
-
- 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
-
- 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 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 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 serviceKeepRight = {
- title: 'keepRight',
- init: function init() {
- _mainFileFetcher.get('keepRight').then(function (d) {
- return _krData = d;
- });
-
- if (!_cache) {
- this.reset();
- }
-
- this.event = utilRebind(this, dispatch$1, 'on');
- },
- reset: function reset() {
- if (_cache) {
- Object.values(_cache.inflightTile).forEach(abortRequest);
- }
-
- _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;
-
- var options = {
- format: 'geojson',
- ch: _krRuleset
- }; // determine the needed tiles to cover the view
-
- var tiles = tiler.zoomExtent([_tileZoom, _tileZoom]).getTiles(projection); // abort inflight requests that are no longer needed
-
- abortUnwantedRequests(_cache, tiles); // issue new requests..
-
- tiles.forEach(function (tile) {
- if (_cache.loadedTile[tile.id] || _cache.inflightTile[tile.id]) return;
-
- 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];
-
- 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 (!data || !data.features || !data.features.length) {
- throw new Error('No Data');
- }
-
- 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.
-
- 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;
-
- case '292':
- case '293':
- description = description.replace('A turn-', 'This turn-');
- break;
-
- 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;
-
- 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
-
-
- var coincident = false;
-
- 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);
-
- 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;
-
- _cache.rtree.insert(encodeIssueRtree(d));
- });
- dispatch$1.call('loaded');
- })["catch"](function () {
- delete _cache.inflightTile[tile.id];
- _cache.loadedTile[tile.id] = true;
- });
- });
- },
- postUpdate: function postUpdate(d, callback) {
- var _this2 = this;
-
- if (_cache.inflightPost[d.id]) {
- return callback({
- message: 'Error update already inflight',
- status: -2
- }, d);
- }
-
- var params = {
- schema: d.schema,
- id: d.id
- };
-
- if (d.newStatus) {
- params.st = d.newStatus;
- }
-
- 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.
-
-
- 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)
-
- d3_json(url, {
- signal: controller.signal
- })["finally"](function () {
- delete _cache.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);
-
- _cache.closed["".concat(d.schema, ":").concat(d.id)] = true;
- } else {
- d = _this2.replaceItem(d.update({
- comment: d.newComment,
- newComment: undefined,
- newState: undefined
- }));
- }
-
- 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];
- },
- // 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
-
- 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
- },
- 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.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;
-
- function abortRequest$1(i) {
- Object.values(i).forEach(function (controller) {
- if (controller) {
- controller.abort();
- }
- });
- }
-
- function abortUnwantedRequests$1(cache, tiles) {
- Object.keys(cache.inflightTile).forEach(function (k) {
- var wanted = tiles.find(function (tile) {
- return k === tile.id;
- });
-
- if (!wanted) {
- abortRequest$1(cache.inflightTile[k]);
- delete cache.inflightTile[k];
- }
- });
- }
-
- 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
-
-
- function updateRtree$1(item, replace) {
- _cache$1.rtree.remove(item, function (a, b) {
- return a.data.id === b.data.id;
- });
-
- if (replace) {
- _cache$1.rtree.insert(item);
- }
- }
-
- function linkErrorObject(d) {
- return "<a class=\"error_object_link\">".concat(d, "</a>");
- }
-
- function linkEntity(d) {
- return "<a class=\"error_entity_link\">".concat(d, "</a>");
- }
-
- 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];
- }
- }
-
- function relativeBearing(p1, p2) {
- var angle = Math.atan2(p2.lon - p1.lon, p2.lat - p1.lat);
-
- if (angle < 0) {
- angle += 2 * Math.PI;
- } // Return degrees
-
-
- return angle * 180 / Math.PI;
- } // Assuming range [0,360)
-
-
- 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 preventCoincident(loc, bumpUp) {
- var coincident = false;
-
- 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 loc;
- }
-
- var serviceImproveOSM = {
- title: 'improveOSM',
- init: function init() {
- _mainFileFetcher.get('qa_data').then(function (d) {
- return _impOsmData = d.improveOSM;
- });
-
- if (!_cache$1) {
- this.reset();
- }
-
- this.event = utilRebind(this, dispatch$2, 'on');
- },
- reset: function reset() {
- if (_cache$1) {
- Object.values(_cache$1.inflightTile).forEach(abortRequest$1);
- }
-
- _cache$1 = {
- data: {},
- loadedTile: {},
- inflightTile: {},
- inflightPost: {},
- closed: {},
- rtree: new RBush()
- };
- },
- loadIssues: function loadIssues(projection) {
- var _this = this;
-
- var options = {
- client: 'iD',
- status: 'OPEN',
- zoom: '19' // Use a high zoom so that clusters aren't returned
-
- }; // determine the needed tiles to cover the view
-
- var tiles = tiler$1.zoomExtent([_tileZoom$1, _tileZoom$1]).getTiles(projection); // abort inflight requests that are no longer needed
-
- abortUnwantedRequests$1(_cache$1, tiles); // issue new requests..
-
- tiles.forEach(function (tile) {
- if (_cache$1.loadedTile[tile.id] || _cache$1.inflightTile[tile.id]) return;
-
- 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];
-
- var params = Object.assign({}, options, {
- east: east,
- south: south,
- west: west,
- north: north
- }); // 3 separate requests to store for each tile
-
- 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];
-
- 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
-
-
- 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
-
- 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
-
-
- 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
-
- 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;
-
- _cache$1.rtree.insert(encodeIssueRtree$1(d));
- });
- } // Tiles at high zoom == missing roads
-
-
- 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
-
- 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
-
- if (numberOfTrips === -1) {
- d.desc = _t('QA.improveOSM.error_types.mr.description_alt', d.replacements);
- }
-
- _cache$1.data[d.id] = d;
-
- _cache$1.rtree.insert(encodeIssueRtree$1(d));
- });
- } // Entities at high zoom == turn restrictions
-
-
- 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 loc = preventCoincident([point.lon, point.lat], true); // Elements are presented in a strange way
-
- 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
-
- 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
-
- 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;
-
- _cache$1.rtree.insert(encodeIssueRtree$1(d));
-
- dispatch$2.call('loaded');
- });
- }
- })["catch"](function () {
- delete _cache$1.inflightTile[tile.id][k];
-
- 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;
-
- // If comments already retrieved no need to do so again
- if (item.comments) {
- return Promise.resolve(item);
- }
-
- var key = item.issueKey;
- var qParams = {};
-
- 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;
- }
-
- var url = "".concat(_impOsmUrls[key], "/retrieveComments?") + utilQsString(qParams);
-
- var cacheComments = function cacheComments(data) {
- // Assign directly for immediate use afterwards
- // comments are served newest to oldest
- item.comments = data.comments ? data.comments.reverse() : [];
-
- _this2.replaceItem(item);
- };
-
- 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
-
-
- serviceOsm.userDetails(sendPayload.bind(this));
-
- function sendPayload(err, user) {
- var _this3 = this;
-
- if (err) {
- return callback(err, d);
- }
-
- var key = d.issueKey;
- var url = "".concat(_impOsmUrls[key], "/comment");
- var payload = {
- username: user.display_name,
- targetIds: [d.identifier]
- };
-
- if (d.newStatus) {
- payload.status = d.newStatus;
- payload.text = 'status changed';
- } // Comment take place of default text
-
-
- if (d.newComment) {
- payload.text = d.newComment;
- }
-
- 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
-
- 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
- });
-
- _this3.replaceItem(d.update({
- comments: comments,
- newComment: undefined
- }));
- } else {
- _this3.removeItem(d);
-
- 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;
- }
- }
-
- 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
-
- 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;
- }
- };
-
- var quot = /"/g;
-
- // B.2.3.2.1 CreateHTML(string, tag, attribute, value)
- // https://tc39.github.io/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, '"') + '"';
- return p1 + '>' + S + '</' + 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;
- });
- };
-
- // `String.prototype.link` method
- // https://tc39.github.io/ecma262/#sec-string.prototype.link
- _export({ target: 'String', proto: true, forced: stringHtmlForced('link') }, {
- link: function link(url) {
- return createHtml(this, 'a', 'href', url);
- }
- });
-
- var getOwnPropertyDescriptor$4 = objectGetOwnPropertyDescriptor.f;
-
-
-
-
-
-
- var nativeEndsWith = ''.endsWith;
- var min$8 = Math.min;
-
- 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$4(String.prototype, 'endsWith');
- return descriptor && !descriptor.writable;
- }();
-
- // `String.prototype.endsWith` method
- // https://tc39.github.io/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$8(toLength(endPosition), len);
- var search = String(searchString);
- return nativeEndsWith
- ? nativeEndsWith.call(that, search, end)
- : that.slice(end - search.length, end) === search;
- }
- });
-
- var getOwnPropertyDescriptor$5 = objectGetOwnPropertyDescriptor.f;
-
-
-
-
-
-
- var nativeStartsWith = ''.startsWith;
- var min$9 = Math.min;
-
- 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$5(String.prototype, 'startsWith');
- return descriptor && !descriptor.writable;
- }();
-
- // `String.prototype.startsWith` method
- // https://tc39.github.io/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$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;
- }
- });
-
- var $trimEnd = stringTrim.end;
-
-
- var FORCED$e = stringTrimForced('trimEnd');
-
- var trimEnd = FORCED$e ? function trimEnd() {
- return $trimEnd(this);
- } : ''.trimEnd;
-
- // `String.prototype.{ trimEnd, trimRight }` methods
- // https://github.com/tc39/ecmascript-string-left-right-trim
- _export({ target: 'String', proto: true, forced: FORCED$e }, {
- trimEnd: trimEnd,
- trimRight: trimEnd
- });
-
- 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 changeDefaults(newDefaults) {
- module.exports.defaults = newDefaults;
- }
-
- module.exports = {
- defaults: getDefaults(),
- getDefaults: getDefaults,
- changeDefaults: changeDefaults
- };
- });
-
- /**
- * Helpers
- */
- var escapeTest = /[&<>"']/;
- var escapeReplace = /[&<>"']/g;
- var escapeTestNoEncode = /[<>"']|&(?!#?\w+;)/;
- var escapeReplaceNoEncode = /[<>"']|&(?!#?\w+;)/g;
- var escapeReplacements = {
- '&': '&',
- '<': '<',
- '>': '>',
- '"': '"',
- "'": '''
- };
-
- var getEscapeReplacement = function getEscapeReplacement(ch) {
- return escapeReplacements[ch];
- };
-
- 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);
- }
- }
-
- return html;
- }
-
- var unescapeTest = /&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/ig;
-
- 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 ':';
-
- if (n.charAt(0) === '#') {
- return n.charAt(1) === 'x' ? String.fromCharCode(parseInt(n.substring(2), 16)) : String.fromCharCode(+n.substring(1));
- }
-
- return '';
- });
- }
-
- var caret = /(^|[^\[])\^/g;
-
- 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 nonWordAndColonTest = /[^\w:]/g;
- var originIndependentUrl = /^$|^[a-z][a-z0-9+.-]*:|^[?#]/i;
-
- function cleanUrl(sanitize, base, href) {
- if (sanitize) {
- var prot;
-
- try {
- prot = decodeURIComponent(unescape$1(href)).replace(nonWordAndColonTest, '').toLowerCase();
- } catch (e) {
- return null;
- }
-
- if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0 || prot.indexOf('data:') === 0) {
- return null;
- }
- }
-
- if (base && !originIndependentUrl.test(href)) {
- href = resolveUrl(base, href);
- }
-
- try {
- href = encodeURI(href).replace(/%25/g, '%');
- } catch (e) {
- return null;
- }
-
- return href;
- }
-
- var baseUrls = {};
- var justDomain = /^[^:]+:\/*[^/]*$/;
- var protocol = /^([^:]+:)[\s\S]*$/;
- var domain = /^([^:]+:\/*[^/]*)[\s\S]*$/;
-
- 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);
- }
- }
-
- base = baseUrls[' ' + base];
- var relativeBase = base.indexOf(':') === -1;
-
- if (href.substring(0, 2) === '//') {
- if (relativeBase) {
- return href;
- }
-
- return base.replace(protocol, '$1') + href;
- } else if (href.charAt(0) === '/') {
- if (relativeBase) {
- return href;
- }
-
- return base.replace(domain, '$1') + href;
- } else {
- return base + href;
- }
- }
-
- var noopTest = {
- exec: function noopTest() {}
- };
-
- function merge$1(obj) {
- var i = 1,
- target,
- key;
-
- for (; i < arguments.length; i++) {
- target = arguments[i];
-
- for (key in target) {
- if (Object.prototype.hasOwnProperty.call(target, key)) {
- obj[key] = target[key];
- }
- }
- }
-
- return obj;
- }
-
- 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;
-
- while (--curr >= 0 && str[curr] === '\\') {
- escaped = !escaped;
- }
-
- 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 (cells.length > count) {
- cells.splice(count);
- } else {
- while (cells.length < count) {
- cells.push('');
- }
- }
-
- for (; i < cells.length; i++) {
- // leading or trailing whitespace is ignored per the gfm spec
- cells[i] = cells[i].trim().replace(/\\\|/g, '|');
- }
-
- 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 rtrim$1(str, c, invert) {
- var l = str.length;
-
- if (l === 0) {
- return '';
- } // Length of suffix matching the invert condition.
-
-
- var suffLen = 0; // Step left until we fail to match the invert condition.
-
- while (suffLen < l) {
- var currChar = str.charAt(l - suffLen - 1);
-
- if (currChar === c && !invert) {
- suffLen++;
- } else if (currChar !== c && invert) {
- suffLen++;
- } else {
- break;
- }
- }
-
- return str.substr(0, l - suffLen);
- }
-
- function findClosingBracket(str, b) {
- if (str.indexOf(b[1]) === -1) {
- return -1;
- }
-
- var l = str.length;
- var level = 0,
- i = 0;
-
- for (; i < l; i++) {
- if (str[i] === '\\') {
- i++;
- } else if (str[i] === b[0]) {
- level++;
- } else if (str[i] === b[1]) {
- level--;
-
- if (level < 0) {
- return i;
- }
- }
- }
-
- return -1;
- }
-
- 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');
- }
- } // copied from https://stackoverflow.com/a/5450113/806777
-
-
- function repeatString(pattern, count) {
- if (count < 1) {
- return '';
- }
-
- var result = '';
-
- while (count > 1) {
- if (count & 1) {
- result += pattern;
- }
-
- count >>= 1;
- pattern += pattern;
- }
-
- return result + pattern;
- }
-
- 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
- };
-
- var defaults$1 = defaults.defaults;
- var rtrim$2 = helpers.rtrim,
- splitCells$1 = helpers.splitCells,
- _escape = helpers.escape,
- findClosingBracket$1 = helpers.findClosingBracket;
-
- 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 indentCodeCompensation(raw, text) {
- var matchIndentToCode = raw.match(/^(\s+)(?:```)/);
-
- if (matchIndentToCode === null) {
- return text;
- }
-
- var indentToCode = matchIndentToCode[1];
- return text.split('\n').map(function (node) {
- var matchIndentInNode = node.match(/^\s+/);
-
- if (matchIndentInNode === null) {
- return node;
- }
-
- var _matchIndentInNode = _slicedToArray(matchIndentInNode, 1),
- indentInNode = _matchIndentInNode[0];
-
- if (indentInNode.length >= indentToCode.length) {
- return node.slice(indentToCode.length);
- }
-
- return node;
- }).join('\n');
- }
- /**
- * Tokenizer
- */
-
-
- var Tokenizer_1 = /*#__PURE__*/function () {
- function Tokenizer(options) {
- _classCallCheck(this, Tokenizer);
-
- this.options = options || defaults$1;
- }
-
- _createClass(Tokenizer, [{
- key: "space",
- value: function space(src) {
- var cap = this.rules.block.newline.exec(src);
-
- if (cap) {
- if (cap[0].length > 1) {
- return {
- type: 'space',
- raw: cap[0]
- };
- }
-
- return {
- raw: '\n'
- };
- }
- }
- }, {
- 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.
-
- if (lastToken && lastToken.type === 'paragraph') {
- return {
- raw: cap[0],
- text: cap[0].trimRight()
- };
- }
-
- var text = cap[0].replace(/^ {4}/gm, '');
- return {
- type: 'code',
- raw: cap[0],
- codeBlockStyle: 'indented',
- text: !this.options.pedantic ? rtrim$2(text, '\n') : text
- };
- }
- }
- }, {
- 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
- };
- }
- }
- }, {
- key: "heading",
- value: function heading(src) {
- var cap = this.rules.block.heading.exec(src);
-
- if (cap) {
- return {
- type: 'heading',
- raw: cap[0],
- depth: cap[1].length,
- text: cap[2]
- };
- }
- }
- }, {
- key: "nptable",
- value: function nptable(src) {
- var cap = this.rules.block.nptable.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') : [],
- raw: cap[0]
- };
-
- if (item.header.length === item.align.length) {
- var l = item.align.length;
- var i;
-
- 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;
- }
- }
-
- l = item.cells.length;
-
- for (i = 0; i < l; i++) {
- item.cells[i] = splitCells$1(item.cells[i], item.header.length);
- }
-
- return item;
- }
- }
- }
- }, {
- key: "hr",
- value: function hr(src) {
- var cap = this.rules.block.hr.exec(src);
-
- 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 isparen = bull[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,
- b,
- addBack,
- loose,
- istask,
- ischecked;
- var l = itemMatch.length;
-
- for (var i = 0; i < l; i++) {
- item = itemMatch[i];
- raw = item; // Remove the list item's bullet
- // so it is seen as the next token.
-
- space = item.length;
- item = item.replace(/^ *([*+-]|\d+[.)]) ?/, ''); // Outdent whatever the
- // list item contains. Hacky.
-
- 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 the next list item belongs here.
- // Backpedal if it does not belong in this list.
-
-
- if (i !== l - 1) {
- b = this.rules.block.bullet.exec(itemMatch[i + 1])[0];
-
- if (isordered ? b.length === 1 || !isparen && b[b.length - 1] === ')' : b.length > 1 || this.options.smartLists && b !== bull) {
- addBack = itemMatch.slice(i + 1).join('\n');
- list.raw = list.raw.substring(0, list.raw.length - addBack.length);
- i = l - 1;
- }
- } // Determine whether item is loose or not.
- // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
- // for discount behavior.
-
-
- loose = next || /\n\n(?!\s*$)/.test(item);
-
- 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
-
-
- istask = /^\[[ xX]\] /.test(item);
- ischecked = undefined;
-
- if (istask) {
- ischecked = item[1] !== ' ';
- item = item.replace(/^\[[ xX]\] +/, '');
- }
-
- 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);
-
- 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 {
- 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 (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') : []
- };
-
- if (item.header.length === item.align.length) {
- item.raw = cap[0];
- var l = item.align.length;
- var i;
-
- 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;
- }
- }
-
- l = item.cells.length;
-
- for (i = 0; i < l; i++) {
- item.cells[i] = splitCells$1(item.cells[i].replace(/^ *\| *| *\| *$/g, ''), item.header.length);
- }
-
- return item;
- }
- }
- }
- }, {
- key: "lheading",
- value: function lheading(src) {
- var cap = this.rules.block.lheading.exec(src);
-
- if (cap) {
- return {
- type: 'heading',
- raw: cap[0],
- depth: cap[2].charAt(0) === '=' ? 1 : 2,
- text: cap[1]
- };
- }
- }
- }, {
- 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, tokens) {
- var cap = this.rules.block.text.exec(src);
-
- if (cap) {
- var lastToken = tokens[tokens.length - 1];
-
- if (lastToken && lastToken.type === 'text') {
- return {
- raw: cap[0],
- text: cap[0]
- };
- }
-
- return {
- type: 'text',
- raw: cap[0],
- text: cap[0]
- };
- }
- }
- }, {
- 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])
- };
- }
- }
- }, {
- 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]
- };
- }
- }
- }, {
- key: "link",
- value: function link(src) {
- var cap = this.rules.inline.link.exec(src);
-
- if (cap) {
- var lastParenIndex = findClosingBracket$1(cap[2], '()');
-
- 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 href = cap[2];
- var title = '';
-
- if (this.options.pedantic) {
- var link = /^([^'"]*[^\s])\s+(['"])(.*)\2/.exec(href);
-
- if (link) {
- href = link[1];
- title = link[3];
- } else {
- title = '';
- }
- } else {
- title = cap[3] ? cap[3].slice(1, -1) : '';
- }
-
- href = href.trim().replace(/^<([\s\S]*)>$/, '$1');
- var token = outputLink(cap, {
- href: href ? href.replace(this.rules.inline._escapes, '$1') : href,
- title: title ? title.replace(this.rules.inline._escapes, '$1') : title
- }, cap[0]);
- return token;
- }
- }
- }, {
- 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
- };
- }
-
- var token = outputLink(cap, link, cap[0]);
- return token;
- }
- }
- }, {
- 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;
-
- while ((match = endReg.exec(maskedSrc)) != null) {
- cap = this.rules.inline.strong.middle.exec(maskedSrc.slice(0, match.index + 3));
-
- if (cap) {
- return {
- type: 'strong',
- raw: src.slice(0, cap[0].length),
- text: src.slice(2, cap[0].length - 2)
- };
- }
- }
- }
- }
- }, {
- 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;
-
- 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)
- };
- }
- }
- }
- }
- }, {
- 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 = text.startsWith(' ') && text.endsWith(' ');
-
- if (hasNonSpaceChars && hasSpaceCharsOnBothEnds) {
- text = text.substring(1, text.length - 1);
- }
-
- 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 (cap) {
- return {
- type: 'br',
- raw: cap[0]
- };
- }
- }
- }, {
- key: "del",
- value: function del(src) {
- var cap = this.rules.inline.del.exec(src);
-
- if (cap) {
- return {
- type: 'del',
- raw: cap[0],
- text: cap[1]
- };
- }
- }
- }, {
- key: "autolink",
- value: function autolink(src, mangle) {
- var cap = this.rules.inline.autolink.exec(src);
-
- if (cap) {
- var text, href;
-
- if (cap[2] === '@') {
- text = _escape(this.options.mangle ? mangle(cap[1]) : cap[1]);
- href = 'mailto:' + text;
- } else {
- text = _escape(cap[1]);
- href = text;
- }
-
- return {
- type: 'link',
- raw: cap[0],
- text: text,
- href: href,
- tokens: [{
- type: 'text',
- raw: text,
- text: text
- }]
- };
- }
- }
- }, {
- key: "url",
- value: function url(src, mangle) {
- var cap;
-
- if (cap = this.rules.inline.url.exec(src)) {
- var text, href;
-
- if (cap[2] === '@') {
- text = _escape(this.options.mangle ? mangle(cap[0]) : cap[0]);
- href = 'mailto:' + text;
- } else {
- // do extended autolink path validation
- var prevCapZero;
-
- do {
- prevCapZero = cap[0];
- cap[0] = this.rules.inline._backpedal.exec(cap[0])[0];
- } while (prevCapZero !== cap[0]);
-
- text = _escape(cap[0]);
-
- if (cap[1] === 'www.') {
- href = 'http://' + text;
- } else {
- href = text;
- }
- }
-
- 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);
-
- if (cap) {
- var text;
-
- 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]);
- }
-
- 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*)+/,
- 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}) +([^\n]*?)(?: +#+)? *(?:\n+|$)/,
- blockquote: /^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/,
- list: /^( {0,3})(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\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]+)*)/,
- 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(?!\1bull ?)[^\n]*)*/;
- block.item = edit$1(block.item, 'gm').replace(/bull/g, 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
- */
-
- 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
-
- });
- 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)
- */
-
- 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]+?) *(?:#+ *)?(?:\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
- */
-
- 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)
-
- },
- 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 = /<(?:\\[<>]?|[^\s<>\\])*>|[^\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
- */
-
- inline.normal = merge$2({}, inline);
- /**
- * Pedantic Inline Grammar
- */
-
- 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
- */
-
- 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)~+/,
- 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
- */
-
- 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
- };
-
- var defaults$2 = defaults.defaults;
- var block$1 = rules.block,
- inline$1 = rules.inline;
- var repeatString$1 = helpers.repeatString;
- /**
- * smartypants text replacement
- */
-
- 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
- */
-
-
- function mangle(text) {
- var out = '',
- i,
- ch;
- var l = text.length;
-
- for (i = 0; i < l; i++) {
- ch = text.charCodeAt(i);
-
- if (Math.random() > 0.5) {
- ch = 'x' + ch.toString(16);
- }
-
- out += '&#' + ch + ';';
- }
-
- return out;
- }
- /**
- * Block Lexer
- */
-
-
- var Lexer_1 = /*#__PURE__*/function () {
- function Lexer(options) {
- _classCallCheck(this, Lexer);
-
- 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
- };
-
- if (this.options.pedantic) {
- rules.block = block$1.pedantic;
- rules.inline = inline$1.pedantic;
- } else if (this.options.gfm) {
- rules.block = block$1.gfm;
-
- if (this.options.breaks) {
- rules.inline = inline$1.breaks;
- } else {
- rules.inline = inline$1.gfm;
- }
- }
-
- this.tokenizer.rules = rules;
- }
- /**
- * Expose Rules
- */
-
-
- _createClass(Lexer, [{
- key: "lex",
-
- /**
- * Preprocessing
- */
- value: 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
- */
-
- }, {
- 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;
- src = src.replace(/^ +$/gm, '');
- var token, i, l, lastToken;
-
- while (src) {
- // newline
- if (token = this.tokenizer.space(src)) {
- src = src.substring(token.raw.length);
-
- if (token.type) {
- tokens.push(token);
- }
-
- continue;
- } // code
-
-
- if (token = this.tokenizer.code(src, tokens)) {
- src = src.substring(token.raw.length);
-
- if (token.type) {
- tokens.push(token);
- } else {
- lastToken = tokens[tokens.length - 1];
- lastToken.raw += '\n' + token.raw;
- lastToken.text += '\n' + token.text;
- }
-
- continue;
- } // fences
-
-
- if (token = this.tokenizer.fences(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // heading
-
-
- if (token = this.tokenizer.heading(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // table no leading pipe (gfm)
-
-
- if (token = this.tokenizer.nptable(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // hr
-
-
- if (token = this.tokenizer.hr(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // blockquote
-
-
- 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 (token = this.tokenizer.list(src)) {
- src = src.substring(token.raw.length);
- l = token.items.length;
-
- for (i = 0; i < l; i++) {
- token.items[i].tokens = this.blockTokens(token.items[i].text, [], false);
- }
-
- tokens.push(token);
- continue;
- } // html
-
-
- if (token = this.tokenizer.html(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // def
-
-
- if (top && (token = this.tokenizer.def(src))) {
- src = src.substring(token.raw.length);
-
- if (!this.tokens.links[token.tag]) {
- this.tokens.links[token.tag] = {
- href: token.href,
- title: token.title
- };
- }
-
- continue;
- } // table (gfm)
-
-
- if (token = this.tokenizer.table(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // lheading
-
-
- if (token = this.tokenizer.lheading(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // top-level paragraph
-
-
- if (top && (token = this.tokenizer.paragraph(src))) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // text
-
-
- if (token = this.tokenizer.text(src, tokens)) {
- src = src.substring(token.raw.length);
-
- if (token.type) {
- tokens.push(token);
- } else {
- lastToken = tokens[tokens.length - 1];
- lastToken.raw += '\n' + token.raw;
- lastToken.text += '\n' + token.text;
- }
-
- continue;
- }
-
- 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: "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];
-
- 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;
-
- for (j = 0; j < l2; j++) {
- token.tokens.header[j] = [];
- this.inlineTokens(token.header[j], token.tokens.header[j]);
- } // cells
-
-
- l2 = token.cells.length;
-
- for (j = 0; j < l2; j++) {
- row = token.cells[j];
- token.tokens.cells[j] = [];
-
- for (k = 0; k < row.length; k++) {
- token.tokens.cells[j][k] = [];
- this.inlineTokens(row[k], token.tokens.cells[j][k]);
- }
- }
-
- break;
- }
-
- case 'blockquote':
- {
- this.inline(token.tokens);
- break;
- }
-
- case 'list':
- {
- l2 = token.items.length;
-
- for (j = 0; j < l2; j++) {
- this.inline(token.items[j].tokens);
- }
-
- break;
- }
- }
- }
-
- return tokens;
- }
- /**
- * Lexing/Compiling
- */
-
- }, {
- 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 prevChar = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : '';
- var token; // String with links masked to avoid interference with em and strong
-
- var maskedSrc = src;
- var match; // Mask out reflinks
-
- if (this.tokens.links) {
- var links = Object.keys(this.tokens.links);
-
- 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);
- }
- }
- }
- } // Mask out other blocks
-
-
- 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);
- }
-
- while (src) {
- // escape
- if (token = this.tokenizer.escape(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // tag
-
-
- 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
-
-
- if (token = this.tokenizer.link(src)) {
- src = src.substring(token.raw.length);
-
- if (token.type === 'link') {
- token.tokens = this.inlineTokens(token.text, [], true, inRawBlock);
- }
-
- tokens.push(token);
- continue;
- } // reflink, nolink
-
-
- if (token = this.tokenizer.reflink(src, this.tokens.links)) {
- src = src.substring(token.raw.length);
-
- if (token.type === 'link') {
- token.tokens = this.inlineTokens(token.text, [], true, inRawBlock);
- }
-
- tokens.push(token);
- continue;
- } // strong
-
-
- 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
-
-
- 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 (token = this.tokenizer.codespan(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // br
-
-
- if (token = this.tokenizer.br(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // del (gfm)
-
-
- 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 (token = this.tokenizer.autolink(src, mangle)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // url (gfm)
-
-
- if (!inLink && (token = this.tokenizer.url(src, mangle))) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
- } // text
-
-
- if (token = this.tokenizer.inlineText(src, inRawBlock, smartypants)) {
- src = src.substring(token.raw.length);
- prevChar = token.raw.slice(-1);
- tokens.push(token);
- continue;
- }
-
- 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: "lex",
-
- /**
- * Static Lex Method
- */
- value: function lex(src, options) {
- var lexer = new Lexer(options);
- return lexer.lex(src);
- }
- /**
- * Static Lex Inline Method
- */
-
- }, {
- key: "lexInline",
- value: function lexInline(src, options) {
- var lexer = new Lexer(options);
- return lexer.inlineTokens(src);
- }
- }, {
- key: "rules",
- get: function get() {
- return {
- block: block$1,
- inline: inline$1
- };
- }
- }]);
-
- return Lexer;
- }();
-
- var defaults$3 = defaults.defaults;
- var cleanUrl$1 = helpers.cleanUrl,
- escape$2 = helpers.escape;
- /**
- * Renderer
- */
-
- var Renderer_1 = /*#__PURE__*/function () {
- function Renderer(options) {
- _classCallCheck(this, Renderer);
-
- this.options = options || defaults$3;
- }
-
- _createClass(Renderer, [{
- key: "code",
- value: function code(_code, infostring, escaped) {
- var lang = (infostring || '').match(/\S*/)[0];
-
- if (this.options.highlight) {
- var out = this.options.highlight(_code, lang);
-
- if (out != null && out !== _code) {
- escaped = true;
- _code = out;
- }
- }
-
- if (!lang) {
- return '<pre><code>' + (escaped ? _code : escape$2(_code, true)) + '</code></pre>\n';
- }
-
- 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
-
- }, {
- 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$1(this.options.sanitize, this.options.baseUrl, href);
-
- if (href === null) {
- return text;
- }
-
- var out = '<a href="' + escape$2(href) + '"';
-
- if (title) {
- out += ' title="' + title + '"';
- }
-
- out += '>' + text + '</a>';
- return out;
- }
- }, {
- key: "image",
- value: function image(href, title, text) {
- href = cleanUrl$1(this.options.sanitize, this.options.baseUrl, href);
-
- if (href === null) {
- return text;
- }
-
- var out = '<img src="' + href + '" alt="' + text + '"';
-
- if (title) {
- out += ' title="' + title + '"';
- }
-
- out += this.options.xhtml ? '/>' : '>';
- return out;
- }
- }, {
- key: "text",
- value: function text(_text) {
- return _text;
- }
- }]);
-
- return Renderer;
- }();
-
- /**
- * TextRenderer
- * returns only the textual part of the token
- */
- var TextRenderer_1 = /*#__PURE__*/function () {
- function TextRenderer() {
- _classCallCheck(this, TextRenderer);
- }
-
- _createClass(TextRenderer, [{
- key: "strong",
- // no need for block level renderers
- value: 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;
- }();
-
- /**
- * Slugger generates header id
- */
- var Slugger_1 = /*#__PURE__*/function () {
- function Slugger() {
- _classCallCheck(this, Slugger);
-
- this.seen = {};
- }
-
- _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, '-');
- }
- /**
- * 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));
- }
-
- 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);
- }
- }]);
-
- return Slugger;
- }();
-
- var defaults$4 = defaults.defaults;
- var unescape$2 = helpers.unescape;
- /**
- * Parsing & Compiling
- */
-
- var Parser_1 = /*#__PURE__*/function () {
- function Parser(options) {
- _classCallCheck(this, Parser);
-
- 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
- */
-
-
- _createClass(Parser, [{
- key: "parse",
-
- /**
- * Parse Loop
- */
- value: 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];
-
- switch (token.type) {
- case 'space':
- {
- continue;
- }
-
- case 'hr':
- {
- out += this.renderer.hr();
- continue;
- }
-
- case 'heading':
- {
- out += this.renderer.heading(this.parseInline(token.tokens), token.depth, unescape$2(this.parseInline(token.tokens, this.textRenderer)), this.slugger);
- continue;
- }
-
- case 'code':
- {
- out += this.renderer.code(token.text, token.lang, token.escaped);
- continue;
- }
-
- case 'table':
- {
- header = ''; // header
-
- cell = '';
- l2 = token.header.length;
-
- for (j = 0; j < l2; j++) {
- cell += this.renderer.tablecell(this.parseInline(token.tokens.header[j]), {
- header: true,
- align: token.align[j]
- });
- }
-
- header += this.renderer.tablerow(cell);
- body = '';
- l2 = token.cells.length;
-
- for (j = 0; j < l2; j++) {
- row = token.tokens.cells[j];
- cell = '';
- l3 = row.length;
-
- for (k = 0; k < l3; k++) {
- cell += this.renderer.tablecell(this.parseInline(row[k]), {
- header: false,
- align: token.align[k]
- });
- }
-
- body += this.renderer.tablerow(cell);
- }
-
- out += this.renderer.table(header, body);
- continue;
- }
-
- case 'blockquote':
- {
- body = this.parse(token.tokens);
- out += this.renderer.blockquote(body);
- continue;
- }
-
- case 'list':
- {
- ordered = token.ordered;
- start = token.start;
- loose = token.loose;
- l2 = token.items.length;
- body = '';
-
- for (j = 0; j < l2; j++) {
- item = token.items[j];
- checked = item.checked;
- task = item.task;
- itemBody = '';
-
- if (item.task) {
- checkbox = this.renderer.checkbox(checked);
-
- if (loose) {
- if (item.tokens.length > 0 && item.tokens[0].type === 'text') {
- item.tokens[0].text = checkbox + ' ' + item.tokens[0].text;
-
- 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;
- }
- }
-
- itemBody += this.parse(item.tokens, loose);
- body += this.renderer.listitem(itemBody, task, checked);
- }
-
- out += this.renderer.list(body, ordered, start);
- continue;
- }
-
- case 'html':
- {
- // TODO parse inline content if parameter markdown=1
- out += this.renderer.html(token.text);
- continue;
- }
-
- case 'paragraph':
- {
- out += this.renderer.paragraph(this.parseInline(token.tokens));
- continue;
- }
-
- case 'text':
- {
- body = token.tokens ? this.parseInline(token.tokens) : token.text;
-
- while (i + 1 < l && tokens[i + 1].type === 'text') {
- token = tokens[++i];
- body += '\n' + (token.tokens ? this.parseInline(token.tokens) : token.text);
- }
-
- out += top ? this.renderer.paragraph(body) : body;
- continue;
- }
-
- default:
- {
- var errMsg = 'Token with "' + token.type + '" type was not found.';
-
- if (this.options.silent) {
- console.error(errMsg);
- return;
- } else {
- throw new Error(errMsg);
- }
- }
- }
- }
-
- return out;
- }
- /**
- * Parse Inline Tokens
- */
-
- }, {
- 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];
-
- switch (token.type) {
- case 'escape':
- {
- out += renderer.text(token.text);
- break;
- }
-
- case 'html':
- {
- out += renderer.html(token.text);
- break;
- }
-
- case 'link':
- {
- out += renderer.link(token.href, token.title, this.parseInline(token.tokens, renderer));
- break;
- }
-
- case 'image':
- {
- out += renderer.image(token.href, token.title, token.text);
- break;
- }
-
- case 'strong':
- {
- out += renderer.strong(this.parseInline(token.tokens, renderer));
- break;
- }
-
- case 'em':
- {
- out += renderer.em(this.parseInline(token.tokens, renderer));
- break;
- }
-
- case 'codespan':
- {
- out += renderer.codespan(token.text);
- break;
- }
-
- case 'br':
- {
- out += renderer.br();
- break;
- }
-
- case 'del':
- {
- out += renderer.del(this.parseInline(token.tokens, renderer));
- break;
- }
-
- case 'text':
- {
- out += renderer.text(token.text);
- break;
- }
-
- default:
- {
- var errMsg = 'Token with "' + token.type + '" type was not found.';
-
- 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
- */
-
- }, {
- key: "parseInline",
- value: function parseInline(tokens, options) {
- var parser = new Parser(options);
- return parser.parseInline(tokens);
- }
- }]);
-
- return Parser;
- }();
-
- 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
- */
-
- 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 (typeof src !== 'string') {
- throw new Error('marked(): input parameter is of type ' + Object.prototype.toString.call(src) + ', string expected');
- }
-
- if (typeof opt === 'function') {
- callback = opt;
- opt = null;
- }
-
- opt = merge$3({}, marked.defaults, opt || {});
- checkSanitizeDeprecation$1(opt);
-
- if (callback) {
- var highlight = opt.highlight;
- var tokens;
-
- try {
- tokens = Lexer_1.lex(src, opt);
- } catch (e) {
- return callback(e);
- }
-
- var done = function done(err) {
- var out;
-
- if (!err) {
- try {
- out = Parser_1.parse(tokens, opt);
- } catch (e) {
- err = e;
- }
- }
-
- opt.highlight = highlight;
- return err ? callback(err) : callback(null, out);
- };
-
- if (!highlight || highlight.length < 3) {
- return done();
- }
-
- 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);
- }
-
- if (code != null && code !== token.text) {
- token.text = code;
- token.escaped = true;
- }
-
- pending--;
-
- if (pending === 0) {
- done();
- }
- });
- }, 0);
- }
- });
-
- if (pending === 0) {
- done();
- }
-
- return;
- }
-
- try {
- var _tokens = Lexer_1.lex(src, opt);
-
- if (opt.walkTokens) {
- marked.walkTokens(_tokens, opt.walkTokens);
- }
-
- return Parser_1.parse(_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$3(e.message + '', true) + '</pre>';
- }
-
- throw e;
- }
- }
- /**
- * Options
- */
-
-
- marked.options = marked.setOptions = function (opt) {
- merge$3(marked.defaults, opt);
- changeDefaults(marked.defaults);
- return marked;
- };
-
- marked.getDefaults = getDefaults;
- marked.defaults = defaults$5;
- /**
- * Use Extension
- */
-
- marked.use = function (extension) {
- var opts = merge$3({}, extension);
-
- if (extension.renderer) {
- (function () {
- var renderer = marked.defaults.renderer || new Renderer_1();
-
- var _loop = function _loop(prop) {
- var prevRenderer = renderer[prop];
-
- renderer[prop] = function () {
- for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
- args[_key] = arguments[_key];
- }
-
- var ret = extension.renderer[prop].apply(renderer, args);
-
- if (ret === false) {
- ret = prevRenderer.apply(renderer, args);
- }
-
- return ret;
- };
- };
-
- for (var prop in extension.renderer) {
- _loop(prop);
- }
-
- opts.renderer = renderer;
- })();
- }
-
- if (extension.tokenizer) {
- (function () {
- var tokenizer = marked.defaults.tokenizer || new Tokenizer_1();
-
- var _loop2 = function _loop2(prop) {
- var prevTokenizer = tokenizer[prop];
-
- tokenizer[prop] = function () {
- for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) {
- args[_key2] = arguments[_key2];
- }
-
- var ret = extension.tokenizer[prop].apply(tokenizer, args);
-
- if (ret === false) {
- ret = prevTokenizer.apply(tokenizer, args);
- }
-
- return ret;
- };
- };
-
- for (var prop in extension.tokenizer) {
- _loop2(prop);
- }
-
- opts.tokenizer = tokenizer;
- })();
- }
-
- if (extension.walkTokens) {
- var walkTokens = marked.defaults.walkTokens;
-
- opts.walkTokens = function (token) {
- extension.walkTokens(token);
-
- if (walkTokens) {
- walkTokens(token);
- }
- };
- }
-
- marked.setOptions(opts);
- };
- /**
- * Run callback for every token
- */
-
-
- marked.walkTokens = function (tokens, callback) {
- var _iterator = _createForOfIteratorHelper(tokens),
- _step;
-
- try {
- for (_iterator.s(); !(_step = _iterator.n()).done;) {
- var token = _step.value;
- callback(token);
-
- switch (token.type) {
- case 'table':
- {
- var _iterator2 = _createForOfIteratorHelper(token.tokens.header),
- _step2;
-
- 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 _iterator3 = _createForOfIteratorHelper(token.tokens.cells),
- _step3;
-
- try {
- for (_iterator3.s(); !(_step3 = _iterator3.n()).done;) {
- var row = _step3.value;
-
- 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();
- }
-
- 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
- */
-
-
- 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$3({}, marked.defaults, opt || {});
- checkSanitizeDeprecation$1(opt);
-
- try {
- var tokens = Lexer_1.lexInline(src, opt);
-
- if (opt.walkTokens) {
- marked.walkTokens(tokens, opt.walkTokens);
- }
-
- 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$3(e.message + '', true) + '</pre>';
- }
-
- throw e;
- }
- };
- /**
- * Expose
- */
-
-
- 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 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
-
- var _cache$2;
-
- function abortRequest$2(controller) {
- if (controller) {
- controller.abort();
- }
- }
-
- function abortUnwantedRequests$2(cache, tiles) {
- Object.keys(cache.inflightTile).forEach(function (k) {
- var wanted = tiles.find(function (tile) {
- return k === tile.id;
- });
-
- if (!wanted) {
- abortRequest$2(cache.inflightTile[k]);
- delete cache.inflightTile[k];
- }
- });
- }
-
- 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 updateRtree$2(item, replace) {
- _cache$2.rtree.remove(item, function (a, b) {
- return a.data.id === b.data.id;
- });
-
- if (replace) {
- _cache$2.rtree.insert(item);
- }
- } // Issues shouldn't obscure each other
-
-
- function preventCoincident$1(loc) {
- var coincident = false;
-
- 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 loc;
- }
-
- 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 (!_cache$2) {
- this.reset();
- }
-
- this.event = utilRebind(this, dispatch$3, 'on');
- },
- reset: function reset() {
- var _strings = {};
- var _colors = {};
-
- if (_cache$2) {
- Object.values(_cache$2.inflightTile).forEach(abortRequest$2); // Strings and colors are static and should not be re-populated
-
- _strings = _cache$2.strings;
- _colors = _cache$2.colors;
- }
-
- _cache$2 = {
- data: {},
- loadedTile: {},
- inflightTile: {},
- inflightPost: {},
- closed: {},
- rtree: new RBush(),
- strings: _strings,
- colors: _colors
- };
- },
- loadIssues: function loadIssues(projection) {
- var _this = this;
-
- 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
-
- var tiles = tiler$2.zoomExtent([_tileZoom$2, _tileZoom$2]).getTiles(projection); // abort inflight requests that are no longer needed
-
- abortUnwantedRequests$2(_cache$2, tiles); // issue new requests..
-
- tiles.forEach(function (tile) {
- if (_cache$2.loadedTile[tile.id] || _cache$2.inflightTile[tile.id]) return;
-
- var _tile$xyz = _slicedToArray(tile.xyz, 3),
- x = _tile$xyz[0],
- y = _tile$xyz[1],
- z = _tile$xyz[2];
-
- 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;
-
- 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) */
-
- var itemType = "".concat(item, "-").concat(cl); // Filter out unsupported issue types (some are too specific or advanced)
-
- if (itemType in _osmoseData.icons) {
- var loc = issue.geometry.coordinates; // lon, lat
-
- loc = preventCoincident$1(loc);
- var d = new QAItem(loc, _this, itemType, id, {
- item: item
- }); // Setting elems here prevents UI detail requests
-
- if (item === 8300 || item === 8360) {
- d.elems = [];
- }
-
- _cache$2.data[d.id] = d;
-
- _cache$2.rtree.insert(encodeIssueRtree$2(d));
- }
- });
- }
-
- 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;
-
- // Issue details only need to be fetched once
- if (issue.elems !== undefined) {
- return Promise.resolve(issue);
- }
-
- var url = "".concat(_osmoseUrlRoot, "/issue/").concat(issue.id, "?langs=").concat(_mainLocalizer.localeCode());
-
- 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
-
- issue.detail = data.subtitle ? marked_1(data.subtitle.auto) : '';
-
- _this2.replaceItem(issue);
- };
-
- 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);
-
- 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
-
-
- 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
-
-
- var allRequests = items.map(function (itemType) {
- // No need to request data we already have
- if (itemType in _cache$2.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$;
-
- 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)
-
-
- if (!cl) {
- /* eslint-disable no-console */
- console.log("Osmose strings request (".concat(itemType, ") had unexpected data"));
- /* eslint-enable no-console */
-
- return;
- } // Cache served item colors to automatically style issue markers later
-
-
- var itemInt = item.item,
- color = item.color;
-
- 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
-
-
- var title = cl.title,
- detail = cl.detail,
- fix = cl.fix,
- trap = cl.trap; // Osmose titles shouldn't contain markdown
-
- 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;
- };
-
- 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
-
-
- 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;
-
- 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'
-
-
- var url = "".concat(_osmoseUrlRoot, "/issue/").concat(issue.id, "/").concat(issue.newStatus);
- var controller = new AbortController();
-
- var after = function after() {
- delete _cache$2.inflightPost[issue.id];
-
- _this3.removeItem(issue);
-
- 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;
- }
-
- _cache$2.closed[issue.item] += 1;
- }
-
- if (callback) callback(null, issue);
- };
-
- _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);
- });
- },
- // 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];
- },
- // 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$2.data[item.id] = item;
- updateRtree$2(encodeIssueRtree$2(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$2.data[item.id];
- updateRtree$2(encodeIssueRtree$2(item), false); // false = remove
- },
- // Used to populate `closed:osmose:*` changeset tags
- getClosedCounts: function getClosedCounts() {
- return _cache$2.closed;
- },
- itemURL: function itemURL(item) {
- return "https://osmose.openstreetmap.fr/en/error/".concat(item.id);
- }
- };
-
- 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;
-
- var _mlyCache;
-
- var _mlyClicks;
-
- var _mlyActiveImage;
-
- var _mlySelectedImageKey;
-
- var _mlyViewer;
-
- var _mlyViewerFilter = ['all'];
-
- var _loadViewerPromise;
-
- var _mlyHighlightedDetection;
-
- var _mlyShowFeatureDetections = false;
- var _mlyShowSignDetections = false;
-
- function abortRequest$3(controller) {
- controller.abort();
- }
-
- 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
-
- var cache = _mlyCache[which];
- Object.keys(cache.inflight).forEach(function (k) {
- var wanted = tiles.find(function (tile) {
- return k.indexOf(tile.id + ',') === 0;
- });
-
- if (!wanted) {
- abortRequest$3(cache.inflight[k]);
- delete cache.inflight[k];
- }
- });
- tiles.forEach(function (tile) {
- loadNextTilePage(which, currZoom, url, tile);
- });
- }
-
- 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 (linkHeader) {
- var pagination = parsePagination(linkHeader);
-
- if (pagination.next) {
- cache.nextURL[tile.id] = pagination.next;
- }
- }
-
- return response.json();
- }).then(function (data) {
- cache.loaded[id] = true;
- delete cache.inflight[id];
-
- if (!data || !data.features || !data.features.length) {
- throw new Error('No Data');
- }
-
- 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
-
- 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
- };
- }
-
- 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
- }
-
- 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];
- });
- }
-
- 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 response.json();
- }).then(function (data) {
- if (!data || !data.features || !data.features.length) {
- throw new Error('No Data');
- }
-
- data.features.forEach(function (feature) {
- var d;
-
- if (which === 'image_detections') {
- d = {
- key: feature.properties.key,
- image_key: feature.properties.image_key,
- value: feature.properties.value,
- shape: feature.properties.shape
- };
-
- if (!cache.forImageKey[d.image_key]) {
- cache.forImageKey[d.image_key] = [];
- }
-
- cache.forImageKey[d.image_key].push(d);
- }
- });
- });
- }
-
- 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 parsePagination(links) {
- return links.split(',').map(function (rel) {
- var elements = rel.split(';');
-
- 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
-
-
- 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.
-
-
- 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;
- }, []);
- }
-
- var serviceMapillary = {
- init: function init() {
- if (!_mlyCache) {
- this.reset();
- }
-
- 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);
- }
-
- _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];
-
- if (sequenceKey) {
- sequenceKeys[sequenceKey] = true;
- }
- }); // 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
-
- if (loadedCount === 2) resolve();
- }
-
- var head = select('head'); // load mapillary-viewercss
-
- 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
-
- 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
-
- }
- },
- showFeatureDetections: function showFeatureDetections(value) {
- _mlyShowFeatureDetections = value;
-
- if (!_mlyShowFeatureDetections && !_mlyShowSignDetections) {
- this.resetTags();
- }
- },
- showSignDetections: function showSignDetections(value) {
- _mlyShowSignDetections = value;
-
- 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]]);
-
- if (fromDate) {
- var fromTimestamp = new Date(fromDate).getTime();
- filter.push(['>=', 'capturedAt', fromTimestamp]);
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- filter.push(['>=', 'capturedAt', toTimestamp]);
- }
-
- if (_mlyViewer) {
- _mlyViewer.setFilter(filter);
- }
-
- _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);
-
- _mlyViewer.resize();
- }
-
- return this;
- },
- hideViewer: function hideViewer(context) {
- _mlyActiveImage = null;
- _mlySelectedImageKey = null;
-
- if (!_mlyFallback && _mlyViewer) {
- _mlyViewer.getComponent('sequence').stop();
- }
-
- 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);
-
- if (imageKey) {
- hash.photo = 'mapillary/' + imageKey;
- } else {
- delete hash.photo;
- }
-
- window.location.replace('#' + utilQsString(hash, true));
- }
- },
- highlightDetection: function highlightDetection(detection) {
- if (detection) {
- _mlyHighlightedDetection = detection.detection_key;
- }
-
- 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
-
- 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
-
- };
- }
-
- _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
-
-
- 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.
- //
-
- function nodeChanged(node) {
- that.resetTags();
- var clicks = _mlyClicks;
- var index = clicks.indexOf(node.key);
- var selectedKey = _mlySelectedImageKey;
- that.setActiveImage(node);
-
- 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..
-
- 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);
- }
-
- dispatch$4.call('nodeChanged');
- }
-
- 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;
-
- if (!fromViewer && imageKey) {
- _mlyClicks.push(imageKey);
- }
-
- this.setStyles(context, null, true);
-
- if (_mlyShowFeatureDetections) {
- this.updateDetections(imageKey, apibase + 'image_detections?layers=points&values=' + mapFeatureConfig.values + '&image_keys=' + imageKey);
- }
-
- if (_mlyShowSignDetections) {
- this.updateDetections(imageKey, apibase + 'image_detections?layers=trafficsigns&image_keys=' + imageKey);
- }
-
- if (_mlyViewer && imageKey) {
- _mlyViewer.moveToKey(imageKey)["catch"](function (e) {
- console.error('mly3', e);
- }); // eslint-disable-line no-console
-
- }
-
- 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
- };
- } 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 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
-
- 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;
- });
- 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 viewfieldPath() {
- var d = this.parentNode.__data__;
-
- 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';
- }
- }
-
- return this;
- },
- updateDetections: function updateDetections(imageKey, url) {
- if (!_mlyViewer || _mlyFallback) return;
- if (!imageKey) return;
-
- 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]);
- }
-
- function showDetections(detections) {
- detections.forEach(function (data) {
- var tag = makeTag(data);
-
- if (tag) {
- var tagComponent = _mlyViewer.getComponent('tag');
-
- tagComponent.add([tag]);
- }
- });
- }
-
- function makeTag(data) {
- var valueParts = data.value.split('--');
- if (!valueParts.length) return;
- var tag;
- var text;
- var color = 0xffffff;
-
- if (_mlyHighlightedDetection === data.key) {
- color = 0xffff00;
- text = valueParts[1];
-
- if (text === 'flat' || text === 'discrete' || text === 'sign') {
- text = valueParts[2];
- }
-
- text = text.replace(/-/g, ' ');
- text = text.charAt(0).toUpperCase() + text.slice(1);
- _mlyHighlightedDetection = null;
- }
-
- 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 tag;
- }
- },
- cache: function cache() {
- return _mlyCache;
- }
- };
-
- function validationIssue(attrs) {
- this.type = attrs.type; // required - name of rule that created the issue (e.g. 'missing_tag')
-
- this.subtype = attrs.subtype; // optional - category of the issue within the type (e.g. 'relation_type' under 'missing_tag')
-
- this.severity = attrs.severity; // required - 'warning' or 'error'
-
- this.message = attrs.message; // required - function returning localized string
-
- 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
-
- 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
-
- this.dynamicFixes = attrs.dynamicFixes; // optional - function(context) returning fixes
-
- this.hash = attrs.hash; // optional - string to further differentiate the issue
-
- this.id = generateID.apply(this); // generated - see below
-
- this.autoFix = null; // generated - if autofix exists, will be set below
- // A unique, deterministic string hash.
- // Issues with identical id values are considered identical.
-
- function generateID() {
- var parts = [this.type];
-
- if (this.hash) {
- // subclasses can pass in their own differentiator
- parts.push(this.hash);
- }
-
- if (this.subtype) {
- parts.push(this.subtype);
- } // include the entities this issue is for
- // (sort them so the id is deterministic)
-
-
- if (this.entityIds) {
- var entityKeys = this.entityIds.slice().sort();
- parts.push.apply(parts, entityKeys);
- }
-
- return parts.join(':');
- }
-
- this.extent = function (resolver) {
- if (this.loc) {
- return geoExtent(this.loc);
- }
-
- if (this.entityIds && this.entityIds.length) {
- return this.entityIds.reduce(function (extent, entityId) {
- return extent.extend(resolver.entity(entityId).extent(resolver));
- }, geoExtent());
- }
-
- return null;
- };
-
- this.fixes = function (context) {
- var fixes = this.dynamicFixes ? this.dynamicFixes(context) : [];
- var issue = this;
-
- 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);
- }
- }));
- }
-
- 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;
-
- if (fix.autoArgs) {
- issue.autoFix = fix;
- }
- });
- return fixes;
- };
- }
- function validationIssueFix(attrs) {
- this.title = attrs.title; // Required
-
- this.onClick = attrs.onClick; // Optional - the function to run to apply the fix
-
- 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
-
- this.entityIds = attrs.entityIds || []; // Optional - used for hover-higlighting.
-
- this.autoArgs = attrs.autoArgs; // Optional - pass [actions, annotation] arglist if this fix can automatically run
-
- this.issue = null; // Generated link - added by validationIssue
- }
-
- 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];
-
- 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];
-
- var expression = _negativeRegex[tagKey].join('|');
-
- var regex = new RegExp(expression);
- return function (tags) {
- return !regex.test(tags[tagKey]);
- };
- }
- };
- };
-
- var buildLineKeys = function buildLineKeys() {
- return {
- highway: {
- rest_area: true,
- services: true
- },
- railway: {
- roundhouse: true,
- station: true,
- traverser: true,
- turntable: true,
- wash: true
- }
- };
- };
-
- 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 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, '');
- });
- };
-
- var tagMap = Object.keys(selector).reduce(function (expectedTags, key) {
- var values;
- var isRegex = /regex/gi.test(key);
- var isEqual = /equals/gi.test(key);
-
- if (isRegex || isEqual) {
- Object.keys(selector[key]).forEach(function (selectorKey) {
- values = isEqual ? [selector[key][selectorKey]] : getRegexValues(selector[key][selectorKey]);
-
- if (expectedTags.hasOwnProperty(selectorKey)) {
- values = values.concat(expectedTags[selectorKey]);
- }
-
- 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 (expectedTags.hasOwnProperty(tagKey)) {
- values = values.concat(expectedTags[tagKey]);
- }
-
- expectedTags[tagKey] = values;
- }
-
- return expectedTags;
- }, {});
- return tagMap;
- },
- // inspired by osmWay#isArea()
- inferGeometry: function inferGeometry(tagMap) {
- var _lineKeys = this._lineKeys;
- var _areaKeys = this._areaKeys;
-
- var keyValueDoesNotImplyArea = function keyValueDoesNotImplyArea(key) {
- return utilArrayIntersection(tagMap[key], Object.keys(_areaKeys[key])).length > 0;
- };
-
- var keyValueImpliesLine = function keyValueImpliesLine(key) {
- return utilArrayIntersection(tagMap[key], Object.keys(_lineKeys[key])).length > 0;
- };
-
- if (tagMap.hasOwnProperty('area')) {
- if (tagMap.area.indexOf('yes') > -1) {
- return 'area';
- }
-
- if (tagMap.area.indexOf('no') > -1) {
- return 'line';
- }
- }
-
- for (var key in tagMap) {
- if (key in _areaKeys && !keyValueDoesNotImplyArea(key)) {
- return 'area';
- }
-
- if (key in _lineKeys && keyValueImpliesLine(key)) {
- return 'area';
- }
- }
-
- 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]
- }));
- }
- }
- };
-
- 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;
- }
- };
-
- var apibase$1 = 'https://nominatim.openstreetmap.org/';
- var _inflight = {};
-
- var _nominatimCache;
-
- 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);
- }
- });
- },
- 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;
- }
-
- 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);
- }
-
- var extent = geoExtent(loc).padByMeters(200);
-
- _nominatimCache.insert(Object.assign(extent.bbox(), {
- data: result
- }));
-
- 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];
-
- if (result && result.error) {
- throw new Error(result.error);
- }
-
- 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;
-
- var _oscSelectedImage;
-
- var _loadViewerPromise$1;
-
- function abortRequest$4(controller) {
- controller.abort();
- }
-
- 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 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 cache = _oscCache[which];
- Object.keys(cache.inflight).forEach(function (k) {
- var wanted = tiles.find(function (tile) {
- return k.indexOf(tile.id + ',') === 0;
- });
-
- if (!wanted) {
- abortRequest$4(cache.inflight[k]);
- delete cache.inflight[k];
- }
- });
- tiles.forEach(function (tile) {
- loadNextTilePage$1(which, currZoom, url, tile);
- });
- }
-
- 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];
-
- if (!data || !data.currentPageItems || !data.currentPageItems.length) {
- throw new Error('No Data');
- }
-
- 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
-
- var seq = _oscCache.sequences[d.sequence_id];
-
- if (!seq) {
- seq = {
- rotation: 0,
- images: []
- };
- _oscCache.sequences[d.sequence_id] = seq;
- }
-
- 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
-
-
- 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 tiler = utilTiler().zoomExtent([z2, z2]);
- return tiler.getTiles(projection).map(function (tile) {
- return tile.extent;
- });
- } // no more than `limit` results per partition.
-
-
- 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();
- }
-
- this.event = utilRebind(this, dispatch$5, 'on');
- },
- reset: function reset() {
- if (_oscCache) {
- Object.values(_oscCache.images.inflight).forEach(abortRequest$4);
- }
-
- _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
-
- _oscCache.images.rtree.search(bbox).forEach(function (d) {
- sequenceKeys[d.data.sequence_id] = true;
- }); // make linestrings from those sequences
-
-
- 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$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
-
- 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();
-
- 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)');
-
- 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 (d.captured_at) {
- attribution.append('span').attr('class', 'captured_at').html(localeDateString(d.captured_at));
- attribution.append('span').html('|');
- }
-
- 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;
-
- 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
-
- 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
-
- context.container().selectAll('.viewfield-group .viewfield').attr('d', viewfieldPath);
-
- function viewfieldPath() {
- var d = this.parentNode.__data__;
-
- 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';
- }
- }
-
- return this;
- },
- updateUrlImage: function updateUrlImage(imageKey) {
- if (!window.mocha) {
- var hash = utilStringQs(window.location.hash);
-
- if (imageKey) {
- hash.photo = 'openstreetcam/' + imageKey;
- } else {
- delete hash.photo;
- }
-
- window.location.replace('#' + utilQsString(hash, true));
- }
- },
- 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;
- });
-
- // `Date.prototype.toJSON` method
- // https://tc39.github.io/ecma262/#sec-date.prototype.tojson
- _export({ target: 'Date', proto: true, forced: FORCED$f }, {
- // eslint-disable-next-line no-unused-vars
- toJSON: function toJSON(key) {
- var O = toObject(this);
- var pv = toPrimitive(O);
- return typeof pv == 'number' && !isFinite(pv) ? null : O.toISOString();
- }
- });
-
- // `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);
- }
- });
-
- /**
- * 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);
-
- return value != null && (type == 'object' || type == 'function');
- }
-
- /** Detect free variable `global` from Node.js. */
- var freeGlobal = (typeof global === "undefined" ? "undefined" : _typeof(global)) == 'object' && global && global.Object === Object && global;
-
- /** Detect free variable `self`. */
-
- var freeSelf = (typeof self === "undefined" ? "undefined" : _typeof(self)) == 'object' && self && self.Object === Object && self;
- /** Used as a reference to the global object. */
-
- var root$1 = freeGlobal || freeSelf || Function('return this')();
-
- /**
- * 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$1 = function now() {
- return root$1.Date.now();
- };
-
- /** Built-in value references. */
-
- var _Symbol = root$1.Symbol;
-
- /** Used for built-in method references. */
-
- var objectProto = Object.prototype;
- /** Used to check objects for own properties. */
-
- var hasOwnProperty$1 = objectProto.hasOwnProperty;
- /**
- * Used to resolve the
- * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
- * of values.
- */
-
- var nativeObjectToString = objectProto.toString;
- /** Built-in value references. */
-
- 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`.
- */
-
- function getRawTag(value) {
- var isOwn = hasOwnProperty$1.call(value, symToStringTag),
- tag = value[symToStringTag];
-
- try {
- value[symToStringTag] = undefined;
- var unmasked = true;
- } catch (e) {}
-
- var result = nativeObjectToString.call(value);
-
- if (unmasked) {
- if (isOwn) {
- value[symToStringTag] = tag;
- } else {
- delete value[symToStringTag];
- }
- }
-
- return result;
- }
-
- /** 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.
- */
-
- 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.
- */
-
- function objectToString$1(value) {
- return nativeObjectToString$1.call(value);
- }
-
- /** `Object#toString` result references. */
-
- var nullTag = '[object Null]',
- undefinedTag = '[object Undefined]';
- /** Built-in value references. */
-
- 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`.
- */
-
- function baseGetTag(value) {
- if (value == null) {
- return value === undefined ? undefinedTag : nullTag;
- }
-
- return symToStringTag$1 && symToStringTag$1 in Object(value) ? getRawTag(value) : objectToString$1(value);
- }
-
- /**
- * 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';
- }
-
- /** `Object#toString` result references. */
-
- 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
- */
-
- function isSymbol$1(value) {
- return _typeof(value) == 'symbol' || isObjectLike(value) && baseGetTag(value) == symbolTag;
- }
-
- /** Used as references for various `Number` constants. */
-
- var NAN = 0 / 0;
- /** Used to match leading and trailing whitespace. */
-
- var reTrim = /^\s+|\s+$/g;
- /** Used to detect bad signed hexadecimal string values. */
-
- 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 toNumber$1(value) {
- if (typeof value == 'number') {
- return value;
- }
-
- if (isSymbol$1(value)) {
- return NAN;
- }
-
- if (isObject$1(value)) {
- var other = typeof value.valueOf == 'function' ? value.valueOf() : value;
- value = isObject$1(other) ? other + '' : other;
- }
-
- if (typeof value != 'string') {
- return value === 0 ? value : +value;
- }
-
- value = value.replace(reTrim, '');
- var isBinary = reIsBinary.test(value);
- return isBinary || reIsOctal.test(value) ? freeParseInt(value.slice(2), isBinary ? 2 : 8) : reIsBadHex.test(value) ? NAN : +value;
- }
-
- /** 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 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;
-
- if (typeof func != 'function') {
- throw new TypeError(FUNC_ERROR_TEXT);
- }
-
- wait = toNumber$1(wait) || 0;
-
- 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 invokeFunc(time) {
- var args = lastArgs,
- thisArg = lastThis;
- lastArgs = lastThis = undefined;
- lastInvokeTime = time;
- result = func.apply(thisArg, args);
- return result;
- }
-
- function leadingEdge(time) {
- // Reset any `maxWait` timer.
- lastInvokeTime = time; // Start the timer for the trailing edge.
-
- timerId = setTimeout(timerExpired, wait); // Invoke the leading edge.
-
- 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;
- }
-
- 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.
-
- return lastCallTime === undefined || timeSinceLastCall >= wait || timeSinceLastCall < 0 || maxing && timeSinceLastInvoke >= maxWait;
- }
-
- function timerExpired() {
- var time = now$1();
-
- if (shouldInvoke(time)) {
- return trailingEdge(time);
- } // Restart the timer.
-
-
- timerId = setTimeout(timerExpired, remainingWait(time));
- }
-
- function trailingEdge(time) {
- timerId = undefined; // Only invoke if we have `lastArgs` which means `func` has been
- // debounced at least once.
-
- if (trailing && lastArgs) {
- return invokeFunc(time);
- }
-
- lastArgs = lastThis = undefined;
- return result;
- }
-
- function cancel() {
- if (timerId !== undefined) {
- clearTimeout(timerId);
- }
-
- lastInvokeTime = 0;
- lastArgs = lastCallTime = lastThis = timerId = undefined;
- }
-
- function flush() {
- return timerId === undefined ? result : trailingEdge(now$1());
- }
-
- function debounced() {
- var time = now$1(),
- isInvoking = shouldInvoke(time);
- lastArgs = arguments;
- lastThis = this;
- lastCallTime = time;
-
- if (isInvoking) {
- if (timerId === undefined) {
- return leadingEdge(lastCallTime);
- }
-
- if (maxing) {
- // Handle invocations in a tight loop.
- clearTimeout(timerId);
- timerId = setTimeout(timerExpired, wait);
- return invokeFunc(lastCallTime);
- }
- }
-
- if (timerId === undefined) {
- timerId = setTimeout(timerExpired, wait);
- }
-
- return result;
- }
-
- debounced.cancel = cancel;
- debounced.flush = flush;
- return debounced;
- }
-
- /** Error message constants. */
-
- 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);
- */
-
- function throttle(func, wait, options) {
- var leading = true,
- trailing = true;
-
- if (typeof func != 'function') {
- throw new TypeError(FUNC_ERROR_TEXT$1);
- }
-
- if (isObject$1(options)) {
- leading = 'leading' in options ? !!options.leading : leading;
- trailing = 'trailing' in options ? !!options.trailing : trailing;
- }
-
- return debounce(func, wait, {
- 'leading': leading,
- 'maxWait': wait,
- 'trailing': trailing
- });
- }
-
- 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;
-
- function utf8Encode(str) {
- var x,
- y,
- output = '',
- i = -1,
- l;
-
- if (str && str.length) {
- l = str.length;
-
- while ((i += 1) < l) {
- /* Decode utf-16 surrogate pairs */
- x = str.charCodeAt(i);
- y = i + 1 < l ? str.charCodeAt(i + 1) : 0;
-
- if (0xD800 <= x && x <= 0xDBFF && 0xDC00 <= y && y <= 0xDFFF) {
- x = 0x10000 + ((x & 0x03FF) << 10) + (y & 0x03FF);
- i += 1;
- }
- /* Encode output as utf-8 */
-
-
- 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);
- }
- }
- }
-
- return output;
- }
-
- function utf8Decode(str) {
- var i,
- ac,
- c1,
- c2,
- c3,
- arr = [],
- l;
- i = ac = c1 = c2 = c3 = 0;
-
- if (str && str.length) {
- l = str.length;
- str += '';
-
- while (i < l) {
- c1 = str.charCodeAt(i);
- ac += 1;
-
- 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 arr.join('');
- }
- /**
- * Add integers, wrapping at 2^32. This uses 16-bit operations internally
- * to work around bugs in some JS interpreters.
- */
-
-
- function safe_add(x, y) {
- var lsw = (x & 0xFFFF) + (y & 0xFFFF),
- msw = (x >> 16) + (y >> 16) + (lsw >> 16);
- return msw << 16 | lsw & 0xFFFF;
- }
- /**
- * Bitwise rotate a 32-bit number to the left.
- */
-
-
- function bit_rol(num, cnt) {
- return num << cnt | num >>> 32 - cnt;
- }
- /**
- * Convert a raw string to a hex string
- */
-
-
- function rstr2hex(input, hexcase) {
- var hex_tab = hexcase ? '0123456789ABCDEF' : '0123456789abcdef',
- output = '',
- x,
- i = 0,
- l = input.length;
-
- for (; i < l; i += 1) {
- x = input.charCodeAt(i);
- output += hex_tab.charAt(x >>> 4 & 0x0F) + hex_tab.charAt(x & 0x0F);
- }
-
- return output;
- }
- /**
- * Convert an array of big-endian words to a string
- */
-
-
- function binb2rstr(input) {
- var i,
- l = input.length * 32,
- output = '';
-
- for (i = 0; i < l; i += 8) {
- output += String.fromCharCode(input[i >> 5] >>> 24 - i % 32 & 0xFF);
- }
-
- return output;
- }
- /**
- * Convert an array of little-endian words to a string
- */
-
-
- function binl2rstr(input) {
- var i,
- l = input.length * 32,
- output = '';
-
- for (i = 0; i < l; i += 8) {
- output += String.fromCharCode(input[i >> 5] >>> i % 32 & 0xFF);
- }
-
- return output;
- }
- /**
- * Convert a raw string to an array of little-endian words
- * Characters >255 have their high-byte silently ignored.
- */
-
-
- function rstr2binl(input) {
- var i,
- l = input.length * 8,
- output = Array(input.length >> 2),
- lo = output.length;
-
- for (i = 0; i < lo; i += 1) {
- output[i] = 0;
- }
-
- for (i = 0; i < l; i += 8) {
- output[i >> 5] |= (input.charCodeAt(i / 8) & 0xFF) << i % 32;
- }
-
- return output;
- }
- /**
- * Convert a raw string to an array of big-endian words
- * Characters >255 have their high-byte silently ignored.
- */
-
-
- function rstr2binb(input) {
- var i,
- l = input.length * 8,
- output = Array(input.length >> 2),
- lo = output.length;
-
- for (i = 0; i < lo; i += 1) {
- output[i] = 0;
- }
-
- for (i = 0; i < l; i += 8) {
- output[i >> 5] |= (input.charCodeAt(i / 8) & 0xFF) << 24 - i % 32;
- }
-
- return output;
- }
- /**
- * Convert a raw string to an arbitrary string encoding
- */
-
-
- 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 */
-
- dividend = Array(Math.ceil(input.length / 2));
- ld = dividend.length;
-
- 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.
- */
-
-
- while (dividend.length > 0) {
- quotient = Array();
- x = 0;
-
- for (i = 0; i < dividend.length; i += 1) {
- x = (x << 16) + dividend[i];
- q = Math.floor(x / divisor);
- x -= q * divisor;
-
- if (quotient.length > 0 || q > 0) {
- quotient[quotient.length] = q;
- }
- }
-
- remainders[remainders.length] = x;
- dividend = quotient;
- }
- /* Convert the remainders to the output string */
-
-
- output = '';
-
- for (i = remainders.length - 1; i >= 0; i--) {
- output += encoding.charAt(remainders[i]);
- }
- /* Append leading zero equivalents */
-
-
- full_length = Math.ceil(input.length * 8 / (Math.log(encoding.length) / Math.log(2)));
-
- for (i = output.length; i < full_length; i += 1) {
- output = encoding[0] + output;
- }
-
- return output;
- }
- /**
- * Convert a raw string to a base-64 string
- */
-
-
- function rstr2b64(input, b64pad) {
- var tab = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/',
- output = '',
- len = input.length,
- i,
- j,
- triplet;
- b64pad = b64pad || '=';
-
- 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 > input.length * 8) {
- output += b64pad;
- } else {
- output += tab.charAt(triplet >>> 6 * (3 - j) & 0x3F);
- }
- }
- }
-
- return output;
- }
-
- Hashes = {
- /**
- * @property {String} version
- * @readonly
- */
- VERSION: '1.0.6',
-
- /**
- * @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
-
- this.encode = function (input) {
- var i,
- j,
- triplet,
- output = '',
- len = input.length;
- pad = pad || '=';
- input = utf8 ? utf8Encode(input) : input;
-
- 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);
- }
- }
- }
-
- return output;
- }; // public method for decoding
-
-
- this.decode = function (input) {
- // var b64 = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
- var i,
- o1,
- o2,
- o3,
- h1,
- h2,
- h3,
- h4,
- bits,
- ac,
- dec = '',
- arr = [];
-
- if (!input) {
- return input;
- }
-
- i = ac = 0;
- input = input.replace(new RegExp('\\' + pad, 'gi'), ''); // use '='
- //input += '';
-
- 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);
-
- dec = arr.join('');
- dec = utf8 ? utf8Decode(dec) : dec;
- return dec;
- }; // set custom pad string
-
-
- this.setPad = function (str) {
- pad = str || pad;
- return this;
- }; // set custom tab string characters
-
-
- this.setTab = function (str) {
- tab = str || tab;
- return this;
- };
-
- this.setUTF8 = function (bool) {
- if (typeof bool === 'boolean') {
- utf8 = bool;
- }
-
- return this;
- };
- },
-
- /**
- * 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 (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)
-
-
- return (crc ^ -1) >>> 0;
- },
-
- /**
- * @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);
- };
-
- this.b64 = function (s) {
- return rstr2b64(rstr(s), b64pad);
- };
-
- this.any = function (s, e) {
- return rstr2any(rstr(s), e);
- };
-
- this.raw = function (s) {
- return rstr(s);
- };
-
- this.hex_hmac = function (k, d) {
- return rstr2hex(rstr_hmac(k, d), hexcase);
- };
-
- 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
- */
-
-
- this.vm_test = function () {
- return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
- };
- /**
- * Enable/disable uppercase hexadecimal returned string
- * @param {Boolean}
- * @return {Object} this
- */
-
-
- this.setUpperCase = function (a) {
- if (typeof a === 'boolean') {
- hexcase = a;
- }
-
- return this;
- };
- /**
- * Defines a base64 pad string
- * @param {String} Pad
- * @return {Object} this
- */
-
-
- this.setPad = function (a) {
- b64pad = a || b64pad;
- return this;
- };
- /**
- * Defines a base64 pad string
- * @param {Boolean}
- * @return {Object} [this]
- */
-
-
- this.setUTF8 = function (a) {
- if (typeof a === 'boolean') {
- utf8 = a;
- }
-
- return this;
- }; // private methods
-
- /**
- * Calculate the MD5 of a raw string
- */
-
-
- 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 rstr_hmac(key, data) {
- var bkey, ipad, opad, hash, i;
- key = utf8 ? utf8Encode(key) : key;
- data = utf8 ? utf8Encode(data) : data;
- bkey = rstr2binl(key);
-
- if (bkey.length > 16) {
- bkey = binl(bkey, key.length * 8);
- }
-
- ipad = Array(16), opad = Array(16);
-
- for (i = 0; i < 16; i += 1) {
- ipad[i] = bkey[i] ^ 0x36363636;
- opad[i] = bkey[i] ^ 0x5C5C5C5C;
- }
-
- 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.
- */
-
-
- function binl(x, len) {
- var i,
- olda,
- oldb,
- oldc,
- oldd,
- a = 1732584193,
- b = -271733879,
- c = -1732584194,
- d = 271733878;
- /* append padding */
-
- x[len >> 5] |= 0x80 << len % 32;
- x[(len + 64 >>> 9 << 4) + 14] = len;
-
- 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.
- */
-
-
- 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);
- }
-
- function md5_ff(a, b, c, d, x, s, t) {
- return md5_cmn(b & c | ~b & d, a, b, x, s, t);
- }
-
- function md5_gg(a, b, c, d, x, s, t) {
- return md5_cmn(b & d | c & ~d, a, b, x, s, t);
- }
-
- function md5_hh(a, b, c, d, x, s, t) {
- return md5_cmn(b ^ c ^ d, a, b, x, s, t);
- }
-
- function md5_ii(a, b, c, d, x, s, t) {
- return md5_cmn(c ^ (b | ~d), a, b, x, s, t);
- }
- },
-
- /**
- * @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
-
- this.hex = function (s) {
- return rstr2hex(rstr(s), hexcase);
- };
-
- this.b64 = function (s) {
- return rstr2b64(rstr(s), b64pad);
- };
-
- this.any = function (s, e) {
- return rstr2any(rstr(s), e);
- };
-
- this.raw = function (s) {
- return rstr(s);
- };
-
- this.hex_hmac = function (k, d) {
- return rstr2hex(rstr_hmac(k, d));
- };
-
- 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
- * @public
- */
-
-
- this.vm_test = function () {
- return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
- };
- /**
- * @description Enable/disable uppercase hexadecimal returned string
- * @param {boolean}
- * @return {Object} this
- * @public
- */
-
-
- this.setUpperCase = function (a) {
- if (typeof a === 'boolean') {
- hexcase = a;
- }
-
- return this;
- };
- /**
- * @description Defines a base64 pad string
- * @param {string} Pad
- * @return {Object} this
- * @public
- */
-
-
- this.setPad = function (a) {
- b64pad = a || b64pad;
- return this;
- };
- /**
- * @description Defines a base64 pad string
- * @param {boolean}
- * @return {Object} this
- * @public
- */
-
-
- this.setUTF8 = function (a) {
- if (typeof a === 'boolean') {
- utf8 = a;
- }
-
- return this;
- }; // private methods
-
- /**
- * Calculate the SHA-512 of a raw string
- */
-
-
- 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 rstr_hmac(key, data) {
- var bkey, ipad, opad, i, hash;
- key = utf8 ? utf8Encode(key) : key;
- data = utf8 ? utf8Encode(data) : data;
- bkey = rstr2binb(key);
-
- if (bkey.length > 16) {
- bkey = binb(bkey, key.length * 8);
- }
-
- ipad = Array(16), opad = Array(16);
-
- for (i = 0; i < 16; i += 1) {
- ipad[i] = bkey[i] ^ 0x36363636;
- opad[i] = bkey[i] ^ 0x5C5C5C5C;
- }
-
- 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
- */
-
-
- 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 */
-
- x[len >> 5] |= 0x80 << 24 - len % 32;
- x[(len + 64 >> 9 << 4) + 15] = len;
-
- for (i = 0; i < x.length; i += 16) {
- olda = a;
- oldb = b;
- oldc = c;
- oldd = d;
- olde = e;
-
- 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);
- }
-
- 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;
- }
-
- 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);
- }
-
- return Array(a, b, c, d, e);
- }
- /**
- * 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;
- }
-
- if (t < 40) {
- return b ^ c ^ d;
- }
-
- if (t < 60) {
- return b & c | b & d | c & d;
- }
-
- return b ^ c ^ d;
- }
- /**
- * Determine the appropriate additive constant for the current iteration
- */
-
-
- 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 : '=',
-
- /* base-64 pad character. Default '=' for strict RFC compliance */
- utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
-
- /* enable/disable utf8 encoding */
- sha256_K;
- /* privileged (public) methods */
-
- this.hex = function (s) {
- return rstr2hex(rstr(s, utf8));
- };
-
- this.b64 = function (s) {
- return rstr2b64(rstr(s, utf8), b64pad);
- };
-
- this.any = function (s, e) {
- return rstr2any(rstr(s, utf8), e);
- };
-
- this.raw = function (s) {
- return rstr(s, utf8);
- };
-
- this.hex_hmac = function (k, d) {
- return rstr2hex(rstr_hmac(k, d));
- };
-
- 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
- * @public
- */
-
-
- this.vm_test = function () {
- return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
- };
- /**
- * Enable/disable uppercase hexadecimal returned string
- * @param {boolean}
- * @return {Object} this
- * @public
- */
-
-
- this.setUpperCase = function (a) {
- if (typeof a === 'boolean') {
- hexcase = a;
- }
-
- return this;
- };
- /**
- * @description Defines a base64 pad string
- * @param {string} Pad
- * @return {Object} this
- * @public
- */
-
-
- this.setPad = function (a) {
- b64pad = a || b64pad;
- return this;
- };
- /**
- * Defines a base64 pad string
- * @param {boolean}
- * @return {Object} this
- * @public
- */
-
-
- this.setUTF8 = function (a) {
- if (typeof a === 'boolean') {
- utf8 = a;
- }
-
- return this;
- }; // private methods
-
- /**
- * Calculate the SHA-512 of a raw string
- */
-
-
- 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)
- */
-
-
- 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);
-
- if (bkey.length > 16) {
- bkey = binb(bkey, key.length * 8);
- }
-
- for (; i < 16; i += 1) {
- ipad[i] = bkey[i] ^ 0x36363636;
- opad[i] = bkey[i] ^ 0x5C5C5C5C;
- }
-
- 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
- */
-
-
- function sha256_S(X, n) {
- return X >>> n | X << 32 - n;
- }
-
- function sha256_R(X, n) {
- return X >>> n;
- }
-
- 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 sha256_Sigma0256(x) {
- return sha256_S(x, 2) ^ sha256_S(x, 13) ^ sha256_S(x, 22);
- }
-
- function sha256_Sigma1256(x) {
- return sha256_S(x, 6) ^ sha256_S(x, 11) ^ sha256_S(x, 25);
- }
-
- function sha256_Gamma0256(x) {
- return sha256_S(x, 7) ^ sha256_S(x, 18) ^ sha256_R(x, 3);
- }
-
- function sha256_Gamma1256(x) {
- return sha256_S(x, 17) ^ sha256_S(x, 19) ^ sha256_R(x, 10);
- }
-
- 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 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;
-
- 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];
-
- 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]);
- }
-
- 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);
- }
-
- 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]);
- }
-
- return HASH;
- }
- },
-
- /**
- * @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 : '=',
-
- /* base-64 pad character. Default '=' for strict RFC compliance */
- utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
-
- /* enable/disable utf8 encoding */
- sha512_k;
- /* privileged (public) methods */
-
- this.hex = function (s) {
- return rstr2hex(rstr(s));
- };
-
- this.b64 = function (s) {
- return rstr2b64(rstr(s), b64pad);
- };
-
- this.any = function (s, e) {
- return rstr2any(rstr(s), e);
- };
-
- this.raw = function (s) {
- return rstr(s);
- };
-
- this.hex_hmac = function (k, d) {
- return rstr2hex(rstr_hmac(k, d));
- };
-
- 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
- * @public
- */
-
-
- this.vm_test = function () {
- return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
- };
- /**
- * @description Enable/disable uppercase hexadecimal returned string
- * @param {boolean}
- * @return {Object} this
- * @public
- */
-
-
- this.setUpperCase = function (a) {
- if (typeof a === 'boolean') {
- hexcase = a;
- }
-
- return this;
- };
- /**
- * @description Defines a base64 pad string
- * @param {string} Pad
- * @return {Object} this
- * @public
- */
-
-
- this.setPad = function (a) {
- b64pad = a || b64pad;
- return this;
- };
- /**
- * @description Defines a base64 pad string
- * @param {boolean}
- * @return {Object} this
- * @public
- */
-
-
- this.setUTF8 = function (a) {
- if (typeof a === 'boolean') {
- utf8 = a;
- }
-
- return this;
- };
- /* private methods */
-
- /**
- * Calculate the SHA-512 of a raw string
- */
-
-
- 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)
- */
-
-
- 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);
- }
-
- for (; i < 32; i += 1) {
- ipad[i] = bkey[i] ^ 0x36363636;
- opad[i] = bkey[i] ^ 0x5C5C5C5C;
- }
-
- 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
- */
-
-
- 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);
-
- 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)];
- }
-
- 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.
-
-
- x[len >> 5] |= 0x80 << 24 - (len & 0x1f);
- x[(len + 128 >> 10 << 5) + 31] = len;
- l = x.length;
-
- 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]);
-
- for (j = 0; j < 16; j += 1) {
- W[j].h = x[i + 2 * j];
- W[j].l = x[i + 2 * j + 1];
- }
-
- 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
-
- 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]);
- }
-
- 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
-
- 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
-
- 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
-
- 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);
- }
-
- 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
-
-
- for (i = 0; i < 8; i += 1) {
- hash[2 * i] = H[i].h;
- hash[2 * i + 1] = H[i].l;
- }
-
- return hash;
- } //A constructor for 64-bit numbers
-
-
- function int64(h, l) {
- this.h = h;
- this.l = l; //this.toString = int64toString;
- } //Copies src into dst, assuming both are 64-bit numbers
-
-
- 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
-
-
- 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 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 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
-
-
- 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 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 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;
- }
- },
-
- /**
- * @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,
-
- /* hexadecimal output case format. false - lowercase; true - uppercase */
- b64pad = options && typeof options.pad === 'string' ? options.pa : '=',
-
- /* base-64 pad character. Default '=' for strict RFC compliance */
- utf8 = options && typeof options.utf8 === 'boolean' ? options.utf8 : true,
-
- /* 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 */
-
- this.hex = function (s) {
- return rstr2hex(rstr(s));
- };
-
- this.b64 = function (s) {
- return rstr2b64(rstr(s), b64pad);
- };
-
- this.any = function (s, e) {
- return rstr2any(rstr(s), e);
- };
-
- this.raw = function (s) {
- return rstr(s);
- };
-
- this.hex_hmac = function (k, d) {
- return rstr2hex(rstr_hmac(k, d));
- };
-
- 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
- * @public
- */
-
-
- this.vm_test = function () {
- return hex('abc').toLowerCase() === '900150983cd24fb0d6963f7d28e17f72';
- };
- /**
- * @description Enable/disable uppercase hexadecimal returned string
- * @param {boolean}
- * @return {Object} this
- * @public
- */
-
-
- this.setUpperCase = function (a) {
- if (typeof a === 'boolean') {
- hexcase = a;
- }
-
- return this;
- };
- /**
- * @description Defines a base64 pad string
- * @param {string} Pad
- * @return {Object} this
- * @public
- */
-
-
- this.setPad = function (a) {
- if (typeof a !== 'undefined') {
- b64pad = a;
- }
-
- return this;
- };
- /**
- * @description Defines a base64 pad string
- * @param {boolean}
- * @return {Object} this
- * @public
- */
-
-
- this.setUTF8 = function (a) {
- if (typeof a === 'boolean') {
- utf8 = a;
- }
-
- return this;
- };
- /* private methods */
-
- /**
- * Calculate the rmd160 of a raw string
- */
-
-
- 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)
- */
-
-
- 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 (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;
- }
-
- 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
- */
-
-
- function binl2rstr(input) {
- var i,
- output = '',
- l = input.length * 32;
-
- for (i = 0; i < l; i += 8) {
- output += String.fromCharCode(input[i >> 5] >>> i % 32 & 0xFF);
- }
-
- return output;
- }
- /**
- * Calculate the RIPE-MD160 of an array of little-endian words, and a bit length.
- */
-
-
- 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 */
-
- 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;
-
- 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;
- }
-
- return [h0, h1, h2, h3, h4];
- } // specific algorithm methods
-
-
- 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';
- }
-
- 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 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 (window, undefined$1) {
- var freeExports = false;
-
- {
- freeExports = exports;
-
- if (exports && _typeof(commonjsGlobal) === 'object' && commonjsGlobal && commonjsGlobal === commonjsGlobal.global) {
- window = commonjsGlobal;
- }
- }
-
- 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
-
- });
-
- var immutable = extend$2;
- var hasOwnProperty$2 = Object.prototype.hasOwnProperty;
-
- function extend$2() {
- var target = {};
-
- for (var i = 0; i < arguments.length; i++) {
- var source = arguments[i];
-
- for (var key in source) {
- if (hasOwnProperty$2.call(source, key)) {
- target[key] = source[key];
- }
- }
- }
-
- return target;
- }
-
- 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$/;
-
- xhr.onreadystatechange = function () {
- if (4 === xhr.readyState && 0 !== xhr.status) {
- if (twoHundred.test(xhr.status)) callback(null, xhr);else return callback(xhr, null);
- }
- };
-
- xhr.onerror = function (e) {
- return callback(e, null);
- };
-
- xhr.open(method, url, true);
-
- for (var h in headers) {
- xhr.setRequestHeader(h, headers[h]);
- }
-
- xhr.send(data);
- return xhr;
- };
-
- 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);
- };
-
- ohauth.nonce = function () {
- for (var o = ''; o.length < 6;) {
- o += '0123456789ABCDEFGHIJKLMNOPQRSTUVWXTZabcdefghiklmnopqrstuvwxyz'[Math.floor(Math.random() * 61)];
- }
-
- return o;
- };
-
- ohauth.authHeader = function (obj) {
- return Object.keys(obj).sort().map(function (key) {
- return encodeURIComponent(key) + '="' + encodeURIComponent(obj[key]) + '"';
- }).join(', ');
- };
-
- 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');
- };
-
- 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.
- */
-
-
- 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 (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);
- };
- };
-
- var ohauth_1 = ohauth;
-
- 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;
-
- if (numUrls === 0) {
- throw new Error("resolveUrl requires at least one argument; got none.");
- }
-
- var base = document.createElement("base");
- base.href = arguments[0];
-
- if (numUrls === 1) {
- return base.href;
- }
-
- var head = document.getElementsByTagName("head")[0];
- head.insertBefore(base, head.firstChild);
- var a = document.createElement("a");
- var resolved;
-
- for (var index = 1; index < numUrls; index++) {
- a.href = arguments[index];
- resolved = a.href;
- base.href = resolved;
- }
-
- head.removeChild(base);
- return resolved;
- }
-
- 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
- };
-
- 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;
- });
- }
-
- return obj;
- };
- }
- }
-
- 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
-
-
- return function create(obj, assignProps1, assignProps2, etc) {
- var assignArgsList = slice$2(arguments, 1);
- F.prototype = obj;
- return assign.apply(this, [new F()].concat(assignArgsList));
- };
- }
- }
-
- 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, '');
- };
- }
- }
-
- function bind$1(obj, fn) {
- return function () {
- return fn.apply(obj, Array.prototype.slice.call(arguments, 0));
- };
- }
-
- function slice$2(arr, index) {
- return Array.prototype.slice.call(arr, index || 0);
- }
-
- function each(obj, fn) {
- pluck(obj, function (val, key) {
- fn(val, key);
- return false;
- });
- }
-
- function map$1(obj, fn) {
- var res = isList(obj) ? [] : {};
- pluck(obj, function (v, k) {
- res[k] = fn(v, k);
- return false;
- });
- return res;
- }
-
- 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];
- }
- }
- }
- }
- }
-
- function isList(val) {
- return val != null && typeof val != 'function' && typeof val.length == 'number';
- }
-
- function isFunction(val) {
- return val && {}.toString.call(val) === '[object Function]';
- }
-
- function isObject$2(val) {
- return val && {}.toString.call(val) === '[object Object]';
- }
-
- 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);
- }
-
- 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, ''));
- });
- },
- // 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);
- }
- };
-
- function _warn() {
- var _console = typeof console == 'undefined' ? null : console;
-
- if (!_console) {
- return;
- }
-
- var fn = _console.warn ? _console.warn : _console.log;
- fn.apply(_console, arguments);
- }
-
- function _createStore(storages, plugins, namespace) {
- if (!namespace) {
- namespace = '';
- }
-
- if (storages && !isList$1(storages)) {
- storages = [storages];
- }
-
- if (plugins && !isList$1(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
-
- 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;
- }
- },
- _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;
- }
-
- each$1(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
-
-
- var val = '';
-
- try {
- val = JSON.parse(strVal);
- } catch (e) {
- val = strVal;
- }
-
- return val !== undefined ? val : defaultVal;
- },
- _addStorage: function _addStorage(storage) {
- if (this.enabled) {
- return;
- }
-
- 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.
-
- 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.
-
-
- var seenPlugin = pluck$1(this.plugins, function (seenPlugin) {
- return plugin === seenPlugin;
- });
-
- if (seenPlugin) {
- return;
- }
-
- this.plugins.push(plugin); // Check that the plugin is properly formed
-
- if (!isFunction$1(plugin)) {
- throw new Error('Plugins must be function values that return objects');
- }
-
- var pluginProperties = plugin.call(this);
-
- if (!isObject$3(pluginProperties)) {
- throw new Error('Plugins must return an object of function properties');
- } // Add the plugin function properties to this store instance.
-
-
- 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.');
- }
-
- 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])');
-
- this._addStorage(storage);
- }
- };
- 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;
- }
-
- var Global$1 = util.Global;
- var localStorage_1 = {
- name: 'localStorage',
- read: read,
- write: write,
- each: each$2,
- remove: remove$2,
- clearAll: clearAll
- };
-
- function localStorage$1() {
- return Global$1.localStorage;
- }
-
- function read(key) {
- return localStorage$1().getItem(key);
- }
-
- function write(key, data) {
- return localStorage$1().setItem(key, data);
- }
-
- function each$2(fn) {
- for (var i = localStorage$1().length - 1; i >= 0; i--) {
- var key = localStorage$1().key(i);
- fn(read(key), key);
- }
- }
-
- function remove$2(key) {
- return localStorage$1().removeItem(key);
- }
-
- function clearAll() {
- return localStorage$1().clear();
- }
-
- // 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;
-
- function read$1(key) {
- return globalStorage[key];
- }
-
- function write$1(key, data) {
- globalStorage[key] = data;
- }
-
- function each$3(fn) {
- for (var i = globalStorage.length - 1; i >= 0; i--) {
- var key = globalStorage.key(i);
- fn(globalStorage[key], key);
- }
- }
-
- function remove$3(key) {
- return globalStorage.removeItem(key);
- }
-
- function clearAll$1() {
- each$3(function (key, _) {
- delete globalStorage[key];
- });
- }
-
- // 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 storageName = 'storejs';
- var doc = Global$3.document;
-
- var _withStorageEl = _makeIEStorageElFunction();
-
- var disable = (Global$3.navigator ? Global$3.navigator.userAgent : '').match(/ (MSIE 8|MSIE 9|MSIE 10)\./); // MSIE 9.x, MSIE 10.x
-
- function write$2(unfixedKey, data) {
- if (disable) {
- return;
- }
-
- var fixedKey = fixKey(unfixedKey);
-
- _withStorageEl(function (storageEl) {
- storageEl.setAttribute(fixedKey, data);
- storageEl.save(storageName);
- });
- }
-
- function read$2(unfixedKey) {
- if (disable) {
- return;
- }
-
- var fixedKey = fixKey(unfixedKey);
- var res = null;
-
- _withStorageEl(function (storageEl) {
- res = storageEl.getAttribute(fixedKey);
- });
-
- return res;
- }
-
- function each$4(callback) {
- _withStorageEl(function (storageEl) {
- var attributes = storageEl.XMLDocument.documentElement.attributes;
-
- for (var i = attributes.length - 1; i >= 0; i--) {
- var attr = attributes[i];
- callback(storageEl.getAttribute(attr.name), attr.name);
- }
- });
- }
-
- function remove$4(unfixedKey) {
- var fixedKey = fixKey(unfixedKey);
-
- _withStorageEl(function (storageEl) {
- storageEl.removeAttribute(fixedKey);
- storageEl.save(storageName);
- });
- }
-
- function clearAll$2() {
- _withStorageEl(function (storageEl) {
- var attributes = storageEl.XMLDocument.documentElement.attributes;
- storageEl.load(storageName);
-
- 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 forbiddenCharsRegex = new RegExp("[!\"#$%&'()*+,/\\\\:;<=>?@[\\]^`{|}~]", "g");
-
- function fixKey(key) {
- return key.replace(/^\d/, '___$&').replace(forbiddenCharsRegex, '___');
- }
-
- function _makeIEStorageElFunction() {
- if (!doc || !doc.documentElement || !doc.documentElement.addBehavior) {
- return null;
- }
-
- 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.createElement('div');
- storageOwner = doc.body;
- }
-
- 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
-
- storageOwner.appendChild(storageEl);
- storageEl.addBehavior('#default#userData');
- storageEl.load(storageName);
- storeFunction.apply(this, args);
- storageOwner.removeChild(storageEl);
- return;
- };
- }
-
- // doesn't work but cookies do. This implementation is adopted from
- // https://developer.mozilla.org/en-US/docs/Web/API/Storage/LocalStorage
-
- 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;
- }
-
- 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);
-
- for (var i = cookies.length - 1; i >= 0; i--) {
- if (!trim$4(cookies[i])) {
- continue;
- }
-
- var kvp = cookies[i].split('=');
- var key = unescape(kvp[0]);
- var val = unescape(kvp[1]);
- callback(val, key);
- }
- }
-
- function write$3(key, data) {
- if (!key) {
- return;
- }
-
- doc$1.cookie = escape(key) + "=" + escape(data) + "; expires=Tue, 19 Jan 2038 03:14:07 GMT; path=/";
- }
-
- function remove$5(key) {
- if (!key || !_has(key)) {
- return;
- }
-
- doc$1.cookie = escape(key) + "=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/";
- }
-
- function clearAll$3() {
- each$5(function (_, key) {
- remove$5(key);
- });
- }
-
- function _has(key) {
- return new RegExp("(?:^|;\\s*)" + escape(key).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=").test(doc$1.cookie);
- }
-
- 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
- };
-
- function sessionStorage() {
- return Global$5.sessionStorage;
- }
-
- function read$4(key) {
- return sessionStorage().getItem(key);
- }
-
- function write$4(key, data) {
- return sessionStorage().setItem(key, data);
- }
-
- function each$6(fn) {
- for (var i = sessionStorage().length - 1; i >= 0; i--) {
- var key = sessionStorage().key(i);
- fn(read$4(key), key);
- }
- }
-
- function remove$6(key) {
- return sessionStorage().removeItem(key);
- }
-
- function clearAll$4() {
- return sessionStorage().clear();
- }
-
- // 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 = {};
-
- function read$5(key) {
- return memoryStorage[key];
- }
-
- function write$5(key, data) {
- memoryStorage[key] = data;
- }
-
- function each$7(callback) {
- for (var key in memoryStorage) {
- if (memoryStorage.hasOwnProperty(key)) {
- callback(memoryStorage[key], key);
- }
- }
- }
-
- function remove$7(key) {
- delete memoryStorage[key];
- }
-
- function clearAll$5(key) {
- memoryStorage = {};
- }
-
- var all = [// Listed in order of usage preference
- localStorage_1, oldFFGlobalStorage, oldIEUserDataStorage, cookieStorage, sessionStorage_1, memoryStorage_1];
-
- /* 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 " "),
- // 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.
-
- /*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 = {};
- }
-
- (function () {
-
- 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;
-
- function f(n) {
- // Format integers to have at least two digits.
- return n < 10 ? "0" + n : n;
- }
-
- function this_value() {
- return this.valueOf();
- }
-
- 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;
- };
-
- Boolean.prototype.toJSON = this_value;
- Number.prototype.toJSON = this_value;
- String.prototype.toJSON = this_value;
- }
-
- 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 + "\"";
- }
-
- function str(key, holder) {
- // Produce a string from holder[key].
- var i; // The loop counter.
-
- var k; // The member key.
-
- 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.
-
- 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 (typeof rep === "function") {
- value = rep.call(holder, key, value);
- } // What happens next depends on the value's type.
-
-
- 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";
-
- 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.
-
- 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.
-
-
- 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;
-
- 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.
-
-
- 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 (rep && _typeof(rep) === "object") {
- length = rep.length;
-
- for (i = 0; i < length; i += 1) {
- if (typeof rep[i] === "string") {
- k = rep[i];
- v = str(k, value);
-
- 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);
-
- if (v) {
- partial.push(quote(k) + (gap ? ": " : ":") + v);
- }
- }
- }
- } // Join all of the member texts together, separated with commas,
- // and wrap them in braces.
-
-
- 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.
-
-
- rep = replacer;
-
- 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.
-
-
- return str("", {
- "": value
- });
- };
- } // If the JSON object does not yet have a parse method, give it one.
-
-
- 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;
-
- 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];
-
- if (value && _typeof(value) === "object") {
- for (k in value) {
- if (Object.prototype.hasOwnProperty.call(value, k)) {
- v = walk(value, k);
-
- if (v !== undefined) {
- value[k] = v;
- } else {
- delete value[k];
- }
- }
- }
- }
-
- 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.
-
-
- text = String(text);
- rx_dangerous.lastIndex = 0;
-
- 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.
-
-
- 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.
-
- return typeof reviver === "function" ? walk({
- "": j
- }, "") : j;
- } // If the text is not JSON parseable, then a SyntaxError is thrown.
-
-
- throw new SyntaxError("JSON.parse");
- };
- }
- })();
-
- var json2 = json2Plugin;
-
- function json2Plugin() {
- return {};
- }
-
- var plugins = [json2];
- var store_legacy = storeEngine.createStore(all, plugins);
-
- //
- // 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.
-
-
- 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
-
- oauth.authenticated = function () {
- return !!(token('oauth_token') && token('oauth_token_secret'));
- };
-
- oauth.logout = function () {
- token('oauth_token', '');
- token('oauth_token_secret', '');
- token('oauth_request_token_secret', '');
- return oauth;
- }; // TODO: detect lack of click event
-
-
- oauth.authenticate = function (callback) {
- if (oauth.authenticated()) return callback();
- oauth.logout(); // ## Getting a request token
-
- 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 (!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;
-
- 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.
-
-
- ohauth_1.xhr('POST', url, params, null, {}, reqTokenDone);
- o.loading();
-
- 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 (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.
-
-
- 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.
-
-
- 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 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);
- }
- };
-
- oauth.bringPopupWindowToFront = function () {
- var brougtPopupToFront = false;
-
- try {
- // This may cause a cross-origin error:
- // `DOMException: Blocked a frame with origin "..." from accessing a cross-origin frame.`
- if (oauth.popupWindow && !oauth.popupWindow.closed) {
- oauth.popupWindow.focus();
- brougtPopupToFront = true;
- }
- } catch (err) {// Bringing popup window to front failed (probably because of the cross-origin error mentioned above)
- }
-
- return brougtPopupToFront;
- };
-
- oauth.bootstrapToken = function (oauth_token, callback) {
- // ## Getting an request token
- // At this point we have an `oauth_token`, brought in from a function
- // 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 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);
- }
-
- get_access_token(oauth_token);
- }; // # xhr
- //
- // A single XMLHttpRequest wrapper that does authenticated calls if the
- // user has logged in.
-
-
- oauth.xhr = function (options, callback) {
- if (!oauth.authenticated()) {
- if (o.auto) {
- return oauth.authenticate(run);
- } else {
- callback('not authenticated', null);
- return;
- }
- } else {
- return run();
- }
-
- 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);
- }
-
- 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
-
-
- 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 () {};
-
- o.done = o.done || function () {};
-
- return oauth.preauth(o);
- }; // 'stamp' an authentication object from `getAuth()`
- // with a [nonce](http://en.wikipedia.org/wiki/Cryptographic_nonce)
- // and timestamp
-
-
- 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 token;
-
- 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
-
-
- function getAuth(o) {
- return {
- oauth_consumer_key: o.oauth_consumer_key,
- oauth_signature_method: 'HMAC-SHA1'
- };
- } // potentially pre-authorize
-
-
- oauth.options(o);
- return oauth;
- };
-
- 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;
- }
-
- if (rIsBool.test(sValue)) {
- return sValue.toLowerCase() === 'true';
- }
-
- if (isFinite(sValue)) {
- return parseFloat(sValue);
- }
-
- if (isFinite(Date.parse(sValue))) {
- return new Date(sValue);
- }
-
- return sValue;
- }
-
- function EmptyTree() {}
-
- EmptyTree.prototype.toString = function () {
- return 'null';
- };
-
- EmptyTree.prototype.valueOf = function () {
- return null;
- };
-
- function objectify(vValue) {
- return vValue === null ? new EmptyTree() : vValue instanceof Object ? vValue : new vValue.constructor(vValue);
- }
-
- 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;
-
- 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();
- }
- /* nodeType is 'Text' (3) */
- else if (oNode.nodeType === 1 && !oNode.prefix) {
- aCache.push(oNode);
- }
- /* nodeType is 'Element' (1) */
-
- }
- }
-
- var nLevelEnd = aCache.length,
- vBuiltVal = parseText(sCollectedTxt);
-
- 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 (vResult.hasOwnProperty(sProp)) {
- if (vResult[sProp].constructor !== Array) {
- vResult[sProp] = [vResult[sProp]];
- }
-
- vResult[sProp].push(vContent);
- } else {
- vResult[sProp] = vContent;
- nLength++;
- }
- }
-
- 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 (bNesteAttr) {
- if (bFreeze) {
- Object.freeze(oAttrParent);
- }
-
- vResult[sAttributesProp] = oAttrParent;
- nLength -= nAttrLen - 1;
- }
- }
-
- 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);
- }
-
- aCache.length = nLevelStart;
- return vResult;
- }
-
- function loadObjTree(oXMLDoc, oParentEl, oParentObj) {
- var vValue, oChild;
-
- 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 (isFinite(sName) || vValue instanceof Function) {
- continue;
- }
- /* verbosity level is 0 */
-
-
- 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 (vValue instanceof Object) {
- loadObjTree(oXMLDoc, oChild, vValue);
- } else if (vValue !== null && vValue !== true) {
- oChild.appendChild(oXMLDoc.createTextNode(vValue.toString()));
- }
-
- 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;
-
- 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;
- };
-
- 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
-
- 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;
-
- var _changeset = {};
-
- var _deferred = new Set();
-
- var _connectionID = 1;
- var _tileZoom$3 = 16;
- var _noteZoom = 12;
-
- var _rateLimitError;
-
- var _userChangesets;
-
- var _userDetails;
-
- var _off; // set a default but also load this from the API status
-
-
- var _maxWayNodes = 2000;
-
- function authLoading() {
- dispatch$6.call('authLoading');
- }
-
- function authDone() {
- dispatch$6.call('authDone');
- }
-
- function abortRequest$5(controllerOrXHR) {
- if (controllerOrXHR) {
- controllerOrXHR.abort();
- }
- }
-
- function hasInflightRequests(cache) {
- return Object.keys(cache.inflight).length;
- }
-
- 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];
- });
- }
-
- function getLoc(attrs) {
- var lon = attrs.lon && attrs.lon.value;
- var lat = attrs.lat && attrs.lat.value;
- return [parseFloat(lon), parseFloat(lat)];
- }
-
- function getNodes(obj) {
- var elems = obj.getElementsByTagName('nd');
- var nodes = new Array(elems.length);
-
- for (var i = 0, l = elems.length; i < l; i++) {
- nodes[i] = 'n' + elems[i].attributes.ref.value;
- }
-
- return nodes;
- }
-
- function getNodesJSON(obj) {
- var elems = obj.nodes;
- var nodes = new Array(elems.length);
-
- for (var i = 0, l = elems.length; i < l; i++) {
- nodes[i] = 'n' + elems[i];
- }
-
- return nodes;
- }
-
- function getTags(obj) {
- var elems = obj.getElementsByTagName('tag');
- var tags = {};
-
- for (var i = 0, l = elems.length; i < l; i++) {
- var attrs = elems[i].attributes;
- tags[attrs.k.value] = attrs.v.value;
- }
-
- return tags;
- }
-
- function getMembers(obj) {
- var elems = obj.getElementsByTagName('member');
- var members = new Array(elems.length);
-
- 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;
- }
-
- function getMembersJSON(obj) {
- var elems = obj.members;
- var members = new Array(elems.length);
-
- 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;
- }
-
- function getVisible(attrs) {
- return !attrs.visible || attrs.visible.value !== 'false';
- }
-
- function parseComments(comments) {
- var parsedComments = []; // for each comment
-
- for (var i = 0; i < comments.length; i++) {
- var comment = comments[i];
-
- if (comment.nodeName === 'comment') {
- var childNodes = comment.childNodes;
- var parsedComment = {};
-
- 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 (uid && !_userCache.user[uid]) {
- _userCache.toLoad[uid] = true;
- }
- }
- }
-
- if (parsedComment) {
- parsedComments.push(parsedComment);
- }
- }
- }
-
- return parsedComments;
- }
-
- function encodeNoteRtree(note) {
- return {
- minX: note.loc[0],
- minY: note.loc[1],
- maxX: note.loc[0],
- maxY: note.loc[1],
- data: note
- };
- }
-
- 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)
- });
- }
- };
-
- function parseJSON(payload, callback, options) {
- options = Object.assign({
- skipSeen: true
- }, options);
-
- if (!payload) {
- return callback({
- message: 'No JSON',
- status: -1
- });
- }
-
- 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);
- }
-
- callback(null, results);
- });
-
- _deferred.add(handle);
-
- 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
-
- _tileCache.seen[uid] = true;
- }
-
- return parser(child, uid);
- }
- }
-
- 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;
-
- do {
- if (coincident) {
- props.loc = geoVecAdd(props.loc, [epsilon, epsilon]);
- }
-
- var bbox = geoExtent(props.loc).bbox();
- coincident = _noteCache.rtree.search(bbox).length;
- } while (coincident); // parse note contents
-
-
- 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 (nodeName === 'comments') {
- props[nodeName] = parseComments(node.childNodes);
- } else {
- props[nodeName] = node.textContent;
- }
- }
-
- var note = new osmNote(props);
- var item = encodeNoteRtree(note);
- _noteCache.note[note.id] = note;
-
- _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'
- };
- var img = obj.getElementsByTagName('img');
-
- if (img && img[0] && img[0].getAttribute('href')) {
- user.image_url = img[0].getAttribute('href');
- }
-
- var changesets = obj.getElementsByTagName('changesets');
-
- if (changesets && changesets[0] && changesets[0].getAttribute('count')) {
- user.changesets_count = changesets[0].getAttribute('count');
- }
-
- var blocks = obj.getElementsByTagName('blocks');
-
- if (blocks && blocks[0]) {
- var received = blocks[0].getElementsByTagName('received');
-
- if (received && received[0] && received[0].getAttribute('active')) {
- user.active_blocks = received[0].getAttribute('active');
- }
- }
-
- _userCache.user[uid] = user;
- delete _userCache.toLoad[uid];
- return user;
- }
- };
-
- function parseXML(xml, callback, options) {
- options = Object.assign({
- skipSeen: true
- }, options);
-
- if (!xml || !xml.childNodes) {
- return callback({
- message: 'No XML',
- status: -1
- });
- }
-
- 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);
- }
-
- callback(null, results);
- });
-
- _deferred.add(handle);
-
- function parseChild(child) {
- var parser = parsers[child.nodeName];
- if (!parser) return null;
- var uid;
-
- 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);
-
- if (options.skipSeen) {
- if (_tileCache.seen[uid]) return null; // avoid reparsing a "seen" entity
-
- _tileCache.seen[uid] = true;
- }
- }
-
- return parser(child, uid);
- }
- } // replace or remove note from rtree
-
-
- function updateRtree$3(item, replace) {
- _noteCache.rtree.remove(item, function isEql(a, b) {
- return a.data.id === b.data.id;
- });
-
- if (replace) {
- _noteCache.rtree.insert(item);
- }
- }
-
- 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();
- }
-
- 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);
- }
- };
- }
-
- var serviceOsm = {
- init: function init() {
- utilRebind(this, dispatch$6, '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$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;
-
- 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..
-
- 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);
- }
- }
- }
- }
- }
-
- 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
-
- var match = err.message.match(/^\d{3}/);
-
- 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
-
- 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;
-
- 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));
- }
-
- function createdChangeset(err, changesetID) {
- _changeset.inflight = null;
-
- if (err) {
- return callback(err, changeset);
- }
-
- _changeset.open = changesetID;
- changeset = changeset.update({
- id: changesetID
- }); // Upload the changeset..
-
- 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));
- }
-
- 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
-
- 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.
-
- 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;
- });
- }
- }
- },
- // 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
- }
-
- utilArrayChunk(toLoad, 150).forEach(function (arr) {
- oauth.xhr({
- method: 'GET',
- path: '/api/0.6/users?users=' + arr.join()
- }, wrapcb(this, done, _connectionID));
- }.bind(this));
-
- function done(err, xml) {
- if (err) {
- return callback(err);
- }
-
- 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]);
- }
-
- oauth.xhr({
- method: 'GET',
- path: '/api/0.6/user/' + uid
- }, wrapcb(this, done, _connectionID));
-
- function done(err, xml) {
- if (err) {
- return callback(err);
- }
-
- 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);
- }
-
- oauth.xhr({
- method: 'GET',
- path: '/api/0.6/user/details'
- }, wrapcb(this, done, _connectionID));
-
- function done(err, xml) {
- if (err) {
- return callback(err);
- }
-
- 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);
- }
-
- this.userDetails(wrapcb(this, gotDetails, _connectionID));
-
- function gotDetails(err, user) {
- if (err) {
- return callback(err);
- }
-
- oauth.xhr({
- method: 'GET',
- path: '/api/0.6/changesets?user=' + user.id
- }, wrapcb(this, done, _connectionID));
- }
-
- function done(err, xml) {
- if (err) {
- return callback(err);
- }
-
- _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);
- }
- },
- // 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);
- });
-
- function done(err, xml) {
- if (err) {
- // the status is null if no response could be retrieved
- return callback(err, null);
- } // update blocklists
-
-
- var elements = xml.getElementsByTagName('blacklist');
- var regexes = [];
-
- for (var i = 0; i < elements.length; i++) {
- var regexString = elements[i].getAttribute('regex'); // needs unencode?
-
- if (regexString) {
- try {
- var regex = new RegExp(regexString);
- regexes.push(regex);
- } catch (e) {
- /* noop */
- }
- }
- }
-
- if (regexes.length) {
- _imageryBlocklists = regexes;
- }
-
- 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);
- }
-
- 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
-
- var tiles = tiler$5.zoomExtent([_tileZoom$3, _tileZoom$3]).getTiles(projection); // abort inflight requests that are no longer needed
-
- var hadRequests = hasInflightRequests(_tileCache);
- abortUnwantedRequests$3(_tileCache, tiles);
-
- if (hadRequests && !hasInflightRequests(_tileCache)) {
- dispatch$6.call('loaded'); // stop the spinner
- } // issue new requests..
-
-
- 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;
-
- if (!hasInflightRequests(_tileCache)) {
- dispatch$6.call('loading'); // start the spinner
- }
-
- var path = '/api/0.6/map.json?bbox=';
- var options = {
- skipSeen: true
- };
- _tileCache.inflight[tile.id] = this.loadFromAPI(path + tile.extent.toParam(), tileCallback, options);
-
- function tileCallback(err, parsed) {
- delete _tileCache.inflight[tile.id];
-
- 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);
- }
-
- if (callback) {
- callback(err, Object.assign({
- data: parsed
- }, tile));
- }
-
- 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=';
-
- 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
-
-
- var tiles = tiler$5.zoomExtent([_noteZoom, _noteZoom]).getTiles(projection); // abort inflight requests that are no longer needed
-
- abortUnwantedRequests$3(_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];
-
- if (!err) {
- _noteCache.loaded[tile.id] = true;
- }
-
- 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);
- }
-
- if (_noteCache.inflightPost[note.id]) {
- return callback({
- message: 'Note update already inflight',
- status: -2
- }, note);
- }
-
- if (!note.loc[0] || !note.loc[1] || !note.newComment) return; // location & description required
-
- var comment = note.newComment;
-
- if (note.newCategory && note.newCategory !== 'None') {
- comment += ' #' + note.newCategory;
- }
-
- 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));
-
- 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);
- 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);
- }
-
- if (_noteCache.inflightPost[note.id]) {
- return callback({
- message: 'Note update already inflight',
- status: -2
- }, note);
- }
-
- var action;
-
- 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
- }
-
- var path = '/api/0.6/notes/' + note.id + '/' + action;
-
- 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];
- }
-
- 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
-
- 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 (!arguments.length) {
- return {
- tile: cloneCache(_tileCache),
- note: cloneCache(_noteCache),
- user: cloneCache(_userCache)
- };
- } // access caches directly for testing (e.g., loading notes rtree)
-
-
- if (obj === 'get') {
- return {
- tile: _tileCache,
- note: _noteCache,
- user: _userCache
- };
- }
-
- if (obj.tile) {
- _tileCache = obj.tile;
- _tileCache.inflight = {};
- }
-
- if (obj.note) {
- _noteCache = obj.note;
- _noteCache.inflight = {};
- _noteCache.inflightPost = {};
- }
-
- if (obj.user) {
- _userCache = obj.user;
- }
-
- 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;
-
- function done(err, res) {
- if (err) {
- if (callback) callback(err);
- return;
- }
-
- if (that.getConnectionId() !== cid) {
- if (callback) callback({
- message: 'Connection Switched',
- status: -1
- });
- return;
- }
-
- _rateLimitError = undefined;
- dispatch$6.call('change');
- if (callback) callback(err, res);
- that.userChangesets(function () {}); // eagerly load user details/changesets
- }
-
- 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
-
- 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 = 'https://wiki.openstreetmap.org/w/api.php';
- var _inflight$1 = {};
- var _wikibaseCache = {};
- var _localeIDs = {
- en: false
- };
-
- 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);
- });
- }
-
- var serviceOsmWikibase = {
- init: function init() {
- _inflight$1 = {};
- _wikibaseCache = {};
- _localeIDs = {};
- },
- reset: function reset() {
- Object.values(_inflight$1).forEach(function (controller) {
- controller.abort();
- });
- _inflight$1 = {};
- },
-
- /**
- * 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;
- }
-
- if (locale && stmt.qualifiers && stmt.qualifiers.P26 && stmt.qualifiers.P26[0].datavalue.value.id === locale) {
- localePick = stmt;
- }
- });
- var result = localePick || preferredPick;
-
- if (result) {
- var datavalue = result.mainsnak.datavalue;
- return datavalue.type === 'wikibase-entityid' ? datavalue.value.id : datavalue.value;
- } else {
- return undefined;
- }
- },
-
- /**
- * 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;
-
- 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);
- }
- });
- }
-
- if (rtypeSitelink) {
- if (_wikibaseCache[rtypeSitelink]) {
- result.rtype = _wikibaseCache[rtypeSitelink];
- } else {
- titles.push(rtypeSitelink);
- }
- }
-
- if (keySitelink) {
- if (_wikibaseCache[keySitelink]) {
- result.key = _wikibaseCache[keySitelink];
- } else {
- titles.push(keySitelink);
- }
- }
-
- 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 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,
-
- };
- 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;
-
- 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
- }
- }
- });
-
- if (localeSitelink) {
- // If locale ID is not found, store false to prevent repeated queries
- that.addLocale(params.langCodes[0], localeID);
- }
-
- 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;
- }
-
- var entity = data.rtype || data.tag || data.key;
-
- if (!entity) {
- callback('No entity');
- return;
- }
-
- var i;
- var description;
-
- for (i in langCodes) {
- var _code = langCodes[i];
-
- if (entity.descriptions[_code] && entity.descriptions[_code].language === _code) {
- description = entity.descriptions[_code];
- break;
- }
- }
-
- 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
-
- if (entity.claims) {
- var imageroot;
- var image = that.claimToValue(entity, 'P4', langCodes[0]);
-
- 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';
- }
- }
-
- 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.
-
-
- 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];
-
- for (i in wikis) {
- var wiki = wikis[i];
-
- 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);
-
- if (info) {
- result.wiki = info;
- break;
- }
- }
-
- if (result.wiki) break;
- }
-
- callback(null, result); // Helper method to get wiki info if a given language exists
-
- 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;
- }
- };
-
- var jsonpCache = {};
- window.jsonpCache = jsonpCache;
- function jsonpRequest(url, callback) {
- var request = {
- abort: function abort() {}
- };
-
- 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);
-
- request.abort = function () {
- window.clearTimeout(t);
- };
- }
-
- return request;
- }
-
- function rand() {
- var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz';
- var c = '';
- var i = -1;
-
- while (++i < 15) {
- c += chars.charAt(Math.floor(Math.random() * 52));
- }
-
- return c;
- }
-
- function create(url) {
- var e = url.match(/callback=(\w+)/);
- var c = e ? e[1] : rand();
-
- jsonpCache[c] = function (data) {
- if (jsonpCache[c]) {
- callback(data);
- }
-
- finalize();
- };
-
- function finalize() {
- delete jsonpCache[c];
- script.remove();
- }
-
- request.abort = finalize;
- return 'jsonpCache.' + c;
- }
-
- 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 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
-
- var maxHfov = 90; // zoom out degrees
-
- var defaultHfov = 45;
- var _hires = false;
- var _resolution = 512; // higher numbers are slower - 512, 1024, 2048, 4096
-
- var _currScene = 0;
-
- var _ssCache;
-
- var _pannellumViewer;
-
- var _sceneOptions = {
- showFullscreenCtrl: false,
- autoLoad: true,
- compass: true,
- yaw: 0,
- minHfov: minHfov,
- maxHfov: maxHfov,
- hfov: defaultHfov,
- type: 'cubemap',
- cubeMap: []
- };
-
- var _loadViewerPromise$2;
- /**
- * abortRequest().
- */
-
-
- function abortRequest$6(i) {
- i.abort();
- }
- /**
- * localeTimeStamp().
- */
-
-
- 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.
- */
-
-
- function loadTiles$2(which, url, projection, margin) {
- var tiles = tiler$6.margin(margin).getTiles(projection); // abort inflight requests that are no longer needed
-
- 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.
- */
-
-
- 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 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
-
- if (bubble.pr === undefined) {
- cache.leaders.push(bubble.id);
- }
-
- return {
- minX: loc[0],
- minY: loc[1],
- maxX: loc[0],
- maxY: loc[1],
- data: d
- };
- }).filter(Boolean);
- cache.rtree.load(features);
- connectSequences();
-
- if (which === 'bubbles') {
- dispatch$7.call('loadedImages');
- }
- });
- } // call this sometimes to connect the bubbles into sequences
-
-
- function connectSequences() {
- var cache = _ssCache.bubbles;
- var keepLeaders = [];
-
- 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 sequence = {
- key: bubble.key,
- bubbles: []
- };
- var complete = false;
-
- 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);
-
- if (complete) {
- _ssCache.sequences[sequence.key] = sequence; // assign bubbles to the sequence
-
- for (var j = 0; j < sequence.bubbles.length; j++) {
- sequence.bubbles[j].sequenceKey = sequence.key;
- } // create a GeoJSON LineString
-
-
- 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
-
-
- cache.leaders = keepLeaders;
- }
- /**
- * getBubbles() handles the request to the server for a tile extent of 'bubbles' (streetside image locations).
- */
-
-
- 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);
- } else {
- callback(data);
- }
- });
- } // 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
-
- var tiler = utilTiler().zoomExtent([z2, z2]);
- return tiler.getTiles(projection).map(function (tile) {
- return tile.extent;
- });
- } // 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;
- }, []);
- }
- /**
- * loadImage()
- */
-
-
- function loadImage(imgInfo) {
- return new Promise(function (resolve) {
- var img = new Image();
-
- 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'
- });
- };
-
- img.setAttribute('crossorigin', '');
- img.src = imgInfo.url;
- });
- }
- /**
- * loadCanvas()
- */
-
-
- 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()
- */
-
-
- function loadFaces(faceGroup) {
- return Promise.all(faceGroup.map(loadCanvas)).then(function () {
- return {
- status: 'loadFaces done'
- };
- });
- }
-
- 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;
- }
-
- return [x, y];
- }
-
- 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'];
- }
-
- return quadKeys;
- }
-
- var serviceStreetside = {
- /**
- * init() initialize streetside.
- */
- init: function init() {
- if (!_ssCache) {
- this.reset();
- }
-
- this.event = utilRebind(this, dispatch$7, 'on');
- },
-
- /**
- * reset() reset the cache.
- */
- reset: function reset() {
- if (_ssCache) {
- Object.values(_ssCache.bubbles.inflight).forEach(abortRequest$6);
- }
-
- _ssCache = {
- bubbles: {
- inflight: {},
- loaded: {},
- nextPage: {},
- rtree: new RBush(),
- points: {},
- leaders: []
- },
- sequences: {}
- };
- },
-
- /**
- * 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
-
- _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);
- }
- });
-
- return results;
- },
-
- /**
- * 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;
- },
- initViewer: function initViewer() {
- if (!window.pannellum) return;
- if (_pannellumViewer) return;
- _currScene += 1;
-
- var sceneID = _currScene.toString();
-
- var options = {
- 'default': {
- firstScene: sceneID
- },
- scenes: {}
- };
- options.scenes[sceneID] = _sceneOptions;
- _pannellumViewer = window.pannellum.viewer('ideditor-viewer-streetside', options);
- },
- 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)
-
- 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
-
- 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.
-
- var t = timer(function (elapsed) {
- dispatch$7.call('viewerChanged');
-
- 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
-
- context.ui().photoviewer.on('resize.streetside', function () {
- if (_pannellumViewer) {
- _pannellumViewer.resize();
- }
- });
- _loadViewerPromise$2 = new Promise(function (resolve, reject) {
- var loadedCount = 0;
-
- function loaded() {
- loadedCount += 1; // wait until both files are loaded
-
- if (loadedCount === 2) resolve();
- }
-
- var head = select('head'); // load streetside pannellum viewer css
-
- 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
-
- 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;
-
- 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;
-
- var yaw = _pannellumViewer.getYaw();
-
- var ca = selected.ca + yaw;
- var origin = selected.loc; // construct a search trapezoid pointing out from current bubble
-
- 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 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
-
- var minDist = Infinity;
-
- _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 (minTheta > 20) {
- dist += 5; // penalize distance if camera angles don't match
- }
-
- if (dist < minDist) {
- nextID = d.data.key;
- minDist = dist;
- }
- });
-
- var nextBubble = nextID && that.cachedImage(nextID);
- if (!nextBubble) return;
- context.map().centerEase(nextBubble.loc);
- that.selectImage(context, nextBubble.key).yaw(yaw).showViewer(context);
- };
- }
- },
- 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 (isHidden) {
- wrap.selectAll('.photo-wrapper:not(.ms-wrapper)').classed('hide', true);
- wrap.selectAll('.photo-wrapper.ms-wrapper').classed('hide', false);
- }
-
- return this;
- },
-
- /**
- * 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);
- },
-
- /**
- * 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
-
- 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
-
- 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('|');
- }
-
- if (d.captured_at) {
- captureInfo.append('span').attr('class', 'captured_at').html(localeTimestamp(d.captured_at));
- } // Add image links
-
-
- 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;
-
- for (var i = 0; i < paddingNeeded; i++) {
- bubbleIdQuadKey = '0' + bubbleIdQuadKey;
- }
-
- 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;
-
- var sceneID = _currScene.toString();
-
- _pannellumViewer.addScene(sceneID, _sceneOptions).loadScene(sceneID); // remove previous scene
-
-
- 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);
- }
-
- 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
-
- 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
-
- context.container().selectAll('.viewfield-group .viewfield').attr('d', viewfieldPath);
-
- function viewfieldPath() {
- var d = this.parentNode.__data__;
-
- 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 this;
- },
- updateUrlImage: function updateUrlImage(imageKey) {
- if (!window.mocha) {
- var hash = utilStringQs(window.location.hash);
-
- if (imageKey) {
- hash.photo = 'streetside/' + imageKey;
- } else {
- delete hash.photo;
- }
-
- window.location.replace('#' + utilQsString(hash, true));
- }
- },
-
- /**
- * cache().
- */
- cache: function cache() {
- return _ssCache;
- }
- };
-
- 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'
- };
-
- function sets(params, n, o) {
- if (params.geometry && o[params.geometry]) {
- params[n] = o[params.geometry];
- }
-
- return params;
- }
-
- function setFilter(params) {
- return sets(params, 'filter', tag_filters);
- }
-
- function setSort(params) {
- return sets(params, 'sortname', tag_sorts);
- }
-
- function setSortMembers(params) {
- return sets(params, 'sortname', tag_sort_members);
- }
-
- function clean(params) {
- return utilObjectOmit(params, ['geometry', 'debounce']);
- }
-
- function filterKeys(type) {
- var count_type = type ? 'count_' + type : 'count_all';
- return function (d) {
- return parseFloat(d[count_type]) > 2500 || d.in_wiki;
- };
- }
-
- 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 filterValues(allowUpperCase) {
- return function (d) {
- if (d.value.match(/[;,]/) !== null) return false; // exclude some punctuation
-
- if (!allowUpperCase && d.value.match(/[A-Z*]/) !== null) return false; // exclude uppercase letters
-
- return parseFloat(d.fraction) > 0.0;
- };
- }
-
- 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
-
- return parseFloat(d[tag_members_fractions[geometry]]) > 0.0;
- };
- }
-
- function valKey(d) {
- return {
- value: d.key,
- title: d.key
- };
- }
-
- function valKeyDescription(d) {
- var obj = {
- value: d.value,
- title: d.description || d.value
- };
-
- if (d.count) {
- obj.count = d.count;
- }
-
- return obj;
- }
-
- function roleKey(d) {
- return {
- value: d.role,
- title: d.role
- };
- } // sort keys with ':' lower than keys without ':'
-
-
- function sortKeys(a, b) {
- return a.key.indexOf(':') === -1 && b.key.indexOf(':') !== -1 ? -1 : a.key.indexOf(':') !== -1 && b.key.indexOf(':') === -1 ? 1 : 0;
- }
-
- var debouncedRequest$1 = debounce(request$1, 300, {
- leading: false
- });
-
- 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);
- });
- }
-
- function checkCache(url, params, exactMatch, callback) {
- var rp = params.rp || 25;
- var testQuery = params.query || '';
- var testUrl = url;
-
- do {
- var hit = _taginfoCache[testUrl]; // exact match, or shorter match yielding fewer than max results (rp)
-
- if (hit && (url === testUrl || hit.length < rp)) {
- callback(null, hit);
- return true;
- } // don't try to shorten the query
-
-
- 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)
-
- testQuery = testQuery.slice(0, -1);
- testUrl = url.replace(/&query=(.*?)&/, '&query=' + testQuery + '&');
- } while (testQuery.length >= 0);
-
- return false;
- }
-
- 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
-
- 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
-
- _popularKeys[d.value] = true;
- });
- });
- },
- reset: function reset() {
- Object.values(_inflight$2).forEach(function (controller) {
- controller.abort();
- });
- _inflight$2 = {};
- },
- 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);
- }
- });
- },
- 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);
- }
- });
- },
- values: function values(params, callback) {
- // Exclude popular keys from values lookups.. see #3955
- var key = params.key;
-
- if (key && _popularKeys[key]) {
- callback(null, []);
- return;
- }
-
- 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);
- }
- });
- },
- 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);
- }
- });
- },
- 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);
- }
- });
- },
- apibase: function apibase(_) {
- if (!arguments.length) return _apibase$1;
- _apibase$1 = _;
- return this;
- }
- };
-
- var helpers$1 = createCommonjsModule(function (module, exports) {
-
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- /**
- * @module helpers
- */
-
- /**
- * Earth Radius used with the Harvesine formula and approximates using a spherical (non-ellipsoid) Earth.
- *
- * @memberof helpers
- * @type {number}
- */
-
- exports.earthRadius = 6371008.8;
- /**
- * Unit of measurement factors using a spherical (non-ellipsoid) earth radius.
- *
- * @memberof helpers
- * @type {Object}
- */
-
- exports.factors = {
- centimeters: exports.earthRadius * 100,
- centimetres: exports.earthRadius * 100,
- degrees: exports.earthRadius / 111325,
- feet: exports.earthRadius * 3.28084,
- inches: exports.earthRadius * 39.370,
- kilometers: exports.earthRadius / 1000,
- kilometres: exports.earthRadius / 1000,
- meters: exports.earthRadius,
- metres: exports.earthRadius,
- miles: exports.earthRadius / 1609.344,
- millimeters: exports.earthRadius * 1000,
- millimetres: exports.earthRadius * 1000,
- nauticalmiles: exports.earthRadius / 1852,
- radians: 1,
- yards: exports.earthRadius / 1.0936
- };
- /**
- * Units of measurement factors based on 1 meter.
- *
- * @memberof helpers
- * @type {Object}
- */
-
- exports.unitsFactors = {
- centimeters: 100,
- centimetres: 100,
- degrees: 1 / 111325,
- feet: 3.28084,
- inches: 39.370,
- kilometers: 1 / 1000,
- kilometres: 1 / 1000,
- meters: 1,
- metres: 1,
- miles: 1 / 1609.344,
- millimeters: 1000,
- millimetres: 1000,
- nauticalmiles: 1 / 1852,
- radians: 1 / exports.earthRadius,
- yards: 1 / 1.0936
- };
- /**
- * Area of measurement factors based on 1 square meter.
- *
- * @memberof helpers
- * @type {Object}
- */
-
- exports.areaFactors = {
- acres: 0.000247105,
- centimeters: 10000,
- centimetres: 10000,
- feet: 10.763910417,
- inches: 1550.003100006,
- kilometers: 0.000001,
- kilometres: 0.000001,
- meters: 1,
- metres: 1,
- miles: 3.86e-7,
- millimeters: 1000000,
- millimetres: 1000000,
- yards: 1.195990046
- };
- /**
- * 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
- */
-
- function feature(geom, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- var feat = {
- type: "Feature"
- };
-
- if (options.id === 0 || options.id) {
- feat.id = options.id;
- }
-
- if (options.bbox) {
- feat.bbox = options.bbox;
- }
-
- feat.properties = properties || {};
- feat.geometry = geom;
- return feat;
- }
-
- exports.feature = feature;
- /**
- * Creates a GeoJSON {@link Geometry} from a Geometry string type & coordinates.
- * For GeometryCollection type use `helpers.geometryCollection`
- *
- * @name geometry
- * @param {string} type Geometry Type
- * @param {Array<any>} coordinates Coordinates
- * @param {Object} [options={}] Optional Parameters
- * @returns {Geometry} a GeoJSON Geometry
- * @example
- * var type = "Point";
- * var coordinates = [110, 50];
- * var geometry = turf.geometry(type, coordinates);
- * // => geometry
- */
-
- function geometry(type, coordinates, options) {
-
- switch (type) {
- case "Point":
- return point(coordinates).geometry;
-
- case "LineString":
- return lineString(coordinates).geometry;
-
- case "Polygon":
- return polygon(coordinates).geometry;
-
- case "MultiPoint":
- return multiPoint(coordinates).geometry;
-
- case "MultiLineString":
- return multiLineString(coordinates).geometry;
-
- case "MultiPolygon":
- return multiPolygon(coordinates).geometry;
-
- default:
- throw new Error(type + " is invalid");
- }
- }
-
- exports.geometry = geometry;
- /**
- * Creates a {@link Point} {@link Feature} from a Position.
- *
- * @name point
- * @param {Array<number>} coordinates longitude, latitude position (each in decimal degrees)
- * @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<Point>} a Point feature
- * @example
- * var point = turf.point([-75.343, 39.984]);
- *
- * //=point
- */
-
- function point(coordinates, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- var geom = {
- type: "Point",
- coordinates: coordinates
- };
- return feature(geom, properties, options);
- }
-
- exports.point = point;
- /**
- * Creates a {@link Point} {@link FeatureCollection} from an Array of Point coordinates.
- *
- * @name points
- * @param {Array<Array<number>>} coordinates an array of Points
- * @param {Object} [properties={}] Translate these properties to each Feature
- * @param {Object} [options={}] Optional Parameters
- * @param {Array<number>} [options.bbox] Bounding Box Array [west, south, east, north]
- * associated with the FeatureCollection
- * @param {string|number} [options.id] Identifier associated with the FeatureCollection
- * @returns {FeatureCollection<Point>} Point Feature
- * @example
- * var points = turf.points([
- * [-75, 39],
- * [-80, 45],
- * [-78, 50]
- * ]);
- *
- * //=points
- */
-
- function points(coordinates, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- return featureCollection(coordinates.map(function (coords) {
- return point(coords, properties);
- }), options);
- }
-
- exports.points = points;
- /**
- * 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 = {};
- }
-
- for (var _i = 0, coordinates_1 = coordinates; _i < coordinates_1.length; _i++) {
- var ring = coordinates_1[_i];
-
- if (ring.length < 4) {
- throw new Error("Each LinearRing of a Polygon must have 4 or more Positions.");
- }
-
- 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.");
- }
- }
- }
-
- var geom = {
- type: "Polygon",
- coordinates: coordinates
- };
- return feature(geom, properties, options);
- }
-
- exports.polygon = polygon;
- /**
- * Creates a {@link Polygon} {@link FeatureCollection} from an Array of Polygon coordinates.
- *
- * @name polygons
- * @param {Array<Array<Array<Array<number>>>>} coordinates an array of Polygon coordinates
- * @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 FeatureCollection
- * @returns {FeatureCollection<Polygon>} Polygon FeatureCollection
- * @example
- * var polygons = turf.polygons([
- * [[[-5, 52], [-4, 56], [-2, 51], [-7, 54], [-5, 52]]],
- * [[[-15, 42], [-14, 46], [-12, 41], [-17, 44], [-15, 42]]],
- * ]);
- *
- * //=polygons
- */
-
- function polygons(coordinates, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- return featureCollection(coordinates.map(function (coords) {
- return polygon(coords, properties);
- }), options);
- }
-
- exports.polygons = polygons;
- /**
- * 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 lineString(coordinates, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- if (coordinates.length < 2) {
- throw new Error("coordinates must be an array of two or more positions");
- }
-
- var geom = {
- type: "LineString",
- coordinates: coordinates
- };
- return feature(geom, properties, options);
- }
-
- exports.lineString = lineString;
- /**
- * Creates a {@link LineString} {@link FeatureCollection} from an Array of LineString coordinates.
- *
- * @name lineStrings
- * @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 FeatureCollection
- * @param {string|number} [options.id] Identifier associated with the FeatureCollection
- * @returns {FeatureCollection<LineString>} LineString FeatureCollection
- * @example
- * var linestrings = turf.lineStrings([
- * [[-24, 63], [-23, 60], [-25, 65], [-20, 69]],
- * [[-14, 43], [-13, 40], [-15, 45], [-10, 49]]
- * ]);
- *
- * //=linestrings
- */
-
- function lineStrings(coordinates, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- return featureCollection(coordinates.map(function (coords) {
- return lineString(coords, properties);
- }), options);
- }
-
- exports.lineStrings = lineStrings;
- /**
- * Takes one or more {@link Feature|Features} and creates a {@link FeatureCollection}.
- *
- * @name featureCollection
- * @param {Feature[]} features input features
- * @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 {FeatureCollection} FeatureCollection of Features
- * @example
- * var locationA = turf.point([-75.343, 39.984], {name: 'Location A'});
- * var locationB = turf.point([-75.833, 39.284], {name: 'Location B'});
- * var locationC = turf.point([-75.534, 39.123], {name: 'Location C'});
- *
- * var collection = turf.featureCollection([
- * locationA,
- * locationB,
- * locationC
- * ]);
- *
- * //=collection
- */
-
- function featureCollection(features, options) {
- if (options === void 0) {
- options = {};
- }
-
- var fc = {
- type: "FeatureCollection"
- };
-
- if (options.id) {
- fc.id = options.id;
- }
-
- if (options.bbox) {
- fc.bbox = options.bbox;
- }
-
- fc.features = features;
- return fc;
- }
-
- exports.featureCollection = featureCollection;
- /**
- * 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
- */
-
- function multiLineString(coordinates, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- var geom = {
- type: "MultiLineString",
- coordinates: coordinates
- };
- return feature(geom, properties, options);
- }
-
- exports.multiLineString = multiLineString;
- /**
- * Creates a {@link Feature<MultiPoint>} based on a
- * coordinate array. Properties can be added optionally.
- *
- * @name multiPoint
- * @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<MultiPoint>} a MultiPoint feature
- * @throws {Error} if no coordinates are passed
- * @example
- * var multiPt = turf.multiPoint([[0,0],[10,10]]);
- *
- * //=multiPt
- */
-
- function multiPoint(coordinates, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- var geom = {
- type: "MultiPoint",
- coordinates: coordinates
- };
- return feature(geom, properties, options);
- }
-
- exports.multiPoint = multiPoint;
- /**
- * 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
- *
- */
-
- function multiPolygon(coordinates, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- var geom = {
- type: "MultiPolygon",
- coordinates: coordinates
- };
- return feature(geom, properties, options);
- }
-
- exports.multiPolygon = multiPolygon;
- /**
- * Creates a {@link Feature<GeometryCollection>} based on a
- * coordinate array. Properties can be added optionally.
- *
- * @name geometryCollection
- * @param {Array<Geometry>} geometries an array of GeoJSON Geometries
- * @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<GeometryCollection>} a GeoJSON GeometryCollection Feature
- * @example
- * var pt = turf.geometry("Point", [100, 0]);
- * var line = turf.geometry("LineString", [[101, 0], [102, 1]]);
- * var collection = turf.geometryCollection([pt, line]);
- *
- * // => collection
- */
-
- function geometryCollection(geometries, properties, options) {
- if (options === void 0) {
- options = {};
- }
-
- var geom = {
- type: "GeometryCollection",
- geometries: geometries
- };
- return feature(geom, properties, options);
- }
-
- exports.geometryCollection = geometryCollection;
- /**
- * Round number to precision
- *
- * @param {number} num Number
- * @param {number} [precision=0] Precision
- * @returns {number} rounded number
- * @example
- * turf.round(120.4321)
- * //=120
- *
- * turf.round(120.4321, 2)
- * //=120.43
- */
-
- function round(num, precision) {
- if (precision === void 0) {
- precision = 0;
- }
-
- if (precision && !(precision >= 0)) {
- throw new Error("precision must be a positive number");
- }
-
- var multiplier = Math.pow(10, precision || 0);
- return Math.round(num * multiplier) / multiplier;
- }
-
- exports.round = round;
- /**
- * Convert a distance measurement (assuming a spherical Earth) from radians to a more friendly unit.
- * Valid units: miles, nauticalmiles, inches, yards, meters, metres, kilometers, centimeters, feet
- *
- * @name radiansToLength
- * @param {number} radians in radians across the sphere
- * @param {string} [units="kilometers"] can be degrees, radians, miles, or kilometers inches, yards, metres,
- * meters, kilometres, kilometers.
- * @returns {number} distance
- */
-
- function radiansToLength(radians, units) {
- if (units === void 0) {
- units = "kilometers";
- }
-
- var factor = exports.factors[units];
-
- if (!factor) {
- throw new Error(units + " units is invalid");
- }
-
- return radians * factor;
- }
-
- exports.radiansToLength = radiansToLength;
- /**
- * Convert a distance measurement (assuming a spherical Earth) from a real-world unit into radians
- * Valid units: miles, nauticalmiles, inches, yards, meters, metres, kilometers, centimeters, feet
- *
- * @name lengthToRadians
- * @param {number} distance in real units
- * @param {string} [units="kilometers"] can be degrees, radians, miles, or kilometers inches, yards, metres,
- * meters, kilometres, kilometers.
- * @returns {number} radians
- */
-
- function lengthToRadians(distance, units) {
- if (units === void 0) {
- units = "kilometers";
- }
-
- var factor = exports.factors[units];
-
- if (!factor) {
- throw new Error(units + " units is invalid");
- }
-
- return distance / factor;
- }
-
- exports.lengthToRadians = lengthToRadians;
- /**
- * Convert a distance measurement (assuming a spherical Earth) from a real-world unit into degrees
- * Valid units: miles, nauticalmiles, inches, yards, meters, metres, centimeters, kilometres, feet
- *
- * @name lengthToDegrees
- * @param {number} distance in real units
- * @param {string} [units="kilometers"] can be degrees, radians, miles, or kilometers inches, yards, metres,
- * meters, kilometres, kilometers.
- * @returns {number} degrees
- */
-
- function lengthToDegrees(distance, units) {
- return radiansToDegrees(lengthToRadians(distance, units));
- }
-
- exports.lengthToDegrees = lengthToDegrees;
- /**
- * Converts any bearing angle from the north line direction (positive clockwise)
- * and returns an angle between 0-360 degrees (positive clockwise), 0 being the north line
- *
- * @name bearingToAzimuth
- * @param {number} bearing angle, between -180 and +180 degrees
- * @returns {number} angle between 0 and 360 degrees
- */
-
- function bearingToAzimuth(bearing) {
- var angle = bearing % 360;
-
- if (angle < 0) {
- angle += 360;
- }
-
- return angle;
- }
-
- exports.bearingToAzimuth = bearingToAzimuth;
- /**
- * Converts an angle in radians to degrees
- *
- * @name radiansToDegrees
- * @param {number} radians angle in radians
- * @returns {number} degrees between 0 and 360 degrees
- */
-
- function radiansToDegrees(radians) {
- var degrees = radians % (2 * Math.PI);
- return degrees * 180 / Math.PI;
- }
-
- exports.radiansToDegrees = radiansToDegrees;
- /**
- * Converts an angle in degrees to radians
- *
- * @name degreesToRadians
- * @param {number} degrees angle between 0 and 360 degrees
- * @returns {number} angle in radians
- */
-
- function degreesToRadians(degrees) {
- var radians = degrees % 360;
- return radians * Math.PI / 180;
- }
-
- exports.degreesToRadians = degreesToRadians;
- /**
- * Converts a length to the requested unit.
- * Valid units: miles, nauticalmiles, inches, yards, meters, metres, kilometers, centimeters, feet
- *
- * @param {number} length to be converted
- * @param {Units} [originalUnit="kilometers"] of the length
- * @param {Units} [finalUnit="kilometers"] returned unit
- * @returns {number} the converted length
- */
-
- function convertLength(length, originalUnit, finalUnit) {
- if (originalUnit === void 0) {
- originalUnit = "kilometers";
- }
-
- if (finalUnit === void 0) {
- finalUnit = "kilometers";
- }
-
- if (!(length >= 0)) {
- throw new Error("length must be a positive number");
- }
-
- return radiansToLength(lengthToRadians(length, originalUnit), finalUnit);
- }
-
- exports.convertLength = convertLength;
- /**
- * Converts a area to the requested unit.
- * Valid units: kilometers, kilometres, meters, metres, centimetres, millimeters, acres, miles, yards, feet, inches
- * @param {number} area to be converted
- * @param {Units} [originalUnit="meters"] of the distance
- * @param {Units} [finalUnit="kilometers"] returned unit
- * @returns {number} the converted distance
- */
-
- function convertArea(area, originalUnit, finalUnit) {
- if (originalUnit === void 0) {
- originalUnit = "meters";
- }
-
- if (finalUnit === void 0) {
- finalUnit = "kilometers";
- }
-
- if (!(area >= 0)) {
- throw new Error("area must be a positive number");
- }
-
- var startFactor = exports.areaFactors[originalUnit];
-
- if (!startFactor) {
- throw new Error("invalid original units");
- }
-
- var finalFactor = exports.areaFactors[finalUnit];
-
- if (!finalFactor) {
- throw new Error("invalid final units");
- }
-
- return area / startFactor * finalFactor;
- }
-
- exports.convertArea = convertArea;
- /**
- * isNumber
- *
- * @param {*} num Number to validate
- * @returns {boolean} true/false
- * @example
- * turf.isNumber(123)
- * //=true
- * turf.isNumber('foo')
- * //=false
- */
-
- function isNumber(num) {
- return !isNaN(num) && num !== null && !Array.isArray(num) && !/^\s*$/.test(num);
- }
-
- exports.isNumber = isNumber;
- /**
- * isObject
- *
- * @param {*} input variable to validate
- * @returns {boolean} true/false
- * @example
- * turf.isObject({elevation: 10})
- * //=true
- * turf.isObject('foo')
- * //=false
- */
-
- function isObject(input) {
- return !!input && input.constructor === Object;
- }
-
- exports.isObject = isObject;
- /**
- * Validate BBox
- *
- * @private
- * @param {Array<number>} bbox BBox to validate
- * @returns {void}
- * @throws Error if BBox is not valid
- * @example
- * validateBBox([-180, -40, 110, 50])
- * //=OK
- * validateBBox([-180, -40])
- * //=Error
- * validateBBox('Foo')
- * //=Error
- * validateBBox(5)
- * //=Error
- * validateBBox(null)
- * //=Error
- * validateBBox(undefined)
- * //=Error
- */
-
- function validateBBox(bbox) {
- if (!bbox) {
- throw new Error("bbox is required");
- }
-
- if (!Array.isArray(bbox)) {
- throw new Error("bbox must be an Array");
- }
-
- if (bbox.length !== 4 && bbox.length !== 6) {
- throw new Error("bbox must be an Array of 4 or 6 numbers");
- }
-
- bbox.forEach(function (num) {
- if (!isNumber(num)) {
- throw new Error("bbox must only contain numbers");
- }
- });
- }
-
- exports.validateBBox = validateBBox;
- /**
- * Validate Id
- *
- * @private
- * @param {string|number} id Id to validate
- * @returns {void}
- * @throws Error if Id is not valid
- * @example
- * validateId([-180, -40, 110, 50])
- * //=Error
- * validateId([-180, -40])
- * //=Error
- * validateId('Foo')
- * //=OK
- * validateId(5)
- * //=OK
- * validateId(null)
- * //=Error
- * validateId(undefined)
- * //=Error
- */
-
- function validateId(id) {
- if (!id) {
- throw new Error("id is required");
- }
-
- if (["string", "number"].indexOf(_typeof(id)) === -1) {
- throw new Error("id must be a number or a string");
- }
- }
-
- exports.validateId = validateId; // Deprecated methods
-
- function radians2degrees() {
- throw new Error("method has been renamed to `radiansToDegrees`");
- }
-
- exports.radians2degrees = radians2degrees;
-
- function degrees2radians() {
- throw new Error("method has been renamed to `degreesToRadians`");
- }
-
- exports.degrees2radians = degrees2radians;
-
- function distanceToDegrees() {
- throw new Error("method has been renamed to `lengthToDegrees`");
- }
-
- exports.distanceToDegrees = distanceToDegrees;
-
- function distanceToRadians() {
- throw new Error("method has been renamed to `lengthToRadians`");
- }
-
- exports.distanceToRadians = distanceToRadians;
-
- function radiansToDistance() {
- throw new Error("method has been renamed to `radiansToLength`");
- }
-
- exports.radiansToDistance = radiansToDistance;
-
- function bearingToAngle() {
- throw new Error("method has been renamed to `bearingToAzimuth`");
- }
-
- exports.bearingToAngle = bearingToAngle;
-
- function convertDistance() {
- throw new Error("method has been renamed to `convertLength`");
- }
-
- exports.convertDistance = convertDistance;
- });
-
- var invariant = createCommonjsModule(function (module, exports) {
-
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- /**
- * Unwrap a coordinate from a Point Feature, Geometry or a single coordinate.
- *
- * @name getCoord
- * @param {Array<number>|Geometry<Point>|Feature<Point>} coord GeoJSON Point or an Array of numbers
- * @returns {Array<number>} coordinates
- * @example
- * var pt = turf.point([10, 10]);
- *
- * var coord = turf.getCoord(pt);
- * //= [10, 10]
- */
-
- function getCoord(coord) {
- if (!coord) {
- throw new Error("coord is required");
- }
-
- if (!Array.isArray(coord)) {
- if (coord.type === "Feature" && coord.geometry !== null && coord.geometry.type === "Point") {
- return coord.geometry.coordinates;
- }
-
- if (coord.type === "Point") {
- return coord.coordinates;
- }
- }
-
- if (Array.isArray(coord) && coord.length >= 2 && !Array.isArray(coord[0]) && !Array.isArray(coord[1])) {
- return coord;
- }
-
- throw new Error("coord must be GeoJSON Point or an Array of numbers");
- }
-
- exports.getCoord = getCoord;
- /**
- * Unwrap coordinates from a Feature, Geometry Object or an Array
- *
- * @name getCoords
- * @param {Array<any>|Geometry|Feature} coords Feature, Geometry Object or an Array
- * @returns {Array<any>} coordinates
- * @example
- * var poly = turf.polygon([[[119.32, -8.7], [119.55, -8.69], [119.51, -8.54], [119.32, -8.7]]]);
- *
- * var coords = turf.getCoords(poly);
- * //= [[[119.32, -8.7], [119.55, -8.69], [119.51, -8.54], [119.32, -8.7]]]
- */
-
- function getCoords(coords) {
- if (Array.isArray(coords)) {
- return coords;
- } // Feature
-
-
- if (coords.type === "Feature") {
- if (coords.geometry !== null) {
- return coords.geometry.coordinates;
- }
- } else {
- // Geometry
- if (coords.coordinates) {
- return coords.coordinates;
- }
- }
-
- throw new Error("coords must be GeoJSON Feature, Geometry Object or an Array");
- }
-
- exports.getCoords = getCoords;
- /**
- * Checks if coordinates contains a number
- *
- * @name containsNumber
- * @param {Array<any>} coordinates GeoJSON Coordinates
- * @returns {boolean} true if Array contains a number
- */
-
- function containsNumber(coordinates) {
- if (coordinates.length > 1 && helpers$1.isNumber(coordinates[0]) && helpers$1.isNumber(coordinates[1])) {
- return true;
- }
-
- if (Array.isArray(coordinates[0]) && coordinates[0].length) {
- return containsNumber(coordinates[0]);
- }
-
- throw new Error("coordinates must only contain numbers");
- }
-
- exports.containsNumber = containsNumber;
- /**
- * Enforce expectations about types of GeoJSON objects for Turf.
- *
- * @name geojsonType
- * @param {GeoJSON} value any GeoJSON object
- * @param {string} type expected GeoJSON type
- * @param {string} name name of calling function
- * @throws {Error} if value is not the expected type.
- */
-
- function geojsonType(value, type, name) {
- if (!type || !name) {
- throw new Error("type and name required");
- }
-
- if (!value || value.type !== type) {
- throw new Error("Invalid input to " + name + ": must be a " + type + ", given " + value.type);
- }
- }
-
- exports.geojsonType = geojsonType;
- /**
- * Enforce expectations about types of {@link Feature} inputs for Turf.
- * Internally this uses {@link geojsonType} to judge geometry types.
- *
- * @name featureOf
- * @param {Feature} feature a feature with an expected geometry type
- * @param {string} type expected GeoJSON type
- * @param {string} name name of calling function
- * @throws {Error} error if value is not the expected type.
- */
-
- function featureOf(feature, type, name) {
- if (!feature) {
- throw new Error("No feature passed");
- }
-
- if (!name) {
- throw new Error(".featureOf() requires a name");
- }
-
- if (!feature || feature.type !== "Feature" || !feature.geometry) {
- throw new Error("Invalid input to " + name + ", Feature with geometry required");
- }
-
- if (!feature.geometry || feature.geometry.type !== type) {
- throw new Error("Invalid input to " + name + ": must be a " + type + ", given " + feature.geometry.type);
- }
- }
-
- exports.featureOf = featureOf;
- /**
- * Enforce expectations about types of {@link FeatureCollection} inputs for Turf.
- * Internally this uses {@link geojsonType} to judge geometry types.
- *
- * @name collectionOf
- * @param {FeatureCollection} featureCollection a FeatureCollection for which features will be judged
- * @param {string} type expected GeoJSON type
- * @param {string} name name of calling function
- * @throws {Error} if value is not the expected type.
- */
-
- function collectionOf(featureCollection, type, name) {
- if (!featureCollection) {
- throw new Error("No featureCollection passed");
- }
-
- if (!name) {
- throw new Error(".collectionOf() requires a name");
- }
-
- if (!featureCollection || featureCollection.type !== "FeatureCollection") {
- throw new Error("Invalid input to " + name + ", FeatureCollection required");
- }
-
- for (var _i = 0, _a = featureCollection.features; _i < _a.length; _i++) {
- var feature = _a[_i];
-
- if (!feature || feature.type !== "Feature" || !feature.geometry) {
- throw new Error("Invalid input to " + name + ", Feature with geometry required");
- }
-
- if (!feature.geometry || feature.geometry.type !== type) {
- throw new Error("Invalid input to " + name + ": must be a " + type + ", given " + feature.geometry.type);
- }
- }
- }
-
- exports.collectionOf = collectionOf;
- /**
- * 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]}
- */
-
- function getGeom(geojson) {
- if (geojson.type === "Feature") {
- return geojson.geometry;
- }
-
- return geojson;
- }
-
- exports.getGeom = getGeom;
- /**
- * Get GeoJSON object's type, Geometry type is prioritize.
- *
- * @param {GeoJSON} geojson GeoJSON object
- * @param {string} [name="geojson"] name of the variable to display in error message
- * @returns {string} GeoJSON type
- * @example
- * var point = {
- * "type": "Feature",
- * "properties": {},
- * "geometry": {
- * "type": "Point",
- * "coordinates": [110, 40]
- * }
- * }
- * var geom = turf.getType(point)
- * //="Point"
- */
-
- function getType(geojson, name) {
- if (geojson.type === "FeatureCollection") {
- return "FeatureCollection";
- }
-
- if (geojson.type === "GeometryCollection") {
- return "GeometryCollection";
- }
-
- if (geojson.type === "Feature" && geojson.geometry !== null) {
- return geojson.geometry.type;
- }
-
- return geojson.type;
- }
-
- exports.getType = getType;
- });
-
- var lineclip_1 = lineclip;
- var _default = lineclip;
- lineclip.polyline = lineclip;
- lineclip.polygon = polygonclip; // 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 = 1; i < len; i++) {
- a = points[i - 1];
- b = points[i];
- codeB = lastCode = bitCode(b, bbox);
-
- 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);
- }
- }
-
- codeA = lastCode;
- }
-
- if (part.length) result.push(part);
- return result;
- } // Sutherland-Hodgeman polygon clipping algorithm
-
-
- function polygonclip(points, bbox) {
- var result, edge, prev, prevInside, i, p, inside; // clip against each side of the clip rectangle
-
- for (edge = 1; edge <= 8; edge *= 2) {
- result = [];
- prev = points[points.length - 1];
- prevInside = !(bitCode(prev, bbox) & edge);
-
- 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
-
- 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;
- }
-
- points = result;
- if (!points.length) break;
- }
-
- return result;
- } // intersect a segment against one of the 4 lines that make up the bbox
-
-
- 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 bitCode(p, bbox) {
- var code = 0;
- if (p[0] < bbox[0]) code |= 1; // left
- else if (p[0] > bbox[2]) code |= 2; // right
-
- if (p[1] < bbox[1]) code |= 4; // bottom
- else if (p[1] > bbox[3]) code |= 8; // top
-
- return code;
- }
- lineclip_1["default"] = _default;
-
- var bboxClip_1 = createCommonjsModule(function (module, exports) {
-
- var __importStar = commonjsGlobal && commonjsGlobal.__importStar || function (mod) {
- if (mod && mod.__esModule) return mod;
- var result = {};
- if (mod != null) for (var k in mod) {
- if (Object.hasOwnProperty.call(mod, k)) result[k] = mod[k];
- }
- result["default"] = mod;
- return result;
- };
-
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
-
- var lineclip = __importStar(lineclip_1);
- /**
- * 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 bboxClip(feature, bbox) {
- var geom = invariant.getGeom(feature);
- var type = geom.type;
- var properties = feature.type === "Feature" ? feature.properties : {};
- var coords = geom.coordinates;
-
- switch (type) {
- case "LineString":
- case "MultiLineString":
- var lines_1 = [];
-
- if (type === "LineString") {
- coords = [coords];
- }
-
- coords.forEach(function (line) {
- lineclip.polyline(line, bbox, lines_1);
- });
-
- if (lines_1.length === 1) {
- return helpers$1.lineString(lines_1[0], properties);
- }
-
- return helpers$1.multiLineString(lines_1, properties);
-
- case "Polygon":
- return helpers$1.polygon(clipPolygon(coords, bbox), properties);
-
- case "MultiPolygon":
- return helpers$1.multiPolygon(coords.map(function (poly) {
- return clipPolygon(poly, bbox);
- }), properties);
-
- default:
- throw new Error("geometry " + type + " not supported");
- }
- }
-
- exports["default"] = bboxClip;
-
- 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 = lineclip.polygon(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]);
- }
-
- if (clipped.length >= 4) {
- outRings.push(clipped);
- }
- }
- }
-
- return outRings;
- }
- });
- var turf_bboxClip = /*@__PURE__*/getDefaultExportFromCjs(bboxClip_1);
-
- 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);
-
- var seen = [];
- return function stringify(node) {
- if (node && node.toJSON && typeof node.toJSON === 'function') {
- node = node.toJSON();
- }
-
- if (node === undefined) return;
- if (typeof node == 'number') return isFinite(node) ? '' + node : 'null';
- if (_typeof(node) !== 'object') return JSON.stringify(node);
- var i, out;
-
- if (Array.isArray(node)) {
- out = '[';
-
- for (i = 0; i < node.length; i++) {
- if (i) out += ',';
- out += stringify(node[i]) || 'null';
- }
-
- return out + ']';
- }
-
- if (node === null) return 'null';
-
- if (seen.indexOf(node) !== -1) {
- if (cycles) return JSON.stringify('__cycle__');
- throw new TypeError('Converting circular structure to JSON');
- }
-
- var seenIndex = seen.push(node) - 1;
- var keys = Object.keys(node).sort(cmp && cmp(node));
- out = '';
-
- 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;
- }
-
- seen.splice(seenIndex, 1);
- return '{' + out + '}';
- }(data);
- };
-
- function DEFAULT_COMPARE(a, b) {
- return a > b ? 1 : a < b ? -1 : 0;
- }
-
- 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;
-
- _classCallCheck(this, SplayTree);
-
- this._compare = compare;
- this._root = null;
- this._size = 0;
- this._noDuplicates = !!noDuplicates;
- }
-
- _createClass(SplayTree, [{
- key: "rotateLeft",
- value: function rotateLeft(x) {
- var y = x.right;
-
- if (y) {
- x.right = y.left;
- if (y.left) y.left.parent = x;
- y.parent = x.parent;
- }
-
- 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;
- }
- }, {
- 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 (!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;
- }
- }, {
- 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);
- }
- }
- }
- }, {
- 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;
- }
-
- 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;
-
- 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;
- }
- }
-
- if (r) {
- p.left = r;
- r.parent = p;
- } else p.left = null;
-
- 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;
-
- 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;
- }
- }, {
- 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;
- }
- }, {
- 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;
- }
- }
-
- 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;
-
- while (z) {
- var cmp = comp(z.key, key);
- if (cmp < 0) z = z.right;else if (cmp > 0) z = z.left;else return z;
- }
-
- return null;
- }
- /**
- * Whether the tree contains a node with the given key
- * @param {Key} key
- * @return {boolean} true/false
- */
-
- }, {
- key: "contains",
- value: function contains(key) {
- var node = this._root;
- var comparator = this._compare;
-
- while (node) {
- var cmp = comparator(key, node.key);
- if (cmp === 0) return true;else if (cmp < 0) node = node.left;else node = node.right;
- }
-
- 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);
-
- if (y.parent !== z) {
- this.replace(y, y.right);
- y.right = z.right;
- y.right.parent = y;
- }
-
- this.replace(z, y);
- y.left = z.left;
- y.left.parent = y;
- }
- this._size--;
- return true;
- }
- }, {
- 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;
- }
-
- this.replace(z, y);
- y.left = z.left;
- y.left.parent = y;
- }
- this._size--;
- return true;
- }
- }, {
- 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;
-
- if (s) {
- s.parent = null;
- sMax = this.maxNode(s);
- this.splay(sMax);
- this._root = sMax;
- }
-
- if (t) {
- if (s) sMax.right = t;else this._root = t;
- t.parent = sMax;
- }
-
- this._size--;
- }
- /**
- * Removes and returns the node with smallest key
- * @return {?Node}
- */
-
- }, {
- key: "pop",
- value: function pop() {
- var node = this._root,
- returnValue = null;
-
- if (node) {
- while (node.left) {
- node = node.left;
- }
-
- returnValue = {
- key: node.key,
- data: node.data
- };
- this.remove(node.key);
- }
-
- return returnValue;
- }
- /* eslint-disable class-methods-use-this */
-
- /**
- * Successor node
- * @param {Node} node
- * @return {?Node}
- */
-
- }, {
- key: "next",
- value: function next(node) {
- var successor = node;
-
- if (successor) {
- if (successor.right) {
- successor = successor.right;
-
- while (successor && successor.left) {
- successor = successor.left;
- }
- } else {
- successor = node.parent;
-
- while (successor && successor.right === node) {
- node = successor;
- successor = successor.parent;
- }
- }
- }
-
- return successor;
- }
- /**
- * 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;
-
- while (predecessor && predecessor.left === node) {
- node = predecessor;
- predecessor = predecessor.parent;
- }
- }
- }
-
- return predecessor;
- }
- /* eslint-enable class-methods-use-this */
-
- /**
- * @param {forEachCallback} callback
- * @return {SplayTree}
- */
-
- }, {
- key: "forEach",
- value: function forEach(callback) {
- var current = this._root;
- var s = [],
- done = false,
- i = 0;
-
- 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
-
- current = current.right;
- } else done = true;
- }
- }
-
- 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);
-
- if (cmp > 0) {
- break;
- } else if (compare(node.key, low) >= 0) {
- if (fn.call(ctx, node)) return this; // stop if smth is returned
- }
-
- node = node.right;
- }
- }
-
- return this;
- }
- /**
- * Returns all keys in order
- * @return {Array<Key>}
- */
-
- }, {
- 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;
- }
- /**
- * 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;
- }
- /**
- * Returns node at given index
- * @param {number} index
- * @return {?Node}
- */
-
- }, {
- 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;
-
- 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;
- }
- }
-
- 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}
- */
-
- }], [{
- key: "createTree",
- value: function createTree(keys, values, comparator, presort, noDuplicates) {
- return new SplayTree(comparator, noDuplicates).load(keys, values, presort);
- }
- }]);
-
- return SplayTree;
- }();
-
- 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;
- }
-
- return null;
- }
-
- 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;
- }
-
- sort(keys, values, left, j, compare);
- sort(keys, values, j + 1, right, compare);
- }
-
- var NORMAL = 0;
- var NON_CONTRIBUTING = 1;
- var SAME_TRANSITION = 2;
- var DIFFERENT_TRANSITION = 3;
-
- var INTERSECTION = 0;
- var UNION = 1;
- var DIFFERENCE = 2;
- var XOR = 3;
-
- /**
- * @param {SweepEvent} event
- * @param {SweepEvent} prev
- * @param {Operation} operation
- */
-
- 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
-
-
- if (prev) {
- event.prevInResult = !inResult(prev, operation) || prev.isVertical() ? prev.prevInResult : prev;
- }
- } // check if the line segment belongs to the Boolean operation
-
-
- var isInResult = inResult(event, operation);
-
- if (isInResult) {
- event.resultTransition = determineResultTransition(event, operation);
- } else {
- event.resultTransition = 0;
- }
- }
- /* eslint-disable indent */
-
- function inResult(event, operation) {
- switch (event.type) {
- case NORMAL:
- switch (operation) {
- case INTERSECTION:
- return !event.otherInOut;
-
- case UNION:
- return event.otherInOut;
-
- case DIFFERENCE:
- // return (event.isSubject && !event.otherInOut) ||
- // (!event.isSubject && event.otherInOut);
- return event.isSubject && event.otherInOut || !event.isSubject && !event.otherInOut;
-
- case XOR:
- return true;
- }
-
- break;
-
- case SAME_TRANSITION:
- return operation === INTERSECTION || operation === UNION;
-
- case DIFFERENT_TRANSITION:
- return operation === DIFFERENCE;
-
- case NON_CONTRIBUTING:
- return false;
- }
-
- return false;
- }
- /* eslint-enable indent */
-
-
- function determineResultTransition(event, operation) {
- var thisIn = !event.inOut;
- var thatIn = !event.otherInOut;
- var isIn;
-
- switch (operation) {
- case INTERSECTION:
- isIn = thisIn && thatIn;
- break;
-
- case UNION:
- isIn = thisIn || thatIn;
- break;
-
- case XOR:
- isIn = thisIn ^ thatIn;
- break;
-
- case DIFFERENCE:
- if (event.isSubject) {
- isIn = thisIn && !thatIn;
- } else {
- isIn = thatIn && !thisIn;
- }
-
- break;
- }
-
- return isIn ? +1 : -1;
- }
-
- 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.inOut = false;
- /**
- * @type {Boolean}
- */
-
- this.otherInOut = false;
- /**
- * Previous event in result?
- * @type {SweepEvent}
- */
-
- this.prevInResult = null;
- /**
- * Type of result transition (0 = not in result, +1 = out-in, -1, in-out)
- * @type {Number}
- */
-
- this.resultTransition = 0; // connection step
-
- /**
- * @type {Number}
- */
-
- this.otherPos = -1;
- /**
- * @type {Number}
- */
-
- this.outputContourId = -1;
- this.isExteriorRing = true; // TODO: Looks unused, remove?
- }
- /**
- * @param {Array.<Number>} p
- * @return {Boolean}
- */
-
-
- _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}
- */
-
- }, {
- key: "isAbove",
- value: function isAbove(p) {
- return !this.isBelow(p);
- }
- /**
- * @return {Boolean}
- */
-
- }, {
- key: "isVertical",
- value: function isVertical() {
- return this.point[0] === this.otherEvent.point[0];
- }
- /**
- * Does event belong to result?
- * @return {Boolean}
- */
-
- }, {
- 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;
- }
- }, {
- key: "inResult",
- get: function get() {
- return this.resultTransition !== 0;
- }
- }]);
-
- return SweepEvent;
- }();
-
- function equals(p1, p2) {
- if (p1[0] === p2[0]) {
- if (p1[1] === p2[1]) {
- return true;
- } else {
- return false;
- }
- }
-
- 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;
- // };
-
- 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
-
- 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;
-
- if (fnow > enow === fnow > -enow) {
- Q = enow;
- enow = e[++eindex];
- } else {
- Q = fnow;
- fnow = f[++findex];
- }
-
- var hindex = 0;
-
- 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];
- }
-
- Q = Qnew;
-
- if (hh !== 0) {
- h[hindex++] = hh;
- }
-
- 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];
- }
-
- Q = Qnew;
-
- if (hh !== 0) {
- h[hindex++] = hh;
- }
- }
- }
-
- while (eindex < elen) {
- Qnew = Q + enow;
- bvirt = Qnew - Q;
- hh = Q - (Qnew - bvirt) + (enow - bvirt);
- enow = e[++eindex];
- Q = Qnew;
-
- if (hh !== 0) {
- h[hindex++] = hh;
- }
- }
-
- 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);
- }
-
- /**
- * Signed area of the triangle (p0, p1, p2)
- * @param {Array.<Number>} p0
- * @param {Array.<Number>} p1
- * @param {Array.<Number>} p2
- * @return {Number}
- */
-
- 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;
- }
-
- /**
- * @param {SweepEvent} e1
- * @param {SweepEvent} e2
- * @return {Number}
- */
-
- function compareEvents(e1, e2) {
- var p1 = e1.point;
- var p2 = e2.point; // Different x-coordinate
-
- 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 (p1[1] !== p2[1]) return p1[1] > p2[1] ? 1 : -1;
- return specialCases(e1, e2, p1);
- }
- /* eslint-disable no-unused-vars */
-
- 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 (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;
- }
-
- return !e1.isSubject && e2.isSubject ? 1 : -1;
- }
- /* eslint-enable no-unused-vars */
-
- /**
- * @param {SweepEvent} se
- * @param {Array.<Number>} p
- * @param {Queue} queue
- * @return {Queue}
- */
-
- 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 (equals(se.point, se.otherEvent.point)) {
- console.warn('what is that, a collapsed segment?', se);
- }
- /* 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) {}
-
-
- se.otherEvent.otherEvent = l;
- se.otherEvent = r;
- queue.push(l);
- queue.push(r);
- return queue;
- }
-
- //const EPS = 1e-9;
-
- /**
- * 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
- */
-
-
- 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.
- */
-
-
- 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:
-
- /* eslint-disable arrow-body-style */
-
- 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.
-
-
- 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.
-
- 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;
- }
-
- var t = crossProduct(e, va) / kross;
-
- if (t < 0 || t > 1) {
- // not on line segment b
- return null;
- }
-
- if (s === 0 || s === 1) {
- // on an endpoint of line segment a
- return noEndpointTouch ? null : [toPoint(a1, s, va)];
- }
-
- if (t === 0 || t === 1) {
- // on an endpoint of line segment b
- return noEndpointTouch ? null : [toPoint(b1, t, vb)];
- }
-
- 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);
-
-
- kross = crossProduct(e, va);
- sqrKross = kross * kross;
-
- if (sqrKross > 0
- /* EPS * sqLenB * sqLenE */
- ) {
- // Lines are just parallel, not the same. No overlap.
- return null;
- }
-
- 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.
-
- if (smin <= 1 && smax >= 0) {
- // overlap on an end point
- if (smin === 1) {
- return noEndpointTouch ? null : [toPoint(a1, smin > 0 ? smin : 0, va)];
- }
-
- if (smax === 0) {
- return noEndpointTouch ? null : [toPoint(a1, smax < 1 ? smax : 1, va)];
- }
-
- if (noEndpointTouch && smin === 0 && smax === 1) return null; // There's overlap on a segment -- two points of intersection. Return both.
-
- return [toPoint(a1, smin > 0 ? smin : 0, va), toPoint(a1, smax < 1 ? smax : 1, va)];
- }
-
- return null;
- }
-
- /**
- * @param {SweepEvent} se1
- * @param {SweepEvent} se2
- * @param {Queue} queue
- * @return {Number}
- */
-
- 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 (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
-
-
- 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
-
-
- if (!equals(se2.point, inter[0]) && !equals(se2.otherEvent.point, inter[0])) {
- divideSegment(se2, inter[0], queue);
- }
-
- return 1;
- } // The line segments associated to se1 and se2 overlap
-
-
- var events = [];
- var leftCoincide = false;
- var rightCoincide = false;
-
- 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);
- }
-
- 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;
-
- 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);
- }
-
- return 2;
- } // the line segments share the right endpoint
-
-
- if (rightCoincide) {
- divideSegment(events[0], events[1].point, queue);
- return 3;
- } // no line segment includes totally the other one
-
-
- 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
-
-
- divideSegment(events[0], events[1].point, queue);
- divideSegment(events[3].otherEvent, events[2].point, queue);
- return 3;
- }
-
- /**
- * @param {SweepEvent} le1
- * @param {SweepEvent} le2
- * @return {Number}
- */
-
- function compareSegments(le1, le2) {
- if (le1 === le2) return 0; // Segments are not collinear
-
- 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
-
- 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 ?
-
- 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
-
- return le1.isBelow(le2.point) ? -1 : 1;
- }
-
- if (le1.isSubject === le2.isSubject) {
- // same polygon
- var p1 = le1.point,
- p2 = le2.point;
-
- 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;
- }
- } else {
- // Segments are collinear, but belong to separate polygons
- return le1.isSubject ? -1 : 1;
- }
-
- return compareEvents(le1, le2) === 1 ? 1 : -1;
- }
-
- 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;
-
- while (eventQueue.length !== 0) {
- var event = eventQueue.pop();
- sortedEvents.push(event); // optimization by bboxes for intersection and difference goes here
-
- 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 (next) {
- if (possibleIntersection(event, next.key, eventQueue) === 2) {
- computeFields(event, prevEvent, operation);
- computeFields(event, next.key, operation);
- }
- }
-
- 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);
- }
- }
- } 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);
-
- if (next && prev) {
- possibleIntersection(prev.key, next.key, eventQueue);
- }
- }
- }
- }
-
- return sortedEvents;
- }
-
- var Contour = /*#__PURE__*/function () {
- /**
- * Contour
- *
- * @class {Contour}
- */
- function Contour() {
- _classCallCheck(this, Contour);
-
- this.points = [];
- this.holeIds = [];
- this.holeOf = null;
- this.depth = null;
- }
-
- _createClass(Contour, [{
- key: "isExterior",
- value: function isExterior() {
- return this.holeOf == null;
- }
- }]);
-
- return Contour;
- }();
-
- /**
- * @param {Array.<SweepEvent>} sortedEvents
- * @return {Array.<SweepEvent>}
- */
-
- function orderEvents(sortedEvents) {
- var event, i, len, tmp;
- var resultEvents = [];
-
- for (i = 0, len = sortedEvents.length; i < len; i++) {
- event = sortedEvents[i];
-
- 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
-
-
- var sorted = false;
-
- while (!sorted) {
- sorted = true;
-
- 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
-
-
- for (i = 0, len = resultEvents.length; i < len; i++) {
- event = resultEvents[i];
-
- if (!event.left) {
- tmp = event.otherPos;
- event.otherPos = event.otherEvent.otherPos;
- event.otherEvent.otherPos = tmp;
- }
- }
-
- return resultEvents;
- }
- /**
- * @param {Number} pos
- * @param {Array.<SweepEvent>} resultEvents
- * @param {Object>} processed
- * @return {Number}
- */
-
-
- 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;
- }
-
- 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".
-
- var lowerContourId = prevInResult.outputContourId;
- var lowerResultTransition = prevInResult.resultTransition;
-
- 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];
-
- 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;
- }
-
- return contour;
- }
- /**
- * @param {Array.<SweepEvent>} sortedEvents
- * @return {Array.<*>} polygons
- */
-
-
- 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);
- };
-
- for (i = 0, len = resultEvents.length; i < len; i++) {
- var _ret = _loop();
-
- if (_ret === "continue") continue;
- }
-
- return contours;
- }
-
- var tinyqueue = TinyQueue;
- var _default$1 = 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;
-
- if (this.length > 0) {
- for (var i = (this.length >> 1) - 1; i >= 0; i--) {
- this._down(i);
- }
- }
- }
-
- function defaultCompare$1(a, b) {
- return a < b ? -1 : a > b ? 1 : 0;
- }
-
- 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];
-
- if (right < this.length && compare(data[right], best) < 0) {
- left = right;
- best = data[right];
- }
-
- if (compare(best, item) >= 0) break;
- data[pos] = best;
- pos = left;
- }
-
- data[pos] = item;
- }
- };
- tinyqueue["default"] = _default$1;
-
- var max$5 = Math.max;
- var min$a = Math.min;
- var contourId = 0;
-
- function processPolygon(contourOrHole, isSubject, depth, Q, bbox, isExteriorRing) {
- var i, len, s1, s2, e1, e2;
-
- 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;
-
- if (s1[0] === s2[0] && s1[1] === s2[1]) {
- continue; // skip collapsed edges, or it breaks
- }
-
- e1.contourId = e2.contourId = depth;
-
- if (!isExteriorRing) {
- e1.isExteriorRing = false;
- e2.isExteriorRing = false;
- }
-
- if (compareEvents(e1, e2) > 0) {
- e2.left = true;
- } else {
- e1.left = true;
- }
-
- var x = s1[0],
- y = s1[1];
- bbox[0] = min$a(bbox[0], x);
- bbox[1] = min$a(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);
- }
- }
-
- function fillQueue(subject, clipping, sbbox, cbbox, operation) {
- var eventQueue = new tinyqueue(null, compareEvents);
- var polygonSet, isExteriorRing, i, ii, j, jj; //, k, kk;
-
- for (i = 0, ii = subject.length; i < ii; i++) {
- polygonSet = subject[i];
-
- for (j = 0, jj = polygonSet.length; j < jj; j++) {
- isExteriorRing = j === 0;
- if (isExteriorRing) contourId++;
- processPolygon(polygonSet[j], true, contourId, eventQueue, sbbox, isExteriorRing);
- }
- }
-
- for (i = 0, ii = clipping.length; i < ii; i++) {
- polygonSet = clipping[i];
-
- 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);
- }
- }
-
- return eventQueue;
- }
-
- var EMPTY = [];
-
- function trivialOperation(subject, clipping, operation) {
- var result = null;
-
- 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;
- }
- }
-
- return result;
- }
-
- function compareBBoxes(subject, clipping, sbbox, cbbox, operation) {
- var result = null;
-
- 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);
- }
- }
-
- return result;
- }
-
- function _boolean(subject, clipping, operation) {
- if (typeof subject[0][0][0] === 'number') {
- subject = [subject];
- }
-
- if (typeof clipping[0][0][0] === 'number') {
- clipping = [clipping];
- }
-
- var trivial = trivialOperation(subject, clipping, operation);
-
- if (trivial) {
- return trivial === EMPTY ? null : trivial;
- }
-
- var sbbox = [Infinity, Infinity, -Infinity, -Infinity];
- var cbbox = [Infinity, Infinity, -Infinity, -Infinity]; // console.time('fill queue');
-
- var eventQueue = fillQueue(subject, clipping, sbbox, cbbox, operation); //console.timeEnd('fill queue');
-
- trivial = compareBBoxes(subject, clipping, sbbox, cbbox, operation);
-
- if (trivial) {
- return trivial === EMPTY ? null : trivial;
- } // console.time('subdivide edges');
-
-
- var sortedEvents = subdivide(eventQueue, subject, clipping, sbbox, cbbox, operation); //console.timeEnd('subdivide edges');
- // console.time('connect vertices');
-
- var contours = connectEdges(sortedEvents); //console.timeEnd('connect vertices');
- // Convert contours to polygons
-
- var polygons = [];
-
- for (var i = 0; i < contours.length; i++) {
- var contour = contours[i];
-
- if (contour.isExterior()) {
- // The exterior ring goes first
- var rings = [contour.points]; // Followed by holes if any
-
- for (var j = 0; j < contour.holeIds.length; j++) {
- var holeId = contour.holeIds[j];
- rings.push(contours[holeId].points);
- }
-
- polygons.push(rings);
- }
- }
-
- return polygons;
- }
-
- function union(subject, clipping) {
- return _boolean(subject, clipping, UNION);
- }
-
- /*! 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;
- }
-
- 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);
-
- if (value * (c = Math.pow(2, -e)) < 1) {
- e--;
- c *= 2;
- }
-
- if (e + eBias >= 1) {
- value += rt / c;
- } else {
- value += rt * Math.pow(2, 1 - eBias);
- }
-
- if (value * c >= 2) {
- e++;
- c /= 2;
- }
-
- 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) {}
-
- e = e << mLen | m;
- eLen += mLen;
-
- for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8) {}
-
- buffer[offset + i - d] |= s * 128;
- };
-
- var ieee754$1 = {
- read: read$6,
- write: write$6
- };
-
- var pbf = Pbf;
-
- 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;
- }
-
- 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
-
- 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;
-
- 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);
- }
-
- 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 (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 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 || [];
-
- while (this.pos < end) {
- arr.push(this.readVarint(isSigned));
- }
-
- return arr;
- },
- readPackedSVarint: function readPackedSVarint(arr) {
- if (this.type !== Pbf.Bytes) return arr.push(this.readSVarint());
- var end = readPackedEnd(this);
- arr = arr || [];
-
- while (this.pos < end) {
- arr.push(this.readSVarint());
- }
-
- return arr;
- },
- readPackedBoolean: function readPackedBoolean(arr) {
- if (this.type !== Pbf.Bytes) return arr.push(this.readBoolean());
- var end = readPackedEnd(this);
- arr = arr || [];
-
- while (this.pos < end) {
- arr.push(this.readBoolean());
- }
-
- return arr;
- },
- readPackedFloat: function readPackedFloat(arr) {
- if (this.type !== Pbf.Bytes) return arr.push(this.readFloat());
- var end = readPackedEnd(this);
- arr = arr || [];
-
- while (this.pos < end) {
- arr.push(this.readFloat());
- }
-
- return arr;
- },
- readPackedDouble: function readPackedDouble(arr) {
- if (this.type !== Pbf.Bytes) return arr.push(this.readDouble());
- var end = readPackedEnd(this);
- arr = arr || [];
-
- while (this.pos < end) {
- arr.push(this.readDouble());
- }
-
- return arr;
- },
- readPackedFixed32: function readPackedFixed32(arr) {
- if (this.type !== Pbf.Bytes) return arr.push(this.readFixed32());
- var end = readPackedEnd(this);
- arr = arr || [];
-
- while (this.pos < end) {
- arr.push(this.readFixed32());
- }
-
- return arr;
- },
- readPackedSFixed32: function readPackedSFixed32(arr) {
- if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed32());
- var end = readPackedEnd(this);
- arr = arr || [];
-
- while (this.pos < end) {
- arr.push(this.readSFixed32());
- }
-
- return arr;
- },
- readPackedFixed64: function readPackedFixed64(arr) {
- if (this.type !== Pbf.Bytes) return arr.push(this.readFixed64());
- var end = readPackedEnd(this);
- arr = arr || [];
-
- while (this.pos < end) {
- arr.push(this.readFixed64());
- }
-
- return arr;
- },
- readPackedSFixed64: function readPackedSFixed64(arr) {
- if (this.type !== Pbf.Bytes) return arr.push(this.readSFixed64());
- var end = readPackedEnd(this);
- arr = arr || [];
-
- while (this.pos < end) {
- arr.push(this.readSFixed64());
- }
-
- 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;
-
- while (length < this.pos + min) {
- length *= 2;
- }
-
- 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;
-
- if (val > 0xfffffff || val < 0) {
- writeBigVarint(val, this);
- return;
- }
-
- 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
-
- 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
-
- 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);
-
- 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 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
-
- 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));
- }
- };
-
- 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');
- }
-
- function readPackedEnd(pbf) {
- return pbf.type === Pbf.Bytes ? pbf.readVarint() + pbf.pos : pbf.pos + 1;
- }
-
- function toNum(low, high, isSigned) {
- if (isSigned) {
- return high * 0x100000000 + (low >>> 0);
- }
-
- return (high >>> 0) * 0x100000000 + (low >>> 0);
- }
-
- function writeBigVarint(val, pbf) {
- var low, high;
-
- if (val >= 0) {
- low = val % 0x100000000 | 0;
- high = val / 0x100000000 | 0;
- } else {
- low = ~(-val % 0x100000000);
- high = ~(-val / 0x100000000);
-
- if (low ^ 0xffffffff) {
- low = low + 1 | 0;
- } else {
- low = 0;
- high = high + 1 | 0;
- }
- }
-
- if (val >= 0x10000000000000000 || val < -0x10000000000000000) {
- throw new Error('Given varint doesn\'t fit into 10 bytes');
- }
-
- pbf.realloc(10);
- writeBigVarintLow(low, high, pbf);
- writeBigVarintHigh(high, pbf);
- }
-
- 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 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;
- }
-
- 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
-
- pbf.realloc(extraLen);
-
- for (var i = pbf.pos - 1; i >= startPos; i--) {
- pbf.buf[i + extraLen] = pbf.buf[i];
- }
- }
-
- 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]);
- }
- }
-
- function _writePackedFloat(arr, pbf) {
- for (var i = 0; i < arr.length; i++) {
- pbf.writeFloat(arr[i]);
- }
- }
-
- function _writePackedDouble(arr, pbf) {
- for (var i = 0; i < arr.length; i++) {
- pbf.writeDouble(arr[i]);
- }
- }
-
- function _writePackedBoolean(arr, pbf) {
- for (var i = 0; i < arr.length; i++) {
- pbf.writeBoolean(arr[i]);
- }
- }
-
- function _writePackedFixed(arr, pbf) {
- for (var i = 0; i < arr.length; i++) {
- pbf.writeFixed32(arr[i]);
- }
- }
-
- function _writePackedSFixed(arr, pbf) {
- for (var i = 0; i < arr.length; i++) {
- pbf.writeSFixed32(arr[i]);
- }
- }
-
- function _writePackedFixed2(arr, pbf) {
- for (var i = 0; i < arr.length; i++) {
- pbf.writeFixed64(arr[i]);
- }
- }
-
- 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 readUInt32(buf, pos) {
- return (buf[pos] | buf[pos + 1] << 8 | buf[pos + 2] << 16) + buf[pos + 3] * 0x1000000;
- }
-
- function writeInt32(buf, val, pos) {
- buf[pos] = val;
- buf[pos + 1] = val >>> 8;
- buf[pos + 2] = val >>> 16;
- buf[pos + 3] = val >>> 24;
- }
-
- function readInt32(buf, pos) {
- return (buf[pos] | buf[pos + 1] << 8 | buf[pos + 2] << 16) + (buf[pos + 3] << 24);
- }
-
- function readUtf8(buf, pos, end) {
- var str = '';
- var i = pos;
-
- while (i < end) {
- var b0 = buf[i];
- var c = null; // codepoint
-
- 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];
-
- 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];
-
- if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80) {
- c = (b0 & 0xF) << 0xC | (b1 & 0x3F) << 0x6 | b2 & 0x3F;
-
- 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];
-
- if ((b1 & 0xC0) === 0x80 && (b2 & 0xC0) === 0x80 && (b3 & 0xC0) === 0x80) {
- c = (b0 & 0xF) << 0x12 | (b1 & 0x3F) << 0xC | (b2 & 0x3F) << 0x6 | b3 & 0x3F;
-
- if (c <= 0xFFFF || c >= 0x110000) {
- c = null;
- }
- }
- }
-
- if (c === null) {
- c = 0xFFFD;
- bytesPerSequence = 1;
- } else if (c > 0xFFFF) {
- c -= 0x10000;
- str += String.fromCharCode(c >>> 10 & 0x3FF | 0xD800);
- c = 0xDC00 | c & 0x3FF;
- }
-
- str += String.fromCharCode(c);
- i += bytesPerSequence;
- }
-
- return str;
- }
-
- function readUtf8TextDecoder(buf, pos, end) {
- return utf8TextDecoder.decode(buf.subarray(pos, end));
- }
-
- function writeUtf8(buf, str, pos) {
- for (var i = 0, c, lead; i < str.length; i++) {
- c = str.charCodeAt(i); // code point
-
- 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;
- }
-
- continue;
- }
- } else if (lead) {
- buf[pos++] = 0xEF;
- buf[pos++] = 0xBF;
- buf[pos++] = 0xBD;
- lead = null;
- }
-
- 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;
- }
-
- buf[pos++] = c >> 0x6 & 0x3F | 0x80;
- }
-
- buf[pos++] = c & 0x3F | 0x80;
- }
- }
-
- return pos;
- }
-
- 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);
- */
-
- function Point(x, y) {
- this.x = x;
- this.y = y;
- }
-
- 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);
- },
-
- /**
- * 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);
- },
-
- /**
- * 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);
- },
-
- /**
- * 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);
- },
-
- /**
- * 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);
- },
-
- /**
- * 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);
- },
-
- /**
- * 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);
- },
-
- /**
- * 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);
- },
-
- /**
- * 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);
- },
-
- /**
- * 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();
- },
-
- /**
- * 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();
- },
-
- /**
- * 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);
- },
-
- /**
- * 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;
- },
-
- /**
- * 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));
- },
-
- /**
- * 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;
- },
-
- /**
- * 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);
- },
-
- /**
- * 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;
- },
- _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());
-
- 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);
- */
-
- Point.convert = function (a) {
- if (a instanceof Point) {
- return a;
- }
-
- if (Array.isArray(a)) {
- return new Point(a[0], a[1]);
- }
-
- return a;
- };
-
- var vectortilefeature = VectorTileFeature;
-
- function VectorTileFeature(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);
- }
-
- 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;
- }
- }
-
- VectorTileFeature.types = ['Unknown', 'Point', 'LineString', 'Polygon'];
-
- 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;
- }
-
- length--;
-
- if (cmd === 1 || cmd === 2) {
- x += pbf.readSVarint();
- y += pbf.readSVarint();
-
- 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);
- }
- }
-
- if (line) lines.push(line);
- return lines;
- };
-
- 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;
-
- while (pbf.pos < end) {
- if (length <= 0) {
- var cmdLen = pbf.readVarint();
- cmd = cmdLen & 0x7;
- length = cmdLen >> 3;
- }
-
- 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);
- }
- }
-
- return [x1, y1, x2, y2];
- };
-
- 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;
-
- 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];
- }
- }
-
- switch (this.type) {
- case 1:
- var points = [];
-
- for (i = 0; i < coords.length; i++) {
- points[i] = coords[i][0];
- }
-
- coords = points;
- project(coords);
- break;
-
- case 2:
- for (i = 0; i < coords.length; i++) {
- project(coords[i]);
- }
-
- break;
-
- case 3:
- coords = classifyRings(coords);
-
- for (i = 0; i < coords.length; i++) {
- for (j = 0; j < coords[i].length; j++) {
- project(coords[i][j]);
- }
- }
-
- break;
- }
-
- if (coords.length === 1) {
- coords = coords[0];
- } else {
- type = 'Multi' + type;
- }
-
- var result = {
- type: "Feature",
- geometry: {
- type: type,
- coordinates: coords
- },
- properties: this.properties
- };
-
- if ('id' in this) {
- result.id = this.id;
- }
-
- return result;
- }; // classifies an array of rings into polygons with outer rings and holes
-
-
- function classifyRings(rings) {
- var len = rings.length;
- if (len <= 1) return [rings];
- var polygons = [],
- polygon,
- ccw;
-
- for (var i = 0; i < len; i++) {
- var area = signedArea$1(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]);
- }
- }
-
- if (polygon) polygons.push(polygon);
- return polygons;
- }
-
- function signedArea$1(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;
- }
-
- var vectortilelayer = VectorTileLayer;
-
- function VectorTileLayer(pbf, end) {
- // Public
- this.version = 1;
- this.name = null;
- this.extent = 4096;
- this.length = 0; // Private
-
- this._pbf = pbf;
- this._keys = [];
- this._values = [];
- this._features = [];
- pbf.readFields(readLayer, this, end);
- this.length = this._features.length;
- }
-
- 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 readValueMessage(pbf) {
- var value = null,
- end = pbf.readVarint() + pbf.pos;
-
- 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;
- }
-
- return value;
- } // return feature `i` from this layer as a `VectorTileFeature`
-
-
- 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];
-
- var end = this._pbf.readVarint() + this._pbf.pos;
-
- return new vectortilefeature(this._pbf, end, this.extent, this._keys, this._values);
- };
-
- var vectortile = VectorTile;
-
- function VectorTile(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;
- }
- }
-
- var VectorTile$1 = vectortile;
- var VectorTileFeature$1 = vectortilefeature;
- var VectorTileLayer$1 = vectortilelayer;
- var vectorTile = {
- VectorTile: VectorTile$1,
- VectorTileFeature: VectorTileFeature$1,
- VectorTileLayer: VectorTileLayer$1
- };
-
- var tiler$7 = utilTiler().tileSize(512).margin(1);
- var dispatch$8 = dispatch('loadedData');
-
- var _vtCache;
-
- function abortRequest$7(controller) {
- controller.abort();
- }
-
- function vtToGeoJSON(data, tile, mergeCache) {
- var vectorTile$1 = new vectorTile.VectorTile(new pbf(data));
- var layers = Object.keys(vectorTile$1.layers);
-
- if (!Array.isArray(layers)) {
- layers = [layers];
- }
-
- var features = [];
- layers.forEach(function (layerID) {
- var layer = vectorTile$1.layers[layerID];
-
- 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 (geometry.type === 'Polygon') {
- geometry.type = 'MultiPolygon';
- geometry.coordinates = [geometry.coordinates];
- }
-
- var isClipped = false; // Clip to tile bounds
-
- if (geometry.type === 'MultiPolygon') {
- var featureClip = turf_bboxClip(feature, tile.extent.rectangle());
-
- if (!fastDeepEqual(feature.geometry, featureClip.geometry)) {
- // feature = featureClip;
- isClipped = true;
- }
-
- 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
-
-
- 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
-
- if (isClipped && geometry.type === 'MultiPolygon') {
- var merged = mergeCache[propertyhash];
-
- if (merged && merged.length) {
- var other = merged[0];
- var coords = union(feature.geometry.coordinates, other.geometry.coordinates);
-
- if (!coords || !coords.length) {
- continue; // something failed in martinez union
- }
-
- merged.push(feature);
-
- 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;
- }
-
- 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);
- }
-
- source.loaded[tile.id] = [];
- delete source.inflight[tile.id];
- return response.arrayBuffer();
- }).then(function (data) {
- if (!data) {
- throw new Error('No Data');
- }
-
- var z = tile.xyz[2];
-
- if (!source.canMerge[z]) {
- source.canMerge[z] = {}; // initialize mergeCache
- }
-
- 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];
- });
- }
-
- var serviceVectorTile = {
- init: function init() {
- if (!_vtCache) {
- this.reset();
- }
-
- this.event = utilRebind(this, dispatch$8, 'on');
- },
- reset: function reset() {
- for (var sourceID in _vtCache) {
- var source = _vtCache[sourceID];
-
- if (source && source.inflight) {
- Object.values(source.inflight).forEach(abortRequest$7);
- }
- }
-
- _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 = [];
-
- for (var i = 0; i < tiles.length; i++) {
- var features = source.loaded[tiles[i].id];
- if (!features || !features.length) continue;
-
- 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
-
- results.push(Object.assign({}, feature)); // shallow copy
- }
- }
-
- return results;
- },
- loadTiles: function loadTiles(sourceID, template, projection) {
- var source = _vtCache[sourceID];
-
- if (!source) {
- source = this.addSource(sourceID, template);
- }
-
- var tiles = tiler$7.getTiles(projection); // abort inflight requests that are no longer needed
-
- Object.keys(source.inflight).forEach(function (k) {
- var wanted = tiles.find(function (tile) {
- return k === tile.id;
- });
-
- if (!wanted) {
- abortRequest$7(source.inflight[k]);
- delete source.inflight[k];
- }
- });
- 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);
- }
-
- 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;
- }
-
- 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 (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;
- }
-
- if (_wikidataCache[qid]) {
- if (callback) callback(null, _wikidataCache[qid]);
- return;
- }
-
- 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);
- }
-
- 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;
- }
-
- var i;
- var description;
-
- for (i in langs) {
- var code = langs[i];
-
- if (entity.descriptions[code] && entity.descriptions[code].language === code) {
- description = entity.descriptions[code];
- break;
- }
- }
-
- if (!description && Object.values(entity.descriptions).length) description = Object.values(entity.descriptions)[0]; // prepare result
-
- var result = {
- title: entity.id,
- description: description ? description.value : '',
- descriptionLocaleCode: description ? description.language : '',
- editURL: 'https://www.wikidata.org/wiki/' + entity.id
- }; // add image
-
- if (entity.claims) {
- var imageroot = 'https://commons.wikimedia.org/w/index.php';
- var props = ['P154', 'P18']; // logo image, image
-
- var prop, image;
-
- for (i = 0; i < props.length; i++) {
- prop = entity.claims[props[i]];
-
- if (prop && Object.keys(prop).length > 0) {
- image = prop[Object.keys(prop)[0]].mainsnak.datavalue.value;
-
- if (image) {
- result.imageURL = imageroot + '?' + utilQsString({
- title: 'Special:Redirect/file/' + image,
- width: 400
- });
- break;
- }
- }
- }
- }
-
- if (entity.sitelinks) {
- var englishLocale = _mainLocalizer.languageCode().toLowerCase() === 'en'; // must be one of these that we requested..
-
- for (i = 0; i < langs.length; i++) {
- // check each, in order of preference
- var w = langs[i] + 'wiki';
-
- if (entity.sitelinks[w]) {
- var title = entity.sitelinks[w].title;
- var tKey = 'inspector.wiki_reference';
-
- if (!englishLocale && langs[i] === 'en') {
- // user's locale isn't English but
- tKey = 'inspector.wiki_en_reference'; // we are sending them to enwiki anyway..
- }
-
- result.wiki = {
- title: title,
- text: tKey,
- url: 'https://' + langs[i] + '.wikipedia.org/wiki/' + title.replace(/ /g, '_')
- };
- break;
- }
- }
- }
-
- callback(null, result);
- });
- }
- };
-
- 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;
- }
-
- 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');
- }
-
- 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;
- }
-
- 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');
- }
-
- 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;
- }
-
- 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');
- }
-
- if (callback) {
- var list = result.query.pages[Object.keys(result.query.pages)[0]];
- var translations = {};
-
- if (list && list.langlinks) {
- list.langlinks.forEach(function (d) {
- translations[d.lang] = d['*'];
- });
- }
-
- callback(null, translations);
- }
- })["catch"](function (err) {
- if (callback) callback(err.message);
- });
- }
- };
-
- 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 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 uiNoteComments() {
- var _note;
-
- function noteComments(selection) {
- if (_note.isNew()) return; // don't draw .comments-container
-
- 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 && d.user) {
- selection = selection.append('a').attr('class', 'comment-author-link').attr('href', osm.userURL(d.user)).attr('target', '_blank');
- }
-
- 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);
- }
-
- 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
-
- _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);
- });
- });
- }
-
- function localeDateString(s) {
- if (!s) return null;
- var options = {
- day: 'numeric',
- month: 'short',
- year: 'numeric'
- };
- s = s.replace(/-/g, '/'); // fix browser-specific Date() 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;
- };
-
- return noteComments;
- }
-
- function uiNoteHeader() {
- var _note;
-
- 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');
- }
-
- return _t('note.note') + ' ' + d.id + ' ' + (d.status === 'closed' ? _t('note.closed') : '');
- });
- }
-
- noteHeader.note = function (val) {
- if (!arguments.length) return _note;
- _note = val;
- return noteHeader;
- };
-
- return noteHeader;
- }
-
- function uiNoteReport() {
- var _note;
-
- function noteReport(selection) {
- var url;
-
- if (services.osm && _note instanceof osmNote && !_note.isNew()) {
- url = services.osm.noteReportURL(_note);
- }
-
- var link = selection.selectAll('.note-report').data(url ? [url] : []); // exit
-
- link.exit().remove(); // enter
-
- 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'));
- }
-
- noteReport.note = function (val) {
- if (!arguments.length) return _note;
- _note = val;
- return noteReport;
- };
-
- return noteReport;
- }
-
- function uiViewOnOSM(context) {
- var _what; // an osmEntity or osmNote
-
-
- function viewOnOSM(selection) {
- var url;
-
- if (_what instanceof osmEntity) {
- url = context.connection().entityURL(_what);
- } else if (_what instanceof osmNote) {
- url = context.connection().noteURL(_what);
- }
-
- var data = !_what || _what.isNew() ? [] : [_what];
- var link = selection.selectAll('.view-on-osm').data(data, function (d) {
- return d.id;
- }); // exit
-
- link.exit().remove(); // enter
-
- 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'));
- }
-
- viewOnOSM.what = function (_) {
- if (!arguments.length) return _what;
- _what = _;
- return viewOnOSM;
- };
-
- return viewOnOSM;
- }
-
- function uiNoteEditor(context) {
- var dispatch$1 = dispatch('change');
- var noteComments = uiNoteComments();
- var noteHeader = uiNoteHeader(); // var formFields = uiFormFields(context);
-
- var _note;
-
- var _newNote; // var _fieldsArr;
-
-
- 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 osm = services.osm;
-
- if (osm) {
- osm.on('change.note-save', function () {
- selection.call(noteEditor);
- });
- }
- }
-
- 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
-
- noteSave.exit().remove(); // enter
-
- 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);
- // }
-
- 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);
-
- if (!commentTextarea.empty() && _newNote) {
- // autofocus the comment field for new notes
- commentTextarea.node().focus();
- } // update
-
-
- noteSave = noteSaveEnter.merge(noteSave).call(userDetails).call(noteSaveButtons); // fast submit if user presses cmd+enter
-
- 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
-
- window.setTimeout(function () {
- if (_note.isNew()) {
- noteSave.selectAll('.save-button').node().focus();
- clickSave();
- } else {
- noteSave.selectAll('.comment-button').node().focus();
- clickComment();
- }
- }, 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 (osm) {
- osm.replaceNote(_note); // update note cache
- }
-
- noteSave.call(noteSaveButtons);
- }
- }
-
- 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
-
- 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');
- }
-
- 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();
-
- var isSelected = _note && _note.id === context.selectedNoteID();
-
- var buttonSection = selection.selectAll('.buttons').data(isSelected ? [_note] : [], function (d) {
- return d.status + d.id;
- }); // exit
-
- buttonSection.exit().remove(); // enter
-
- var buttonEnter = buttonSection.enter().append('div').attr('class', 'buttons');
-
- 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
-
-
- 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 isSaveDisabled(d) {
- return hasAuth && d.status === 'open' && d.newComment ? null : true;
- }
- }
-
- function clickCancel(d3_event, d) {
- this.blur(); // avoid keeping focus on the button - #4641
-
- var osm = services.osm;
-
- if (osm) {
- osm.removeNote(d);
- }
-
- context.enter(modeBrowse(context));
- dispatch$1.call('change');
- }
-
- function clickSave(d3_event, d) {
- this.blur(); // avoid keeping focus on the button - #4641
-
- var osm = services.osm;
-
- if (osm) {
- osm.postNoteCreate(d, function (err, note) {
- dispatch$1.call('change', note);
- });
- }
- }
-
- function clickStatus(d3_event, d) {
- this.blur(); // avoid keeping focus on the button - #4641
-
- var osm = services.osm;
-
- 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 osm = services.osm;
-
- if (osm) {
- osm.postNoteUpdate(d, d.status, function (err, note) {
- dispatch$1.call('change', note);
- });
- }
- }
-
- noteEditor.note = function (val) {
- if (!arguments.length) return _note;
- _note = val;
- return noteEditor;
- };
-
- noteEditor.newNote = function (val) {
- if (!arguments.length) return _newNote;
- _newNote = val;
- return noteEditor;
- };
-
- return utilRebind(noteEditor, dispatch$1, 'on');
- }
-
- function modeSelectNote(context, selectedNoteID) {
- var mode = {
- id: 'select-note',
- button: 'browse'
- };
-
- var _keybinding = utilKeybinding('select-note');
-
- var _noteEditor = uiNoteEditor(context).on('change', function () {
- context.map().pan([0, 0]); // trigger a redraw
-
- var note = checkSelectedID();
- if (!note) return;
- context.ui().sidebar.show(_noteEditor.note(note));
- });
-
- var _behaviors = [behaviorBreathe(), behaviorHover(context), behaviorSelect(context), behaviorLasso(context), modeDragNode(context).behavior, modeDragNote(context).behavior];
- var _newFeature = false;
-
- function checkSelectedID() {
- if (!services.osm) return;
- var note = services.osm.getNote(selectedNoteID);
-
- if (!note) {
- context.enter(modeBrowse(context));
- }
-
- return note;
- } // class the note as selected, or return to browse mode if the note is gone
-
-
- 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);
- }
- }
-
- 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);
-
- if (note) {
- context.map().centerZoomEase(note.loc, 20);
- }
- };
-
- mode.newFeature = function (val) {
- if (!arguments.length) return _newFeature;
- _newFeature = val;
- return mode;
- };
-
- 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);
- };
-
- mode.exit = function () {
- _behaviors.forEach(context.uninstall);
-
- 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);
- };
-
- return mode;
- }
-
- function modeDragNote(context) {
- var mode = {
- id: 'drag-note',
- button: 'browse'
- };
- var edit = behaviorEdit(context);
-
- var _nudgeInterval;
-
- var _lastLoc;
-
- var _note; // most current note.. dragged note may have stale datum.
-
-
- function startNudge(d3_event, nudge) {
- if (_nudgeInterval) window.clearInterval(_nudgeInterval);
- _nudgeInterval = window.setInterval(function () {
- context.map().pan(nudge);
- doMove(d3_event, nudge);
- }, 50);
- }
-
- function stopNudge() {
- if (_nudgeInterval) {
- window.clearInterval(_nudgeInterval);
- _nudgeInterval = null;
- }
- }
-
- function origin(note) {
- return context.projection(note.loc);
- }
-
- function start(d3_event, note) {
- _note = note;
- var osm = services.osm;
-
- 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);
- }
-
- context.surface().selectAll('.note-' + _note.id).classed('active', true);
- context.perform(actionNoop());
- context.enter(mode);
- context.selectedNoteID(_note.id);
- }
-
- 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 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;
-
- if (osm) {
- osm.replaceNote(_note); // update note cache
- }
-
- 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 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;
- };
-
- 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 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;
-
- 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;
- });
- }));
- };
-
- 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
-
- input.node().focus(); // focus the input as if it was clicked
-
- mousedown(d3_event);
- }).on('mouseup.combo-caret', function (d3_event) {
- d3_event.preventDefault(); // don't steal focus from input
-
- mouseup(d3_event);
- });
- });
-
- function mousedown(d3_event) {
- if (d3_event.button !== 0) return; // left click only
-
- _tDown = +new Date(); // clear selection
-
- var start = input.property('selectionStart');
- var end = input.property('selectionEnd');
-
- if (start !== end) {
- var val = utilGetSetValue(input);
- input.node().setSelectionRange(val.length, val.length);
- return;
- }
-
- 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
-
- if (input.node() !== document.activeElement) return; // exit if this input is not focused
-
- 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 combo = container.selectAll('.combobox');
-
- 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();
- }
- }
-
- function focus() {
- fetchComboData(''); // prefetch values (may warm taginfo cache)
- }
-
- function blur() {
- _comboHideTimerID = window.setTimeout(hide, 75);
- }
-
- function show() {
- hide(); // remove any existing
-
- 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 hide() {
- if (_comboHideTimerID) {
- window.clearTimeout(_comboHideTimerID);
- _comboHideTimerID = undefined;
- }
-
- container.selectAll('.combobox').remove();
- container.on('scroll.combo-scroll', null);
- }
-
- function keydown(d3_event) {
- var shown = !container.selectAll('.combobox').empty();
- var tagName = input.node() ? input.node().tagName.toLowerCase() : '';
-
- switch (d3_event.keyCode) {
- case 8: // ⌫ Backspace
-
- 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;
-
- case 9:
- // ⇥ Tab
- accept();
- break;
-
- case 13:
- // ↩ Return
- d3_event.preventDefault();
- d3_event.stopPropagation();
- break;
-
- case 38:
- // ↑ Up arrow
- if (tagName === 'textarea' && !shown) return;
- d3_event.preventDefault();
-
- if (tagName === 'input' && !shown) {
- show();
- }
-
- nav(-1);
- break;
-
- case 40:
- // ↓ Down arrow
- if (tagName === 'textarea' && !shown) return;
- d3_event.preventDefault();
-
- if (tagName === 'input' && !shown) {
- show();
- }
-
- nav(+1);
- break;
- }
- }
-
- 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)
-
-
- function change() {
- fetchComboData(value(), function () {
- _selected = null;
- var val = input.property('value');
-
- if (_suggestions.length) {
- if (input.property('selectionEnd') === val.length) {
- _selected = tryAutocomplete();
- }
-
- if (!_selected) {
- _selected = val;
- }
- }
-
- if (val.length) {
- var combo = container.selectAll('.combobox');
-
- if (combo.empty()) {
- show();
- }
- } else {
- hide();
- }
-
- render();
- });
- } // Called when the user presses up/down arrows to navigate the list
-
-
- 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
-
-
- index = Math.max(Math.min(index + dir, _suggestions.length - 1), 0);
- _selected = _suggestions[index].value;
- input.property('value', _selected);
- }
-
- render();
- ensureVisible();
- }
-
- function ensureVisible() {
- var combo = container.selectAll('.combobox');
- if (combo.empty()) return;
- var containerRect = container.node().getBoundingClientRect();
- var comboRect = combo.node().getBoundingClientRect();
-
- 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
-
-
- var selected = combo.selectAll('.combobox-option.selected').node();
-
- if (selected) {
- selected.scrollIntoView({
- behavior: 'smooth',
- block: 'nearest'
- });
- }
- }
-
- function value() {
- var value = input.property('value');
- var start = input.property('selectionStart');
- var end = input.property('selectionEnd');
-
- if (start && end) {
- value = value.substring(0, start);
- }
-
- return value;
- }
-
- function fetchComboData(v, cb) {
- _cancelFetch = false;
-
- _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 (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;
-
- 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 (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 (bestIndex !== -1) {
- var bestVal = _suggestions[bestIndex].value;
- input.property('value', bestVal);
- input.node().setSelectionRange(val.length, bestVal.length);
- return bestVal;
- }
- }
-
- function render() {
- if (_suggestions.length < _minItems || document.activeElement !== input.node()) {
- hide();
- return;
- }
-
- 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
-
- 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.
-
-
- function accept(d3_event, d) {
- _cancelFetch = true;
- var thiz = input.node();
-
- if (d) {
- // user clicked on a suggestion
- utilGetSetValue(input, d.value); // replace field contents
-
- utilTriggerEvent(input, 'change');
- } // clear (and keep) selection
-
-
- 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.
-
-
- 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();
- }
- };
-
- 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;
- };
-
- combobox.minItems = function (val) {
- if (!arguments.length) return _minItems;
- _minItems = val;
- return combobox;
- };
-
- combobox.itemsMouseEnter = function (val) {
- if (!arguments.length) return _mouseEnterHandler;
- _mouseEnterHandler = val;
- return combobox;
- };
-
- combobox.itemsMouseLeave = function (val) {
- if (!arguments.length) return _mouseLeaveHandler;
- _mouseLeaveHandler = val;
- return combobox;
- };
-
- return utilRebind(combobox, dispatch$1, 'on');
- }
-
- 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);
- };
-
- // 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.
-
- 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 uiDisclosure(context, key, expandedDefault) {
- var dispatch$1 = dispatch('toggled');
-
- var _expanded;
-
- var _label = utilFunctor('');
-
- var _updatePreference = true;
-
- var _content = function _content() {};
-
- 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';
- }
-
- 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
-
- 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 (_expanded) {
- wrap.call(_content);
- }
-
- function toggle(d3_event) {
- d3_event.preventDefault();
- _expanded = !_expanded;
-
- if (_updatePreference) {
- corePreferences('disclosure.' + key + '.expanded', _expanded);
- }
-
- 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 (_expanded) {
- wrap.call(_content);
- }
-
- dispatch$1.call('toggled', this, _expanded);
- }
- };
-
- disclosure.label = function (val) {
- if (!arguments.length) return _label;
- _label = utilFunctor(val);
- return disclosure;
- };
-
- disclosure.expanded = function (val) {
- if (!arguments.length) return _expanded;
- _expanded = val;
- return disclosure;
- };
-
- disclosure.updatePreference = function (val) {
- if (!arguments.length) return _updatePreference;
- _updatePreference = val;
- return disclosure;
- };
-
- disclosure.content = function (val) {
- if (!arguments.length) return _content;
- _content = val;
- return disclosure;
- };
-
- return utilRebind(disclosure, dispatch$1, 'on');
- }
-
- // Can be labeled and collapsible.
-
- function uiSection(id, context) {
- var _classes = utilFunctor('');
-
- var _shouldDisplay;
-
- var _content;
-
- var _disclosure;
-
- var _label;
-
- var _expandedByDefault = utilFunctor(true);
-
- var _disclosureContent;
-
- var _disclosureExpanded;
-
- var _containerSelection = select(null);
-
- var section = {
- id: id
- };
-
- section.classes = function (val) {
- if (!arguments.length) return _classes;
- _classes = utilFunctor(val);
- return section;
- };
-
- section.label = function (val) {
- if (!arguments.length) return _label;
- _label = utilFunctor(val);
- return section;
- };
-
- section.expandedByDefault = function (val) {
- if (!arguments.length) return _expandedByDefault;
- _expandedByDefault = utilFunctor(val);
- return section;
- };
-
- section.shouldDisplay = function (val) {
- if (!arguments.length) return _shouldDisplay;
- _shouldDisplay = utilFunctor(val);
- return section;
- };
-
- section.content = function (val) {
- if (!arguments.length) return _content;
- _content = val;
- return section;
- };
-
- section.disclosureContent = function (val) {
- if (!arguments.length) return _disclosureContent;
- _disclosureContent = val;
- return section;
- };
-
- section.disclosureExpanded = function (val) {
- if (!arguments.length) return _disclosureExpanded;
- _disclosureExpanded = val;
- return section;
- }; // may be called multiple times
-
-
- section.render = function (selection) {
- _containerSelection = selection.selectAll('.section-' + id).data([0]);
-
- var sectionEnter = _containerSelection.enter().append('div').attr('class', 'section section-' + id + ' ' + (_classes && _classes() || ''));
-
- _containerSelection = sectionEnter.merge(_containerSelection);
-
- _containerSelection.call(renderContent);
- };
-
- section.reRender = function () {
- _containerSelection.call(renderContent);
- };
-
- section.selection = function () {
- return _containerSelection;
- };
-
- section.disclosure = function () {
- return _disclosure;
- }; // may be called multiple times
-
-
- function renderContent(selection) {
- if (_shouldDisplay) {
- var shouldDisplay = _shouldDisplay();
-
- selection.classed('hide', !shouldDisplay);
-
- if (!shouldDisplay) {
- selection.html('');
- return;
- }
- }
-
- 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;
- }
-
- selection.call(_disclosure);
- return;
- }
-
- if (_content) {
- selection.call(_content);
- }
- }
-
- return section;
- }
-
- // {
- // 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 = {};
-
- var _button = select(null);
-
- var _body = select(null);
-
- var _loaded;
-
- var _showing;
-
- function load() {
- if (!wikibase) return;
-
- _button.classed('tag-reference-loading', true);
-
- wikibase.getDocs(what, gotDocs);
- }
-
- function gotDocs(err, docs) {
- _body.html('');
-
- if (!docs || !docs.title) {
- _body.append('p').attr('class', 'tag-reference-description').html(_t.html('inspector.no_documentation_key'));
-
- done();
- return;
- }
-
- 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();
- }
-
- _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
-
-
- 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'));
- }
- }
-
- function done() {
- _loaded = true;
-
- _button.classed('tag-reference-loading', false);
-
- _body.classed('expanded', true).transition().duration(200).style('max-height', '200px').style('opacity', '1');
-
- _showing = true;
-
- _button.selectAll('svg.icon use').each(function () {
- var iconUse = select(this);
-
- if (iconUse.attr('href') === '#iD-icon-info') {
- iconUse.attr('href', '#iD-icon-info-filled');
- }
- });
- }
-
- function hide() {
- _body.transition().duration(200).style('max-height', '0px').style('opacity', '0').on('end', function () {
- _body.classed('expanded', false);
- });
-
- _showing = false;
-
- _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');
- }
- });
- }
-
- 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);
-
- _button.on('click', function (d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- this.blur(); // avoid keeping focus on the button - #4641
-
- if (_showing) {
- hide();
- } else if (_loaded) {
- done();
- } else {
- load();
- }
- });
- };
-
- 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();
- }
- };
-
- tagReference.showing = function (val) {
- if (!arguments.length) return _showing;
- _showing = val;
- return tagReference;
- };
-
- return tagReference;
- }
-
- 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'
- }];
-
- var _tagView = corePreferences('raw-tag-editor-view') || 'list'; // 'list, 'text'
-
-
- 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;
-
- var _entityIDs;
-
- var _didInteract = false;
-
- function interacted() {
- _didInteract = true;
- }
-
- 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
-
-
- var rowData = _orderedKeys.map(function (key, i) {
- return {
- index: i,
- key: key,
- value: _tags[key]
- };
- }); // append blank row last, if necessary
-
-
- if (!rowData.length || _showBlank) {
- _showBlank = false;
- rowData.push({
- index: rowData.length,
- key: '',
- value: ''
- });
- } // View Options
-
-
- 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;
- });
- 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 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
-
- 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
-
- 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
-
- var items = list.selectAll('.tag-row').data(rowData, function (d) {
- return d.key;
- });
- items.exit().each(unbind).remove(); // Enter
-
- 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
-
- 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
-
- var value = row.select('input.value'); // propagate bound data
-
- if (_entityIDs && taginfo && _state !== 'hover') {
- bindTypeahead(key, value);
- }
-
- var referenceOptions = {
- key: d.key
- };
-
- if (typeof d.value === 'string') {
- referenceOptions.value = d.value;
- }
-
- var reference = uiTagReference(referenceOptions);
-
- if (_state === 'hover') {
- reference.showing(false);
- }
-
- 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;
- }
- }
-
- return false;
- }
-
- 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 stringify(s) {
- return JSON.stringify(s).slice(1, -1); // without leading/trailing "
- }
-
- function unstringify(s) {
- var leading = '';
- var trailing = '';
-
- if (s.length < 1 || s.charAt(0) !== '"') {
- leading = '"';
- }
-
- if (s.length < 2 || s.charAt(s.length - 1) !== '"' || s.charAt(s.length - 1) === '"' && s.charAt(s.length - 2) === '\\') {
- trailing = '"';
- }
-
- return JSON.parse(leading + s + trailing);
- }
-
- 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';
- }
-
- return str;
- }
-
- function textChanged() {
- var newText = this.value.trim();
- var newTags = {};
- newText.split('\n').forEach(function (row) {
- var m = row.match(/^\s*([^=]+)=(.*)$/);
-
- 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
-
- if (change.newVal === '*' && typeof change.oldVal !== 'string') return;
-
- if (change.type === '-') {
- _pendingChange[change.key] = undefined;
- } else if (change.type === '+') {
- _pendingChange[change.key] = change.newVal || '';
- }
- });
-
- if (Object.keys(_pendingChange).length === 0) {
- _pendingChange = null;
- return;
- }
-
- scheduleChange();
- }
-
- 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();
- }
- }
-
- 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;
-
- var data = _tags[keyString].filter(Boolean).map(function (tagValue) {
- return {
- value: tagValue,
- title: tagValue
- };
- });
-
- 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));
- }
- });
- }));
- 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);
- }
- }
-
- function unbind() {
- var row = select(this);
- row.selectAll('input.key').call(uiCombobox.off, context);
- row.selectAll('input.value').call(uiCombobox.off, context);
- }
-
- 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 (_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 (isReadOnly({
- key: kNew
- })) {
- this.value = kOld;
- return;
- }
-
- 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();
- }
- });
- 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;
- }
-
- _pendingChange[kNew] = vNew; // update the ordered key index so this row doesn't change position
-
- var existingKeyIndex = _orderedKeys.indexOf(kOld);
-
- if (existingKeyIndex !== -1) _orderedKeys[existingKeyIndex] = kNew;
- d.key = kNew; // update datum to avoid exit/enter on tag update
-
- d.value = vNew;
- this.value = kNew;
- utilGetSetValue(inputVal, vNew);
- scheduleChange();
- }
-
- 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
-
- if (_pendingChange && _pendingChange.hasOwnProperty(d.key) && _pendingChange[d.key] === undefined) return;
- _pendingChange = _pendingChange || {};
- _pendingChange[d.key] = context.cleanTagValue(this.value);
- scheduleChange();
- }
-
- function removeTag(d3_event, d) {
- if (isReadOnly(d)) return;
-
- 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();
- }
- }
-
- 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
-
- window.setTimeout(function () {
- if (!_pendingChange) return;
- dispatch$1.call('change', this, entityIDs, _pendingChange);
- _pendingChange = null;
- }, 10);
- }
-
- section.state = function (val) {
- if (!arguments.length) return _state;
-
- if (_state !== val) {
- _orderedKeys = [];
- _state = val;
- }
-
- return section;
- };
-
- section.presets = function (val) {
- if (!arguments.length) return _presets;
- _presets = val;
-
- 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);
- }
-
- return section;
- };
-
- section.tags = function (val) {
- if (!arguments.length) return _tags;
- _tags = val;
- return section;
- };
-
- section.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
-
- if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
- _entityIDs = val;
- _orderedKeys = [];
- }
-
- 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$1, 'on');
- }
-
- function uiDataEditor(context) {
- var dataHeader = uiDataHeader();
- var rawTagEditor = uiSectionRawTagEditor('custom-data-tag-editor', context).expandedByDefault(true).readOnlyTags([/./]);
-
- var _datum;
-
- 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
-
- 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
-
- 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);
- }
-
- dataEditor.datum = function (val) {
- if (!arguments.length) return _datum;
- _datum = val;
- return this;
- };
-
- return dataEditor;
- }
-
- 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 selectData(d3_event, drawn) {
- var selection = context.surface().selectAll('.layer-mapdata .data' + selectedDatum.__featurehash__);
-
- 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);
- }
- }
-
- 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));
- };
-
- 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 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');
- }
-
- 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)
- });
- });
- 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
- });
- }
-
- 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
-
- 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);
- }
-
- 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)
-
- 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 error - #5880
-
- 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;
- };
-
- 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 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);
- }
-
- improveOsmHeader.issue = function (val) {
- if (!arguments.length) return _qaItem;
- _qaItem = val;
- return improveOsmHeader;
- };
-
- return improveOsmHeader;
- }
-
- function uiImproveOsmEditor(context) {
- var dispatch$1 = dispatch('change');
- var qaDetails = uiImproveOsmDetails(context);
- var qaComments = uiImproveOsmComments();
- var qaHeader = uiImproveOsmHeader();
-
- var _qaItem;
-
- 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 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
-
- 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('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
-
- saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
-
- function changeInput() {
- var input = select(this);
- var val = input.property('value').trim();
-
- if (val === '') {
- val = undefined;
- } // store the unsaved comment with the issue itself
-
-
- _qaItem = _qaItem.update({
- newComment: val
- });
- var qaService = services.improveOSM;
-
- if (qaService) {
- qaService.replaceItem(_qaItem);
- }
-
- saveSection.call(qaSaveButtons);
- }
- }
-
- function qaSaveButtons(selection) {
- var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
-
- 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$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
-
- var qaService = services.improveOSM;
-
- if (qaService) {
- d.newStatus = 'SOLVED';
- qaService.postUpdate(d, function (err, item) {
- return dispatch$1.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
-
- 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
-
-
- improveOsmEditor.error = function (val) {
- if (!arguments.length) return _qaItem;
- _qaItem = val;
- return improveOsmEditor;
- };
-
- return utilRebind(improveOsmEditor, dispatch$1, 'on');
- }
-
- 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 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 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
-
- 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);
- }
-
- context.map().centerZoomEase(_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)
-
- 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
-
- 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;
- };
-
- return keepRightDetails;
- }
-
- 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
-
- var title = _t.html("QA.keepRight.errorTypes.".concat(itemType, ".title"), replacements);
-
- if (title === unknown) {
- title = _t.html("QA.keepRight.errorTypes.".concat(parentIssueType, ".title"), replacements);
- }
-
- return title;
- }
-
- 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;
- });
- 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);
- }
-
- keepRightHeader.issue = function (val) {
- if (!arguments.length) return _qaItem;
- _qaItem = val;
- return keepRightHeader;
- };
-
- return keepRightHeader;
- }
-
- function uiViewOnKeepRight() {
- var _qaItem;
-
- function viewOnKeepRight(selection) {
- var url;
-
- 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'));
- }
-
- viewOnKeepRight.what = function (val) {
- if (!arguments.length) return _qaItem;
- _qaItem = val;
- return viewOnKeepRight;
- };
-
- return viewOnKeepRight;
- }
-
- function uiKeepRightEditor(context) {
- var dispatch$1 = dispatch('change');
- var qaDetails = uiKeepRightDetails(context);
- var qaHeader = uiKeepRightHeader();
-
- 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 keepRightSaveSection(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
-
- 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
-
- saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
-
- function changeInput() {
- var input = select(this);
- var val = input.property('value').trim();
-
- if (val === _qaItem.comment) {
- val = undefined;
- } // store the unsaved comment with the issue itself
-
-
- _qaItem = _qaItem.update({
- newComment: val
- });
- var qaService = services.keepRight;
-
- if (qaService) {
- qaService.replaceItem(_qaItem); // update keepright cache
- }
-
- saveSection.call(qaSaveButtons);
- }
- }
-
- function qaSaveButtons(selection) {
- var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
-
- 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') // 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;
-
- if (qaService) {
- qaService.postUpdate(d, function (err, item) {
- return dispatch$1.call('change', item);
- });
- }
- });
- 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);
- });
- }
- });
- 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;
-
- if (qaService) {
- d.newStatus = 'ignore'; // ignore permanently (false positive)
-
- qaService.postUpdate(d, function (err, item) {
- return dispatch$1.call('change', item);
- });
- }
- });
- } // NOTE: Don't change method name until UI v3 is merged
-
-
- keepRightEditor.error = function (val) {
- if (!arguments.length) return _qaItem;
- _qaItem = val;
- return keepRightEditor;
- };
-
- return utilRebind(keepRightEditor, dispatch$1, 'on');
- }
-
- function uiOsmoseDetails(context) {
- var _qaItem;
-
- function issueString(d, type) {
- if (!d) return ''; // Issue strings are cached from Osmose API
-
- 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);
- });
- details.exit().remove();
- var detailsEnter = details.enter().append('div').attr('class', 'error-details qa-details-container'); // Description
-
- 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)
-
-
- 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)
-
- if (issueString(_qaItem, 'fix')) {
- var _div = detailsEnter.append('div').attr('class', 'qa-details-subsection');
-
- _div.append('h4').html(_t.html('QA.osmose.fix_title'));
-
- _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)
-
-
- 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
-
-
- 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 (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
-
-
- 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
-
- 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);
- }
-
- 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)
-
- if (entity) {
- var name = utilDisplayName(entity); // try to use common name
-
- 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
-
- context.features().forceVisible(d.elems);
- context.map().pan([0, 0]); // trigger a redraw
- })["catch"](function (err) {
- console.log(err); // eslint-disable-line no-console
- });
- }
-
- osmoseDetails.issue = function (val) {
- if (!arguments.length) return _qaItem;
- _qaItem = val;
- return osmoseDetails;
- };
-
- return osmoseDetails;
- }
-
- function uiOsmoseHeader() {
- var _qaItem;
-
- 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;
- }
-
- 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;
-
- 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);
- }
-
- osmoseHeader.issue = function (val) {
- if (!arguments.length) return _qaItem;
- _qaItem = val;
- return osmoseHeader;
- };
-
- return osmoseHeader;
- }
-
- function uiViewOnOsmose() {
- var _qaItem;
-
- function viewOnOsmose(selection) {
- var url;
-
- if (services.osmose && _qaItem instanceof QAItem) {
- url = services.osmose.itemURL(_qaItem);
- }
-
- 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 viewOnOsmose;
- }
-
- 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
-
- saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
- }
-
- function qaSaveButtons(selection) {
- var isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
-
- 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 close-button action');
- buttonEnter.append('button').attr('class', 'button ignore-button action'); // update
-
- 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$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 qaService = services.osmose;
-
- 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
-
-
- osmoseEditor.error = function (val) {
- if (!arguments.length) return _qaItem;
- _qaItem = val;
- return osmoseEditor;
- };
-
- return utilRebind(osmoseEditor, dispatch$1, 'on');
- }
-
- function modeSelectError(context, selectedErrorID, selectedErrorService) {
- var mode = {
- id: 'select-error',
- button: 'browse'
- };
- var keybinding = utilKeybinding('select-error');
- var errorService = services[selectedErrorService];
- var errorEditor;
-
- switch (selectedErrorService) {
- case 'improveOSM':
- errorEditor = uiImproveOsmEditor(context).on('change', function () {
- context.map().pan([0, 0]); // trigger a redraw
-
- var error = checkSelectedID();
- if (!error) return;
- context.ui().sidebar.show(errorEditor.error(error));
- });
- break;
-
- case 'keepRight':
- errorEditor = uiKeepRightEditor(context).on('change', function () {
- context.map().pan([0, 0]); // trigger a redraw
-
- var error = checkSelectedID();
- if (!error) return;
- context.ui().sidebar.show(errorEditor.error(error));
- });
- break;
-
- case 'osmose':
- errorEditor = uiOsmoseEditor(context).on('change', function () {
- context.map().pan([0, 0]); // trigger a redraw
-
- var error = checkSelectedID();
- if (!error) return;
- context.ui().sidebar.show(errorEditor.error(error));
- });
- break;
- }
-
- 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);
-
- if (!error) {
- context.enter(modeBrowse(context));
- }
-
- return error;
- }
-
- mode.zoomToSelected = function () {
- if (!errorService) return;
- var error = errorService.getError(selectedErrorID);
-
- if (error) {
- context.map().centerZoomEase(error.loc, 20);
- }
- };
-
- 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 selectError(d3_event, drawn) {
- if (!checkSelectedID()) return;
- var selection = context.surface().selectAll('.itemId-' + selectedErrorID + '.' + selectedErrorService);
-
- 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.selectedErrorID(selectedErrorID);
- }
- }
-
- function esc() {
- if (context.container().select('.combobox').size()) return;
- context.enter(modeBrowse(context));
- }
- };
-
- 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([]);
- };
-
- return mode;
- }
-
- function behaviorSelect(context) {
- var _tolerancePx = 4; // see also behaviorDrag
-
- 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 _multiselectionPointerId = null; // use pointer events on supported platforms; fallback to mouse events
-
- var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
-
- 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;
- }
-
- if (d3_event.keyCode === 93 || // context menu key
- d3_event.keyCode === 32) {
- // spacebar
- d3_event.preventDefault();
- }
-
- 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
-
- cancelLongPress();
-
- if (d3_event.shiftKey) {
- context.surface().classed('behavior-multiselect', true);
- }
-
- 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 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;
-
- if (pointer) {
- delete _downPointers.spacebar;
- if (pointer.done) return;
- d3_event.preventDefault();
- _lastInteractionType = 'spacebar';
- click(pointer.firstEvent, pointer.lastEvent, 'spacebar');
- }
- }
- }
-
- 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 didLongPress(id, interactionType) {
- var pointer = _downPointers[id];
- if (!pointer) return;
-
- 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
-
-
- _longPressTimeout = null;
- _lastInteractionType = interactionType;
- _showMenu = true;
- click(pointer.firstEvent, pointer.lastEvent, id);
- }
-
- 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 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;
- }
-
- if (pointer.done) return;
- click(pointer.firstEvent, d3_event, id);
- }
-
- 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();
-
- if (!+d3_event.clientX && !+d3_event.clientY) {
- if (_lastMouseEvent) {
- d3_event.sourceEvent = _lastMouseEvent;
- } else {
- return;
- }
- } else {
- _lastMouseEvent = d3_event;
- _lastInteractionType = 'rightclick';
- }
-
- _showMenu = true;
- click(d3_event, d3_event);
- }
-
- 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 mapContains(event) {
- var rect = mapNode.getBoundingClientRect();
- return event.clientX >= rect.left && event.clientX <= rect.right && event.clientY >= rect.top && event.clientY <= rect.bottom;
- }
-
- function pointerDownOnSelection(skipPointerId) {
- var mode = context.mode();
- var selectedIDs = mode.id === 'select' ? mode.selectedIDs() : [];
-
- 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
- };
- }
-
- return null;
- }
- }
-
- 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;
-
- if (datum && datum.type === 'midpoint') {
- // treat targeting midpoints as if targeting the parent way
- datum = datum.parents[0];
- }
-
- var newMode;
-
- if (datum instanceof osmEntity) {
- // targeting an entity
- var selectedIDs = context.selectedIDs();
- context.selectedNoteID(null);
- context.selectedErrorID(null);
-
- 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
-
- 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);
-
- if (!isMultiselect && mode.id !== 'browse') {
- context.enter(modeBrowse(context));
- }
- }
-
- context.ui().closeEditMenu(); // always request to show the edit menu in case the mode needs it
-
- if (showMenu) context.ui().showEditMenu(point, interactionType);
- resetProperties();
- }
-
- 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();
- }
- });
- 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 behavior;
- }
-
- 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.
-
- var _nodeIndex;
-
- var _origWay;
-
- var _wayGeometry;
-
- var _headNodeID;
-
- var _annotation;
-
- var _pointerHasMoved = false; // The osmNode to be placed.
- // This is temporary and just follows the mouse cursor until an "add" event occurs.
-
- var _drawNode;
-
- var _didResolveTempEdit = false;
-
- 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 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 keydown(d3_event) {
- if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
- if (context.surface().classed('nope')) {
- context.surface().classed('nope-suppressed', true);
- }
-
- 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);
- }
-
- context.surface().classed('nope-suppressed', false).classed('nope-disabled', false);
- }
- }
-
- 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 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;
-
- 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);
-
- if (choice) {
- loc = choice.loc;
- }
- }
-
- 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
-
-
- function checkGeometry(includeDrawNode) {
- var nopeDisabled = context.surface().classed('nope-disabled');
- var isInvalid = isInvalidGeometry(includeDrawNode);
-
- if (nopeDisabled) {
- context.surface().classed('nope', false).classed('nope-suppressed', isInvalid);
- } else {
- context.surface().classed('nope', isInvalid).classed('nope-suppressed', false);
- }
- }
-
- function isInvalidGeometry(includeDrawNode) {
- var testNode = _drawNode; // we only need to test the single way we're drawing
-
- var parentWay = context.graph().entity(wayID);
- var nodes = context.graph().childNodes(parentWay).slice(); // shallow copy
-
- 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;
- }
- }
-
- return testNode && geoHasSelfIntersections(nodes, testNode.id);
- }
-
- function undone() {
- // undoing removed the temp edit
- _didResolveTempEdit = true;
- context.pauseChangeDispatch();
- var nextMode;
-
- 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
-
- nextMode = mode;
- } // clear the redo stack by adding and removing a blank edit
-
-
- context.perform(actionNoop());
- context.pop(1);
- context.resumeChangeDispatch();
- context.enter(nextMode);
- }
-
- function setActiveElements() {
- if (!_drawNode) return;
- context.surface().selectAll('.' + _drawNode.id).classed('active', true);
- }
-
- function resetToStartGraph() {
- while (context.graph() !== startGraph) {
- context.pop();
- }
- }
-
- 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.
-
- 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();
- }
-
- _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);
- };
-
- 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);
- }
-
- checkGeometry(true
- /* includeDrawNode */
- );
-
- if (d && d.properties && d.properties.nope || context.surface().classed('nope')) {
- if (!_pointerHasMoved) {
- // prevent the temporary draw node from appearing on touch devices
- removeDrawNode();
- }
-
- dispatch$1.call('rejectedSelfIntersection', this);
- return; // can't click here
- }
-
- context.pauseChangeDispatch();
- doAdd(); // we just replaced the temporary edit with the real one
-
- _didResolveTempEdit = true;
- context.resumeChangeDispatch();
- context.enter(mode);
- } // Accept the current position of the drawing node
-
-
- drawWay.add = function (loc, d) {
- attemptAdd(d, loc, function () {// don't need to do anything extra
- });
- }; // Connect the way to an existing way
-
-
- 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
-
-
- 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.
-
-
- drawWay.finish = function () {
- checkGeometry(false
- /* includeDrawNode */
- );
-
- if (context.surface().classed('nope')) {
- dispatch$1.call('rejectedSelfIntersection', this);
- return; // can't click here
- }
-
- context.pauseChangeDispatch(); // remove the temporary edit
-
- context.pop(1);
- _didResolveTempEdit = true;
- context.resumeChangeDispatch();
- var way = context.hasEntity(wayID);
-
- 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.
-
-
- 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;
- };
-
- drawWay.activeID = function () {
- if (!arguments.length) return _drawNode && _drawNode.id; // no assign
-
- return drawWay;
- };
-
- return utilRebind(drawWay, dispatch$1, 'on');
- }
-
- 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);
- };
-
- mode.exit = function () {
- context.uninstall(behavior);
- };
-
- mode.selectedIDs = function () {
- return [wayID];
- };
-
- mode.activeID = function () {
- return behavior && behavior.activeID() || [];
- };
-
- return mode;
- }
-
- function operationContinue(context, selectedIDs) {
- var _entities = selectedIDs.map(function (id) {
- return context.graph().entity(id);
- });
-
- var _geometries = Object.assign({
- line: [],
- vertex: []
- }, utilArrayGroupBy(_entities, function (entity) {
- return entity.geometry(context.graph());
- }));
-
- var _vertex = _geometries.vertex.length && _geometries.vertex[0];
-
- 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);
- }) : [];
- }
-
- var _candidates = candidateWays();
-
- 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] : [];
- };
-
- operation.available = function () {
- return _geometries.vertex.length === 1 && _geometries.line.length <= 1 && !context.features().hasHiddenConnections(_vertex, context.graph());
- };
-
- operation.disabled = function () {
- if (_candidates.length === 0) {
- return 'not_eligible';
- } else if (_candidates.length > 1) {
- return 'multiple';
- }
-
- return false;
- };
-
- operation.tooltip = function () {
- var disable = operation.disabled();
- return disable ? _t('operations.continue.' + disable) : _t('operations.continue.description');
- };
-
- operation.annotation = function () {
- return _t('operations.continue.annotation.line');
- };
-
- 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';
- });
- }
-
- 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];
-
- if (!skip[entity.id] && entity.isComplete(graph)) {
- canCopy.push(entity.id);
- skip = getDescendants(entity.id, graph, skip);
- }
- }
-
- for (i = 0; i < selected.way.length; i++) {
- entity = selected.way[i];
-
- if (!skip[entity.id]) {
- canCopy.push(entity.id);
- skip = getDescendants(entity.id, graph, skip);
- }
- }
-
- for (i = 0; i < selected.node.length; i++) {
- entity = selected.node[i];
-
- if (!skip[entity.id]) {
- canCopy.push(entity.id);
- }
- }
-
- 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));
- } else {
- context.copyLonLat(null);
- }
- };
-
- function groupEntities(ids, graph) {
- var entities = ids.map(function (id) {
- return graph.entity(id);
- });
- return Object.assign({
- relation: [],
- way: [],
- node: []
- }, utilArrayGroupBy(entities, 'type'));
- }
-
- 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;
- });
- } 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);
- }
- }
-
- return descendants;
- }
-
- operation.available = function () {
- return getFilteredIdsToCopy().length > 0;
- };
-
- operation.disabled = function () {
- var extent = utilTotalExtent(getFilteredIdsToCopy(), context.graph());
-
- if (extent.percentContainedIn(context.map().extent()) < 0.8) {
- return 'too_large';
- }
-
- return false;
- };
-
- operation.availableForKeypress = function () {
- var selection = window.getSelection && window.getSelection(); // if the user has text selected then let them copy that, not the selected feature
-
- return !selection || !selection.toString();
- };
-
- operation.tooltip = function () {
- var disable = operation.disabled();
- return disable ? _t('operations.copy.' + disable, {
- n: selectedIDs.length
- }) : _t('operations.copy.description', {
- n: selectedIDs.length
- });
- };
-
- operation.annotation = function () {
- return _t('operations.copy.annotation', {
- n: selectedIDs.length
- });
- };
-
- var _point;
-
- operation.point = function (val) {
- _point = val;
- return operation;
- };
-
- operation.id = 'copy';
- operation.keys = [uiCmd('⌘C')];
- operation.title = _t('operations.copy.title');
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
- }
-
- 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);
- }
- });
-
- var _coords,
- _descriptionID = '',
- _annotationID = 'features';
-
- var _disconnectingVertexIds = [];
- var _disconnectingWayIds = [];
-
- if (_vertexIDs.length > 0) {
- // At the selected vertices, disconnect the selected ways, if any, else
- // disconnect all connected ways
- _disconnectingVertexIds = _vertexIDs;
-
- _vertexIDs.forEach(function (vertexID) {
- var action = actionDisconnect(vertexID);
-
- if (_wayIDs.length > 0) {
- var waysIDsForVertex = _wayIDs.filter(function (wayID) {
- var way = context.entity(wayID);
- return way.nodes.indexOf(vertexID) !== -1;
- });
-
- action.limitWays(waysIDsForVertex);
- }
-
- _actions.push(action);
-
- _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.';
-
- 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 nodes = utilGetAllNodes(_wayIDs, context.graph());
- _coords = nodes.map(function (n) {
- return n.loc;
- }); // actions for connected nodes shared by at least two selected ways
-
- var sharedActions = [];
- var sharedNodes = []; // actions for connected nodes
-
- var unsharedActions = [];
- var unsharedNodes = [];
- nodes.forEach(function (node) {
- var action = actionDisconnect(node.id).limitWays(_wayIDs);
-
- if (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;
- }
-
- 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;
- });
- _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 (_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 (_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';
- }
-
- return false;
-
- function someMissing() {
- if (context.inIntro()) return false;
- var osm = context.connection();
-
- if (osm) {
- var missing = _coords.filter(function (loc) {
- return !osm.isDataLoaded(loc);
- });
-
- if (missing.length) {
- missing.forEach(function (loc) {
- context.loadTileAtLoc(loc);
- });
- return true;
- }
- }
-
- return false;
- }
- };
-
- operation.tooltip = function () {
- var disable = operation.disabled();
-
- if (disable) {
- return _t('operations.disconnect.' + disable);
- }
-
- 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';
-
- function downgradeTypeForEntityIDs(entityIds) {
- var downgradeType;
- _affectedFeatureCount = 0;
-
- for (var i in entityIds) {
- var entityID = entityIds[i];
- var type = downgradeTypeForEntityID(entityID);
-
- if (type) {
- _affectedFeatureCount += 1;
-
- if (downgradeType && type !== downgradeType) {
- if (downgradeType !== 'generic' && type !== 'generic') {
- downgradeType = 'building_address';
- } else {
- downgradeType = 'generic';
- }
- } else {
- downgradeType = type;
- }
- }
- }
-
- return downgradeType;
- }
-
- function downgradeTypeForEntityID(entityID) {
- var graph = context.graph();
- var entity = graph.entity(entityID);
- var preset = _mainPresetIndex.match(entity, graph);
- if (!preset || preset.isFallback()) return null;
-
- if (entity.type === 'node' && preset.id !== 'address' && Object.keys(entity.tags).some(function (key) {
- return key.match(/^addr:.{1,}/);
- })) {
- return 'address';
- }
-
- var geometry = entity.geometry(graph);
-
- if (geometry === 'area' && entity.tags.building && !preset.tags.building) {
- return 'building';
- }
-
- if (geometry === 'vertex' && Object.keys(entity.tags).length) {
- return 'generic';
- }
-
- return null;
- }
-
- var buildingKeysToKeep = ['architect', 'building', 'height', 'layer', 'source', 'type', 'wheelchair'];
- var addressKeysToKeep = ['source'];
-
- 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
-
- for (var key in tags) {
- if (type === 'address' && addressKeysToKeep.indexOf(key) !== -1) continue;
-
- if (type === 'building') {
- if (buildingKeysToKeep.indexOf(key) !== -1 || key.match(/^building:.{1,}/) || key.match(/^roof:.{1,}/)) continue;
- }
-
- if (type !== 'generic') {
- if (key.match(/^addr:.{1,}/) || key.match(/^source:.{1,}/)) continue;
- }
-
- delete tags[key];
- }
-
- graph = actionChangeTags(entityID, tags)(graph);
- }
-
- return graph;
- }, operation.annotation());
- context.validator().validate(); // refresh the select mode to enable the delete operation
-
- context.enter(modeSelect(context, selectedIDs));
- };
-
- operation.available = function () {
- return _downgradeType;
- };
-
- operation.disabled = function () {
- if (selectedIDs.some(hasWikidataTag)) {
- return 'has_wikidata_tag';
- }
-
- return false;
-
- function hasWikidataTag(id) {
- var entity = context.entity(id);
- return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
- }
- };
-
- operation.tooltip = function () {
- var disable = operation.disabled();
- return disable ? _t('operations.downgrade.' + disable + '.' + _multi) : _t('operations.downgrade.description.' + _downgradeType);
- };
-
- operation.annotation = function () {
- var suffix;
-
- if (_downgradeType === 'building_address') {
- suffix = 'generic';
- } else {
- suffix = _downgradeType;
- }
-
- return _t('operations.downgrade.annotation.' + suffix, {
- n: _affectedFeatureCount
- });
- };
-
- operation.id = 'downgrade';
- operation.keys = [uiCmd('⌫')];
- operation.title = _t('operations.downgrade.title');
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
- }
-
- function operationExtract(context, selectedIDs) {
- var _amount = selectedIDs.length === 1 ? 'single' : 'multiple';
-
- var _geometries = utilArrayUniq(selectedIDs.map(function (entityID) {
- return context.graph().hasEntity(entityID) && context.graph().geometry(entityID);
- }).filter(Boolean));
-
- var _geometryID = _geometries.length === 1 ? _geometries[0] : 'feature';
-
- var _extent;
-
- 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;
- }
-
- _extent = _extent ? _extent.extend(entity.extent(graph)) : entity.extent(graph);
- return actionExtract(entityID);
- }).filter(Boolean);
-
- var operation = function operation() {
- var combinedAction = function combinedAction(graph) {
- _actions.forEach(function (action) {
- graph = action(graph);
- });
-
- return graph;
- };
-
- context.perform(combinedAction, operation.annotation()); // do the extract
-
- var extractedNodeIDs = _actions.map(function (action) {
- return action.getExtractedNodeID();
- });
-
- context.enter(modeSelect(context, extractedNodeIDs));
- };
-
- operation.available = function () {
- return _actions.length && selectedIDs.length === _actions.length;
- };
-
- 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';
- }
-
- return false;
- };
-
- operation.tooltip = function () {
- var disableReason = operation.disabled();
-
- if (disableReason) {
- return _t('operations.extract.' + disableReason + '.' + _amount);
- } else {
- return _t('operations.extract.description.' + _geometryID + '.' + _amount);
- }
- };
-
- operation.annotation = function () {
- return _t('operations.extract.annotation', {
- n: selectedIDs.length
- });
- };
-
- 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
-
- 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 operation = function operation() {
- if (operation.disabled()) return;
- context.perform(_action, operation.annotation());
- context.validator().validate();
- var resultIDs = selectedIDs.filter(context.hasEntity);
-
- 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));
- };
-
- operation.available = function () {
- return selectedIDs.length >= 2;
- };
-
- operation.disabled = function () {
- var actionDisabled = _action.disabled(context.graph());
-
- if (actionDisabled) return actionDisabled;
- var osm = context.connection();
-
- if (osm && _action.resultingWayNodesLength && _action.resultingWayNodesLength(context.graph()) > osm.maxWayNodes()) {
- return 'too_many_vertices';
- }
-
- return false;
- };
-
- operation.tooltip = function () {
- var disabled = operation.disabled();
-
- if (disabled) {
- if (disabled === 'restriction') {
- return _t('operations.merge.restriction', {
- relation: _mainPresetIndex.item('type/restriction').name()
- });
- }
-
- return _t('operations.merge.' + disabled);
- }
-
- return _t('operations.merge.description');
- };
-
- operation.annotation = function () {
- return _t('operations.merge.annotation', {
- n: selectedIDs.length
- });
- };
-
- operation.id = 'merge';
- operation.keys = [_t('operations.merge.key')];
- operation.title = _t('operations.merge.title');
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
- }
-
- function operationPaste(context) {
- var _pastePoint;
-
- 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);
- });
-
- for (var id in copies) {
- var oldEntity = oldGraph.entity(id);
- var newEntity = copies[id];
-
- extent._extend(oldEntity.extent(oldGraph)); // Exclude child nodes from newIDs if their parent way was also copied.
-
-
- 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
-
-
- 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
-
- context.replace(actionMove(newIDs, delta, projection), operation.annotation());
- context.enter(modeSelect(context, newIDs));
- };
-
- operation.point = function (val) {
- _pastePoint = val;
- return operation;
- };
-
- operation.available = function () {
- return context.mode().id === 'browse';
- };
-
- operation.disabled = function () {
- return !context.copyIDs().length;
- };
-
- operation.tooltip = function () {
- var oldGraph = context.copyGraph();
- var ids = context.copyIDs();
-
- if (!ids.length) {
- return _t('operations.paste.nothing_copied');
- }
-
- return _t('operations.paste.description', {
- feature: utilDisplayLabel(oldGraph.entity(ids[0]), oldGraph),
- n: ids.length
- });
- };
-
- operation.annotation = function () {
- var ids = context.copyIDs();
- return _t('operations.paste.annotation', {
- n: ids.length
- });
- };
-
- operation.id = 'paste';
- operation.keys = [uiCmd('⌘V')];
- operation.title = _t('operations.paste.title');
- return operation;
- }
-
- 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 actions(situation) {
- return selectedIDs.map(function (entityID) {
- var entity = context.hasEntity(entityID);
- if (!entity) return null;
-
- if (situation === 'toolbar') {
- if (entity.type === 'way' && !entity.isOneWay() && !entity.isSided()) return null;
- }
-
- 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 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';
- }
-
- operation.available = function (situation) {
- return actions(situation).length > 0;
- };
-
- operation.disabled = function () {
- return false;
- };
-
- operation.tooltip = function () {
- return _t('operations.reverse.description.' + reverseTypeID());
- };
-
- operation.annotation = function () {
- var acts = actions();
- return _t('operations.reverse.annotation.' + reverseTypeID(), {
- n: acts.length
- });
- };
-
- operation.id = 'reverse';
- operation.keys = [_t('operations.reverse.key')];
- operation.title = _t('operations.reverse.title');
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
- }
-
- function operationSplit(context, selectedIDs) {
- var _vertexIds = selectedIDs.filter(function (id) {
- return context.graph().geometry(id) === 'vertex';
- });
-
- var _selectedWayIds = selectedIDs.filter(function (id) {
- var entity = context.graph().hasEntity(id);
- return entity && entity.type === 'way';
- });
-
- var _isAvailable = _vertexIds.length > 0 && _vertexIds.length + _selectedWayIds.length === selectedIDs.length;
-
- var _action = actionSplit(_vertexIds);
-
- var _ways = [];
- var _geometry = 'feature';
- var _waysAmount = 'single';
-
- var _nodesAmount = _vertexIds.length === 1 ? 'single' : 'multiple';
-
- if (_isAvailable) {
- if (_selectedWayIds.length) _action.limitWays(_selectedWayIds);
- _ways = _action.ways(context.graph());
- var geometries = {};
-
- _ways.forEach(function (way) {
- geometries[way.geometry(context.graph())] = true;
- });
-
- if (Object.keys(geometries).length === 1) {
- _geometry = Object.keys(geometries)[0];
- }
-
- _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
-
- var idsToSelect = _vertexIds.concat(difference.extantIDs().filter(function (id) {
- // filter out relations that may have had member additions
- return context.entity(id).type === 'way';
- }));
-
- context.enter(modeSelect(context, idsToSelect));
- };
-
- operation.relatedEntityIds = function () {
- return _selectedWayIds.length ? [] : _ways.map(function (way) {
- return way.id;
- });
- };
-
- operation.available = function () {
- return _isAvailable;
- };
-
- operation.disabled = function () {
- var reason = _action.disabled(context.graph());
-
- if (reason) {
- return reason;
- } else if (selectedIDs.some(context.hasHiddenConnections)) {
- return 'connected_to_hidden';
- }
-
- return false;
- };
-
- 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
- });
- };
-
- operation.id = 'split';
- operation.keys = [_t('operations.split.key')];
- operation.title = _t('operations.split.title');
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
- }
-
- 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';
- });
-
- var _amount = (_wayIDs.length ? _wayIDs : _nodeIDs).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 _action = chooseAction();
-
- var _geometry;
-
- 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 = [];
-
- 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
- }
-
- startNodeIDs.push(entity.first());
- endNodeIDs.push(entity.last());
- } // Remove duplicate end/startNodeIDs (duplicate nodes cannot be at the line end)
-
-
- 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 (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
-
- if (_nodeIDs.length === 2 && (wayNodeIDs.indexOf(_nodeIDs[0]) === -1 || wayNodeIDs.indexOf(_nodeIDs[1]) === -1)) return null;
-
- if (_nodeIDs.length) {
- // If we're only straightenting between two points, we only need that extent visible
- _extent = utilTotalExtent(_nodeIDs, context.graph());
- }
-
- _geometry = 'line';
- return actionStraightenWay(selectedIDs, context.projection);
- }
-
- 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);
- };
-
- operation.disabled = function () {
- var reason = _action.disabled(context.graph());
-
- 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;
-
- function someMissing() {
- if (context.inIntro()) return false;
- var osm = context.connection();
-
- if (osm) {
- var missing = _coords.filter(function (loc) {
- return !osm.isDataLoaded(loc);
- });
-
- if (missing.length) {
- missing.forEach(function (loc) {
- context.loadTileAtLoc(loc);
- });
- return true;
- }
- }
-
- return false;
- }
- };
-
- 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
- });
- };
-
- 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
- });
-
- var _relatedParent;
-
- function modeSelect(context, selectedIDs) {
- var mode = {
- id: 'select',
- button: 'browse'
- };
- var keybinding = utilKeybinding('select');
-
- var _breatheBehavior = behaviorBreathe();
-
- var _modeDragNode = modeDragNode(context);
-
- var _selectBehavior;
-
- var _behaviors = [];
- var _operations = [];
- var _newFeature = false;
- var _follow = false;
-
- function singular() {
- if (selectedIDs && selectedIDs.length === 1) {
- return context.hasEntity(selectedIDs[0]);
- }
- }
-
- function selectedEntities() {
- return selectedIDs.map(function (id) {
- return context.hasEntity(id);
- }).filter(Boolean);
- }
-
- function checkSelectedIDs() {
- var ids = [];
-
- if (Array.isArray(selectedIDs)) {
- ids = selectedIDs.filter(function (id) {
- return context.hasEntity(id);
- });
- }
-
- 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;
- }
-
- selectedIDs = ids;
- return true;
- } // find the common parent ways for nextVertex, previousVertex
-
-
- function commonParents() {
- var graph = context.graph();
- var commonParents = [];
-
- for (var i = 0; i < selectedIDs.length; i++) {
- var entity = context.hasEntity(selectedIDs[i]);
-
- if (!entity || entity.geometry(graph) !== 'vertex') {
- return []; // selection includes some not vertices
- }
-
- var currParents = graph.parentWays(entity).map(function (w) {
- return w.id;
- });
-
- if (!commonParents.length) {
- commonParents = currParents;
- continue;
- }
-
- commonParents = utilArrayIntersection(commonParents, currParents);
-
- if (!commonParents.length) {
- return [];
- }
- }
-
- 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.
-
-
- if (parents.length === 1) {
- _relatedParent = parents[0]; // remember this parent for later
-
- return _relatedParent;
- }
-
- if (parents.indexOf(_relatedParent) !== -1) {
- return _relatedParent; // prefer the previously seen parent
- }
-
- return parents[0];
- }
-
- mode.selectedIDs = function (val) {
- if (!arguments.length) return selectedIDs;
- selectedIDs = val;
- return mode;
- };
-
- mode.zoomToSelected = function () {
- context.map().zoomToEase(selectedEntities());
- };
-
- mode.newFeature = function (val) {
- if (!arguments.length) return _newFeature;
- _newFeature = val;
- return mode;
- };
-
- mode.selectBehavior = function (val) {
- if (!arguments.length) return _selectBehavior;
- _selectBehavior = val;
- return mode;
- };
-
- mode.follow = function (val) {
- if (!arguments.length) return _follow;
- _follow = val;
- return mode;
- };
-
- function loadOperations() {
- _operations.forEach(function (operation) {
- if (operation.behavior) {
- context.uninstall(operation.behavior);
- }
- });
-
- _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();
- });
-
- _operations.forEach(function (operation) {
- if (operation.behavior) {
- context.install(operation.behavior);
- }
- }); // remove any displayed menu
-
-
- context.ui().closeEditMenu();
- }
-
- mode.operations = function () {
- return _operations;
- };
-
- mode.enter = function () {
- if (!checkSelectedIDs()) return;
- context.features().forceVisible(selectedIDs);
-
- _modeDragNode.restoreSelectedIDs(selectedIDs);
-
- loadOperations();
-
- 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];
- }
-
- _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'], 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
-
- selectElements();
- }).on('undone.select', checkSelectedIDs).on('redone.select', checkSelectedIDs);
- context.map().on('drawn.select', selectElements).on('crossEditableZoom.select', function () {
- selectElements();
-
- _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);
-
- 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
-
- _follow = false;
- }
-
- function nudgeSelection(delta) {
- return function () {
- // prevent nudging during low zoom selection
- if (!context.map().withinEditableZoom()) return;
- var moveOp = operationMove(context, selectedIDs);
-
- 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();
- }
- };
- }
-
- 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
-
- 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.
-
- 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 false;
-
- 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 someMissing() {
- if (context.inIntro()) return false;
- var osm = context.connection();
-
- if (osm) {
- var missing = nodes.filter(function (n) {
- return !osm.isDataLoaded(n.loc);
- });
-
- 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());
- }
- }
-
- 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 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 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'));
- }
- }
-
- 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();
-
- if (_relatedParent) {
- surface.selectAll(utilEntitySelector([_relatedParent])).classed('related', true);
- }
-
- 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);
- }
- }
-
- function esc() {
- if (context.container().select('.combobox').size()) return;
- context.enter(modeBrowse(context));
- }
-
- function firstVertex(d3_event) {
- d3_event.preventDefault();
- var entity = singular();
- var parent = singularParent();
- var way;
-
- if (entity && entity.type === 'way') {
- way = entity;
- } else if (parent) {
- way = context.entity(parent);
- }
-
- 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;
-
- if (entity && entity.type === 'way') {
- way = entity;
- } else if (parent) {
- way = context.entity(parent);
- }
-
- if (way) {
- context.enter(modeSelect(context, [way.last()]).follow(true));
- }
- }
-
- 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;
-
- if (curr > 0) {
- index = curr - 1;
- } else if (way.isClosed()) {
- index = length - 2;
- }
-
- if (index !== -1) {
- context.enter(modeSelect(context, [way.nodes[index]]).follow(true));
- }
- }
-
- 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 (curr < length - 1) {
- index = curr + 1;
- } else if (way.isClosed()) {
- index = 0;
- }
-
- 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 (index < 0 || index > parents.length - 2) {
- _relatedParent = parents[0];
- } else {
- _relatedParent = parents[index + 1];
- }
-
- var surface = context.surface();
- surface.selectAll('.related').classed('related', false);
-
- if (_relatedParent) {
- surface.selectAll(utilEntitySelector([_relatedParent])).classed('related', true);
- }
- }
- };
-
- mode.exit = function () {
- _newFeature = false;
-
- _operations.forEach(function (operation) {
- if (operation.behavior) {
- context.uninstall(operation.behavior);
- }
- });
-
- _operations = [];
-
- _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();
-
- 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'));
- }
- };
-
- return mode;
- }
-
- function uiLasso(context) {
- var group, polygon;
- lasso.coordinates = [];
-
- 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));
- }
-
- function draw() {
- if (polygon) {
- polygon.data([lasso.coordinates]).attr('d', function (d) {
- return 'M' + d.join(' L') + ' Z';
- });
- }
- }
-
- lasso.extent = function () {
- return lasso.coordinates.reduce(function (extent, point) {
- return extent.extend(geoExtent(point));
- }, geoExtent());
- };
-
- lasso.p = function (_) {
- if (!arguments.length) return lasso;
- lasso.coordinates.push(_);
- draw();
- return lasso;
- };
-
- lasso.close = function () {
- if (group) {
- group.call(uiToggle(false, function () {
- select(this).remove();
- }));
- }
-
- context.container().classed('lasso', false);
- };
-
- return lasso;
- }
-
- function behaviorLasso(context) {
- // use pointer events on supported platforms; fallback to mouse events
- var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
-
- var behavior = function behavior(selection) {
- var lasso;
-
- 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();
- }
- }
-
- function pointermove() {
- if (!lasso) {
- lasso = uiLasso(context);
- context.surface().call(lasso);
- }
-
- 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])]];
- }
-
- function lassoed() {
- if (!lasso) return [];
- var graph = context.graph();
- var limitToNodes;
-
- 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 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
-
- 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);
-
- 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
-
-
- 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();
-
- if (ids.length) {
- context.enter(modeSelect(context, ids));
- }
- }
-
- selection.on(_pointerPrefix + 'down.lasso', pointerdown);
- };
-
- behavior.off = function (selection) {
- selection.on(_pointerPrefix + 'down.lasso', null);
- };
-
- return behavior;
- }
-
- function modeBrowse(context) {
- var mode = {
- button: 'browse',
- id: 'browse',
- title: _t('modes.browse.title'),
- description: _t('modes.browse.description')
- };
- var sidebar;
-
- var _selectBehavior;
-
- var _behaviors = [];
-
- mode.selectBehavior = function (val) {
- if (!arguments.length) return _selectBehavior;
- _selectBehavior = val;
- return mode;
- };
-
- 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];
- }
-
- _behaviors.forEach(context.install); // Get focus on the body.
-
-
- if (document.activeElement && document.activeElement.blur) {
- document.activeElement.blur();
- }
-
- if (sidebar) {
- context.ui().sidebar.show(sidebar);
- } else {
- context.ui().sidebar.select(null);
- }
- };
-
- mode.exit = function () {
- context.ui().sidebar.hover.cancel();
-
- _behaviors.forEach(context.uninstall);
-
- if (sidebar) {
- context.ui().sidebar.hide();
- }
- };
-
- mode.sidebar = function (_) {
- if (!arguments.length) return sidebar;
- sidebar = _;
- return mode;
- };
-
- mode.operations = function () {
- return [operationPaste(context)];
- };
-
- return mode;
- }
-
- function behaviorAddWay(context) {
- var dispatch$1 = dispatch('start', 'startFromWay', 'startFromNode');
- var draw = behaviorDraw(context);
-
- 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);
- }
-
- behavior.off = function (surface) {
- surface.call(draw.off);
- };
-
- behavior.cancel = function () {
- window.setTimeout(function () {
- context.map().dblclickZoomEnable(true);
- }, 1000);
- context.enter(modeBrowse(context));
- };
-
- return utilRebind(behavior, dispatch$1, 'on');
- }
-
- function behaviorHash(context) {
- // cached window.location.hash
- var _cachedHash = null; // allowable latitude range
-
- var _latitudeLimit = 90 - 1e-8;
-
- 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);
- });
-
- if (selected.length) {
- newParams.id = selected.join(',');
- }
-
- newParams.map = zoom.toFixed(2) + '/' + center[1].toFixed(precision) + '/' + center[0].toFixed(precision);
- return Object.assign(oldParams, newParams);
- }
-
- 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);
- });
-
- if (selected.length) {
- var firstLabel = utilDisplayLabel(context.entity(selected[0]), context.graph());
-
- if (selected.length > 1) {
- contextual = _t('title.labeled_and_more', {
- labeled: firstLabel,
- count: selected.length - 1
- });
- } else {
- contextual = firstLabel;
- }
-
- titleID = 'context';
- }
-
- if (includeChangeCount) {
- changeCount = context.history().difference().summary().length;
-
- if (changeCount > 0) {
- titleID = contextual ? 'changes_context' : 'changes';
- }
- }
-
- if (titleID) {
- return _t('title.format.' + titleID, {
- changes: changeCount,
- base: baseTitle,
- context: contextual
- });
- }
-
- return baseTitle;
- }
-
- function updateTitle(includeChangeCount) {
- if (!context.setsDocumentTitle()) return;
- var newTitle = computedTitle(includeChangeCount);
-
- if (document.title !== newTitle) {
- document.title = newTitle;
- }
- }
-
- 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 */
- );
- }
- }
-
- var _throttledUpdate = throttle(updateHashIfNeeded, 500);
-
- 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);
-
- 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]);
-
- if (q.id && mode) {
- var ids = q.id.split(',').filter(function (id) {
- return context.hasEntity(id);
- });
-
- if (ids.length && (mode.id === 'browse' || mode.id === 'select' && !utilArrayIdentical(mode.selectedIDs(), ids))) {
- context.enter(modeSelect(context, ids));
- return;
- }
- }
-
- 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
-
- if (mode && mode.id.match(/^draw/) !== null && dist > maxdist) {
- context.enter(modeBrowse(context));
- return;
- }
- }
- }
-
- 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 (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 (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 behavior;
- }
-
- /*
- 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.
- */
-
- function coreDifference(base, head) {
- var _changes = {};
- var _didChange = {}; // 'addition', 'deletion', 'geometry', 'properties'
-
- var _diff = {};
-
- function checkEntityID(id) {
- var h = head.entities[id];
- var b = base.entities[id];
- if (h === b) return;
- if (_changes[id]) return;
-
- if (!h && b) {
- _changes[id] = {
- base: b,
- head: h
- };
- _didChange.deletion = true;
- return;
- }
-
- 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;
- }
-
- if (h.loc && b.loc && !geoVecEqual(h.loc, b.loc)) {
- _changes[id] = {
- base: b,
- head: h
- };
- _didChange.geometry = true;
- }
-
- 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;
- }
- }
- }
-
- 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)));
-
- for (var i = 0; i < ids.length; i++) {
- checkEntityID(ids[i]);
- }
- }
-
- load();
-
- _diff.length = function length() {
- return Object.keys(_changes).length;
- };
-
- _diff.changes = function changes() {
- return _changes;
- };
-
- _diff.didChange = _didChange; // pass true to include affected relation members
-
- _diff.extantIDs = function extantIDs(includeRelMembers) {
- var result = new Set();
- Object.keys(_changes).forEach(function (id) {
- if (_changes[id].head) {
- result.add(id);
- }
-
- var h = _changes[id].head;
- var b = _changes[id].base;
- var entity = h || b;
-
- 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);
- };
-
- _diff.modified = function modified() {
- var result = [];
- Object.values(_changes).forEach(function (change) {
- if (change.base && change.head) {
- result.push(change.head);
- }
- });
- return result;
- };
-
- _diff.created = function created() {
- var result = [];
- Object.values(_changes).forEach(function (change) {
- if (!change.base && change.head) {
- result.push(change.head);
- }
- });
- return result;
- };
-
- _diff.deleted = function deleted() {
- var result = [];
- Object.values(_changes).forEach(function (change) {
- if (change.base && !change.head) {
- result.push(change.base);
- }
- });
- return result;
- };
-
- _diff.summary = function summary() {
- var relevant = {};
- var keys = Object.keys(_changes);
-
- for (var i = 0; i < keys.length; i++) {
- var change = _changes[keys[i]];
-
- 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);
-
- 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 Object.values(relevant);
-
- function addEntity(entity, graph, changeType) {
- relevant[entity.id] = {
- entity: entity,
- graph: graph,
- changeType: changeType
- };
- }
-
- function addParents(entity) {
- var parents = head.parentWays(entity);
-
- 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`)
-
-
- _diff.complete = function complete(extent) {
- var result = {};
- var id, change;
-
- 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;
-
- if (entity.type === 'way') {
- var nh = h ? h.nodes : [];
- var nb = b ? b.nodes : [];
- var diff;
- diff = utilArrayDifference(nh, nb);
-
- for (i = 0; i < diff.length; i++) {
- result[diff[i]] = head.hasEntity(diff[i]);
- }
-
- 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);
- }
- }
- };
-
- return _diff;
- }
-
- function coreTree(head) {
- // tree for entities
- var _rtree = new RBush();
-
- var _bboxes = {}; // maintain a separate tree for granular way segments
-
- var _segmentsRTree = new RBush();
-
- var _segmentsBBoxes = {};
- var _segmentsByWayId = {};
- var tree = {};
-
- function entityBBox(entity) {
- var bbox = entity.extent(head).bbox();
- bbox.id = entity.id;
- _bboxes[entity.id] = bbox;
- return bbox;
- }
-
- function segmentBBox(segment) {
- var extent = segment.extent(head); // extent can be null if the node entities aren't in the graph for some reason
-
- if (!extent) return null;
- var bbox = extent.bbox();
- bbox.segment = segment;
- _segmentsBBoxes[segment.id] = bbox;
- return bbox;
- }
-
- function removeEntity(entity) {
- _rtree.remove(_bboxes[entity.id]);
-
- delete _bboxes[entity.id];
-
- if (_segmentsByWayId[entity.id]) {
- _segmentsByWayId[entity.id].forEach(function (segment) {
- _segmentsRTree.remove(_segmentsBBoxes[segment.id]);
-
- delete _segmentsBBoxes[segment.id];
- });
-
- delete _segmentsByWayId[entity.id];
- }
- }
-
- 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
-
- _segmentsByWayId[entity.id] = entitySegments;
- segments = segments.concat(entitySegments);
- }
- });
- if (segments.length) _segmentsRTree.load(segments.map(segmentBBox).filter(Boolean));
- }
-
- 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 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);
- }
-
- 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 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 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 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');
-
- 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 coreHistory(context) {
- var dispatch$1 = dispatch('reset', 'change', 'merge', 'restore', 'undone', 'redone');
-
- var _lock = utilSessionMutex('lock'); // restorable if iD not open in another window/tab and a saved history exists in localStorage
-
-
- var _hasUnresolvedRestorableChanges = _lock.lock() && !!corePreferences(getKey('saved_history'));
-
- var duration = 150;
- var _imageryUsed = [];
- var _photoOverlaysUsed = [];
- var _checkpoints = {};
-
- var _pausedGraph;
-
- var _stack;
-
- var _index;
-
- var _tree; // internal _act, accepts list of actions and eased time
-
-
- function _act(actions, t) {
- actions = Array.prototype.slice.call(actions);
- var annotation;
-
- if (typeof actions[actions.length - 1] !== 'function') {
- annotation = actions.pop();
- }
-
- var graph = _stack[_index].graph;
-
- for (var i = 0; i < actions.length; i++) {
- graph = actions[i](graph, t);
- }
-
- return {
- graph: graph,
- annotation: annotation,
- imageryUsed: _imageryUsed,
- photoOverlaysUsed: _photoOverlaysUsed,
- transform: context.projection.transform(),
- selectedIDs: context.selectedIDs()
- };
- } // internal _perform with eased time
-
-
- function _perform(args, t) {
- var previous = _stack[_index].graph;
- _stack = _stack.slice(0, _index + 1);
-
- var actionResult = _act(args, t);
-
- _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$1.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;
- });
-
- _stack[0].graph.rebase(entities, stack, false);
-
- _tree.rebase(entities, false);
-
- 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;
- }
-
- 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;
-
- if (isNaN(+n) || +n < 0) {
- n = 1;
- }
-
- while (n-- > 0 && _index > 0) {
- _index--;
-
- _stack.pop();
- }
-
- 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;
-
- while (_index > 0) {
- _index--;
- if (_stack[_index].annotation) break;
- }
-
- 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;
-
- while (tryIndex < _stack.length - 1) {
- tryIndex++;
-
- if (_stack[tryIndex].annotation) {
- _index = tryIndex;
- dispatch$1.call('redone', this, _stack[_index], previousStack);
- break;
- }
- }
-
- 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;
-
- while (i >= 0) {
- if (_stack[i].annotation) return _stack[i].annotation;
- i--;
- }
- },
- redoAnnotation: function redoAnnotation() {
- var i = _index + 1;
-
- 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 (action) {
- head = action(head);
- }
-
- 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();
-
- _stack.slice(1, _index + 1).forEach(function (state) {
- if (state.photoOverlaysUsed && Array.isArray(state.photoOverlaysUsed)) {
- state.photoOverlaysUsed.forEach(function (photoOverlay) {
- s.add(photoOverlay);
- });
- }
- });
-
- 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 = {};
- }
-
- 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..
-
- Object.keys(graph.entities).forEach(function (id) {
- var entity = graph.entities[id];
-
- if (entity) {
- var copy = copyIntroEntity(entity);
- baseEntities[copy.id] = copy;
- } else {
- delete baseEntities[id];
- }
- }); // swap temporary for permanent ids..
-
- Object.values(baseEntities).forEach(function (entity) {
- if (Array.isArray(entity.nodes)) {
- entity.nodes = entity.nodes.map(function (node) {
- return permIDs[node] || node;
- });
- }
-
- 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
- });
-
- 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 (copy.tags && !Object.keys(copy.tags)) {
- delete copy.tags;
- }
-
- if (Array.isArray(copy.loc)) {
- copy.loc[0] = +copy.loc[0].toFixed(6);
- copy.loc[1] = +copy.loc[1].toFixed(6);
- }
-
- var match = source.id.match(/([nrw])-\d*/); // temporary id
-
- if (match !== null) {
- var nrw = match[1];
- var permID;
-
- do {
- permID = nrw + ++nextID[nrw];
- } while (baseEntities.hasOwnProperty(permID));
-
- copy.id = permIDs[source.id] = permID;
- }
-
- return copy;
- }
- },
- toJSON: function toJSON() {
- if (!this.hasChanges()) return;
- var allEntities = {};
- var baseEntities = {};
- var base = _stack[0];
-
- var s = _stack.map(function (i) {
- var modified = [];
- var deleted = [];
- Object.keys(i.graph.entities).forEach(function (id) {
- var entity = i.graph.entities[id];
-
- 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.
-
-
- if (id in base.graph.entities) {
- baseEntities[id] = base.graph.entities[id];
- }
-
- 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
-
-
- var baseParents = base.graph._parentWays[id];
-
- 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;
- });
-
- 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 (h.version === 2 || h.version === 3) {
- var allEntities = {};
- h.entities.forEach(function (entity) {
- allEntities[osmEntity.key(entity)] = osmEntity(entity);
- });
-
- 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);
- });
-
- var stack = _stack.map(function (state) {
- return state.graph;
- });
-
- _stack[0].graph.rebase(baseEntities, stack, true);
-
- _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
-
-
- 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);
- });
-
- if (missing.length && osm) {
- loadComplete = false;
- context.map().redrawEnable(false);
- var loading = uiLoading(context).blocking(true);
- context.container().call(loading);
-
- var childNodesLoaded = function childNodesLoaded(err, result) {
- if (!err) {
- var visibleGroups = utilArrayGroupBy(result.data, 'visible');
- var visibles = visibleGroups["true"] || []; // alive nodes
-
- var invisibles = visibleGroups["false"] || []; // deleted nodes
-
- if (visibles.length) {
- var visibleIDs = visibles.map(function (entity) {
- return entity.id;
- });
-
- var stack = _stack.map(function (state) {
- return state.graph;
- });
-
- missing = utilArrayDifference(missing, visibleIDs);
-
- _stack[0].graph.rebase(visibles, stack, true);
-
- _tree.rebase(visibles, true);
- } // fetch older versions of nodes that were deleted..
-
-
- invisibles.forEach(function (entity) {
- osm.loadEntityVersion(entity.id, +entity.version - 1, childNodesLoaded);
- });
- }
-
- if (err || !missing.length) {
- loading.close();
- context.map().redrawEnable(true);
- dispatch$1.call('change');
- dispatch$1.call('restore', this);
- }
- };
-
- osm.loadMultiple(missing, childNodesLoaded);
- }
- }
- }
-
- _stack = h.stack.map(function (d) {
- var entities = {},
- entity;
-
- if (d.modified) {
- d.modified.forEach(function (key) {
- entity = allEntities[key];
- entities[entity.id] = entity;
- });
- }
-
- 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 = {};
-
- for (var i in d.entities) {
- var entity = d.entities[i];
- entities[i] = entity === 'undefined' ? undefined : osmEntity(entity);
- }
-
- d.graph = coreGraph(_stack[0].graph).load(entities);
- return d;
- });
- }
-
- var transform = _stack[_index].transform;
-
- if (transform) {
- context.map().transformEase(transform, 0); // 0 = immediate, no easing
- }
-
- if (loadComplete) {
- dispatch$1.call('change');
- dispatch$1.call('restore', this);
- }
-
- 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();
-
- if (_lock.locked()) {
- _hasUnresolvedRestorableChanges = false;
- corePreferences(getKey('saved_history'), null); // clear the changeset metadata associated with the saved history
-
- corePreferences('comment', null);
- corePreferences('hashtags', null);
- corePreferences('source', null);
- }
-
- 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');
- }
-
- /**
- * Look for roads that can be connected to other roads with a short extension
- */
-
- 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 isHighway(entity) {
- return entity.type === 'way' && osmRoutableHighwayTagValues[entity.tags.highway];
- }
-
- 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]);
-
- 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;
-
- 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');
-
- var _this$issue$entityIds = _slicedToArray(this.issue.entityIds, 3),
- endNodeId = _this$issue$entityIds[1],
- crossWayId = _this$issue$entityIds[2];
-
- 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 (nearEndNodes.length > 0) {
- var collinear = findSmallJoinAngle(midNode, endNode, nearEndNodes);
-
- if (collinear) {
- context.perform(actionMergeNodes([collinear.id, endNode.id], collinear.loc), annotation);
- return;
- }
- }
-
- 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 (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]);
-
- 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'));
- }
- }));
- }
-
- return fixes;
- }
-
- 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'));
- }
-
- function isExtendableCandidate(node, way) {
- // can not accurately test vertices on tiles not downloaded from osm - #5938
- var osm = services.osm;
-
- if (osm && !osm.isDataLoaded(node.loc)) {
- return false;
- }
-
- if (isTaggedAsNotContinuing(node) || graph.parentWays(node).length !== 1) {
- return false;
- }
-
- var occurrences = 0;
-
- for (var index in way.nodes) {
- if (way.nodes[index] === node.id) {
- occurrences += 1;
-
- if (occurrences > 1) {
- return false;
- }
- }
- }
-
- return true;
- }
-
- 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
-
- testNodes[nodeIndex] = testNodes[nodeIndex].move(connectionInfo.cross_loc); // don't flag issue if connecting the ways would cause self-intersection
-
- if (geoHasSelfIntersections(testNodes, nodeID)) return;
- results.push(connectionInfo);
- });
- return results;
- }
-
- 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;
- });
- }
-
- 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);
-
- if (diff < minAngle) {
- joinTo = endNode;
- minAngle = diff;
- }
- });
- /* Threshold set by considering right angle triangle
- based on node joining threshold and extension distance */
-
- if (minAngle <= SIG_ANGLE_TH) return joinTo;
- return null;
- }
-
- function hasTag(tags, key) {
- return tags[key] !== undefined && tags[key] !== 'no';
- }
-
- 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
-
- 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 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 canConnectByExtend(way, endNodeIdx) {
- var tipNid = way.nodes[endNodeIdx]; // the 'tip' node for extension point
-
- var midNid = endNodeIdx === 0 ? way.nodes[1] : way.nodes[way.nodes.length - 2]; // the other node of the edge
-
- 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
-
- 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
-
- var segmentInfos = tree.waySegments(queryExtent, graph);
-
- 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;
- }
- };
-
- validation.type = type;
- return validation;
- }
-
- 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);
- }
-
- return [];
-
- function getIssuesForNode(node) {
- var parentWays = graph.parentWays(node);
-
- if (parentWays.length) {
- return getIssuesForVertex(node, parentWays);
- } else {
- return getIssuesForDetachedPoint(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);
-
- for (var i in parentRelations) {
- var relation = parentRelations[i];
- if (relation.tags.type === 'boundary') return 'boundary';
-
- 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';
- }
- }
-
- return 'other';
- }
-
- 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;
- }
-
- function getIssuesForWay(way) {
- if (!shouldCheckWay(way)) return [];
- var issues = [],
- nodes = graph.childNodes(way);
-
- 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);
- }
-
- for (var i = 0; i < parentWays.length; i++) {
- var parentWay = parentWays[i];
- if (!shouldCheckWay(parentWay)) continue;
- var lastIndex = parentWay.nodes.length - 1;
-
- 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 (j !== lastIndex) {
- if (parentWay.nodes[j + 1] === node.id) {
- checkForCloseness(graph.entity(parentWay.nodes[j]), node, parentWay);
- }
- }
- }
- }
-
- return issues;
- }
-
- 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 (wayType === 'boundary') return 0; // expect some features to be mapped with higher levels of detail
-
- if (wayType === 'indoor') return 0.01;
- if (wayType === 'building') return 0.05;
- if (wayType === 'path') return 0.1;
- return 0.2;
- }
-
- 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;
-
- 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')
- })];
- }
- }));
- }
- }
-
- return issues;
-
- 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);
- }
- }
-
- function getWayIssueIfAny(node1, node2, way) {
- if (node1.id === node2.id || node1.hasInterestingTags() && node2.hasInterestingTags()) {
- return null;
- }
-
- 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;
- }
-
- 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);
- }
- }
- };
-
- validation.type = type;
- return validation;
- }
-
- 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);
-
- for (var i = 0; i < parentRels.length; i++) {
- var rel = parentRels[i];
-
- if (getFeatureType(rel, graph) !== null) {
- return rel;
- }
- }
- }
-
- return way;
- }
-
- function hasTag(tags, key) {
- return tags[key] !== undefined && tags[key] !== 'no';
- }
-
- function taggedAsIndoor(tags) {
- return hasTag(tags, 'indoor') || hasTag(tags, 'level') || tags.highway === 'corridor';
- }
-
- function allowsBridge(featureType) {
- return featureType === 'highway' || featureType === 'railway' || featureType === 'waterway';
- }
-
- function allowsTunnel(featureType) {
- return featureType === 'highway' || featureType === 'railway' || featureType === 'waterway';
- } // discard
-
-
- 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;
- }
-
- function isLegitCrossing(tags1, featureType1, tags2, featureType2) {
- // assume 0 by default
- var level1 = tags1.level || '0';
- var level2 = tags2.level || '0';
-
- 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 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 (featureType1 === featureType2) {
- if (featureType1 === 'highway') {
- var entity1IsPath = osmPathHighwayTagValues[entity1.tags.highway];
- var entity2IsPath = osmPathHighwayTagValues[entity2.tags.highway];
-
- if ((entity1IsPath || entity2IsPath) && entity1IsPath !== entity2IsPath) {
- // one feature is a path but not both
- var roadFeature = entity1IsPath ? entity2 : entity1;
-
- if (nonCrossingHighways[roadFeature.tags.highway]) {
- // don't mark path connections with certain roads as crossings
- return {};
- }
-
- var pathFeature = entity1IsPath ? entity1 : entity2;
-
- 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
-
-
- return bothLines ? {
- highway: 'crossing'
- } : {};
- }
-
- return {};
- }
-
- 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
-
- 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 {
- railway: 'level_crossing'
- };
- }
- }
-
- 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;
-
- if (highwaysDisallowingFords[entity1.tags.highway] || highwaysDisallowingFords[entity2.tags.highway]) {
- // do not allow fords on major highways
- return null;
- }
-
- return bothLines ? {
- ford: 'yes'
- } : {};
- }
- }
- }
-
- return null;
- }
-
- 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 = {};
-
- 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
-
- segmentInfos = tree.waySegments(extent, graph);
-
- for (j = 0; j < segmentInfos.length; j++) {
- segment2Info = segmentInfos[j]; // don't check for self-intersection in this validation
-
- if (segment2Info.wayId === way1.id) continue; // skip if this way was already checked and only one issue is needed
-
- if (checkedSingleCrossingWays[segment2Info.wayId]) continue; // mark this way as checked even if there are no crossings
-
- comparedWays[segment2Info.wayId] = true;
- way2 = graph.hasEntity(segment2Info.wayId);
- if (!way2) continue;
- taggedFeature2 = getFeatureWithFeatureTypeTagsForWay(way2, graph); // only check crossing highway, waterway, building, and railway
-
- way2FeatureType = getFeatureType(taggedFeature2, graph);
-
- if (way2FeatureType === null || isLegitCrossing(taggedFeature1.tags, way1FeatureType, taggedFeature2.tags, way2FeatureType)) {
- continue;
- } // create only one issue for building crossings
-
-
- oneOnly = way1FeatureType === 'building' || way2FeatureType === 'building';
- nAId = segment2Info.nodes[0];
- nBId = segment2Info.nodes[1];
-
- if (nAId === n1.id || nAId === n2.id || nBId === n1.id || nBId === n2.id) {
- // n1 or n2 is a connection node; skip
- continue;
- }
-
- 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);
-
- 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
- });
-
- if (oneOnly) {
- checkedSingleCrossingWays[way2.id] = true;
- break;
- }
- }
- }
- }
-
- return edgeCrossInfos;
- }
-
- function waysToCheck(entity, graph) {
- var featureType = getFeatureType(entity, graph);
- if (!featureType) return [];
-
- 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);
- }
- }
-
- return array;
- }, []);
- }
-
- 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
-
- var wayIndex, crossingIndex, crossings;
-
- for (wayIndex in ways) {
- crossings = findCrossingsByWay(ways[wayIndex], graph, tree);
-
- for (crossingIndex in crossings) {
- issues.push(createIssue(crossings[crossingIndex], graph));
- }
- }
-
- return issues;
- };
-
- 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;
-
- if (type1 === type2) {
- return utilDisplayLabel(way1Info.way, graph) > utilDisplayLabel(way2Info.way, graph);
- } else if (type1 === 'waterway') {
- return true;
- } else if (type2 === 'waterway') {
- return false;
- }
-
- 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';
- }
-
- 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 = [];
-
- if (connectionTags) {
- fixes.push(makeConnectWaysFix(this.data.connectionTags));
- }
-
- 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 skipTunnelFix = otherFeatureType === 'waterway' && selectedFeatureType !== 'waterway';
-
- if (allowsTunnel(selectedFeatureType) && !skipTunnelFix) {
- fixes.push(makeAddBridgeOrTunnelFix('add_a_tunnel', 'temaki-tunnel', 'tunnel'));
- }
- } // repositioning the features is always an option
-
-
- fixes.push(new validationIssueFix({
- icon: 'iD-operation-move',
- title: _t.html('issues.fix.reposition_features.title')
- }));
- return fixes;
- }
- });
-
- 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;
-
- 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;
-
- 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
-
- 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();
- }
-
- 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;
- }
-
- 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
-
- structLengthMeters += 4; // clamp the length to a reasonable range
-
- structLengthMeters = Math.min(Math.max(structLengthMeters, 4), 50);
-
- function geomToProj(geoPoint) {
- return [geoLonToMeters(geoPoint[0], geoPoint[1]), geoLatToMeters(geoPoint[1])];
- }
-
- function projToGeom(projPoint) {
- var lat = geoMetersToLat(projPoint[1]);
- return [geoMetersToLon(projPoint[0], lat), lat];
- }
-
- 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]);
- }
-
- var endpointLocGetter1 = function endpointLocGetter1(lengthMeters) {
- return locSphericalDistanceFromCrossingLoc(projectedAngle, lengthMeters);
- };
-
- var endpointLocGetter2 = function endpointLocGetter2(lengthMeters) {
- return locSphericalDistanceFromCrossingLoc(projectedAngle + Math.PI, lengthMeters);
- }; // avoid creating very short edges from splitting too close to another node
-
-
- var minEdgeLengthMeters = 0.55; // decide where to bound the structure along the way, splitting as necessary
-
- 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
-
- 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;
- }
- }
- });
- });
-
- 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
-
-
- if (!newNode) newNode = endNode;
- var splitAction = actionSplit([newNode.id]).limitWays(resultWayIDs); // only split selected or created ways
- // do the split
-
- graph = splitAction(graph);
-
- if (splitAction.getCreatedWayIDs().length) {
- resultWayIDs.push(splitAction.getCreatedWayIDs()[0]);
- }
-
- return newNode;
- }
-
- 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 (bridgeOrTunnel === 'bridge') {
- tags.bridge = 'yes';
- tags.layer = '1';
- } else {
- var tunnelValue = 'yes';
-
- 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
-
-
- graph = actionChangeTags(structureWay.id, tags)(graph);
- return graph;
- };
-
- context.perform(action, _t('issues.fix.' + fixTitleID + '.annotation'));
- context.enter(modeSelect(context, resultWayIDs));
- }
- });
- }
-
- function makeConnectWaysFix(connectionTags) {
- var fixTitleID = 'connect_features';
-
- if (connectionTags.ford) {
- fixTitleID = 'connect_using_ford';
- }
-
- 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
-
- 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);
- }
- });
-
- if (nodesToMerge.length > 1) {
- // if we're using nearby nodes, merge them with the new node
- graph = actionMergeNodes(nodesToMerge, loc)(graph);
- }
-
- return graph;
- }, _t('issues.fix.connect_crossing_features.annotation'));
- }
- });
- }
-
- 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
-
- var layer = tags.layer && Number(tags.layer);
-
- if (layer && !isNaN(layer)) {
- if (higherOrLower === 'higher') {
- layer += 1;
- } else {
- layer -= 1;
- }
- } else {
- if (higherOrLower === 'higher') {
- layer = 1;
- } else {
- layer = -1;
- }
- }
-
- tags.layer = layer.toString();
- context.perform(actionChangeTags(entity.id, tags), _t('operations.change_tags.annotation'));
- }
- });
- }
-
- validation.type = type;
- return validation;
- }
-
- function validationDisconnectedWay() {
- var type = 'disconnected_way';
-
- function isTaggedAsHighway(entity) {
- return osmRoutableHighwayTagValues[entity.tags.highway];
- }
-
- 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
- })];
-
- function makeFixes(context) {
- var fixes = [];
- var singleEntity = this.entityIds.length === 1 && context.hasEntity(this.entityIds[0]);
-
- 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 (!fixes.length) {
- fixes.push(new validationIssueFix({
- title: _t.html('issues.fix.connect_feature.title')
- }));
- }
-
- 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 showReference(selection) {
- selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.disconnected_way.routable.reference'));
- }
-
- function routingIslandForEntity(entity) {
- var routingIsland = new Set(); // the interconnected routable features
-
- 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);
- }
- });
- }
-
- 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;
- }
-
- while (waysToCheck.length) {
- var wayToCheck = waysToCheck.pop();
- var childNodes = graph.childNodes(wayToCheck);
-
- 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;
- }
-
- if (isRoutableNode(vertex)) {
- routingIsland.add(vertex);
- }
-
- queueParentWays(vertex);
- }
- } // no network link found, this is a routing island, return its members
-
-
- return routingIsland;
- }
-
- 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 (vertex.tags.entrance && vertex.tags.entrance !== 'no') return true;
- if (vertex.tags.amenity === 'parking_entrance') return true;
- return false;
- }
-
- function isRoutableNode(node) {
- // treat elevators as distinct features in the highway network
- if (node.tags.highway === 'elevator') return true;
- return false;
- }
-
- 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;
- });
- }
-
- 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
-
- var map = context.map();
-
- if (!context.editable() || !map.trimmedExtent().contains(vertex.loc)) {
- map.zoomToEase(vertex);
- }
-
- context.enter(modeDrawLine(context, wayId, context.graph(), 'line', way.affix(vertexId), true));
- }
- });
- }
- };
-
- validation.type = type;
- return validation;
- }
-
- function validationFormatting() {
- var type = 'invalid_format';
-
- var validation = function validation(entity) {
- var issues = [];
-
- 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
-
- return !email || valid_email.test(email);
- }
- /*
- function isSchemePresent(url) {
- var valid_scheme = /^https?:\/\//i;
- return (!url || valid_scheme.test(url));
- }
- */
-
-
- 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' : ''
- }));
- }
- }
- */
-
-
- 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 (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' : ''
- }));
- }
- }
-
- return issues;
- };
-
- validation.type = type;
- return validation;
- }
-
- function validationHelpRequest(context) {
- var type = 'help_request';
-
- var validation = function checkFixmeTag(entity) {
- if (!entity.tags.fixme) return []; // don't flag fixmes on features added by the user
-
- if (entity.version === undefined) return [];
-
- if (entity.v !== undefined) {
- var baseEntity = context.history().base().hasEntity(entity.id); // don't flag fixmes added by the user on existing features
-
- if (!baseEntity || !baseEntity.tags.fixme) return [];
- }
-
- 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]
- })];
-
- function showReference(selection) {
- selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.fixme_tag.reference'));
- }
- };
-
- validation.type = type;
- return validation;
- }
-
- function validationImpossibleOneway() {
- var type = 'impossible_oneway';
-
- 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);
-
- 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;
- }
-
- function isOneway(way) {
- if (way.tags.oneway === 'yes') return true;
- if (way.tags.oneway) return false;
-
- for (var key in way.tags) {
- if (osmOneWayTags[key] && osmOneWayTags[key][way.tags[key]]) {
- return true;
- }
- }
-
- return false;
- }
-
- function nodeOccursMoreThanOnce(way, nodeID) {
- var occurrences = 0;
-
- for (var index in way.nodes) {
- if (way.nodes[index] === nodeID) {
- occurrences += 1;
- if (occurrences > 1) return true;
- }
- }
-
- return false;
- }
-
- function isConnectedViaOtherTypes(way, node) {
- var wayType = typeForWay(way);
-
- 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;
- }
- }
-
- return graph.parentWays(node).some(function (parentWay) {
- if (parentWay.id === way.id) return false;
-
- 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
-
- 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
-
- 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;
- }
-
- return false;
- });
- }
-
- 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
-
- 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
-
- 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
-
- if (attachedOneways.length < attachedWaysOfSameType.length) return [];
-
- 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 + '.';
-
- if (wayType === 'waterway') {
- messageID += 'connected.' + placement;
- referenceID += 'connected';
- } else {
- messageID += placement;
- referenceID += placement;
- }
-
- 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
- }));
- }
- }));
- }
-
- 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);
- }
- }));
- }
-
- 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 continueDrawing(way, vertex, context) {
- // make sure the vertex is actually visible and editable
- var map = context.map();
-
- if (!context.editable() || !map.trimmedExtent().contains(vertex.loc)) {
- map.zoomToEase(vertex);
- }
-
- context.enter(modeDrawLine(context, way.id, context.graph(), 'line', way.affix(vertex.id), true));
- }
-
- validation.type = type;
- return validation;
- }
-
- function validationIncompatibleSource() {
- var type = 'incompatible_source';
- var invalidSources = [{
- id: 'google',
- regex: 'google',
- exceptRegex: 'books.google|Google Books|drive.google|googledrive|Google Drive'
- }];
-
- 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;
-
- 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'));
- };
- }
- };
-
- validation.type = type;
- return validation;
- }
-
- function validationMaprules() {
- var type = 'maprules';
-
- var validation = function checkMaprules(entity, graph) {
- if (!services.maprules) return [];
- var rules = services.maprules.validationRules();
- var issues = [];
-
- for (var i = 0; i < rules.length; i++) {
- var rule = rules[i];
- rule.findIssues(entity, graph, issues);
- }
-
- return issues;
- };
-
- validation.type = type;
- return validation;
- }
-
- function validationMismatchedGeometry() {
- var type = 'mismatched_geometry';
-
- function tagSuggestingLineIsArea(entity) {
- if (entity.type !== 'way' || entity.isClosed()) return null;
- var tagSuggestingArea = entity.tagSuggestingArea();
-
- if (!tagSuggestingArea) {
- return null;
- }
-
- var asLine = _mainPresetIndex.matchTags(tagSuggestingArea, 'line');
- var asArea = _mainPresetIndex.matchTags(tagSuggestingArea, 'area');
-
- if (asLine && asArea && asLine === asArea) {
- // these tags also allow lines and making this an area wouldn't matter
- return null;
- }
-
- 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
-
- if (firstToLastDistanceMeters < 0.75) {
- testNodes = nodes.slice(); // shallow copy
-
- testNodes.pop();
- testNodes.push(testNodes[0]); // make sure this will not create a self-intersection
-
- 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
-
-
- testNodes = nodes.slice(); // shallow copy
-
- testNodes.push(testNodes[0]); // make sure this will not create a self-intersection
-
- 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'));
- };
- }
- }
-
- 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
-
- for (var key in tagSuggestingArea) {
- delete tags[key];
- }
-
- context.perform(actionChangeTags(entityId, tags), _t('issues.fix.remove_tag.annotation'));
- }
- }));
- return fixes;
- }
- });
-
- function showReference(selection) {
- selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.tag_suggests_area.reference'));
- }
- }
-
- 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
-
- if (entity.isOnAddressLine(graph)) return null;
- var geometry = entity.geometry(graph);
- var allowedGeometries = osmNodeGeometriesForTags(entity.tags);
-
- 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()]));
- };
- }
-
- return [new validationIssueFix({
- icon: 'iD-operation-extract',
- title: _t.html('issues.fix.extract_point.title'),
- onClick: extractOnClick
- })];
- }
- });
- }
-
- return null;
- }
-
- 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 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);
- }
-
- return issues;
-
- 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 validation = function checkMismatchedGeometry(entity, graph) {
- var issues = [vertexTaggedAsPointIssue(entity, graph), lineTaggedAsAreaIssue(entity)];
- issues = issues.concat(unclosedMultipolygonPartIssues(entity, graph));
- return issues.filter(Boolean);
- };
-
- validation.type = type;
- return validation;
- }
-
- function validationMissingRole() {
- var type = 'missing_role';
-
- var validation = function checkMissingRole(entity, graph) {
- var issues = [];
-
- if (entity.type === 'way') {
- graph.parentRelations(entity).forEach(function (relation) {
- if (!relation.isMultipolygon()) return;
- var member = relation.memberById(entity.id);
-
- 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 (way && isMissingRole(member)) {
- issues.push(makeIssue(way, entity, member));
- }
- });
- }
-
- return issues;
- };
-
- 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
- }));
- }
- })];
- }
- });
-
- function showReference(selection) {
- selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.missing_role.multipolygon.reference'));
- }
- }
-
- 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
- }));
- }
- });
- }
-
- validation.type = type;
- return validation;
- }
-
- function validationMissingTag(context) {
- var type = 'missing_tag';
-
- function hasDescriptiveTags(entity, graph) {
- var keys = Object.keys(entity.tags).filter(function (k) {
- if (k === 'area' || k === 'name') {
- return false;
- } else {
- return osmIsInterestingTag(k);
- }
- });
-
- 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 keys.length > 0;
- }
-
- function isUnknownRoad(entity) {
- return entity.type === 'way' && entity.tags.highway === 'road';
- }
-
- function isUntypedRelation(entity) {
- return entity.type === 'relation' && !entity.tags.type;
- }
-
- 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 (!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 (!subtype && isUnknownRoad(entity)) {
- subtype = 'highway_classification';
- }
-
- 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..
-
- 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 (!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'));
- }
- };
-
- validation.type = type;
- return validation;
- }
-
- 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());
- };
-
- // {
- // 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"
- // }
-
- 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 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
- };
-
- var matchGroups$1 = require$$0.matchGroups;
-
- var matcher$1 = function matcher() {
- var _warnings = []; // array of match conflict pairs
-
- var _ambiguous = {};
- var _matchIndex = {};
- var matcher = {}; // Create an index of all the keys/simplenames for fast matching
-
- 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.
-
- if (which === 'secondary' && parts.d) return;
-
- if (obj.countryCodes) {
- parts.countryCodes = obj.countryCodes.slice(); // copy
- }
-
- var nomatches = obj.nomatch || [];
-
- if (nomatches.some(function (s) {
- return s === kvnd;
- })) {
- console.log("WARNING match/nomatch conflict for ".concat(kvnd));
- return;
- }
-
- 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 = [];
-
- 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);
- }
-
- if (!match_nsimple.length) return; // nothing to do
-
- 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
- }
- }
- });
- });
- }
- }; // pass a `key`, `value`, `name` and return the best match,
- // `countryCode` optional (if supplied, must match that too)
-
-
- 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)
-
-
- matcher.matchParts = function (parts, countryCode) {
- var match = null;
- var inGroup = false; // fixme: we currently return a single match for ambiguous
-
- match = _ambiguous[parts.kv] && _ambiguous[parts.kv][parts.nsimple];
- if (match && matchesCountryCode(match)) return match; // try to return an exact match
-
- match = _matchIndex[parts.kv] && _matchIndex[parts.kv][parts.nsimple];
- if (match && matchesCountryCode(match)) return match; // look in match groups
-
- for (var mg in matchGroups$1) {
- var matchGroup = matchGroups$1[mg];
- match = null;
- inGroup = false;
-
- for (var i = 0; i < matchGroup.length; i++) {
- var otherkv = matchGroup[i].toLowerCase();
-
- if (!inGroup) {
- inGroup = otherkv === parts.kv;
- }
-
- if (!match) {
- // fixme: we currently return a single match for ambiguous
- match = _ambiguous[otherkv] && _ambiguous[otherkv][parts.nsimple];
- }
-
- if (!match) {
- match = _matchIndex[otherkv] && _matchIndex[otherkv][parts.nsimple];
- }
-
- if (match && !matchesCountryCode(match)) {
- match = null;
- }
-
- if (inGroup && match) {
- return match;
- }
- }
- }
-
- return null;
-
- function matchesCountryCode(match) {
- if (!countryCode) return true;
- if (!match.countryCodes) return true;
- return match.countryCodes.indexOf(countryCode) !== -1;
- }
- };
-
- matcher.getWarnings = function () {
- return _warnings;
- };
-
- return matcher;
- };
-
- var fromCharCode = String.fromCharCode;
- var nativeFromCodePoint = String.fromCodePoint;
-
- // length should be 1, old FF problem
- var INCORRECT_LENGTH = !!nativeFromCodePoint && nativeFromCodePoint.length != 1;
-
- // `String.fromCodePoint` method
- // https://tc39.github.io/ecma262/#sec-string.fromcodepoint
- _export({ target: 'String', stat: true, forced: INCORRECT_LENGTH }, {
- fromCodePoint: function fromCodePoint(x) { // eslint-disable-line no-unused-vars
- 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 quickselect(arr, k, left, right, compare) {
- quickselectStep(arr, k, left || 0, right || arr.length - 1, compare || defaultCompare);
- }
-
- 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);
-
- while (i < j) {
- swap(arr, i, j);
- i++;
- j--;
-
- while (compare(arr[i], t) < 0) {
- i++;
- }
-
- while (compare(arr[j], t) > 0) {
- j--;
- }
- }
-
- 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 swap(arr, i, j) {
- var tmp = arr[i];
- arr[i] = arr[j];
- arr[j] = tmp;
- }
-
- function defaultCompare(a, b) {
- return a < b ? -1 : a > b ? 1 : 0;
- }
-
- return quickselect;
- });
- });
-
- var rbush_1 = rbush;
- var _default$2 = rbush;
-
- 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
-
- this._maxEntries = Math.max(4, maxEntries || 9);
- this._minEntries = Math.max(2, Math.ceil(this._maxEntries * 0.4));
-
- if (format) {
- this._initFormat(format);
- }
-
- this.clear();
- }
-
- 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;
-
- while (node) {
- for (i = 0, len = node.children.length; i < len; i++) {
- child = node.children[i];
- childBBox = node.leaf ? toBBox(child) : child;
-
- 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);
- }
- }
-
- node = nodesToSearch.pop();
- }
-
- 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;
-
- if (intersects$1(bbox, childBBox)) {
- if (node.leaf || contains$1(bbox, childBBox)) return true;
- nodesToSearch.push(child);
- }
- }
-
- node = nodesToSearch.pop();
- }
-
- return false;
- },
- load: function load(data) {
- if (!(data && data.length)) return this;
-
- if (data.length < this._minEntries) {
- for (var i = 0, len = data.length; i < len; i++) {
- this.insert(data[i]);
- }
-
- return this;
- } // recursively build the tree with the given data from scratch using OMT algorithm
-
-
- 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
-
-
- this._insert(node, this.data.height - node.height - 1, true);
- }
-
- 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;
- }
-
- if (node.leaf) {
- // check current node
- index = findItem$1(item, node.children, equalsFn);
-
- if (index !== -1) {
- // item found, remove the item and condense tree upwards
- node.children.splice(index, 1);
- path.push(node);
-
- this._condense(path);
-
- return this;
- }
- }
-
- 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
-
- }
-
- 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 = [];
-
- while (node) {
- if (node.leaf) result.push.apply(result, node.children);else nodesToSearch.push.apply(nodesToSearch, node.children);
- node = nodesToSearch.pop();
- }
-
- 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;
- }
-
- 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
-
- 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 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 (i = left; i <= right; i += N1) {
- right2 = Math.min(i + N1 - 1, right);
- multiSelect$1(items, i, right2, N2, this.compareMinY);
-
- for (j = i; j <= right2; j += N2) {
- right3 = Math.min(j + N2 - 1, right2); // pack each entry recursively
-
- node.children.push(this._build(items, j, right3, height - 1));
- }
- }
-
- calcBBox$1(node, this.toBBox);
- return node;
- },
- _chooseSubtree: function _chooseSubtree(bbox, node, level, path) {
- var i, len, child, targetNode, area, enlargement, minArea, minEnlargement;
-
- 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
-
- 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;
- }
- }
- }
-
- 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 node = this._chooseSubtree(bbox, this.data, level, insertPath); // put the item into the node
-
-
- node.children.push(item);
- extend$3(node, bbox); // split on node overflow; propagate upwards if necessary
-
- while (level >= 0) {
- if (insertPath[level].children.length > this._maxEntries) {
- this._split(insertPath, level);
-
- level--;
- } else break;
- } // adjust bboxes along the insertion path
-
-
- 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;
-
- this._chooseSplitAxis(node, m, M);
-
- var splitIndex = this._chooseSplitIndex(node, m, M);
-
- 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
-
- 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 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
-
-
- 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;
-
- 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);
- }
-
- 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);
- }
- },
- _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);
-
- 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 calcBBox$1(node, toBBox) {
- distBBox$1(node, 0, node.children.length, toBBox, node);
- } // min bounding rectangle of node children from k to p-1
-
-
- 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;
-
- for (var i = k, child; i < p; i++) {
- child = node.children[i];
- extend$3(destNode, node.leaf ? toBBox(child) : child);
- }
-
- return destNode;
- }
-
- 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;
- }
-
- function compareNodeMinX$1(a, b) {
- return a.minX - b.minX;
- }
-
- function compareNodeMinY$1(a, b) {
- return a.minY - b.minY;
- }
-
- function bboxArea$1(a) {
- return (a.maxX - a.minX) * (a.maxY - a.minY);
- }
-
- 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));
- }
-
- 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);
- }
-
- function contains$1(a, b) {
- return a.minX <= b.minX && a.minY <= b.minY && b.maxX <= a.maxX && b.maxY <= a.maxY;
- }
-
- function intersects$1(a, b) {
- return b.minX <= a.maxX && b.minY <= a.maxY && b.maxX >= a.minX && b.maxY >= a.minY;
- }
-
- 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
-
-
- function multiSelect$1(arr, left, right, n, compare) {
- var stack = [left, right],
- mid;
-
- 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$2;
-
- var lineclip_1$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 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);
-
- 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$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);
- }
- }
-
- codeA = lastCode;
- }
-
- if (part.length) result.push(part);
- return result;
- } // Sutherland-Hodgeman polygon clipping algorithm
-
-
- function polygonclip$1(points, bbox) {
- var result, edge, prev, prevInside, i, p, inside; // clip against each side of the clip rectangle
-
- 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
-
- if (inside !== prevInside) result.push(intersect$1(prev, p, edge, bbox));
- if (inside) result.push(p); // add a point if it's inside
-
- prev = p;
- prevInside = inside;
- }
-
- points = result;
- if (!points.length) break;
- }
-
- return result;
- } // intersect a segment against one of the 4 lines that make up the bbox
-
-
- 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
-
-
- 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
-
- if (p[1] < bbox[1]) code |= 4; // bottom
- else if (p[1] > bbox[3]) code |= 8; // top
-
- return code;
- }
-
- var whichPolygon_1 = whichPolygon;
-
- function whichPolygon(data) {
- var bboxes = [];
-
- for (var i = 0; i < data.features.length; i++) {
- var feature = data.features[i];
- var coords = feature.geometry.coordinates;
-
- 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));
- }
- }
- }
-
- var tree = rbush_1().load(bboxes);
-
- function query(p, multi) {
- var output = [],
- result = tree.search({
- minX: p[0],
- minY: p[1],
- maxX: p[0],
- maxY: p[1]
- });
-
- 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;
- }
- }
-
- return multi && output.length ? output : null;
- }
-
- query.tree = tree;
-
- query.bbox = function queryBBox(bbox) {
- var output = [];
- var result = tree.search({
- minX: bbox[0],
- minY: bbox[1],
- maxX: bbox[2],
- maxY: bbox[3]
- });
-
- for (var i = 0; i < result.length; i++) {
- if (polygonIntersectsBBox(result[i].coords, bbox)) {
- output.push(result[i].props);
- }
- }
-
- return output;
- };
-
- return query;
- }
-
- function polygonIntersectsBBox(polygon, bbox) {
- var bboxCenter = [(bbox[0] + bbox[2]) / 2, (bbox[1] + bbox[3]) / 2];
- if (insidePolygon(polygon, bboxCenter)) return true;
-
- for (var i = 0; i < polygon.length; i++) {
- if (lineclip_1$1(polygon[i], bbox).length > 0) return true;
- }
-
- return false;
- } // ray casting algorithm for detecting if point is in polygon
-
-
- function insidePolygon(rings, p) {
- var inside = false;
-
- for (var i = 0, len = rings.length; i < len; i++) {
- var ring = rings[i];
-
- for (var j = 0, len2 = ring.length, k = len2 - 1; j < len2; k = j++) {
- if (rayIntersect(p, ring[j], ring[k])) inside = !inside;
- }
- }
-
- return inside;
- }
-
- 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];
- }
-
- function treeItem(coords, props) {
- var item = {
- minX: Infinity,
- minY: Infinity,
- maxX: -Infinity,
- maxY: -Infinity,
- coords: coords,
- props: props
- };
-
- 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]);
- }
-
- return item;
- }
-
- 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 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);
-
- function loadDerivedDataAndCaches(borders) {
- var identifierProps = ['iso1A2', 'iso1A3', 'm49', 'wikidata', 'emojiFlag', 'nameEn'];
- var geometryFeatures = [];
-
- 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);
- }
-
- 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);
- });
-
- loadMembersForGroupsOf(_feature2);
- }
-
- var geometryOnlyCollection = {
- type: 'RegionFeatureCollection',
- features: geometryFeatures
- };
- whichPolygonGetter = whichPolygon_1(geometryOnlyCollection);
-
- function loadGroups(feature) {
- var props = feature.properties;
-
- if (!props.groups) {
- props.groups = [];
- }
-
- if (props.country) {
- props.groups.push(props.country);
- }
-
- if (props.m49 !== '001') {
- props.groups.push('001');
- }
- }
-
- function loadM49(feature) {
- var props = feature.properties;
-
- if (!props.m49 && props.iso1N3) {
- props.m49 = props.iso1N3;
- }
- }
-
- function loadIsoStatus(feature) {
- var props = feature.properties;
-
- if (!props.isoStatus && props.iso1A2) {
- props.isoStatus = 'official';
- }
- }
-
- 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 loadRoadSpeedUnit(feature) {
- var props = feature.properties;
-
- if (props.roadSpeedUnit === undefined && props.iso1A2 && props.iso1A2 !== 'EU') {
- props.roadSpeedUnit = 'km/h';
- }
- }
-
- function loadDriveSide(feature) {
- var props = feature.properties;
-
- if (props.driveSide === undefined && props.iso1A2 && props.iso1A2 !== 'EU') {
- props.driveSide = 'right';
- }
- }
-
- 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 = [];
-
- for (var j in feature.properties.groups) {
- var groupID = feature.properties.groups[j];
- var groupFeature = featuresByCode[groupID];
- standardizedGroupIDs.push(groupFeature.properties.id);
-
- if (groupFeature.properties.members) {
- groupFeature.properties.members.push(featureID);
- } else {
- groupFeature.properties.members = [featureID];
- }
- }
-
- feature.properties.groups = standardizedGroupIDs;
- }
-
- function cacheFeatureByIDs(feature) {
- for (var k in identifierProps) {
- var prop = identifierProps[k];
- var id = prop && feature.properties[prop];
-
- if (id) {
- id = id.replace(idFilterRegex, '').toUpperCase();
- featuresByCode[id] = feature;
- }
- }
-
- if (feature.properties.aliases) {
- for (var j in feature.properties.aliases) {
- var alias = feature.properties.aliases[j].replace(idFilterRegex, '').toUpperCase();
- featuresByCode[alias] = feature;
- }
- }
- }
- }
-
- function locArray(loc) {
- if (Array.isArray(loc)) {
- return loc;
- } else if (loc.coordinates) {
- return loc.coordinates;
- }
-
- return loc.geometry.coordinates;
- }
-
- function smallestFeature(loc) {
- var query = locArray(loc);
- var featureProperties = whichPolygonGetter(query);
- if (!featureProperties) return null;
- return featuresByCode[featureProperties.id];
- }
-
- function countryFeature(loc) {
- var feature = smallestFeature(loc);
- if (!feature) return null;
- var countryCode = feature.properties.country || feature.properties.iso1A2;
- return featuresByCode[countryCode];
- }
-
- 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;
-
- for (var i in features) {
- var _feature3 = features[i];
-
- if (_feature3.properties.level === targetLevel || levels.indexOf(_feature3.properties.level) > targetLevelIndex) {
- return _feature3;
- }
- }
-
- return null;
- }
-
- return countryFeature(loc);
- }
-
- function featureForID(id) {
- var stringID;
-
- if (typeof id === 'number') {
- stringID = id.toString();
-
- if (stringID.length === 1) {
- stringID = '00' + stringID;
- } else if (stringID.length === 2) {
- stringID = '0' + stringID;
- }
- } else {
- stringID = id.replace(idFilterRegex, '').toUpperCase();
- }
-
- return featuresByCode[stringID] || null;
- }
-
- function smallestOrMatchingFeature(query) {
- if (_typeof(query) === 'object') {
- return smallestFeature(query);
- }
-
- return featureForID(query);
- }
-
- function feature(query, opts) {
- if (_typeof(query) === 'object') {
- return featureForLoc(query, opts);
- }
-
- return featureForID(query);
- }
- function iso1A2Code(query, opts) {
- var match = feature(query, opts);
- if (!match) return null;
- return match.properties.iso1A2 || null;
- }
- function featuresContaining(query, strict) {
- var feature = smallestOrMatchingFeature(query);
- if (!feature) return [];
- var features = [];
-
- if (!strict || _typeof(query) === 'object') {
- features.push(feature);
- }
-
- var properties = feature.properties;
-
- 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;
- }
-
- var _dataDeprecated;
-
- var _nsi;
-
- 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
-
- _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
-
- _nsi.matcher.buildMatchIndex(d.brands); // index all known wikipedia and wikidata tags
-
-
- 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 (wd) {
- _nsi.wikidata[wd] = kvnd;
- }
-
- if (wp) {
- _nsi.wikipedia[wp] = kvnd;
- }
- });
- return _nsi;
- })["catch"](function () {
- /* ignore */
- });
-
- function oldTagIssues(entity, graph) {
- var oldTags = Object.assign({}, entity.tags); // shallow copy
-
- 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..
-
-
- if (_dataDeprecated) {
- var deprecatedTags = entity.deprecatedTags(_dataDeprecated);
-
- if (deprecatedTags.length) {
- deprecatedTags.forEach(function (tag) {
- graph = actionUpgradeTags(entity.id, tag.old, tag.replace)(graph);
- });
- entity = graph.entity(entity.id);
- }
- } // add missing addTags..
-
-
- 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];
- }
- }
- });
- }
-
- 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;
-
- if (newTags.wikidata) {
- // try matching `wikidata`
- isBrand = _nsi.wikidata[newTags.wikidata];
- }
-
- if (!isBrand && newTags.wikipedia) {
- // fallback to `wikipedia`
- isBrand = _nsi.wikipedia[newTags.wikipedia];
- }
-
- if (isBrand && !newTags.office) {
- // but avoid doing this for corporate offices
- if (newTags.wikidata) {
- newTags['brand:wikidata'] = newTags.wikidata;
- delete newTags.wikidata;
- }
-
- 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.
-
- } // try key/value|name match against name-suggestion-index
-
-
- 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);
-
- var match = _nsi.matcher.matchKVN(k, newTags[k], newTags.name, countryCode && countryCode.toLowerCase());
-
- if (!match) continue; // for now skip ambiguous matches (like Target~(USA) vs Target~(Australia))
-
- if (match.d) continue;
- var brand = _nsi.brands[match.kvnd];
-
- 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];
- }
-
- return acc;
- }, {});
- nsiKeys.forEach(function (k) {
- return delete newTags[k];
- });
- Object.assign(newTags, brand.tags, keepTags);
- break;
- }
- }
- }
- } // determine diff
-
-
- var tagDiff = utilTagDiff(oldTags, newTags);
- if (!tagDiff.length) return [];
- 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;
- 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
-
- 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 _t.html(messageID, {
- feature: utilDisplayLabel(currEntity, context.graph())
- });
- }
-
- 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 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 [];
- }
-
- 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 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'));
- }
- }
-
- 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 validationPrivateData() {
- var type = 'private_data'; // assume that some buildings are private
-
- 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
-
- var publicKeys = {
- amenity: true,
- craft: true,
- historic: true,
- leisure: true,
- office: true,
- shop: true,
- tourism: true
- }; // these tags may contain personally identifying info
-
- var personalTags = {
- 'contact:email': true,
- 'contact:fax': true,
- 'contact:phone': true,
- email: true,
- fax: true,
- phone: true
- };
-
- var validation = function checkPrivateData(entity) {
- var tags = entity.tags;
- if (!tags.building || !privateBuildingValues[tags.building]) return [];
- var keepTags = {};
-
- for (var k in tags) {
- if (publicKeys[k]) return []; // probably a public feature
-
- if (!personalTags[k]) {
- keepTags[k] = tags[k];
- }
- }
-
- 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'));
- }
- })];
- }
- })];
-
- function doUpgrade(graph) {
- var currEntity = graph.hasEntity(entity.id);
- if (!currEntity) return graph;
- var newTags = Object.assign({}, currEntity.tags); // shallow copy
-
- 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(this.entityIds[0]);
- if (!currEntity) return '';
- return _t.html('issues.private_data.contact.message', {
- feature: utilDisplayLabel(currEntity, context.graph())
- });
- }
-
- 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;
- });
- }
- };
-
- validation.type = type;
- return validation;
- }
-
- 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 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")
-
-
- function nameMatchesRawTag(lowercaseName, tags) {
- for (var i = 0; i < keysToTestForGenericValues.length; i++) {
- var key = keysToTestForGenericValues[i];
- var val = tags[key];
-
- if (val) {
- val = val.toLowerCase();
-
- if (key === lowercaseName || val === lowercaseName || key.replace(/\_/g, ' ') === lowercaseName || val.replace(/\_/g, ' ') === lowercaseName) {
- return true;
- }
- }
- }
-
- return false;
- }
-
- function isGenericName(name, tags) {
- name = name.toLowerCase();
- return nameMatchesRawTag(name, tags) || isDiscardedSuggestionName(name);
- }
-
- 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
-
- delete tags[nameKey];
- context.perform(actionChangeTags(entityId, tags), _t('issues.fix.remove_generic_name.annotation'));
- }
- })];
- }
- });
-
- function showReference(selection) {
- selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.generic_name.reference'));
- }
- }
-
- 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 showReference(selection) {
- selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.generic_name.reference'));
- }
- }
-
- 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];
-
- if (notNames.length) {
- for (var i in notNames) {
- var notName = notNames[i];
-
- if (notName && value === notName) {
- issues.push(makeIncorrectNameIssue(entity.id, key, value, langCode));
- continue;
- }
- }
- }
-
- if (isGenericName(value, entity.tags)) {
- issues.push(makeGenericNameIssue(entity.id, key, value, langCode));
- }
- }
-
- return issues;
- };
-
- validation.type = type;
- return validation;
- }
-
- 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
-
- var epsilon = 0.05;
- var nodeThreshold = 10;
-
- function isBuilding(entity, graph) {
- if (entity.type !== 'way' || entity.geometry(graph) !== 'area') return false;
- return entity.tags.building && entity.tags.building !== 'no';
- }
-
- var validation = function checkUnsquareWay(entity, graph) {
- if (!isBuilding(entity, graph)) return []; // don't flag ways marked as physically unsquare
-
- 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
-
- var nodes = graph.childNodes(entity).slice(); // shallow copy
-
- if (nodes.length > nodeThreshold + 1) return []; // +1 because closing node appears twice
- // ignore if not all nodes are fully downloaded
-
- 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
-
- 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';
- });
- });
- });
- 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: 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
-
- context.perform(actionOrthogonalize(entityId, context.projection, undefined, degreeThreshold), _t('operations.orthogonalize.annotation.feature', {
- n: 1
- })); // run after the squaring transition (currently 150ms)
-
- 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')
- );
- }
- })
- */
- ];
- }
- })];
-
- function showReference(selection) {
- selection.selectAll('.issue-reference').data([0]).enter().append('div').attr('class', 'issue-reference').html(_t.html('issues.unsquare_way.buildings.reference'));
- }
- };
-
- validation.type = type;
- return validation;
- }
-
- 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 coreValidator(context) {
- var dispatch$1 = dispatch('validated', 'focusedIssue');
- var validator = utilRebind({}, dispatch$1, 'on');
- var _rules = {};
- var _disabledRules = {};
- var _ignoredIssueIDs = {}; // issue.id -> true
-
- var _baseCache = validationCache(); // issues before any user edits
-
-
- var _headCache = validationCache(); // issues after all user edits
-
-
- var _validatedGraph = null;
-
- var _deferred = new Set(); //
- // initialize the validator rulesets
- //
-
-
- 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');
-
- if (disabledRules) {
- disabledRules.split(',').forEach(function (key) {
- _disabledRules[key] = true;
- });
- }
- };
-
- function reset(resetIgnored) {
- Array.from(_deferred).forEach(function (handle) {
- window.cancelIdleCallback(handle);
-
- _deferred["delete"](handle);
- }); // clear caches
-
- if (resetIgnored) _ignoredIssueIDs = {};
- _baseCache = validationCache();
- _headCache = validationCache();
- _validatedGraph = null;
- } //
- // clear caches, called whenever iD resets after a save
- //
-
-
- validator.reset = function () {
- reset(true);
- };
-
- validator.resetIgnoredIssues = function () {
- _ignoredIssueIDs = {}; // reload UI
-
- dispatch$1.call('validated');
- }; // must update issues when the user changes the unsquare thereshold
-
-
- validator.reloadUnsquareIssues = function () {
- reloadUnsquareIssues(_headCache, context.graph());
- reloadUnsquareIssues(_baseCache, context.history().base());
- dispatch$1.call('validated');
- };
-
- function reloadUnsquareIssues(cache, graph) {
- var checkUnsquareWay = _rules.unsquare_way;
- if (typeof checkUnsquareWay !== 'function') return; // uncache existing
-
- 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
-
- buildings.forEach(function (entity) {
- var detected = checkUnsquareWay(entity, graph);
- if (detected.length !== 1) return;
- var issue = detected[0];
-
- if (!cache.issuesByEntityID[entity.id]) {
- cache.issuesByEntityID[entity.id] = new Set();
- }
-
- 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'
- // };
-
-
- 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..
-
- var entityIds = issue.entityIds || [];
-
- for (var i = 0; i < entityIds.length; i++) {
- var entityId = entityIds[i];
-
- if (!context.hasEntity(entityId)) {
- delete _headCache.issuesByEntityID[entityId];
- delete _headCache.issuesByIssueID[issue.id];
- return false;
- }
- }
-
- if (opts.what === 'edited' && _baseCache.issuesByIssueID[issue.id]) return false;
-
- if (opts.where === 'visible') {
- var extent = issue.extent(context.graph());
- if (!view.intersects(extent)) return false;
- }
-
- return true;
- });
- };
-
- validator.getResolvedIssues = function () {
- var baseIssues = Object.values(_baseCache.issuesByIssueID);
- return baseIssues.filter(function (issue) {
- return !_headCache.issuesByIssueID[issue.id];
- });
- };
-
- validator.focusIssue = function (issue) {
- var extent = issue.extent(context.graph());
-
- if (extent) {
- var setZoom = Math.max(context.map().zoom(), 19);
- context.map().unobscuredCenterZoomEase(extent.center(), setZoom); // select the first entity
-
- 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
- }
- }
- };
-
- 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
-
-
- 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
-
- validator.getSharedEntityIssues = function (entityIDs, options) {
- var cache = _headCache; // gather the issues that are common to all the entities
-
- var issueIDs = entityIDs.reduce(function (acc, entityID) {
- var entityIssueIDs = cache.issuesByEntityID[entityID] || new Set();
-
- if (!acc) {
- return new Set(entityIssueIDs);
- }
-
- 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;
- }
-
- var index1 = orderedIssueTypes.indexOf(issue1.type);
- var index2 = orderedIssueTypes.indexOf(issue2.type);
-
- 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;
- }
- });
- };
-
- validator.getEntityIssues = function (entityID, options) {
- return validator.getSharedEntityIssues([entityID], options);
- };
-
- validator.getRuleKeys = function () {
- return Object.keys(_rules);
- };
-
- validator.isRuleEnabled = function (key) {
- return !_disabledRules[key];
- };
-
- validator.toggleRule = function (key) {
- if (_disabledRules[key]) {
- delete _disabledRules[key];
- } else {
- _disabledRules[key] = true;
- }
-
- corePreferences('validate-disabledRules', Object.keys(_disabledRules).join(','));
- validator.validate();
- };
-
- validator.disableRules = function (keys) {
- _disabledRules = {};
- keys.forEach(function (k) {
- _disabledRules[k] = true;
- });
- 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 validateEntity(entity, graph) {
- var entityIssues = []; // runs validation and appends resulting issues
-
- function runValidation(key) {
- var fn = _rules[key];
-
- if (typeof fn !== 'function') {
- console.error('no such validation rule = ' + key); // eslint-disable-line no-console
-
- return;
- }
-
- var detected = fn(entity, graph);
- entityIssues = entityIssues.concat(detected);
- } // run all rules
-
-
- Object.keys(_rules).forEach(runValidation);
- return entityIssues;
- }
-
- 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];
-
- if (entity.type === 'node') {
- graph.parentWays(entity).forEach(function (parentWay) {
- acc.add(parentWay.id); // include parent ways
-
- 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
-
- graph._parentWays[nodeID].forEach(function (wayID) {
- acc.add(wayID); // include connected ways
- });
- });
- }
-
- 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`
- //
-
-
- function validateEntities(entityIDs, graph, cache) {
- // clear caches for existing issues related to these entities
- entityIDs.forEach(cache.uncacheEntityID); // detect new issues and update caches
-
- entityIDs.forEach(function (entityID) {
- var entity = graph.hasEntity(entityID); // don't validate deleted entities
-
- 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.
- //
-
-
- validator.validate = function () {
- var currGraph = context.graph();
- _validatedGraph = _validatedGraph || context.history().base();
-
- if (currGraph === _validatedGraph) {
- dispatch$1.call('validated');
- return;
- }
-
- var oldGraph = _validatedGraph;
- var difference = coreDifference(oldGraph, currGraph);
- _validatedGraph = currGraph;
- var createdAndModifiedEntityIDs = difference.extantIDs(true); // created/modified (true = w/relation members)
-
- 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)
-
- var modifiedAndDeletedEntityIDs = difference.deleted().concat(difference.modified()).map(function (entity) {
- return entity.id;
- });
- var entityIDsToCheckForOldGraph = entityIDsToValidate(modifiedAndDeletedEntityIDs, oldGraph); // concat the sets
-
- entityIDsToCheckForOldGraph.forEach(entityIDsToCheck.add, entityIDsToCheck);
- validateEntities(entityIDsToCheck, context.graph(), _headCache);
- dispatch$1.call('validated');
- };
-
- 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:
-
- 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:
-
- context.on('exit.validator', validator.validate); // When merging fetched data:
-
- context.history().on('merge.validator', function (entities) {
- if (!entities) return;
- var handle = window.requestIdleCallback(function () {
- var entityIDs = entities.map(function (entity) {
- return entity.id;
- });
- 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;
- }
-
- function validationCache() {
- var cache = {
- issuesByIssueID: {},
- // issue.id -> issue
- issuesByEntityID: {} // entity.id -> set(issue.id)
-
- };
-
- 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();
- }
-
- cache.issuesByEntityID[entityId].add(issue.id);
- });
- cache.issuesByIssueID[issue.id] = issue;
- });
- };
-
- 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);
- }
- });
- 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;
- });
- cache.uncacheIssues(issuesOfType);
- }; //
- // Remove a single entity and all its related issues from the caches
- //
-
-
- cache.uncacheEntityID = function (entityID) {
- var issueIDs = cache.issuesByEntityID[entityID];
- if (!issueIDs) return;
- issueIDs.forEach(function (issueID) {
- var issue = cache.issuesByIssueID[issueID];
-
- if (issue) {
- cache.uncacheIssue(issue);
- } else {
- delete cache.issuesByIssueID[issueID];
- }
- });
- delete cache.issuesByEntityID[entityID];
- };
-
- return cache;
- }
-
- 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 = [];
-
- var _origChanges;
-
- var _discardTags = {};
- _mainFileFetcher.get('discarded').then(function (d) {
- _discardTags = d;
- })["catch"](function () {
- /* ignore */
- });
- var uploader = utilRebind({}, dispatch$1, 'on');
-
- uploader.isSaving = function () {
- return _isSaving;
- };
-
- uploader.save = function (changeset, tryAgain, checkConflicts) {
- // Guard against accidentally entering save code twice - #4641
- if (_isSaving && !tryAgain) {
- return;
- }
-
- 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.
-
- if (!osm.authenticated()) {
- osm.authenticate(function (err) {
- if (!err) {
- uploader.save(changeset, tryAgain, checkConflicts); // continue where we left off..
- }
- });
- return;
- }
-
- if (!_isSaving) {
- _isSaving = true;
- dispatch$1.call('saveStarted', this);
- }
-
- var history = context.history();
- _conflicts = [];
- _errors = []; // Store original changes, in case user wants to download them as an .osc file
-
- _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 (!tryAgain) {
- history.perform(actionNoop());
- } // Attempt a fast upload.. If there are conflicts, re-enter with `checkConflicts = true`
-
-
- if (!checkConflicts) {
- upload(changeset); // Do the full (slow) conflict check..
- } else {
- performFullConflictCheck(changeset);
- }
- };
-
- 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 = [];
-
- for (var i = 0; i < summary.length; i++) {
- var item = summary[i];
-
- if (item.changeType === 'modified') {
- _toCheck.push(item.entity.id);
- }
- }
-
- var _toLoad = withChildNodes(_toCheck, localGraph);
-
- var _loaded = {};
- var _toLoadCount = 0;
- var _toLoadTotal = _toLoad.length;
-
- if (_toCheck.length) {
- dispatch$1.call('progressChanged', this, _toLoadCount, _toLoadTotal);
-
- _toLoad.forEach(function (id) {
- _loaded[id] = false;
- });
-
- 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);
- }
- });
- });
- 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
- })]
- });
-
- 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..
-
- var i, id;
-
- if (entity.type === 'way') {
- for (i = 0; i < entity.nodes.length; i++) {
- id = entity.nodes[i];
-
- 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 (_loaded[id] === undefined) {
- _loaded[id] = false;
- loadMore.push(id);
- }
- }
- }
- });
- _toLoadCount += result.data.length;
- _toLoadTotal += loadMore.length;
- dispatch$1.call('progressChanged', this, _toLoadCount, _toLoadTotal);
-
- if (loadMore.length) {
- _toLoad.push.apply(_toLoad, loadMore);
-
- osm.loadMultiple(loadMore, loaded);
- }
-
- if (!_toLoad.length) {
- detectConflicts();
- upload(changeset);
- }
- }
- }
-
- 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;
- }
-
- function sameVersions(local, remote) {
- if (local.version !== remote.version) return false;
-
- if (local.type === 'way') {
- var children = utilArrayUnion(local.nodes, remote.nodes);
-
- 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;
- }
- }
-
- return true;
- }
-
- _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 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'));
-
- _conflicts.push({
- id: id,
- name: entityName(local),
- details: mergeConflicts,
- chosen: 1,
- choices: [choice(id, keepMine, forceLocal), choice(id, keepTheirs, forceRemote)]
- });
- });
- }
- }
-
- function upload(changeset) {
- var osm = context.connection();
-
- if (!osm) {
- _errors.push({
- msg: 'No OSM Service'
- });
- }
-
- 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$1.call('willAttemptUpload', this);
- osm.putChangeset(changeset, changes, uploadCallback);
- } else {
- // changes were insignificant or reverted by user
- didResultInNoChanges();
- }
- }
- }
-
- 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
- })]
- });
-
- didResultInErrors();
- }
- } else {
- didResultInSuccess(changeset);
- }
- }
-
- function didResultInNoChanges() {
- dispatch$1.call('resultNoChanges', this);
- endSave();
- context.flush(); // reset iD
- }
-
- function didResultInErrors() {
- context.history().pop();
- dispatch$1.call('resultErrors', this, _errors);
- endSave();
- }
-
- function didResultInConflicts(changeset) {
- _conflicts.sort(function (a, b) {
- return b.id.localeCompare(a.id);
- });
-
- dispatch$1.call('resultConflicts', this, changeset, _conflicts, _origChanges);
- endSave();
- }
-
- 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
-
- window.setTimeout(function () {
- endSave();
- context.flush(); // reset iD
- }, 2500);
- }
-
- function endSave() {
- _isSaving = false;
- dispatch$1.call('saveEnded', this);
- }
-
- uploader.cancelConflictResolution = function () {
- context.history().pop();
- };
-
- uploader.processResolvedConflicts = function (changeset) {
- var history = context.history();
-
- 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));
- }
- }
-
- uploader.save(changeset, true, false); // tryAgain = true, checkConflicts = false
- };
-
- uploader.reset = function () {};
-
- return uploader;
- }
-
- 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;
- });
-
- // `Math.sinh` method
- // https://tc39.github.io/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
-
- 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 vintageRange(vintage) {
- var s;
-
- if (vintage.start || vintage.end) {
- s = vintage.start || '?';
-
- if (vintage.start !== vintage.end) {
- s += ' - ' + (vintage.end || '?');
- }
- }
-
- return s;
- }
-
- function rendererBackgroundSource(data) {
- var source = Object.assign({}, data); // shallow copy
-
- 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;
-
- 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;
- };
-
- source.nudge = function (val, zoomlevel) {
- _offset[0] += val[0] / Math.pow(2, zoomlevel);
- _offset[1] += val[1] / Math.pow(2, zoomlevel);
- return source;
- };
-
- 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
- });
- };
-
- source.description = function () {
- var id_safe = source.id.replace(/\./g, '<TX_DOT>');
- return _t.html('imagery.' + id_safe + '.description', {
- "default": _description
- });
- };
-
- source.best = function () {
- return _best;
- };
-
- source.area = function () {
- if (!data.polygon) return Number.MAX_VALUE; // worldwide
-
- var area = d3_geoArea({
- type: 'MultiPolygon',
- coordinates: [data.polygon]
- });
- return isNaN(area) ? 0 : area;
- };
-
- source.imageryUsed = function () {
- return _name || source.id;
- };
-
- source.template = function (val) {
- if (!arguments.length) return _template;
-
- if (source.id === 'custom') {
- _template = val;
- }
-
- return source;
- };
-
- 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 (!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';
- }
- }
-
- 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;
- };
-
- 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)));
-
- switch (source.projection) {
- case 'EPSG:4326':
- return {
- x: lon * 180 / Math.PI,
- y: lat * 180 / Math.PI
- };
-
- 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 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;
-
- case 'proj':
- return projection;
-
- case 'wkid':
- return projection.replace(/^EPSG:/, '');
-
- 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;
- }
-
- case 'w':
- return minXmaxY.x;
-
- case 's':
- return maxXminY.y;
-
- case 'n':
- return maxXminY.x;
-
- case 'e':
- return minXmaxY.y;
-
- 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 = '';
-
- 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();
- }
-
- return u;
- });
- } // these apply to any type..
-
-
- result = result.replace(/\{switch:([^}]+)\}/, function (s, r) {
- var subdomains = r.split(',');
- return subdomains[(coord[0] + coord[1]) % subdomains.length];
- });
- return result;
- };
-
- source.validZoom = function (z) {
- return source.zoomExtent[0] <= z && (source.overzoom || source.zoomExtent[1] > z);
- };
-
- source.isLocatorOverlay = function () {
- return source.id === 'mapbox_locator_overlay';
- };
- /* hides a source from the list, but leaves it available for use */
-
-
- source.isHidden = function () {
- return source.id === 'DigitalGlobe-Premium-vintage' || source.id === 'DigitalGlobe-Standard-vintage';
- };
-
- source.copyrightNotices = function () {};
-
- 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;
- }
-
- 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
-
- var key = 'Ak5oTE46TUbjRp08OFVcGpkARErDobfpuyNKa-W2mQ8wbt1K1KL8p1bIRwWwcF-Q'; // iD
-
- 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 */
- });
-
- 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(', ');
- };
-
- 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;
-
- if (!cache[tileID]) {
- cache[tileID] = {};
- }
-
- if (cache[tileID] && cache[tileID].metadata) {
- return callback(null, cache[tileID].metadata);
- }
-
- inflight[tileID] = true;
- d3_json(url).then(function (result) {
- delete inflight[tileID];
-
- if (!result) {
- throw new Error('Unknown Error');
- }
-
- 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);
- });
- };
-
- bing.terms_url = 'https://blog.openstreetmap.org/2010/11/30/microsoft-imagery-details';
- return bing;
- };
-
- 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 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/
-
-
- 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
-
- var z = 20; // first generate a random url using the template
-
- var dummyUrl = esri.url([1, 2, 3]); // calculate url z/y/x from the lat/long of the center of the map
-
- 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
-
- 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
-
- d3_json(tilemapUrl).then(function (tilemap) {
- if (!tilemap) {
- throw new Error('Unknown Error');
- }
-
- var hasTiles = true;
-
- 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
-
-
- esri.zoomExtent[1] = hasTiles ? 22 : 19;
- })["catch"](function () {
- /* ignore */
- });
- };
-
- 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 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;
-
- case zoom >= 19:
- metadataLayer = 3;
- break;
-
- case zoom >= 17:
- metadataLayer = 2;
- break;
-
- case zoom >= 13:
- metadataLayer = 0;
- break;
-
- default:
- metadataLayer = 99;
- }
-
- 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/';
- }
-
- url += metadataLayer + '/query?returnGeometry=false&geometry=' + centerPoint + '&inSR=4326&geometryType=esriGeometryPoint&outFields=*&f=json';
-
- if (!cache[tileID]) {
- cache[tileID] = {};
- }
-
- if (cache[tileID] && cache[tileID].metadata) {
- return callback(null, cache[tileID].metadata);
- } // accurate metadata is only available >= 13
-
-
- 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];
-
- 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
-
-
- 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 (isFinite(metadata.resolution)) {
- metadata.resolution += ' m';
- }
-
- if (isFinite(metadata.accuracy)) {
- metadata.accuracy += ' m';
- }
-
- 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;
- };
-
- source.area = function () {
- return -1; // sources in background pane are sorted by area
- };
-
- return source;
- };
-
- rendererBackgroundSource.Custom = function (template) {
- var source = rendererBackgroundSource({
- id: 'custom',
- template: template
- });
-
- source.name = function () {
- return _t('background.custom');
- };
-
- source.label = function () {
- return _t.html('background.custom');
- };
-
- source.imageryUsed = function () {
- // sanitize personal connection tokens - #6801
- var cleaned = source.template(); // from query string parameters
-
- 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
-
-
- cleaned = cleaned.replace(/token\/(\w+)/, 'token/{apikey}');
- return 'Custom (' + cleaned + ' )';
- };
-
- source.area = function () {
- return -2; // sources in background pane are sorted by area
- };
-
- return source;
- };
-
- function rendererTileLayer(context) {
- var transformProp = utilPrefixCSSProperty('Transform');
- var tiler = utilTiler();
- var _tileSize = 256;
-
- var _projection;
-
- var _cache = {};
-
- var _tileOrigin;
-
- var _zoom;
-
- var _source;
-
- function tileSizeAtZoom(d, z) {
- var EPSILON = 0.002; // close seams
-
- return _tileSize * Math.pow(2, z - d[2]) / _tileSize + EPSILON;
- }
-
- function atZoom(t, distance) {
- var power = Math.pow(2, distance);
- return [Math.floor(t[0] * power), Math.floor(t[1] * power), t[2] + distance];
- }
-
- function lookUp(d) {
- for (var up = -1; up > -d[2]; up--) {
- var tile = atZoom(d, up);
-
- if (_cache[_source.url(tile)] !== false) {
- return tile;
- }
- }
- }
-
- function uniqueBy(a, n) {
- var o = [];
- var seen = {};
-
- for (var i = 0; i < a.length; i++) {
- if (seen[a[i][n]] === undefined) {
- o.push(a[i]);
- seen[a[i][n]] = true;
- }
- }
-
- return o;
- }
-
- function addSource(d) {
- d.push(_source.url(d));
- return d;
- } // Update tiles based on current state of `projection`.
-
-
- 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 {
- pixelOffset = [0, 0];
- }
-
- 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 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;
- });
- }
-
- 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);
- }
-
- function imageTransform(d) {
- var ts = _tileSize * Math.pow(2, _zoom - d[2]);
-
- var scale = tileSizeAtZoom(d, _zoom);
- return 'translate(' + (d[0] * ts - _tileOrigin[0]) + 'px,' + (d[1] * ts - _tileOrigin[1]) + 'px) ' + 'scale(' + scale + ',' + scale + ')';
- }
-
- function tileCenter(d) {
- var ts = _tileSize * Math.pow(2, _zoom - d[2]);
-
- return [d[0] * ts - _tileOrigin[0] + ts / 2, d[1] * ts - _tileOrigin[1] + ts / 2];
- }
-
- 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)
-
-
- 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);
-
- 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();
-
- 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));
-
- _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'));
- });
- });
- }
- }
-
- background.projection = function (val) {
- if (!arguments.length) return _projection;
- _projection = val;
- return background;
- };
-
- background.dimensions = function (val) {
- if (!arguments.length) return tiler.size();
- tiler.size(val);
- return background;
- };
-
- 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;
- }
-
- 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 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
-
- 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]] ]
-
- 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
-
- _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'
-
- _imageryIndex.backgrounds.unshift(rendererBackgroundSource.None()); // Add 'Custom'
-
-
- var template = corePreferences('background-custom-template') || '';
- var custom = rendererBackgroundSource.Custom(template);
-
- _imageryIndex.backgrounds.unshift(custom);
-
- return _imageryIndex;
- });
- }
-
- 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
-
- 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 sources = background.sources(context.map().extent());
- var wasValid = _isValid;
- _isValid = !!sources.filter(function (d) {
- return d === currSource;
- }).length;
-
- if (wasValid !== _isValid) {
- // change in valid status
- background.updateImagery();
- }
-
- var baseFilter = '';
-
- if (detected.cssfilters) {
- if (_brightness !== 1) {
- baseFilter += " brightness(".concat(_brightness, ")");
- }
-
- if (_contrast !== 1) {
- baseFilter += " contrast(".concat(_contrast, ")");
- }
-
- if (_saturation !== 1) {
- baseFilter += " saturate(".concat(_saturation, ")");
- }
-
- if (_sharpness < 1) {
- // gaussian blur
- var blur = d3_interpolateNumber(0.5, 5)(1 - _sharpness);
- baseFilter += " blur(".concat(blur, "px)");
- }
- }
-
- var base = selection.selectAll('.layer-background').data([0]);
- base = base.enter().insert('div', '.layer-data').attr('class', 'layer layer-background').merge(base);
-
- if (detected.cssfilters) {
- base.style('filter', baseFilter || null);
- } else {
- base.style('opacity', _brightness);
- }
-
- 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 (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);
- });
- }
-
- background.updateImagery = function () {
- var currSource = baseLayer.source();
- if (context.inIntro() || !currSource) return;
-
- var o = _overlayLayers.filter(function (d) {
- return !d.source().isLocatorOverlay() && !d.source().isHidden();
- }).map(function (d) {
- return d.source().id;
- }).join(',');
-
- 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;
-
- if (id === 'custom') {
- id = "custom:".concat(currSource.template());
- }
-
- if (id) {
- hash.background = id;
- } else {
- delete hash.background;
- }
-
- if (o) {
- hash.overlays = o;
- } else {
- delete hash.overlays;
- }
-
- if (Math.abs(x) > EPSILON || Math.abs(y) > EPSILON) {
- hash.offset = "".concat(x, ",").concat(y);
- } else {
- delete hash.offset;
- }
-
- if (!window.mocha) {
- window.location.replace('#' + utilQsString(hash, true));
- }
-
- var imageryUsed = [];
- var photoOverlaysUsed = [];
- var currUsed = currSource.imageryUsed();
-
- if (currUsed && _isValid) {
- imageryUsed.push(currUsed);
- }
-
- _overlayLayers.filter(function (d) {
- return !d.source().isLocatorOverlay() && !d.source().isHidden();
- }).forEach(function (d) {
- return imageryUsed.push(d.source().imageryUsed());
- });
-
- var dataLayer = context.layers().layer('data');
-
- if (dataLayer && dataLayer.enabled() && dataLayer.hasData()) {
- imageryUsed.push(dataLayer.getSrc());
- }
-
- var photoOverlayLayers = {
- streetside: 'Bing Streetside',
- mapillary: 'Mapillary Images',
- 'mapillary-map-features': 'Mapillary Map Features',
- 'mapillary-signs': 'Mapillary Signs',
- openstreetcam: 'OpenStreetCam Images'
- };
-
- for (var layerID in photoOverlayLayers) {
- var layer = context.layers().layer(layerID);
-
- if (layer && layer.enabled()) {
- photoOverlaysUsed.push(layerID);
- imageryUsed.push(photoOverlayLayers[layerID]);
- }
- }
-
- context.history().imageryUsed(imageryUsed);
- context.history().photoOverlaysUsed(photoOverlaysUsed);
- };
-
- var _checkedBlocklists;
-
- background.sources = function (extent, zoom, includeCurrent) {
- if (!_imageryIndex) return []; // called before init()?
-
- 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();
-
- if (blocklists && blocklists !== _checkedBlocklists) {
- _imageryIndex.backgrounds.forEach(function (source) {
- source.isBlocked = blocklists.some(function (blocklist) {
- return blocklist.test(source.template());
- });
- });
-
- _checkedBlocklists = blocklists;
- }
-
- return _imageryIndex.backgrounds.filter(function (source) {
- if (includeCurrent && currSource === source) return true; // optionally always include the current imagery
-
- if (source.isBlocked) return false; // even bundled sources may be blocked - #7905
-
- if (!source.polygon) return true; // always include imagery with worldwide coverage
-
- if (zoom && zoom < 6) return false; // optionally exclude local imagery at low zooms
-
- return visible[source.id]; // include imagery visible in given extent
- });
- };
-
- background.dimensions = function (val) {
- if (!val) return;
- baseLayer.dimensions(val);
-
- _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 osm = context.connection();
- if (!osm) return background;
- var blocklists = osm.imageryBlocklists();
- var template = d.template();
- var fail = false;
- var tested = 0;
- var regex;
-
- 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$1.call('change');
- background.updateImagery();
- return background;
- };
-
- background.findSource = function (id) {
- if (!id || !_imageryIndex) return null; // called before init()?
-
- return _imageryIndex.backgrounds.find(function (d) {
- return d.id && d.id === id;
- });
- };
-
- background.bing = function () {
- background.baseLayerSource(background.findSource('Bing'));
- };
-
- 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;
- });
- };
-
- background.overlayLayerSources = function () {
- return _overlayLayers.map(function (layer) {
- return layer.source();
- });
- };
-
- background.toggleOverlayLayer = function (d) {
- var layer;
-
- for (var i = 0; i < _overlayLayers.length; i++) {
- layer = _overlayLayers[i];
-
- if (layer.source() === d) {
- _overlayLayers.splice(i, 1);
-
- dispatch$1.call('change');
- background.updateImagery();
- return;
- }
- }
-
- layer = rendererTileLayer(context).source(d).projection(context.projection).dimensions(baseLayer.dimensions());
-
- _overlayLayers.push(layer);
-
- dispatch$1.call('change');
- background.updateImagery();
- };
-
- background.nudge = function (d, zoom) {
- var currSource = baseLayer.source();
-
- if (currSource) {
- currSource.nudge(d, zoom);
- dispatch$1.call('change');
- background.updateImagery();
- }
-
- return background;
- };
-
- background.offset = function (d) {
- var currSource = baseLayer.source();
-
- if (!arguments.length) {
- return currSource && currSource.offset() || [0, 0];
- }
-
- if (currSource) {
- currSource.offset(d);
- dispatch$1.call('change');
- background.updateImagery();
- }
-
- return background;
- };
-
- background.brightness = function (d) {
- if (!arguments.length) return _brightness;
- _brightness = d;
- if (context.mode()) dispatch$1.call('change');
- return background;
- };
-
- background.contrast = function (d) {
- if (!arguments.length) return _contrast;
- _contrast = d;
- if (context.mode()) dispatch$1.call('change');
- return background;
- };
-
- background.saturation = function (d) {
- if (!arguments.length) return _saturation;
- _saturation = d;
- if (context.mode()) dispatch$1.call('change');
- return background;
- };
-
- background.sharpness = function (d) {
- if (!arguments.length) return _sharpness;
- _sharpness = d;
- if (context.mode()) dispatch$1.call('change');
- return background;
- };
-
- var _loadPromise;
-
- 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
- }
-
- 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;
-
- if (!requested && extent) {
- best = background.sources(extent).find(function (s) {
- return s.best();
- });
- } // Decide which background layer to display
-
-
- 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'));
- }
-
- var locator = imageryIndex.backgrounds.find(function (d) {
- return d.overlay && d["default"];
- });
-
- if (locator) {
- background.toggleOverlayLayer(locator);
- }
-
- 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');
- }
- }
-
- 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 */
- });
- };
-
- return utilRebind(background, dispatch$1, 'on');
- }
-
- function rendererFeatures(context) {
- var dispatch$1 = dispatch('change', 'redraw');
- var features = utilRebind({}, dispatch$1, 'on');
-
- var _deferred = new Set();
-
- 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 = {};
-
- function update() {
- if (!window.mocha) {
- var hash = utilStringQs(window.location.hash);
- var disabled = features.disabled();
-
- if (disabled.length) {
- hash.disable_features = disabled.join(',');
- } else {
- delete hash.disable_features;
- }
-
- window.location.replace('#' + utilQsString(hash, true));
- corePreferences('disabled-features', disabled.join(','));
- }
-
- _hidden = features.hidden();
- dispatch$1.call('change');
- dispatch$1.call('redraw');
- }
-
- function defineRule(k, filter, max) {
- var isEnabled = true;
-
- _keys.push(k);
-
- _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;
- }
- };
- }
-
- 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..
-
- 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);
-
- for (var i = 0; i < strings.length; i++) {
- var s = strings[i];
-
- if (past_futures[s] || past_futures[tags[s]]) {
- return true;
- }
- }
-
- 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`
-
- defineRule('others', function isOther(tags, geometry) {
- return geometry === 'line' || geometry === 'area';
- });
-
- features.features = function () {
- return _rules;
- };
-
- features.keys = function () {
- return _keys;
- };
-
- features.enabled = function (k) {
- if (!arguments.length) {
- return _keys.filter(function (k) {
- return _rules[k].enabled;
- });
- }
-
- return _rules[k] && _rules[k].enabled;
- };
-
- features.disabled = function (k) {
- if (!arguments.length) {
- return _keys.filter(function (k) {
- return !_rules[k].enabled;
- });
- }
-
- return _rules[k] && !_rules[k].enabled;
- };
-
- features.hidden = function (k) {
- if (!arguments.length) {
- return _keys.filter(function (k) {
- return _rules[k].hidden();
- });
- }
-
- return _rules[k] && _rules[k].hidden();
- };
-
- features.autoHidden = function (k) {
- if (!arguments.length) {
- return _keys.filter(function (k) {
- return _rules[k].autoHidden();
- });
- }
-
- return _rules[k] && _rules[k].autoHidden();
- };
-
- features.enable = function (k) {
- if (_rules[k] && !_rules[k].enabled) {
- _rules[k].enable();
-
- update();
- }
- };
-
- features.enableAll = function () {
- var didEnable = false;
-
- for (var k in _rules) {
- if (!_rules[k].enabled) {
- didEnable = true;
-
- _rules[k].enable();
- }
- }
-
- if (didEnable) update();
- };
-
- features.disable = function (k) {
- if (_rules[k] && _rules[k].enabled) {
- _rules[k].disable();
-
- update();
- }
- };
-
- features.disableAll = function () {
- var didDisable = false;
-
- for (var k in _rules) {
- if (_rules[k].enabled) {
- didDisable = true;
-
- _rules[k].disable();
- }
- }
-
- if (didDisable) update();
- };
-
- features.toggle = function (k) {
- if (_rules[k]) {
- (function (f) {
- return f.enabled ? f.disable() : f.enable();
- })(_rules[k]);
-
- update();
- }
- };
-
- features.resetStats = function () {
- for (var i = 0; i < _keys.length; i++) {
- _rules[_keys[i]].count = 0;
- }
-
- dispatch$1.call('change');
- };
-
- 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;
-
- 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..
-
-
- _cullFactor = dimensions[0] * dimensions[1] / 1000000;
-
- for (i = 0; i < entities.length; i++) {
- geometry = entities[i].geometry(resolver);
- matches = Object.keys(features.getMatches(entities[i], resolver, geometry));
-
- for (j = 0; j < matches.length; j++) {
- _rules[matches[j]].count++;
- }
- }
-
- currHidden = features.hidden();
-
- if (currHidden !== _hidden) {
- _hidden = currHidden;
- needsRedraw = true;
- dispatch$1.call('change');
- }
-
- return needsRedraw;
- };
-
- features.stats = function () {
- for (var i = 0; i < _keys.length; i++) {
- _stats[_keys[i]] = _rules[_keys[i]].count;
- }
-
- return _stats;
- };
-
- features.clear = function (d) {
- for (var i = 0; i < d.length; i++) {
- features.clearEntity(d[i]);
- }
- };
-
- features.clearEntity = function (entity) {
- delete _cache[osmEntity.key(entity)];
- };
-
- features.reset = function () {
- Array.from(_deferred).forEach(function (handle) {
- window.cancelIdleCallback(handle);
-
- _deferred["delete"](handle);
- });
- _cache = {};
- }; // only certain relations are worth checking
-
-
- function relationShouldBeChecked(relation) {
- // multipolygon features have `area` geometry and aren't checked here
- return relation.tags.type === 'boundary';
- }
-
- features.getMatches = function (entity, resolver, geometry) {
- if (geometry === 'vertex' || geometry === 'relation' && !relationShouldBeChecked(entity)) return {};
- var ent = osmEntity.key(entity);
-
- if (!_cache[ent]) {
- _cache[ent] = {};
- }
-
- if (!_cache[ent].matches) {
- var matches = {};
- var hasMatch = false;
-
- 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,
-
- if (entity.type === 'way') {
- var parents = features.getParents(entity, resolver, geometry); // 2a. belongs only to a single multipolygon relation
-
- 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]);
-
- if (_cache[pkey] && _cache[pkey].matches) {
- matches = Object.assign({}, _cache[pkey].matches); // shallow copy
-
- continue;
- }
- }
- }
- }
-
- if (_rules[_keys[i]].filter(entity.tags, geometry)) {
- matches[_keys[i]] = hasMatch = true;
- }
- }
-
- _cache[ent].matches = matches;
- }
-
- return _cache[ent].matches;
- };
-
- features.getParents = function (entity, resolver, geometry) {
- if (geometry === 'point') return [];
- var ent = osmEntity.key(entity);
-
- if (!_cache[ent]) {
- _cache[ent] = {};
- }
-
- if (!_cache[ent].parents) {
- var parents = [];
-
- if (geometry === 'vertex') {
- parents = resolver.parentWays(entity);
- } else {
- // 'line', 'area', 'relation'
- parents = resolver.parentRelations(entity);
- }
-
- _cache[ent].parents = parents;
- }
-
- return _cache[ent].parents;
- };
-
- features.isHiddenPreset = function (preset, geometry) {
- if (!_hidden.length) return false;
- if (!preset.tags) return false;
- var test = preset.setTags({}, geometry);
-
- for (var key in _rules) {
- if (_rules[key].filter(test, geometry)) {
- if (_hidden.indexOf(key) !== -1) {
- return key;
- }
-
- return false;
- }
- }
-
- 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);
- });
- };
-
- 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;
-
- for (var i = 0; i < parents.length; i++) {
- if (!features.isHidden(parents[i], resolver, parents[i].geometry(resolver))) {
- return false;
- }
- }
-
- return true;
- };
-
- features.hasHiddenConnections = function (entity, resolver) {
- if (!_hidden.length) return false;
- var childNodes, connections;
-
- 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..
-
-
- 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));
- });
- };
-
- 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);
- };
-
- features.filter = function (d, resolver) {
- if (!_hidden.length) return d;
- var result = [];
-
- for (var i = 0; i < d.length; i++) {
- var entity = d[i];
-
- if (!features.isHidden(entity, resolver, entity.geometry(resolver))) {
- result.push(entity);
- }
- }
-
- return result;
- };
-
- features.forceVisible = function (entityIDs) {
- if (!arguments.length) return Object.keys(_forceVisible);
- _forceVisible = {};
-
- for (var i = 0; i < entityIDs.length; i++) {
- _forceVisible[entityIDs[i]] = true;
- var entity = context.hasEntity(entityIDs[i]);
-
- if (entity && entity.type === 'relation') {
- // also show relation members (one level deep)
- for (var j in entity.members) {
- _forceVisible[entity.members[j].id] = true;
- }
- }
- }
-
- return features;
- };
-
- features.init = function () {
- var storage = corePreferences('disabled-features');
-
- if (storage) {
- var storageDisabled = storage.replace(/;/g, ',').split(',');
- storageDisabled.forEach(features.disable);
- }
-
- var hash = utilStringQs(window.location.hash);
-
- 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
-
-
- 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
-
- var entities = [].concat(types.relation || [], types.way || [], types.node || []);
-
- for (var i = 0; i < entities.length; i++) {
- var geometry = entities[i].geometry(graph);
- features.getMatches(entities[i], graph, geometry);
- }
- });
-
- _deferred.add(handle);
- });
- return features;
- }
-
- //
- // - 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
- //
-
- 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;
-
- for (i = 0; i < parents.length; i++) {
- nodes = parents[i].nodes;
- isClosed = parents[i].isClosed();
-
- 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;
-
- 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
- }
- }
- }
-
- 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;
-
- 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];
-
- if (a) {
- var span = geoVecLength(a, b) - offset;
-
- 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
-
- var coord = [a, p];
-
- for (span -= dt; span >= 0; span -= dt) {
- p = geoVecAdd(p, [dx, dy]);
- coord.push(p);
- }
-
- coord.push(b); // generate svg paths
-
- var segment = '';
- var j;
-
- 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
- });
-
- if (bothDirections(entity)) {
- segment = '';
-
- 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
- });
- }
- }
-
- offset = -span;
- }
-
- 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));
- }
- };
-
- 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);
- }
- };
-
- 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;
-
- 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 getWaySegments() {
- var isActiveWay = way.nodes.indexOf(activeID) !== -1;
- var features = {
- passive: [],
- active: []
- };
- var start = {};
- var end = {};
- var node, type;
-
- 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 (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);
- }
- }
-
- start = end;
- }
-
- return features;
-
- 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 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 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'];
-
- var _tags = function _tags(entity) {
- return entity.tags;
- };
-
- var tagClasses = function tagClasses(selection) {
- selection.each(function tagClassesEach(entity) {
- var value = this.className;
-
- if (value.baseVal !== undefined) {
- value = value.baseVal;
- }
-
- var t = _tags(entity);
-
- var computed = tagClasses.getClassesString(t, value);
-
- if (computed !== value) {
- select(this).attr('class', computed);
- }
- });
- };
-
- 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
-
- var overrideGeometry;
-
- if (/\bstroke\b/.test(value)) {
- if (!!t.barrier && t.barrier !== 'no') {
- overrideGeometry = 'line';
- }
- } // preserve base classes (nothing with `tag-`)
-
-
- 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..
-
- for (i = 0; i < primaries.length; i++) {
- k = primaries[i];
- v = t[k];
- if (!v || v === 'no') continue;
-
- 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';
- }
-
- primary = k;
-
- 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);
- }
-
- break;
- }
-
- 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`
-
- 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 (!status) {
- for (i = 0; i < statuses.length; i++) {
- k = statuses[i];
- v = t[k];
- if (!v || v === 'no') continue;
-
- 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 (status) break;
- }
- }
-
- if (status) {
- classes.push('tag-status');
- classes.push('tag-status-' + status);
- } // add any secondary tags
-
-
- 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';
- }
- }
-
- classes.push('tag-' + surface);
- } // If this is a wikidata-tagged item, add a class for that..
-
-
- if (t.wikidata || t['brand:wikidata']) {
- classes.push('tag-wikidata');
- }
-
- return classes.join(' ').trim();
- };
-
- tagClasses.tags = function (val) {
- if (!arguments.length) return _tags;
- _tags = val;
- return tagClasses;
- };
-
- return tagClasses;
- }
-
- // 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;
- }
-
- for (var tag in patterns) {
- var entityValue = tags[tag];
- if (!entityValue) continue;
-
- if (typeof patterns[tag] === 'string') {
- // extra short syntax (just tag) - pattern name
- return 'pattern-' + patterns[tag];
- } else {
- var values = patterns[tag];
-
- for (var value in values) {
- if (entityValue !== value) continue;
- var rules = values[value];
-
- 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 null;
- }
-
- function svgAreas(projection, context) {
- function getPatternStyle(tags) {
- var imageID = svgTagPattern(tags);
-
- if (imageID) {
- return 'url("#ideditor-' + imageID + '")';
- }
-
- return '';
- }
-
- 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
-
- 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('.area.target-allowed').filter(function (d) {
- return filter(d.properties.entity);
- }).data(targetData, function key(d) {
- return d.id;
- }); // exit
-
- targets.exit().remove();
-
- var segmentWasEdited = function segmentWasEdited(d) {
- var wayID = d.properties.entity.id; // if the whole line was edited, don't draw segment changes
-
- if (!base.entities[wayID] || !fastDeepEqual(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
- return false;
- }
-
- 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
-
-
- 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
-
- 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
-
- 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);
- }
-
- function drawAreas(selection, graph, entities, filter) {
- var path = svgPath(projection, graph, true);
- var areas = {};
- var multipolygon;
- var base = context.history().base();
-
- for (var i = 0; i < entities.length; i++) {
- var entity = entities[i];
- if (entity.geometry(graph) !== 'area') continue;
- multipolygon = osmIsOldMultipolygonOuterMember(entity, graph);
-
- 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))
- };
- }
- }
-
- 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..
-
- 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 sortedByArea(entity) {
- if (this._parent.__data__ === 'fill') {
- return fillpaths[bisect(fillpaths, -entity.area(graph))];
- }
- }
-
- 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);
-
- 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..
-
- touchLayer.call(drawTargets, graph, data.stroke, filter);
- }
-
- return drawAreas;
- }
-
- //[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
-
- 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
-
- var S_TAG = 0; //tag name offerring
-
- var S_ATTR = 1; //attr name offerring
-
- var S_ATTR_SPACE = 2; //attr name end and space offer
-
- var S_EQ = 3; //=space?
-
- var S_ATTR_NOQUOT_VALUE = 4; //attr value(no quot value only)
-
- var S_ATTR_END = 5; //attr value end and no space(quot end)
-
- var S_TAG_SPACE = 6; //(attr value end || tag end ) && (space offer)
-
- var S_TAG_CLOSE = 7; //closed el<el />
-
- function XMLReader() {}
-
- XMLReader.prototype = {
- parse: function parse(source, defaultNSMap, entityMap) {
- var domBuilder = this.domBuilder;
- domBuilder.startDocument();
-
- _copy(defaultNSMap, defaultNSMap = {});
-
- _parse(source, defaultNSMap, entityMap, domBuilder, this.errorHandler);
-
- domBuilder.endDocument();
- }
- };
-
- 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);
- }
- }
-
- function entityReplacer(a) {
- var k = a.slice(1, -1);
-
- 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 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;
- }
- }
-
- 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)
- }
-
- locator.columnNumber = p - lineStart + 1;
- }
-
- 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;
-
- 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;
- }
-
- return;
- }
-
- if (tagStart > start) {
- appendText(tagStart);
- }
-
- switch (source.charAt(tagStart + 1)) {
- case '/':
- var end = source.indexOf('>', tagStart + 3);
- var tagName = source.substring(tagStart + 2, end);
- var config = parseStack.pop();
-
- if (end < 0) {
- tagName = source.substring(tagStart + 2).replace(/[\s<].*/, ''); //console.error('#@@@@@@'+tagName)
-
- 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);
-
-
- var localNSMap = config.localNSMap;
- var endMatch = config.tagName == tagName;
- var endIgnoreCaseMach = endMatch || config.tagName && config.tagName.toLowerCase() == tagName.toLowerCase();
-
- if (endIgnoreCaseMach) {
- domBuilder.endElement(config.uri, config.localName, tagName);
-
- if (localNSMap) {
- for (var prefix in localNSMap) {
- domBuilder.endPrefixMapping(prefix);
- }
- }
-
- if (!endMatch) {
- errorHandler.fatalError("end tag name: " + tagName + ' is not match the current start tagName:' + config.tagName);
- }
- } else {
- parseStack.push(config);
- }
-
- end++;
- break;
- // end elment
-
- case '?':
- // <?...?>
- locator && position(tagStart);
- end = parseInstruction(source, tagStart, domBuilder);
- break;
-
- case '!':
- // <!doctype,<![CDATA,<!--
- locator && position(tagStart);
- end = parseDCC(source, tagStart, domBuilder, errorHandler);
- break;
-
- default:
- locator && position(tagStart);
- var el = new ElementAttributes();
- var currentNSMap = parseStack[parseStack.length - 1].currentNSMap; //elStartEnd
-
- var end = parseElementStartPart(source, tagStart, el, currentNSMap, entityReplacer, errorHandler);
- var len = el.length;
-
- if (!el.closed && fixSelfClosed(source, end, el.tagName, closeMap)) {
- el.closed = true;
-
- if (!entityMap.nbsp) {
- errorHandler.warning('unclosed xml attribute');
- }
- }
-
- if (locator && len) {
- var locator2 = copyLocator(locator, {}); //try{//attribute position fixed
-
- for (var i = 0; i < len; i++) {
- var a = el[i];
- position(a.offset);
- a.locator = copyLocator(locator, {});
- } //}catch(e){console.error('@@@@@'+e)}
-
-
- domBuilder.locator = locator2;
-
- if (appendElement(el, domBuilder, currentNSMap)) {
- parseStack.push(el);
- }
-
- 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++;
- }
-
- }
- } catch (e) {
- errorHandler.error('element parse error: ' + e); //errorHandler.error('element parse error: '+e);
-
- end = -1; //throw e;
- }
-
- if (end > start) {
- start = end;
- } else {
- //TODO: 这里有可能sax回退,有位置错误风险
- appendText(Math.max(tagStart, start) + 1);
- }
- }
- }
-
- 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)
- */
-
-
- function parseElementStartPart(source, start, el, currentNSMap, entityReplacer, errorHandler) {
- var attrName;
- var value;
- var p = ++start;
- var s = S_TAG; //status
-
- while (true) {
- var c = source.charAt(p);
-
- 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');
- }
-
- break;
-
- 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);
- }
-
- start = p + 1;
- p = source.indexOf(c, start);
-
- 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)
-
- 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 "="');
- }
-
- break;
-
- case '/':
- switch (s) {
- case S_TAG:
- el.setTagName(source.slice(start, p));
-
- case S_ATTR_END:
- case S_TAG_SPACE:
- case S_TAG_CLOSE:
- s = S_TAG_CLOSE;
- el.closed = true;
-
- case S_ATTR_NOQUOT_VALUE:
- case S_ATTR:
- case S_ATTR_SPACE:
- break;
- //case S_EQ:
-
- default:
- throw new Error("attribute invalid close char('/')");
- }
-
- break;
-
- case '':
- //end document
- //throw new Error('unexpected end of input')
- errorHandler.error('unexpected end of input');
-
- if (s == S_TAG) {
- el.setTagName(source.slice(start, p));
- }
-
- return p;
-
- 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
-
- case S_ATTR_NOQUOT_VALUE: //Compatible state
-
- case S_ATTR:
- value = source.slice(start, p);
-
- if (value.slice(-1) === '/') {
- el.closed = true;
- value = value.slice(0, -1);
- }
-
- case S_ATTR_SPACE:
- if (s === S_ATTR_SPACE) {
- value = attrName;
- }
-
- 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!!');
- }
-
- el.add(value, value, start);
- }
-
- break;
-
- case S_EQ:
- throw new Error('attribute value missed!!');
- } // console.log(tagName,tagNamePattern,tagNamePattern.test(tagName))
-
-
- return p;
-
- /*xml space '\x20' | #x9 | #xD | #xA; */
-
- case "\x80":
- c = ' ';
-
- default:
- if (c <= ' ') {
- //space
- switch (s) {
- case S_TAG:
- el.setTagName(source.slice(start, p)); //tagName
-
- s = S_TAG_SPACE;
- break;
-
- case S_ATTR:
- attrName = source.slice(start, p);
- s = S_ATTR_SPACE;
- break;
-
- 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);
-
- 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 (currentNSMap[''] !== 'http://www.w3.org/1999/xhtml' || !attrName.match(/^(?:disabled|checked|selected)$/i)) {
- errorHandler.warning('attribute "' + attrName + '" missed value!! "' + attrName + '" instead2!!');
- }
-
- el.add(attrName, attrName, start);
- start = p;
- s = S_ATTR;
- break;
-
- case S_ATTR_END:
- errorHandler.warning('attribute space is required"' + attrName + '"!!');
-
- case S_TAG_SPACE:
- s = S_ATTR;
- start = p;
- break;
-
- case S_EQ:
- s = S_ATTR_NOQUOT_VALUE;
- start = p;
- break;
-
- case S_TAG_CLOSE:
- throw new Error("elements closed character '/' and '>' must be connected to");
- }
- }
-
- } //end outer switch
- //console.log('p++',p)
-
-
- p++;
- }
- }
- /**
- * @return true if has new namespace define
- */
-
-
- function appendElement(el, domBuilder, currentNSMap) {
- var tagName = el.tagName;
- var localNSMap = null; //var currentNSMap = parseStack[parseStack.length-1].currentNSMap;
-
- var i = el.length;
-
- while (i--) {
- var a = el[i];
- var qName = a.qName;
- var value = a.value;
- var nsp = qName.indexOf(':');
-
- 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 !== ''
-
-
- a.localName = localName; //prefix == null for no ns prefix attribute
-
- if (nsPrefix !== false) {
- //hack!!
- if (localNSMap == null) {
- localNSMap = {}; //console.log(currentNSMap,0)
-
- _copy(currentNSMap, currentNSMap = {}); //console.log(currentNSMap,1)
-
- }
-
- currentNSMap[nsPrefix] = localNSMap[nsPrefix] = value;
- a.uri = 'http://www.w3.org/2000/xmlns/';
- domBuilder.startPrefixMapping(nsPrefix, value);
- }
- }
-
- var i = el.length;
-
- while (i--) {
- a = el[i];
- var prefix = a.prefix;
-
- if (prefix) {
- //no prefix attribute has no namespace
- if (prefix === 'xml') {
- a.uri = 'http://www.w3.org/XML/1998/namespace';
- }
-
- if (prefix !== 'xmlns') {
- a.uri = currentNSMap[prefix || '']; //{console.log('###'+a.qName,domBuilder.locator.systemId+'',currentNSMap,a.uri)}
- }
- }
- }
-
- var nsp = tagName.indexOf(':');
-
- 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
-
-
- var ns = el.uri = currentNSMap[prefix || ''];
- domBuilder.startElement(ns, localName, tagName, el); //endPrefixMapping and startPrefixMapping have not any help for dom builder
- //localNSMap = null
-
- if (el.closed) {
- domBuilder.endElement(ns, localName, tagName);
-
- if (localNSMap) {
- for (prefix in localNSMap) {
- domBuilder.endPrefixMapping(prefix);
- }
- }
- } else {
- el.currentNSMap = currentNSMap;
- el.localNSMap = localNSMap; //parseStack.push(el);
-
- return true;
- }
- }
-
- 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 (/[&<]/.test(text)) {
- if (/^script$/i.test(tagName)) {
- //if(!/\]\]>/.test(text)){
- //lexHandler.startCDATA();
- domBuilder.characters(text, 0, text.length); //lexHandler.endCDATA();
-
- return elEndStart; //}
- } //}else{//text area
-
-
- text = text.replace(/&#?\w+;/g, entityReplacer);
- domBuilder.characters(text, 0, text.length);
- return elEndStart; //}
- }
- }
-
- 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 + '>');
-
- if (pos < elStartEnd) {
- //忘记闭合
- pos = source.lastIndexOf('</' + tagName);
- }
-
- closeMap[tagName] = pos;
- }
-
- return pos < elStartEnd; //}
- }
-
- function _copy(source, target) {
- for (var n in source) {
- target[n] = source[n];
- }
- }
-
- function parseDCC(source, start, domBuilder, errorHandler) {
- //sure start with '<!'
- var next = source.charAt(start + 2);
-
- switch (next) {
- case '-':
- if (source.charAt(start + 3) === '-') {
- var end = source.indexOf('-->', start + 4); //append comment source.substring(4,end)//<!--
-
- 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;
- }
-
- 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)
-
-
- var matchs = split$1(source, start);
- var len = matchs.length;
-
- 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;
- }
-
- }
-
- return -1;
- }
-
- function parseInstruction(source, start, domBuilder) {
- var end = source.indexOf('?>', start);
-
- if (end) {
- var match = source.substring(start, end).match(/^<\?(\S*)\s*([\s\S]*?)\s*$/);
-
- if (match) {
- var len = match[0].length;
- domBuilder.processingInstruction(match[1], match[2]);
- return end + 2;
- } else {
- //error
- return -1;
- }
- }
-
- return -1;
- }
- /**
- * @param source
- */
-
-
- function ElementAttributes(source) {}
-
- ElementAttributes.prototype = {
- setTagName: function setTagName(tagName) {
- if (!tagNamePattern.test(tagName)) {
- throw new Error('invalid tagName:' + tagName);
- }
-
- this.tagName = tagName;
- },
- add: function add(qName, value, offset) {
- if (!tagNamePattern.test(qName)) {
- throw new Error('invalid attribute:' + qName);
- }
-
- 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 _set_proto_(thiz, parent) {
- thiz.__proto__ = parent;
- return thiz;
- }
-
- 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];
- }
-
- return p;
- };
- }
-
- function split$1(source, start) {
- var match;
- var buf = [];
- var reg = /'[^']+'|"[^"]+"|[^\s<>\/=]+=?|(\/?\s*>|<)/g;
- reg.lastIndex = start;
- reg.exec(source); //skip <
-
- 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$1(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 _extends(Class, Super) {
- var pt = Class.prototype;
-
- if (Object.create) {
- var ppt = Object.create(Super.prototype);
- pt.__proto__ = ppt;
- }
-
- if (!(pt instanceof Super)) {
- var t = function t() {};
- t.prototype = Super.prototype;
- t = new t();
- copy$1(pt, t);
- Class.prototype = pt = t;
- }
-
- if (pt.constructor != Class) {
- if (typeof Class != 'function') {
- console.error("unknow Class:" + Class);
- }
-
- pt.constructor = Class;
- }
- }
-
- var htmlns = 'http://www.w3.org/1999/xhtml'; // Node Types
-
- 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 = {};
- 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
-
- 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);
-
- 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);
- }
-
- error.code = code;
- if (message) this.message = this.message + ": " + message;
- return error;
- }
- 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,
-
- /**
- * 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);
- }
-
- return buf.join('');
- }
- };
-
- function LiveNodeList(node, refresh) {
- this._node = node;
- this._refresh = refresh;
-
- _updateLiveList(this);
- }
-
- function _updateLiveList(list) {
- var inc = list._node._inc || list._node.ownerDocument._inc;
-
- if (list._inc != inc) {
- var ls = list._refresh(list._node); //console.log(ls.length)
-
-
- __set__(list, 'length', ls.length);
-
- copy$1(ls, list);
- list._inc = inc;
- }
- }
-
- LiveNodeList.prototype.item = function (i) {
- _updateLiveList(this);
-
- return this[i];
- };
-
- _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
- */
-
-
- function NamedNodeMap() {}
-
- function _findNodeIndex(list, node) {
- var i = list.length;
-
- while (i--) {
- if (list[i] === node) {
- return i;
- }
- }
- }
-
- function _addNamedNode(el, list, newAttr, oldAttr) {
- if (oldAttr) {
- list[_findNodeIndex(list, oldAttr)] = newAttr;
- } else {
- list[list.length++] = newAttr;
- }
-
- if (el) {
- newAttr.ownerElement = el;
- var doc = el.ownerDocument;
-
- if (doc) {
- oldAttr && _onRemoveAttribute(doc, el, oldAttr);
-
- _onAddAttribute(doc, el, newAttr);
- }
- }
- }
-
- function _removeNamedNode(el, list, attr) {
- //console.log('remove attr:'+attr)
- var i = _findNodeIndex(list, attr);
-
- if (i >= 0) {
- var lastIndex = list.length - 1;
-
- while (i < lastIndex) {
- list[i] = list[++i];
- }
-
- list.length = lastIndex;
-
- if (el) {
- var doc = el.ownerDocument;
-
- if (doc) {
- _onRemoveAttribute(doc, el, attr);
-
- attr.ownerElement = 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;
-
- while (i--) {
- var attr = this[i]; //console.log(attr.nodeName,key)
-
- if (attr.nodeName == key) {
- return attr;
- }
- }
- },
- setNamedItem: function setNamedItem(attr) {
- var el = attr.ownerElement;
-
- if (el && el != this._ownerElement) {
- throw new DOMException$2(INUSE_ATTRIBUTE_ERR);
- }
-
- var oldAttr = this.getNamedItem(attr.nodeName);
-
- _addNamedNode(this._ownerElement, this, attr, oldAttr);
-
- return oldAttr;
- },
-
- /* returns Node */
- setNamedItemNS: function setNamedItemNS(attr) {
- // raises: WRONG_DOCUMENT_ERR,NO_MODIFICATION_ALLOWED_ERR,INUSE_ATTRIBUTE_ERR
- var el = attr.ownerElement,
- oldAttr;
-
- if (el && el != this._ownerElement) {
- throw new DOMException$2(INUSE_ATTRIBUTE_ERR);
- }
-
- oldAttr = this.getNamedItemNS(attr.namespaceURI, attr.localName);
-
- _addNamedNode(this._ownerElement, this, attr, oldAttr);
-
- return oldAttr;
- },
-
- /* returns Node */
- removeNamedItem: function removeNamedItem(key) {
- var attr = this.getNamedItem(key);
-
- _removeNamedNode(this._ownerElement, this, attr);
-
- return attr;
- },
- // raises: NOT_FOUND_ERR,NO_MODIFICATION_ALLOWED_ERR
- //for level2
- removeNamedItemNS: function removeNamedItemNS(namespaceURI, localName) {
- var attr = this.getNamedItemNS(namespaceURI, localName);
-
- _removeNamedNode(this._ownerElement, this, attr);
-
- return attr;
- },
- getNamedItemNS: function getNamedItemNS(namespaceURI, localName) {
- var i = this.length;
-
- while (i--) {
- var node = this[i];
-
- if (node.localName == localName && node.namespaceURI == namespaceURI) {
- return node;
- }
- }
-
- return null;
- }
- };
- /**
- * @see http://www.w3.org/TR/REC-DOM-Level-1/level-one-core.html#ID-102161490
- */
-
- 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()];
-
- 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 (doctype) {
- doc.appendChild(doctype);
- }
-
- if (qualifiedName) {
- var root = doc.createElementNS(namespaceURI, qualifiedName);
- doc.appendChild(root);
- }
-
- 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;
-
- return node;
- }
- };
- /**
- * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html#ID-1950641247
- */
-
- 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);
-
- 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;
-
- 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;
-
- while (el) {
- var map = el._nsMap; //console.dir(map)
-
- if (map) {
- for (var n in map) {
- if (map[n] == namespaceURI) {
- return n;
- }
- }
- }
-
- el = el.nodeType == ATTRIBUTE_NODE ? el.ownerDocument : el.parentNode;
- }
-
- return null;
- },
- // Introduced in DOM Level 3:
- lookupNamespaceURI: function lookupNamespaceURI(prefix) {
- var el = this;
-
- while (el) {
- var map = el._nsMap; //console.dir(map)
-
- if (map) {
- if (prefix in map) {
- return map[prefix];
- }
- }
-
- el = el.nodeType == ATTRIBUTE_NODE ? el.ownerDocument : el.parentNode;
- }
-
- return null;
- },
- // Introduced in DOM Level 3:
- isDefaultNamespace: function isDefaultNamespace(namespaceURI) {
- var prefix = this.lookupPrefix(namespaceURI);
- return prefix == null;
- }
- };
-
- function _xmlEncoder(c) {
- return c == '<' && '<' || c == '>' && '>' || c == '&' && '&' || c == '"' && '"' || '&#' + c.charCodeAt() + ';';
- }
-
- copy$1(NodeType, Node);
- copy$1(NodeType, Node.prototype);
- /**
- * @param callback return true for continue,false for break
- * @return boolean true: break visit;
- */
-
- function _visitNode(node, callback) {
- if (callback(node)) {
- return true;
- }
-
- if (node = node.firstChild) {
- do {
- if (_visitNode(node, callback)) {
- return true;
- }
- } while (node = node.nextSibling);
- }
- }
-
- function Document() {}
-
- function _onAddAttribute(doc, el, newAttr) {
- doc && doc._inc++;
- var ns = newAttr.namespaceURI;
-
- if (ns == 'http://www.w3.org/2000/xmlns/') {
- //update namespace
- el._nsMap[newAttr.prefix ? newAttr.localName : ''] = newAttr.value;
- }
- }
-
- function _onRemoveAttribute(doc, el, newAttr, remove) {
- doc && doc._inc++;
- var ns = newAttr.namespaceURI;
-
- if (ns == 'http://www.w3.org/2000/xmlns/') {
- //update namespace
- delete el._nsMap[newAttr.prefix ? newAttr.localName : ''];
- }
- }
-
- function _onUpdateChild(doc, el, newChild) {
- if (doc && doc._inc) {
- doc._inc++; //update childNodes
-
- var cs = el.childNodes;
-
- if (newChild) {
- cs[cs.length++] = newChild;
- } else {
- //console.log(1)
- var child = el.firstChild;
- var i = 0;
-
- while (child) {
- cs[i++] = child;
- child = child.nextSibling;
- }
-
- cs.length = i;
- }
- }
- }
- /**
- * attributes;
- * children;
- *
- * writeable properties:
- * nodeValue,Attr:value,CharacterData:data
- * prefix
- */
-
-
- function _removeChild(parentNode, child) {
- var previous = child.previousSibling;
- var next = child.nextSibling;
-
- if (previous) {
- previous.nextSibling = next;
- } else {
- parentNode.firstChild = next;
- }
-
- if (next) {
- next.previousSibling = previous;
- } else {
- parentNode.lastChild = previous;
- }
-
- _onUpdateChild(parentNode.ownerDocument, parentNode);
-
- return child;
- }
- /**
- * preformance key(refChild == null)
- */
-
-
- function _insertBefore(parentNode, newChild, nextChild) {
- var cp = newChild.parentNode;
-
- if (cp) {
- cp.removeChild(newChild); //remove and update
- }
-
- if (newChild.nodeType === DOCUMENT_FRAGMENT_NODE) {
- var newFirst = newChild.firstChild;
-
- if (newFirst == null) {
- return newChild;
- }
-
- var newLast = newChild.lastChild;
- } else {
- newFirst = newLast = newChild;
- }
-
- var pre = nextChild ? nextChild.previousSibling : parentNode.lastChild;
- newFirst.previousSibling = pre;
- newLast.nextSibling = nextChild;
-
- if (pre) {
- pre.nextSibling = newFirst;
- } else {
- parentNode.firstChild = newFirst;
- }
-
- if (nextChild == null) {
- parentNode.lastChild = newLast;
- } else {
- nextChild.previousSibling = newLast;
- }
-
- do {
- newFirst.parentNode = parentNode;
- } while (newFirst !== newLast && (newFirst = newFirst.nextSibling));
-
- _onUpdateChild(parentNode.ownerDocument || parentNode, parentNode); //console.log(parentNode.lastChild.nextSibling == null)
-
-
- if (newChild.nodeType == DOCUMENT_FRAGMENT_NODE) {
- newChild.firstChild = newChild.lastChild = null;
- }
-
- return newChild;
- }
-
- function _appendSingleChild(parentNode, newChild) {
- var cp = newChild.parentNode;
-
- if (cp) {
- var pre = parentNode.lastChild;
- cp.removeChild(newChild); //remove and update
-
- var pre = parentNode.lastChild;
- }
-
- 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 newChild;
- }
-
- if (this.documentElement == null && newChild.nodeType == ELEMENT_NODE) {
- this.documentElement = newChild;
- }
-
- return _insertBefore(this, newChild, refChild), newChild.ownerDocument = this, newChild;
- },
- removeChild: function removeChild(oldChild) {
- if (this.documentElement == oldChild) {
- this.documentElement = null;
- }
-
- 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;
-
- if (pl.length == 2) {
- node.prefix = pl[0];
- node.localName = pl[1];
- } else {
- //el.prefix = null;
- node.localName = qualifiedName;
- }
-
- 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;
-
- if (pl.length == 2) {
- node.prefix = pl[0];
- node.localName = pl[1];
- } else {
- //el.prefix = null;
- node.localName = qualifiedName;
- }
-
- return node;
- }
- };
-
- _extends(Document, Node);
-
- 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 = [];
-
- _visitNode(base, function (node) {
- if (node !== base && node.nodeType == ELEMENT_NODE && (tagName === '*' || node.tagName == tagName)) {
- ls.push(node);
- }
- });
-
- return ls;
- });
- },
- getElementsByTagNameNS: function getElementsByTagNameNS(namespaceURI, localName) {
- return new LiveNodeList(this, function (base) {
- var ls = [];
-
- _visitNode(base, function (node) {
- if (node !== base && node.nodeType === ELEMENT_NODE && (namespaceURI === '*' || node.namespaceURI === namespaceURI) && (localName === '*' || node.localName == localName)) {
- ls.push(node);
- }
- });
-
- return ls;
- });
- }
- };
- Document.prototype.getElementsByTagName = Element.prototype.getElementsByTagName;
- Document.prototype.getElementsByTagNameNS = Element.prototype.getElementsByTagNameNS;
-
- _extends(Element, Node);
-
- function Attr() {}
- Attr.prototype.nodeType = ATTRIBUTE_NODE;
-
- _extends(Attr, Node);
-
- 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);
- }
-
- return newNode;
- }
- };
-
- _extends(Text, CharacterData);
-
- function Comment() {}
- Comment.prototype = {
- nodeName: "#comment",
- nodeType: COMMENT_NODE
- };
-
- _extends(Comment, CharacterData);
-
- function CDATASection() {}
- CDATASection.prototype = {
- nodeName: "#cdata-section",
- nodeType: CDATA_SECTION_NODE
- };
-
- _extends(CDATASection, CharacterData);
-
- function DocumentType() {}
- DocumentType.prototype.nodeType = DOCUMENT_TYPE_NODE;
-
- _extends(DocumentType, Node);
-
- function Notation() {}
- Notation.prototype.nodeType = NOTATION_NODE;
-
- _extends(Notation, Node);
-
- function Entity() {}
- Entity.prototype.nodeType = ENTITY_NODE;
-
- _extends(Entity, Node);
-
- function EntityReference() {}
- EntityReference.prototype.nodeType = ENTITY_REFERENCE_NODE;
-
- _extends(EntityReference, Node);
-
- function DocumentFragment() {}
- DocumentFragment.prototype.nodeName = "#document-fragment";
- DocumentFragment.prototype.nodeType = DOCUMENT_FRAGMENT_NODE;
-
- _extends(DocumentFragment, Node);
-
- function ProcessingInstruction() {}
-
- ProcessingInstruction.prototype.nodeType = PROCESSING_INSTRUCTION_NODE;
-
- _extends(ProcessingInstruction, Node);
-
- function XMLSerializer$1() {}
-
- XMLSerializer$1.prototype.serializeToString = function (node, isHtml, nodeFilter) {
- return nodeSerializeToString.call(node, isHtml, nodeFilter);
- };
-
- Node.prototype.toString = nodeSerializeToString;
-
- function nodeSerializeToString(isHtml, nodeFilter) {
- var buf = [];
- var refNode = this.nodeType == 9 ? this.documentElement : this;
- var prefix = refNode.prefix;
- var uri = refNode.namespaceURI;
-
- if (uri && prefix == null) {
- //console.log(prefix)
- var prefix = refNode.lookupPrefix(uri);
-
- if (prefix == null) {
- //isHTML = true;
- var visibleNamespaces = [{
- namespace: uri,
- prefix: null
- } //{namespace:uri,prefix:''}
- ];
- }
- }
-
- serializeToString(this, buf, isHtml, nodeFilter, visibleNamespaces); //console.log('###',this.nodeType,uri,prefix,buf.join(''))
-
- return buf.join('');
- }
-
- function needNamespaceDefine(node, isHTML, visibleNamespaces) {
- var prefix = node.prefix || '';
- var uri = node.namespaceURI;
-
- if (!prefix && !uri) {
- return false;
- }
-
- if (prefix === "xml" && uri === "http://www.w3.org/XML/1998/namespace" || uri == 'http://www.w3.org/2000/xmlns/') {
- return false;
- }
-
- var i = visibleNamespaces.length; //console.log('@@@@',node.tagName,prefix,uri,visibleNamespaces)
-
- while (i--) {
- var ns = visibleNamespaces[i]; // get namespace prefix
- //console.log(node.nodeType,node.tagName,ns.prefix,prefix)
-
- 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)
-
-
- return true;
- }
-
- function serializeToString(node, buf, isHTML, nodeFilter, visibleNamespaces) {
- if (nodeFilter) {
- node = nodeFilter(node);
-
- if (node) {
- if (typeof node == 'string') {
- buf.push(node);
- return;
- }
- } else {
- return;
- } //buf.sort.apply(attrs, attributeSorter);
-
- }
-
- 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);
-
- if (attr.prefix == 'xmlns') {
- visibleNamespaces.push({
- prefix: attr.localName,
- namespace: attr.value
- });
- } else if (attr.nodeName == 'xmlns') {
- visibleNamespaces.push({
- prefix: '',
- namespace: attr.value
- });
- }
- }
-
- for (var i = 0; i < len; i++) {
- var attr = attrs.item(i);
-
- 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
- });
- }
-
- serializeToString(attr, buf, isHTML, nodeFilter, visibleNamespaces);
- } // add namespace for current node
-
-
- 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
- });
- }
-
- 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);
- }
-
- child = child.nextSibling;
- }
- } else {
- while (child) {
- serializeToString(child, buf, isHTML, nodeFilter, visibleNamespaces);
- child = child.nextSibling;
- }
- }
-
- buf.push('</', nodeName, '>');
- } else {
- buf.push('/>');
- } // remove added visible namespaces
- //visibleNamespaces.length = startVisibleNamespaces;
-
-
- return;
-
- case DOCUMENT_NODE:
- case DOCUMENT_FRAGMENT_NODE:
- var child = node.firstChild;
-
- while (child) {
- serializeToString(child, buf, isHTML, nodeFilter, visibleNamespaces);
- child = child.nextSibling;
- }
-
- 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, ']]>');
-
- 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);
-
- if (pubid) {
- buf.push(' PUBLIC "', pubid);
-
- if (sysid && sysid != '.') {
- buf.push('" "', sysid);
- }
-
- buf.push('">');
- } else if (sysid && sysid != '.') {
- buf.push(' SYSTEM "', sysid, '">');
- } else {
- var sub = node.internalSubset;
-
- if (sub) {
- buf.push(" [", sub, "]");
- }
-
- buf.push(">");
- }
-
- return;
-
- case PROCESSING_INSTRUCTION_NODE:
- return buf.push("<?", node.target, " ", node.data, "?>");
-
- case ENTITY_REFERENCE_NODE:
- return buf.push('&', node.nodeName, ';');
- //case ENTITY_NODE:
- //case NOTATION_NODE:
-
- default:
- buf.push('??', node.nodeName);
- }
- }
-
- function _importNode(doc, node, deep) {
- var node2;
-
- 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;
-
- 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 (!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 node2;
- } //
- //var _relationMap = {firstChild:1,lastChild:1,previousSibling:1,nextSibling:1,
- // attributes:1,childNodes:1,parentNode:1,documentElement:1,doctype,};
-
-
- 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;
- }
- }
- }
-
- if (node.childNodes) {
- node2.childNodes = new NodeList();
- }
-
- node2.ownerDocument = doc;
-
- switch (node2.nodeType) {
- case ELEMENT_NODE:
- var attrs = node.attributes;
- var attrs2 = node2.attributes = new NamedNodeMap();
- var len = attrs.length;
- attrs2._ownerElement = node2;
-
- for (var i = 0; i < len; i++) {
- node2.setAttributeNode(_cloneNode(doc, attrs.item(i), true));
- }
-
- break;
-
- case ATTRIBUTE_NODE:
- deep = true;
- }
-
- if (deep) {
- var child = node.firstChild;
-
- while (child) {
- node2.appendChild(_cloneNode(doc, child, deep));
- child = child.nextSibling;
- }
- }
-
- return node2;
- }
-
- function __set__(object, key, value) {
- object[key] = value;
- } //do dynamic
-
-
- try {
- if (Object.defineProperty) {
- var getTextContent = function getTextContent(node) {
- switch (node.nodeType) {
- case ELEMENT_NODE:
- case DOCUMENT_FRAGMENT_NODE:
- var buf = [];
- node = node.firstChild;
-
- while (node) {
- if (node.nodeType !== 7 && node.nodeType !== 8) {
- buf.push(getTextContent(node));
- }
-
- node = node.nextSibling;
- }
-
- return buf.join('');
-
- default:
- return node.nodeValue;
- }
- };
-
- 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;
- }
- }
- });
-
- __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; //}
-
- var dom = {
- DOMImplementation: DOMImplementation_1,
- XMLSerializer: XMLSerializer_1
- };
-
- var domParser = createCommonjsModule(function (module, exports) {
- function DOMParser(options) {
- this.options = options || {
- locator: {}
- };
- }
-
- DOMParser.prototype.parseFromString = function (source, mimeType) {
- var options = this.options;
- var sax = new XMLReader();
- var domBuilder = options.domBuilder || new DOMHandler(); //contentHandler and LexicalHandler
-
- var errorHandler = options.errorHandler;
- var locator = options.locator;
- var defaultNSMap = options.xmlns || {};
- var entityMap = {
- 'lt': '<',
- 'gt': '>',
- 'amp': '&',
- 'quot': '"',
- 'apos': "'"
- };
-
- if (locator) {
- domBuilder.setDocumentLocator(locator);
- }
-
- 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';
- }
-
- 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");
- }
-
- return domBuilder.doc;
- };
-
- 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];
-
- 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 () {};
- }
-
- 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 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
- */
-
-
- DOMHandler.prototype = {
- startDocument: function startDocument() {
- this.doc = new DOMImplementation().createDocument(null, null, null);
-
- 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);
-
- 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;
- 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)
-
- if (chars) {
- if (this.cdata) {
- var charNode = this.doc.createCDATASection(chars);
- } 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
- }
-
- 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);
- }
- },
-
- /**
- * @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 _locator(l) {
- if (l) {
- return '\n@' + (l.systemId || '') + '#[line:' + l.lineNumber + ',col:' + l.columnNumber + ']';
- }
- }
-
- 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 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) {};
- */
-
-
- "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 */
-
- 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'){
-
-
- var XMLReader = sax.XMLReader;
- var DOMImplementation = exports.DOMImplementation = dom.DOMImplementation;
- exports.XMLSerializer = dom.XMLSerializer;
- exports.DOMParser = DOMParser; //}
- });
-
- var togeojson = createCommonjsModule(function (module, exports) {
- var toGeoJSON = function () {
-
- 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;
-
- for (var i = 0, h = 0; i < x.length; i++) {
- h = (h << 5) - h + x.charCodeAt(i) | 0;
- }
-
- return h;
- } // all Y children of X
-
-
- function get(x, y) {
- return x.getElementsByTagName(y);
- }
-
- function attr(x, y) {
- return x.getAttribute(y);
- }
-
- function attrf(x, y) {
- return parseFloat(attr(x, y));
- } // one Y child of X, if any, otherwise null
-
-
- 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
-
-
- function norm(el) {
- if (el.normalize) {
- el.normalize();
- }
-
- return el;
- } // cast array x into numbers
-
-
- function numarray(x) {
- for (var j = 0, o = []; j < x.length; j++) {
- o[j] = parseFloat(x[j]);
- }
-
- return o;
- } // get the content of a text node, if any
-
-
- function nodeVal(x) {
- if (x) {
- norm(x);
- }
-
- return x && x.textContent || '';
- } // get the contents of multiple text nodes, if present
-
-
- function getMulti(x, ys) {
- var o = {},
- n,
- k;
-
- for (k = 0; k < ys.length; k++) {
- n = get1(x, ys[k]);
- if (n) o[ys[k]] = nodeVal(n);
- }
-
- return o;
- } // add properties of Y to X, overwriting if present in both
-
-
- function extend(x, y) {
- for (var k in y) {
- x[k] = y[k];
- }
- } // get one coordinate from a coordinate array, if any
-
-
- function coord1(v) {
- return numarray(v.replace(removeSpace, '').split(','));
- } // get all coordinates from a coordinate array as [[],[]]
-
-
- function coord(v) {
- var coords = v.replace(trimSpace, '').split(splitSpace),
- o = [];
-
- for (var i = 0; i < coords.length; i++) {
- o.push(coord1(coords[i]));
- }
-
- return o;
- }
-
- 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));
-
- if (!isNaN(e)) {
- ll.push(e);
- }
- }
-
- return {
- coordinates: ll,
- time: time ? nodeVal(time) : null,
- heartRate: heartRate ? parseFloat(nodeVal(heartRate)) : null
- };
- } // create a new feature collection parent object
-
-
- function fc() {
- return {
- type: 'FeatureCollection',
- features: []
- };
- }
-
- var serializer;
-
- 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 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
-
- /* istanbul ignore next */
- if (str.xml !== undefined) return str.xml;
- return serializer.serializeToString(str);
- }
-
- 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');
-
- 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];
- }
-
- 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'));
- }
-
- styleMapIndex['#' + attr(styleMaps[l], 'id')] = pairsMap;
- }
-
- for (var j = 0; j < placemarks.length; j++) {
- gj.features = gj.features.concat(getPlacemark(placemarks[j]));
- }
-
- function kmlColor(v) {
- var color, opacity;
- v = v || '';
-
- if (v.substr(0, 1) === '#') {
- v = v.substr(1);
- }
-
- if (v.length === 6 || v.length === 3) {
- color = v;
- }
-
- 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);
- }
-
- return [color, isNaN(opacity) ? undefined : opacity];
- }
-
- function gxCoord(v) {
- return numarray(v.split(' '));
- }
-
- function gxCoords(root) {
- var elems = get(root, 'coord'),
- coords = [],
- times = [];
- if (elems.length === 0) elems = get(root, 'gx:coord');
-
- for (var i = 0; i < elems.length; i++) {
- coords.push(gxCoord(nodeVal(elems[i])));
- }
-
- var timeElems = get(root, 'when');
-
- for (var j = 0; j < timeElems.length; j++) {
- times.push(nodeVal(timeElems[j]));
- }
-
- return {
- coords: coords,
- times: times
- };
- }
-
- function getGeometry(root) {
- var geomNode,
- geomNodes,
- i,
- j,
- k,
- geoms = [],
- coordTimes = [];
-
- if (get1(root, 'MultiGeometry')) {
- return getGeometry(get1(root, 'MultiGeometry'));
- }
-
- if (get1(root, 'MultiTrack')) {
- return getGeometry(get1(root, 'MultiTrack'));
- }
-
- if (get1(root, 'gx:MultiTrack')) {
- return getGeometry(get1(root, 'gx:MultiTrack'));
- }
-
- for (i = 0; i < geotypes.length; i++) {
- geomNodes = get(root, geotypes[i]);
-
- if (geomNodes) {
- for (j = 0; j < geomNodes.length; j++) {
- geomNode = geomNodes[j];
-
- 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 = [];
-
- 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);
- }
- }
- }
- }
-
- return {
- geoms: geoms,
- coordTimes: coordTimes
- };
- }
-
- 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;
-
- if (styleUrl) {
- if (styleUrl[0] !== '#') {
- styleUrl = '#' + styleUrl;
- }
-
- properties.styleUrl = styleUrl;
-
- 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');
- }
- }
-
- 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
- };
- }
-
- if (timeStamp) {
- properties.timestamp = nodeVal(get1(timeStamp, 'when'));
- }
-
- 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 (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 (extendedData) {
- var datas = get(extendedData, 'Data'),
- simpleDatas = get(extendedData, 'SimpleData');
-
- for (i = 0; i < datas.length; i++) {
- properties[datas[i].getAttribute('name')] = nodeVal(get1(datas[i], 'value'));
- }
-
- for (i = 0; i < simpleDatas.length; i++) {
- properties[simpleDatas[i].getAttribute('name')] = nodeVal(simpleDatas[i]);
- }
- }
-
- if (visibility) {
- properties.visibility = nodeVal(visibility);
- }
-
- if (geomsAndTimes.coordTimes.length) {
- properties.coordTimes = geomsAndTimes.coordTimes.length === 1 ? geomsAndTimes.coordTimes[0] : geomsAndTimes.coordTimes;
- }
-
- 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];
- }
-
- 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;
-
- for (i = 0; i < tracks.length; i++) {
- feature = getTrack(tracks[i]);
- if (feature) gj.features.push(feature);
- }
-
- for (i = 0; i < routes.length; i++) {
- feature = getRoute(routes[i]);
- if (feature) gj.features.push(feature);
- }
-
- for (i = 0; i < waypoints.length; i++) {
- gj.features.push(getPoint(waypoints[i]));
- }
-
- 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);
- }
-
- return {
- line: line,
- times: times,
- heartRates: heartRates
- };
- }
-
- function getTrack(node) {
- var segments = get(node, 'trkseg'),
- track = [],
- times = [],
- heartRates = [],
- line;
-
- for (var i = 0; i < segments.length; i++) {
- line = getPoints(segments[i], 'trkpt');
-
- 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);
- }
- }
-
- 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 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 getPoint(node) {
- var prop = getProperties(node);
- extend(prop, getMulti(node, ['sym']));
- return {
- type: 'Feature',
- properties: prop,
- geometry: {
- type: 'Point',
- coordinates: coordPair(node).coordinates
- }
- };
- }
-
- function getLineStyle(extensions) {
- var style = {};
-
- if (extensions) {
- var lineStyle = get1(extensions, 'line');
-
- 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 (!isNaN(width)) style['stroke-width'] = width * 96 / 25.4;
- }
- }
-
- return style;
- }
-
- function getProperties(node) {
- var prop = getMulti(node, ['name', 'cmt', 'desc', 'type', 'time', 'keywords']),
- links = get(node, 'link');
- if (links.length) prop.links = [];
-
- 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);
- }
-
- return prop;
- }
-
- return gj;
- }
- };
- return t;
- }();
-
- module.exports = toGeoJSON;
- });
-
- var _initialized = false;
- var _enabled = false;
-
- var _geojson;
-
- function svgData(projection, context, dispatch) {
- var throttledRedraw = throttle(function () {
- dispatch.call('change');
- }, 1000);
-
- var _showLabels = true;
- var detected = utilDetect();
- var layer = select(null);
-
- var _vtService;
-
- var _fileList;
-
- var _template;
-
- var _src;
-
- function init() {
- if (_initialized) return; // run once
-
- _geojson = {};
- _enabled = true;
-
- 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;
- }
-
- function getService() {
- if (services.vectorTile && !_vtService) {
- _vtService = services.vectorTile;
-
- _vtService.event.on('loadedData', throttledRedraw);
- } else if (!services.vectorTile && _vtService) {
- _vtService = null;
- }
-
- return _vtService;
- }
-
- function showLayer() {
- layerOn();
- layer.style('opacity', 0).transition().duration(250).style('opacity', 1).on('end', function () {
- dispatch.call('change');
- });
- }
-
- function hideLayer() {
- throttledRedraw.cancel();
- layer.transition().duration(250).style('opacity', 0).on('end', layerOff);
- }
-
- function layerOn() {
- layer.style('display', 'block');
- }
-
- function layerOff() {
- layer.selectAll('.viewfield-group').remove();
- layer.style('display', 'none');
- } // ensure that all geojson features in a collection have IDs
-
-
- function ensureIDs(gj) {
- if (!gj) return null;
-
- if (gj.type === 'FeatureCollection') {
- for (var i = 0; i < gj.features.length; i++) {
- ensureFeatureID(gj.features[i]);
- }
- } else {
- ensureFeatureID(gj);
- }
-
- return gj;
- } // ensure that each single Feature object has a unique ID
-
-
- function ensureFeatureID(feature) {
- if (!feature) return;
- feature.__featurehash__ = utilHashcode(fastJsonStableStringify(feature));
- return feature;
- } // Prefer an array of Features instead of a FeatureCollection
-
-
- function getFeatures(gj) {
- if (!gj) return [];
-
- if (gj.type === 'FeatureCollection') {
- return gj.features;
- } else {
- return [gj];
- }
- }
-
- function featureKey(d) {
- return d.__featurehash__;
- }
-
- function isPolygon(d) {
- return d.geometry.type === 'Polygon' || d.geometry.type === 'MultiPolygon';
- }
-
- function clipPathID(d) {
- return 'ideditor-data-' + d.__featurehash__ + '-clippath';
- }
-
- function featureClasses(d) {
- return ['data' + d.__featurehash__, d.geometry.type, isPolygon(d) ? 'area' : '', d.__layerID__ || ''].filter(Boolean).join(' ');
- }
-
- 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 geoData, polygonData;
-
- 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);
- }
-
- geoData = geoData.filter(getPath);
- polygonData = geoData.filter(isPolygon); // Draw clip paths for polygons
-
- 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
-
- 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
-
- var pathData = {
- fill: polygonData,
- shadow: geoData,
- stroke: geoData
- };
- var paths = datagroups.selectAll('path').data(function (layer) {
- return pathData[layer];
- }, featureKey); // exit
-
- paths.exit().remove(); // enter/update
-
- 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
-
- layer.call(drawLabels, 'label-halo', geoData).call(drawLabels, 'label', geoData);
-
- 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
-
- labels.exit().remove(); // enter/update
-
- 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];
- });
- }
- }
-
- 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 xmlToDom(textdata) {
- return new DOMParser().parseFromString(textdata, 'text/xml');
- }
-
- 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;
-
- case '.geojson':
- case '.json':
- gj = JSON.parse(data);
- break;
- }
-
- gj = gj || {};
-
- if (Object.keys(gj).length) {
- _geojson = ensureIDs(gj);
- _src = extension + ' data file';
- this.fitZoom();
- }
-
- dispatch.call('change');
- return this;
- };
-
- drawData.showLabels = function (val) {
- if (!arguments.length) return _showLabels;
- _showLabels = val;
- return this;
- };
-
- drawData.enabled = function (val) {
- if (!arguments.length) return _enabled;
- _enabled = val;
-
- if (_enabled) {
- showLayer();
- } else {
- hideLayer();
- }
-
- dispatch.call('change');
- return this;
- };
-
- drawData.hasData = function () {
- var gj = _geojson || {};
- return !!(_template || Object.keys(gj).length);
- };
-
- drawData.template = function (val, src) {
- if (!arguments.length) return _template; // test source against OSM imagery blocklists..
-
- var osm = context.connection();
-
- if (osm) {
- var blocklists = osm.imageryBlocklists();
- var fail = false;
- var tested = 0;
- var regex;
-
- for (var i = 0; i < blocklists.length; i++) {
- regex = blocklists[i];
- fail = regex.test(val);
- tested++;
- if (fail) break;
- } // ensure at least one test was run.
-
-
- if (!tested) {
- regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
- fail = regex.test(val);
- }
- }
-
- _template = val;
- _fileList = null;
- _geojson = null; // strip off the querystring/hash from the template,
- // it often includes the access token
-
- _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';
- }
-
- dispatch.call('change');
- return this;
- };
-
- 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();
-
- reader.onload = function () {
- return function (e) {
- drawData.setFile(extension, e.target.result);
- };
- }();
-
- reader.readAsText(f);
- return this;
- };
-
- drawData.url = function (url, defaultExtension) {
- _template = null;
- _fileList = null;
- _geojson = null;
- _src = null; // strip off any querystring/hash from the url before checking extension
-
- var testUrl = url.split(/[?#]/)[0];
- var extension = getExtension(testUrl) || defaultExtension;
-
- if (extension) {
- _template = null;
- d3_text(url).then(function (data) {
- drawData.setFile(extension, data);
- })["catch"](function () {
- /* ignore */
- });
- } else {
- drawData.template(url);
- }
-
- return this;
- };
-
- drawData.getSrc = function () {
- return _src || '';
- };
-
- 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 */
-
- switch (geom.type) {
- case 'Point':
- c = [c];
-
- case 'MultiPoint':
- case 'LineString':
- break;
-
- case 'MultiPolygon':
- c = utilArrayFlatten(c);
-
- case 'Polygon':
- case 'MultiLineString':
- c = utilArrayFlatten(c);
- break;
- }
- /* eslint-enable no-fallthrough */
-
-
- 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 this;
- };
-
- init();
- return drawData;
- }
-
- 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 = [];
-
- if (showTile) {
- debugData.push({
- "class": 'red',
- label: 'tile'
- });
- }
-
- if (showCollision) {
- debugData.push({
- "class": 'yellow',
- label: 'collision'
- });
- }
-
- if (showImagery) {
- debugData.push({
- "class": 'orange',
- label: 'imagery'
- });
- }
-
- if (showTouchTargets) {
- debugData.push({
- "class": 'pink',
- label: 'touchTargets'
- });
- }
-
- if (showDownloaded) {
- debugData.push({
- "class": 'purple',
- label: 'downloaded'
- });
- }
-
- 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 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.
-
-
- 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;
- }
- };
-
- return drawDebug;
- }
-
- /*
- 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.
- */
-
- function svgDefs(context) {
- var _defsSelection = select(null);
-
- var _spritesheetIds = ['iD-sprite', 'maki-sprite', 'temaki-sprite', 'fa-sprite', 'community-sprite'];
-
- 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
-
- 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');
-
- _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
-
-
- 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');
-
- 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
-
- _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
-
-
- addSprites(_spritesheetIds, true);
- }
-
- function addSprites(ids, overrideColors) {
- _spritesheetIds = utilArrayUniq(_spritesheetIds.concat(ids));
-
- var spritesheets = _defsSelection.selectAll('.spritesheet').data(_spritesheetIds);
-
- 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 (overrideColors && d !== 'iD-sprite') {
- // allow icon colors to be overridden..
- select(node).selectAll('path').attr('fill', 'currentColor');
- }
- })["catch"](function () {
- /* ignore */
- });
- });
- spritesheets.exit().remove();
- }
-
- drawDefs.addSprites = addSprites;
- return drawDefs;
- }
-
- var _layerEnabled = false;
-
- var _qaService;
-
- function svgKeepRight(projection, context, dispatch) {
- var throttledRedraw = throttle(function () {
- return dispatch.call('change');
- }, 1000);
-
- 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(-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 getService() {
- if (services.keepRight && !_qaService) {
- _qaService = services.keepRight;
-
- _qaService.on('loaded', throttledRedraw);
- } else if (!services.keepRight && _qaService) {
- _qaService = null;
- }
-
- return _qaService;
- } // Show the markers
-
-
- function editOn() {
- if (!layerVisible) {
- layerVisible = true;
- drawLayer.style('display', 'block');
- }
- } // Immediately remove the markers and their touch targets
-
-
- 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.
-
-
- 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 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
-
-
- 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 markers = drawLayer.selectAll('.qaItem.keepRight').data(data, function (d) {
- return d.id;
- }); // exit
-
- 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.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
-
- 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.keepRight').data(data, function (d) {
- return d.id;
- }); // exit
-
- targets.exit().remove(); // enter/update
-
- 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 drawKeepRight(selection) {
- var service = getService();
- var surface = context.surface();
-
- if (surface && !surface.empty()) {
- touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
- }
-
- 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);
-
- if (_layerEnabled) {
- if (service && ~~context.map().zoom() >= minZoom) {
- editOn();
- service.loadIssues(projection);
- updateMarkers();
- } else {
- editOff();
- }
- }
- } // Toggles the layer on and off
-
-
- drawKeepRight.enabled = function (val) {
- if (!arguments.length) return _layerEnabled;
- _layerEnabled = val;
-
- if (_layerEnabled) {
- layerOn();
- } else {
- layerOff();
-
- if (context.selectedErrorID()) {
- context.enter(modeBrowse(context));
- }
- }
-
- dispatch.call('change');
- return this;
- };
-
- drawKeepRight.supported = function () {
- return !!getService();
- };
-
- return drawKeepRight;
- }
-
- function svgGeolocate(projection) {
- var layer = select(null);
-
- var _position;
-
- function init() {
- if (svgGeolocate.initialized) return; // run once
-
- svgGeolocate.enabled = false;
- svgGeolocate.initialized = true;
- }
-
- function showLayer() {
- layer.style('display', 'block');
- }
-
- function hideLayer() {
- layer.transition().duration(250).style('opacity', 0);
- }
-
- function layerOn() {
- layer.style('opacity', 0).transition().duration(250).style('opacity', 1);
- }
-
- function layerOff() {
- layer.style('display', 'none');
- }
-
- function transform(d) {
- return svgPointTransform(projection)(d);
- }
-
- 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 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));
- }
-
- 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 (enabled) {
- update();
- } else {
- layerOff();
- }
- }
-
- drawLocation.enabled = function (position, enabled) {
- if (!arguments.length) return svgGeolocate.enabled;
- _position = position;
- svgGeolocate.enabled = enabled;
-
- if (svgGeolocate.enabled) {
- showLayer();
- layerOn();
- } else {
- hideLayer();
- }
-
- 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 _rskipped = new RBush();
-
- var _textWidthCache = {};
- var _entitybboxes = {}; // Listed from highest to lowest priority
-
- 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;
- });
- }
-
- function get(array, prop) {
- return function (d, i) {
- return array[i][prop];
- };
- }
-
- 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);
- }
- }
- }
-
- function drawLinePaths(selection, entities, filter, classes, labels) {
- var paths = selection.selectAll('path').filter(filter).data(entities, osmEntity.key); // exit
-
- paths.exit().remove(); // enter/update
-
- 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'));
- }
-
- function drawLineLabels(selection, entities, filter, classes, labels) {
- var texts = selection.selectAll('text.' + classes).filter(filter).data(entities, osmEntity.key); // exit
-
- texts.exit().remove(); // enter
-
- 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
-
- 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 drawPointLabels(selection, entities, filter, classes, labels) {
- var texts = selection.selectAll('text.' + classes).filter(filter).data(entities, osmEntity.key); // exit
-
- 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');
- }
- }
-
- function drawAreaIcons(selection, entities, filter, classes, labels) {
- var icons = selection.selectAll('use.' + classes).filter(filter).data(entities, osmEntity.key); // exit
-
- icons.exit().remove(); // enter/update
-
- 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 (!picon) {
- return '';
- } else {
- var isMaki = /^maki-/.test(picon);
- return '#' + picon + (isMaki ? '-15' : '');
- }
- });
- }
-
- 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]]]
- };
- });
- }
-
- 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([]);
- }
-
- if (fullRedraw) {
- _rdrawn.clear();
-
- _rskipped.clear();
-
- _entitybboxes = {};
- } else {
- for (i = 0; i < entities.length; i++) {
- entity = entities[i];
- var toRemove = [].concat(_entitybboxes[entity.id] || []).concat(_entitybboxes[entity.id + 'I'] || []);
-
- for (j = 0; j < toRemove.length; j++) {
- _rdrawn.remove(toRemove[j]);
-
- _rskipped.remove(toRemove[j]);
- }
- }
- } // 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
- } else {
- renderNodeAs[entity.id] = 'vertex';
- markerPadding = 0;
- }
-
- 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;
- }
- }
- }
-
- 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;
-
- 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);
- }
-
- if (p) {
- if (geometry === 'vertex') {
- geometry = 'point';
- } // treat vertex like point
-
-
- p.classes = geometry + ' tag-' + labelStack[k][1];
- positions[geometry].push(p);
- labelled[geometry].push(entity);
- }
- }
- }
-
- 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 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;
-
- 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
- };
- }
-
- if (tryInsert([bbox], entity.id, true)) {
- return p;
- }
- }
-
- 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();
- }
-
- var bboxes = [];
- var boxsize = (height + 2) / 2;
-
- 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
-
- 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 + '%'
- };
- }
- }
-
- 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 lineString(points) {
- return 'M' + points.join('L');
- }
-
- function subpath(points, from, to) {
- var sofar = 0;
- var start, end, i0, i1;
-
- 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 (!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 (!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;
- }
-
- sofar += current;
- }
-
- var result = points.slice(i0, i1);
- result.unshift(start);
- result.push(end);
- return result;
- }
- }
-
- 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 = {};
-
- if (picon) {
- // icon and label..
- if (addIcon()) {
- addLabel(iconSize + padding);
- return p;
- }
- } else {
- // label only..
- if (addLabel(0)) {
- return p;
- }
- }
-
- 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
- };
-
- if (tryInsert([bbox], entity.id + 'I', true)) {
- p.transform = 'translate(' + iconX + ',' + iconY + ')';
- return true;
- }
-
- return false;
- }
-
- 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
- };
-
- if (tryInsert([bbox], entity.id, true)) {
- p.x = labelX;
- p.y = labelY;
- p.textAnchor = 'middle';
- p.height = height;
- return true;
- }
- }
-
- return false;
- }
- } // force insert a singular bounding box
- // singular box only, no array, id better be unique
-
-
- function doInsert(bbox, id) {
- bbox.id = id;
- var oldbox = _entitybboxes[id];
-
- if (oldbox) {
- _rdrawn.remove(oldbox);
- }
-
- _entitybboxes[id] = bbox;
-
- _rdrawn.insert(bbox);
- }
-
- function tryInsert(bboxes, id, saveSkipped) {
- var skipped = false;
-
- for (var i = 0; i < bboxes.length; i++) {
- var bbox = bboxes[i];
- bbox.id = id; // Check that label is visible
-
- 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;
- }
- }
-
- _entitybboxes[id] = bboxes;
-
- if (skipped) {
- if (saveSkipped) {
- _rskipped.load(bboxes);
- }
- } else {
- _rdrawn.load(bboxes);
- }
-
- return !skipped;
- }
-
- 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
-
- drawPointLabels(label, labelled.point, filter, 'pointlabel', positions.point);
- drawPointLabels(halo, labelled.point, filter, 'pointlabel-halo', positions.point); // lines
-
- 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
-
- 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);
- }
-
- 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
-
- if (mouse) {
- pad = 20;
- bbox = {
- minX: mouse[0] - pad,
- minY: mouse[1] - pad,
- maxX: mouse[0] + pad,
- maxY: mouse[1] + pad
- };
-
- var nearMouse = _rdrawn.search(bbox).map(function (entity) {
- return entity.id;
- });
-
- ids.push.apply(ids, nearMouse);
- } // hide labels on selected nodes (they look weird when dragging / haloed)
-
-
- for (var i = 0; i < selectedIDs.length; i++) {
- var entity = graph.hasEntity(selectedIDs[i]);
-
- if (entity && entity.type === 'node') {
- ids.push(selectedIDs[i]);
- }
- }
-
- layers.selectAll(utilEntitySelector(ids)).classed('nolabel', true); // draw the mouse bbox if debugging is on..
-
- var debug = selection.selectAll('.labels-group.debug');
- var gj = [];
-
- 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());
- }
-
- var throttleFilterLabels = throttle(filterLabels, 100);
-
- drawLabels.observe = function (selection) {
- var listener = function listener() {
- throttleFilterLabels(selection);
- };
-
- selection.on('mousemove.hidelabels', listener);
- context.on('enter.hidelabels', listener);
- };
-
- drawLabels.off = function (selection) {
- throttleFilterLabels.cancel();
- selection.on('mousemove.hidelabels', null);
- context.on('enter.hidelabels', null);
- };
-
- return drawLabels;
- }
-
- var _layerEnabled$1 = false;
-
- var _qaService$1;
-
- function svgImproveOSM(projection, context, dispatch) {
- var throttledRedraw = throttle(function () {
- return dispatch.call('change');
- }, 1000);
-
- 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
-
-
- function getService() {
- if (services.improveOSM && !_qaService$1) {
- _qaService$1 = services.improveOSM;
-
- _qaService$1.on('loaded', throttledRedraw);
- } else if (!services.improveOSM && _qaService$1) {
- _qaService$1 = null;
- }
-
- return _qaService$1;
- } // Show the markers
-
-
- function editOn() {
- if (!layerVisible) {
- layerVisible = true;
- drawLayer.style('display', 'block');
- }
- } // Immediately remove the markers and their touch targets
-
-
- 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 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 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 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
-
- 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
-
- 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 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 drawImproveOSM(selection) {
- var service = getService();
- var surface = context.surface();
-
- if (surface && !surface.empty()) {
- touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
- }
-
- 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 {
- editOff();
- }
- }
- } // Toggles the layer on and off
-
-
- drawImproveOSM.enabled = function (val) {
- if (!arguments.length) return _layerEnabled$1;
- _layerEnabled$1 = val;
-
- if (_layerEnabled$1) {
- layerOn();
- } else {
- layerOff();
-
- if (context.selectedErrorID()) {
- context.enter(modeBrowse(context));
- }
- }
-
- dispatch.call('change');
- return this;
- };
-
- drawImproveOSM.supported = function () {
- return !!getService();
- };
-
- return drawImproveOSM;
- }
-
- var _layerEnabled$2 = false;
-
- var _qaService$2;
-
- function svgOsmose(projection, context, dispatch) {
- var throttledRedraw = throttle(function () {
- return dispatch.call('change');
- }, 1000);
-
- 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 osmose service for fetching issues
-
-
- function getService() {
- if (services.osmose && !_qaService$2) {
- _qaService$2 = services.osmose;
-
- _qaService$2.on('loaded', throttledRedraw);
- } else if (!services.osmose && _qaService$2) {
- _qaService$2 = null;
- }
-
- return _qaService$2;
- } // Show the markers
-
-
- function editOn() {
- if (!layerVisible) {
- layerVisible = true;
- drawLayer.style('display', 'block');
- }
- } // Immediately remove the markers and their touch targets
-
-
- 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.
-
-
- 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 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
-
-
- 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..
-
- var markers = drawLayer.selectAll('.qaItem.osmose').data(data, function (d) {
- return d.id;
- }); // exit
-
- 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', 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;
-
- 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.osmose').data(data, function (d) {
- return d.id;
- }); // exit
-
- targets.exit().remove(); // enter/update
-
- 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 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.
-
-
- function drawOsmose(selection) {
- var service = getService();
- var surface = context.surface();
-
- if (surface && !surface.empty()) {
- touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
- }
-
- 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);
-
- if (_layerEnabled$2) {
- if (service && ~~context.map().zoom() >= minZoom) {
- editOn();
- service.loadIssues(projection);
- updateMarkers();
- } else {
- editOff();
- }
- }
- } // Toggles the layer on and off
-
-
- drawOsmose.enabled = function (val) {
- if (!arguments.length) return _layerEnabled$2;
- _layerEnabled$2 = val;
-
- 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
- });
- } else {
- layerOff();
-
- if (context.selectedErrorID()) {
- context.enter(modeBrowse(context));
- }
- }
-
- dispatch.call('change');
- return this;
- };
-
- drawOsmose.supported = function () {
- return !!getService();
- };
-
- return drawOsmose;
- }
-
- function svgStreetside(projection, context, dispatch) {
- var throttledRedraw = throttle(function () {
- dispatch.call('change');
- }, 1000);
-
- var minZoom = 14;
- var minMarkerZoom = 16;
- var minViewfieldZoom = 18;
- var layer = select(null);
- var _viewerYaw = 0;
- var _selectedSequence = null;
-
- var _streetside;
- /**
- * init().
- */
-
-
- function init() {
- if (svgStreetside.initialized) return; // run once
-
- svgStreetside.enabled = false;
- svgStreetside.initialized = true;
- }
- /**
- * getService().
- */
-
-
- function getService() {
- if (services.streetside && !_streetside) {
- _streetside = services.streetside;
-
- _streetside.event.on('viewerChanged.svgStreetside', viewerChanged).on('loadedImages.svgStreetside', throttledRedraw);
- } else if (!services.streetside && _streetside) {
- _streetside = null;
- }
-
- return _streetside;
- }
- /**
- * showLayer().
- */
-
-
- 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');
- });
- }
- /**
- * hideLayer().
- */
-
-
- function hideLayer() {
- throttledRedraw.cancel();
- layer.transition().duration(250).style('opacity', 0).on('end', editOff);
- }
- /**
- * editOn().
- */
-
-
- function editOn() {
- layer.style('display', 'block');
- }
- /**
- * editOff().
- */
-
-
- function editOff() {
- layer.selectAll('.viewfield-group').remove();
- layer.style('display', 'none');
- }
- /**
- * click() Handles 'bubble' point click event.
- */
-
-
- function click(d3_event, d) {
- var service = getService();
- if (!service) return; // try to preserve the viewer rotation when staying on the same sequence
-
- if (d.sequenceKey !== _selectedSequence) {
- _viewerYaw = 0; // reset
- }
-
- _selectedSequence = d.sequenceKey;
- service.ensureViewerLoaded(context).then(function () {
- service.selectImage(context, d.key).yaw(_viewerYaw).showViewer(context);
- });
- context.map().centerEase(d.loc);
- }
- /**
- * mouseover().
- */
-
-
- function mouseover(d3_event, d) {
- var service = getService();
- if (service) service.setStyles(context, d);
- }
- /**
- * mouseout().
- */
-
-
- function mouseout() {
- var service = getService();
- if (service) service.setStyles(context, null);
- }
- /**
- * transform().
- */
-
-
- function transform(d) {
- var t = svgPointTransform(projection)(d);
- var rot = d.ca + _viewerYaw;
-
- if (rot) {
- t += ' rotate(' + Math.floor(rot) + ',0,0)';
- }
-
- return t;
- }
-
- function viewerChanged() {
- var service = getService();
- if (!service) return;
- var viewer = service.viewer();
- if (!viewer) return; // update viewfield rotation
-
- _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);
- }
-
- 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;
- });
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- bubbles = bubbles.filter(function (bubble) {
- return new Date(bubble.captured_at).getTime() <= toTimestamp;
- });
- }
-
- if (usernames) {
- bubbles = bubbles.filter(function (bubble) {
- return usernames.indexOf(bubble.captured_by) !== -1;
- });
- }
-
- 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;
- });
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- sequences = sequences.filter(function (sequences) {
- return new Date(sequences.properties.captured_at).getTime() <= toTimestamp;
- });
- }
-
- if (usernames) {
- sequences = sequences.filter(function (sequences) {
- return usernames.indexOf(sequences.properties.captured_by) !== -1;
- });
- }
-
- return sequences;
- }
- /**
- * update().
- */
-
-
- 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 (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
-
- traces.exit().remove(); // enter/update
-
- 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
-
- groups.exit().remove(); // enter
-
- 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 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
-
- viewfields.enter().insert('path', 'circle').attr('class', 'viewfield').attr('transform', 'scale(1.5,1.5),translate(-8, -13)').attr('d', viewfieldPath);
-
- function viewfieldPath() {
- var d = this.parentNode.__data__;
-
- 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 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);
-
- if (enabled) {
- if (service && ~~context.map().zoom() >= minZoom) {
- editOn();
- update();
- service.loadBubbles(projection);
- } else {
- editOff();
- }
- }
- }
- /**
- * drawImages.enabled().
- */
-
-
- drawImages.enabled = function (_) {
- if (!arguments.length) return svgStreetside.enabled;
- svgStreetside.enabled = _;
-
- if (svgStreetside.enabled) {
- showLayer();
- context.photos().on('change.streetside', update);
- } else {
- hideLayer();
- context.photos().on('change.streetside', null);
- }
-
- dispatch.call('change');
- return this;
- };
- /**
- * drawImages.supported().
- */
-
-
- drawImages.supported = function () {
- return !!getService();
- };
-
- init();
- return drawImages;
- }
-
- function svgMapillaryImages(projection, context, dispatch) {
- var throttledRedraw = throttle(function () {
- dispatch.call('change');
- }, 1000);
-
- var minZoom = 12;
- var minMarkerZoom = 16;
- var minViewfieldZoom = 18;
- var layer = select(null);
-
- var _mapillary;
-
- var viewerCompassAngle;
-
- function init() {
- if (svgMapillaryImages.initialized) return; // run once
-
- svgMapillaryImages.enabled = false;
- svgMapillaryImages.initialized = true;
- }
-
- function getService() {
- if (services.mapillary && !_mapillary) {
- _mapillary = services.mapillary;
-
- _mapillary.event.on('loadedImages', throttledRedraw);
- } else if (!services.mapillary && _mapillary) {
- _mapillary = null;
- }
-
- return _mapillary;
- }
-
- 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');
- });
- }
-
- 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, 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 mouseover(d) {
- var service = getService();
- if (service) service.setStyles(context, d);
- }
-
- function mouseout() {
- var service = getService();
- if (service) service.setStyles(context, null);
- }
-
- function transform(d) {
- var t = svgPointTransform(projection)(d);
-
- 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)';
- }
-
- return t;
- }
-
- 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();
-
- if (!showsPano || !showsFlat) {
- images = images.filter(function (image) {
- if (image.pano) return showsPano;
- return showsFlat;
- });
- }
-
- if (fromDate) {
- var fromTimestamp = new Date(fromDate).getTime();
- images = images.filter(function (image) {
- return new Date(image.captured_at).getTime() >= fromTimestamp;
- });
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- images = images.filter(function (image) {
- return new Date(image.captured_at).getTime() <= toTimestamp;
- });
- }
-
- if (usernames) {
- images = images.filter(function (image) {
- return usernames.indexOf(image.captured_by) !== -1;
- });
- }
-
- return images;
- }
-
- 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();
-
- 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;
-
- 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);
-
- if (image && image.hasOwnProperty('pano')) {
- if (image.pano) return showsPano;
- return showsFlat;
- }
- }
- }
- }
-
- return false;
- });
- }
-
- if (fromDate) {
- var fromTimestamp = new Date(fromDate).getTime();
- sequences = sequences.filter(function (sequence) {
- return new Date(sequence.properties.captured_at).getTime() >= fromTimestamp;
- });
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- sequences = sequences.filter(function (sequence) {
- return new Date(sequence.properties.captured_at).getTime() <= toTimestamp;
- });
- }
-
- if (usernames) {
- sequences = sequences.filter(function (sequence) {
- return usernames.indexOf(sequence.properties.username) !== -1;
- });
- }
-
- 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);
- service.filterViewer(context);
- var traces = layer.selectAll('.sequences').selectAll('.sequence').data(sequences, function (d) {
- return d.properties.key;
- }); // exit
-
- traces.exit().remove(); // enter/update
-
- 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
-
- groups.exit().remove(); // enter
-
- 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 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__;
-
- 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';
- }
- }
- }
-
- 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();
- }
- }
- }
-
- 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);
- }
-
- dispatch.call('change');
- return this;
- };
-
- drawImages.supported = function () {
- return !!getService();
- };
-
- init();
- return drawImages;
- }
-
- function svgMapillaryPosition(projection, context) {
- var throttledRedraw = throttle(function () {
- update();
- }, 1000);
-
- var minZoom = 12;
- var minViewfieldZoom = 18;
- var layer = select(null);
-
- var _mapillary;
-
- var viewerCompassAngle;
-
- function init() {
- if (svgMapillaryPosition.initialized) return; // run once
-
- svgMapillaryPosition.initialized = true;
- }
-
- function getService() {
- if (services.mapillary && !_mapillary) {
- _mapillary = services.mapillary;
-
- _mapillary.event.on('nodeChanged', throttledRedraw);
-
- _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);
- });
- } else if (!services.mapillary && _mapillary) {
- _mapillary = null;
- }
-
- return _mapillary;
- }
-
- function editOn() {
- layer.style('display', 'block');
- }
-
- function editOff() {
- layer.selectAll('.viewfield-group').remove();
- layer.style('display', 'none');
- }
-
- function transform(d) {
- var t = svgPointTransform(projection)(d);
-
- 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)';
- }
-
- 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
-
- groups.exit().remove(); // enter
-
- var groupsEnter = groups.enter().append('g').attr('class', 'viewfield-group currentView highlighted');
- groupsEnter.append('g').attr('class', 'viewfield-scale'); // update
-
- 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);
-
- function viewfieldPath() {
- var d = this.parentNode.__data__;
-
- 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';
- }
- }
- }
-
- 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;
- };
-
- 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;
- }
-
- 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;
- }
- });
-
- if (imageKey === selectedImageKey) {
- service.highlightDetection(highlightedDetection).selectImage(context, imageKey);
- } else {
- service.ensureViewerLoaded(context).then(function () {
- service.highlightDetection(highlightedDetection).selectImage(context, imageKey).showViewer(context);
- });
- }
- }
-
- function filterData(detectedFeatures) {
- var service = getService();
- var fromDate = context.photos().fromDate();
- var toDate = context.photos().toDate();
- var usernames = context.photos().usernames();
-
- if (fromDate) {
- var fromTimestamp = new Date(fromDate).getTime();
- detectedFeatures = detectedFeatures.filter(function (feature) {
- return new Date(feature.last_seen_at).getTime() >= fromTimestamp;
- });
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- detectedFeatures = detectedFeatures.filter(function (feature) {
- return new Date(feature.first_seen_at).getTime() <= toTimestamp;
- });
- }
-
- 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;
- });
- });
- }
-
- return detectedFeatures;
- }
-
- 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
-
- signs.exit().remove(); // enter
-
- 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).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 (aSelected === bSelected) {
- return b.loc[1] - a.loc[1]; // sort Y
- } else if (aSelected) {
- return 1;
- }
-
- return -1;
- });
- }
-
- 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);
- }
- }
-
- drawSigns.enabled = function (_) {
- if (!arguments.length) return svgMapillarySigns.enabled;
- svgMapillarySigns.enabled = _;
-
- if (svgMapillarySigns.enabled) {
- showLayer();
- context.photos().on('change.mapillary_signs', update);
- } else {
- hideLayer();
- context.photos().on('change.mapillary_signs', null);
- }
-
- dispatch.call('change');
- return this;
- };
-
- drawSigns.supported = function () {
- return !!getService();
- };
-
- init();
- return drawSigns;
- }
-
- function svgMapillaryMapFeatures(projection, context, dispatch) {
- var throttledRedraw = throttle(function () {
- dispatch.call('change');
- }, 1000);
-
- var minZoom = 12;
- var layer = select(null);
-
- var _mapillary;
-
- function init() {
- if (svgMapillaryMapFeatures.initialized) return; // run once
-
- svgMapillaryMapFeatures.enabled = false;
- svgMapillaryMapFeatures.initialized = true;
- }
-
- function getService() {
- if (services.mapillary && !_mapillary) {
- _mapillary = services.mapillary;
-
- _mapillary.event.on('loadedMapFeatures', throttledRedraw);
- } else if (!services.mapillary && _mapillary) {
- _mapillary = null;
- }
-
- return _mapillary;
- }
-
- function showLayer() {
- var service = getService();
- if (!service) return;
- service.loadObjectResources(context);
- editOn();
- }
-
- function hideLayer() {
- throttledRedraw.cancel();
- editOff();
- }
-
- function editOn() {
- layer.style('display', 'block');
- }
-
- function editOff() {
- layer.selectAll('.icon-map-feature').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 map feature 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;
- }
- });
-
- if (imageKey === selectedImageKey) {
- service.highlightDetection(highlightedDetection).selectImage(context, imageKey);
- } else {
- service.ensureViewerLoaded(context).then(function () {
- service.highlightDetection(highlightedDetection).selectImage(context, imageKey).showViewer(context);
- });
- }
- }
-
- function filterData(detectedFeatures) {
- var service = getService();
- var fromDate = context.photos().fromDate();
- var toDate = context.photos().toDate();
- var usernames = context.photos().usernames();
-
- if (fromDate) {
- var fromTimestamp = new Date(fromDate).getTime();
- detectedFeatures = detectedFeatures.filter(function (feature) {
- return new Date(feature.last_seen_at).getTime() >= fromTimestamp;
- });
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- detectedFeatures = detectedFeatures.filter(function (feature) {
- return new Date(feature.first_seen_at).getTime() <= toTimestamp;
- });
- }
-
- 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;
- });
- });
- }
-
- return detectedFeatures;
- }
-
- 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
-
- mapFeatures.exit().remove(); // enter
-
- 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';
- }
-
- return '#' + d.value;
- });
- enter.append('rect').attr('width', '24px').attr('height', '24px').attr('x', '-12px').attr('y', '-12px'); // update
-
- 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;
- });
-
- if (aSelected === bSelected) {
- return b.loc[1] - a.loc[1]; // sort Y
- } else if (aSelected) {
- return 1;
- }
-
- return -1;
- });
- }
-
- 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);
- }
- }
-
- 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;
- };
-
- drawMapFeatures.supported = function () {
- return !!getService();
- };
-
- init();
- return drawMapFeatures;
- }
-
- function svgOpenstreetcamImages(projection, context, dispatch) {
- var throttledRedraw = throttle(function () {
- dispatch.call('change');
- }, 1000);
-
- var minZoom = 12;
- var minMarkerZoom = 16;
- var minViewfieldZoom = 18;
- var layer = select(null);
-
- var _openstreetcam;
-
- function init() {
- if (svgOpenstreetcamImages.initialized) return; // run once
-
- svgOpenstreetcamImages.enabled = false;
- svgOpenstreetcamImages.initialized = true;
- }
-
- function getService() {
- if (services.openstreetcam && !_openstreetcam) {
- _openstreetcam = services.openstreetcam;
-
- _openstreetcam.event.on('loadedImages', throttledRedraw);
- } else if (!services.openstreetcam && _openstreetcam) {
- _openstreetcam = null;
- }
-
- return _openstreetcam;
- }
-
- 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');
- });
- }
-
- 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, 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 mouseover(d3_event, d) {
- var service = getService();
- if (service) service.setStyles(context, d);
- }
-
- 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 t;
- }
-
- function filterImages(images) {
- var fromDate = context.photos().fromDate();
- var toDate = context.photos().toDate();
- var usernames = context.photos().usernames();
-
- if (fromDate) {
- var fromTimestamp = new Date(fromDate).getTime();
- images = images.filter(function (item) {
- return new Date(item.captured_at).getTime() >= fromTimestamp;
- });
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- images = images.filter(function (item) {
- return new Date(item.captured_at).getTime() <= toTimestamp;
- });
- }
-
- if (usernames) {
- images = images.filter(function (item) {
- return usernames.indexOf(item.captured_by) !== -1;
- });
- }
-
- return images;
- }
-
- 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 (image) {
- return new Date(image.properties.captured_at).getTime() >= fromTimestamp;
- });
- }
-
- if (toDate) {
- var toTimestamp = new Date(toDate).getTime();
- sequences = sequences.filter(function (image) {
- return new Date(image.properties.captured_at).getTime() <= toTimestamp;
- });
- }
-
- if (usernames) {
- sequences = sequences.filter(function (image) {
- return usernames.indexOf(image.properties.captured_by) !== -1;
- });
- }
-
- return sequences;
- }
-
- 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 (context.photos().showsFlat()) {
- sequences = service ? service.sequences(projection) : [];
- images = service && showMarkers ? service.images(projection) : [];
- sequences = filterSequences(sequences);
- images = filterImages(images);
- }
-
- var traces = layer.selectAll('.sequences').selectAll('.sequence').data(sequences, function (d) {
- return d.properties.key;
- }); // exit
-
- traces.exit().remove(); // enter/update
-
- 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
-
- groups.exit().remove(); // enter
-
- 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 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');
- }
-
- 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);
-
- if (enabled) {
- if (service && ~~context.map().zoom() >= minZoom) {
- editOn();
- update();
- service.loadImages(projection);
- } else {
- editOff();
- }
- }
- }
-
- drawImages.enabled = function (_) {
- if (!arguments.length) return svgOpenstreetcamImages.enabled;
- svgOpenstreetcamImages.enabled = _;
-
- if (svgOpenstreetcamImages.enabled) {
- showLayer();
- context.photos().on('change.openstreetcam_images', update);
- } else {
- hideLayer();
- context.photos().on('change.openstreetcam_images', null);
- }
-
- dispatch.call('change');
- return this;
- };
-
- drawImages.supported = function () {
- return !!getService();
- };
-
- init();
- return drawImages;
- }
-
- function svgOsm(projection, context, dispatch) {
- var enabled = true;
-
- 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 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');
- });
- }
-
- 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');
- });
- }
-
- drawOsm.enabled = function (val) {
- if (!arguments.length) return enabled;
- enabled = val;
-
- if (enabled) {
- showLayer();
- } else {
- hideLayer();
- }
-
- dispatch.call('change');
- return this;
- };
-
- return drawOsm;
- }
-
- var _notesEnabled = false;
-
- var _osmService;
-
- function svgNotes(projection, context, dispatch$1) {
- if (!dispatch$1) {
- dispatch$1 = dispatch('change');
- }
-
- var throttledRedraw = throttle(function () {
- dispatch$1.call('change');
- }, 1000);
-
- var minZoom = 12;
- var touchLayer = select(null);
- var drawLayer = select(null);
- var _notesVisible = false;
-
- 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 getService() {
- if (services.osm && !_osmService) {
- _osmService = services.osm;
-
- _osmService.on('loadedNotes', throttledRedraw);
- } else if (!services.osm && _osmService) {
- _osmService = null;
- }
-
- return _osmService;
- } // Show the notes
-
-
- function editOn() {
- if (!_notesVisible) {
- _notesVisible = true;
- drawLayer.style('display', 'block');
- }
- } // Immediately remove the notes and their touch targets
-
-
- 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.
-
-
- 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.
-
-
- 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 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 notes = drawLayer.selectAll('.note').data(data, function (d) {
- return d.status + d.id;
- }); // exit
-
- notes.exit().remove(); // enter
-
- 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
-
- notes.merge(notesEnter).sort(sortY).classed('selected', function (d) {
- var mode = context.mode();
- var isMoving = mode && mode.id === 'drag-note'; // no shadows when dragging
-
- return !isMoving && d.id === selectedID;
- }).attr('transform', getTransform); // Draw targets..
-
- if (touchLayer.empty()) return;
- var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
- var targets = touchLayer.selectAll('.note').data(data, function (d) {
- return d.id;
- }); // exit
-
- targets.exit().remove(); // enter/update
-
- 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);
-
- function sortY(a, b) {
- return a.id === selectedID ? 1 : b.id === selectedID ? -1 : b.loc[1] - a.loc[1];
- }
- } // Draw the notes layer and schedule loading notes and updating markers.
-
-
- function drawNotes(selection) {
- var service = getService();
- var surface = context.surface();
-
- if (surface && !surface.empty()) {
- touchLayer = surface.selectAll('.data-layer.touch .layer-touch.markers');
- }
-
- 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);
-
- if (_notesEnabled) {
- if (service && ~~context.map().zoom() >= minZoom) {
- editOn();
- service.loadNotes(projection);
- updateMarkers();
- } else {
- editOff();
- }
- }
- } // Toggles the layer on and off
-
-
- drawNotes.enabled = function (val) {
- if (!arguments.length) return _notesEnabled;
- _notesEnabled = val;
-
- if (_notesEnabled) {
- layerOn();
- } else {
- layerOff();
-
- if (context.selectedNoteID()) {
- context.enter(modeBrowse(context));
- }
- }
-
- dispatch$1.call('change');
- return this;
- };
-
- return drawNotes;
- }
-
- 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;
- });
- }
-
- return drawTouch;
- }
-
- function refresh(selection, node) {
- var cr = node.getBoundingClientRect();
- var prop = [cr.width, cr.height];
- selection.property('__dimensions__', prop);
- return prop;
- }
-
- function utilGetDimensions(selection, force) {
- if (!selection || selection.empty()) {
- return [0, 0];
- }
-
- 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 node = selection.node();
-
- if (dimensions === null) {
- refresh(selection, node);
- return selection;
- }
-
- return selection.property('__dimensions__', [dimensions[0], dimensions[1]]).attr('width', dimensions[0]).attr('height', dimensions[1]);
- }
-
- 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)
- }, {
- id: 'touch',
- layer: svgTouch()
- }];
-
- 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);
- });
- }
-
- drawLayers.all = function () {
- return _layers;
- };
-
- drawLayers.layer = function (id) {
- var obj = _layers.find(function (o) {
- return o.id === id;
- });
-
- return obj && obj.layer;
- };
-
- drawLayers.only = function (what) {
- var arr = [].concat(what);
-
- var all = _layers.map(function (layer) {
- return layer.id;
- });
-
- 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$1.call('change');
- return this;
- };
-
- 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;
- };
-
- drawLayers.dimensions = function (val) {
- if (!arguments.length) return utilGetDimensions(svg);
- utilSetDimensions(svg, val);
- return this;
- };
-
- return utilRebind(drawLayers, dispatch$1, 'on');
- }
-
- 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
-
- 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) {
- return d.id;
- }); // exit
-
- targets.exit().remove();
-
- var segmentWasEdited = function segmentWasEdited(d) {
- var wayID = d.properties.entity.id; // if the whole line was edited, don't draw segment changes
-
- if (!base.entities[wayID] || !fastDeepEqual(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
- return false;
- }
-
- 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
-
-
- 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 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
-
- nopes.exit().remove(); // enter/update
-
- 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);
- }
-
- function drawLines(selection, graph, entities, filter) {
- var base = context.history().base();
-
- 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;
-
- if (a.tags.highway) {
- scoreA -= highway_stack[a.tags.highway];
- }
-
- if (b.tags.highway) {
- scoreB -= highway_stack[b.tags.highway];
- }
-
- return scoreA - scoreB;
- }
-
- 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
-
- 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
-
- 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 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;
- }
-
- 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;
- });
- };
- }
-
- 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;
- });
-
- if (detected.ie) {
- markers.each(function () {
- this.parentNode.insertBefore(this, this);
- });
- }
- }
-
- var getPath = svgPath(projection, graph);
- var ways = [];
- var onewaydata = {};
- var sideddata = {};
- var oldMultiPolygonOuters = {};
-
- 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);
- }
- }
-
- 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 uncovered = selection.selectAll('.layer-osm.lines'); // over areas
-
- var touchLayer = selection.selectAll('.layer-touch.lines'); // Draw lines..
-
- [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..
-
- touchLayer.call(drawTargets, graph, ways, filter);
- }
-
- 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
-
- targets.exit().remove(); // enter/update
-
- targets.enter().append('circle').attr('r', targetRadius).merge(targets).attr('class', function (d) {
- return 'node midpoint target ' + fillClass + d.id;
- }).attr('transform', getTransform);
- }
-
- 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 (mode && mode.id !== 'select' || !context.map().withinEditableZoom()) {
- drawLayer.selectAll('.midpoint').remove();
- touchLayer.selectAll('.midpoint.target').remove();
- return;
- }
-
- var poly = extent.polygon();
- var midpoints = {};
-
- 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);
-
- 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;
-
- 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 (point && geoVecLength(projection(a.loc), projection(point)) > 20 && geoVecLength(projection(b.loc), projection(point)) > 20) {
- loc = point;
- break;
- }
- }
- }
-
- if (loc) {
- midpoints[id] = {
- type: 'midpoint',
- id: id,
- loc: loc,
- edge: [a.id, b.id],
- parents: [entity]
- };
- }
- }
- }
- }
-
- 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;
- }
- }
-
- return false;
- }
-
- 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..
-
- touchLayer.call(drawTargets, graph, Object.values(midpoints), midpointFilter);
- }
-
- return drawMidpoints;
- }
-
- 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.
-
-
- 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);
- }
-
- 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
-
- 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
-
- targets.exit().remove(); // enter/update
-
- 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 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..
-
- 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 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..
-
- 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
-
- groups.select('.stroke'); // propagate bound data
-
- groups.select('.icon') // propagate bound data
- .attr('xlink:href', function (entity) {
- var preset = _mainPresetIndex.match(entity, graph);
- var picon = preset && preset.icon;
-
- if (!picon) {
- return '';
- } else {
- var isMaki = /^maki-/.test(picon);
- return '#' + picon + (isMaki ? '-11' : '');
- }
- }); // Draw touch targets..
-
- touchLayer.call(drawTargets, graph, points, filter);
- }
-
- return drawPoints;
- }
-
- 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 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 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
-
- return 'translate(' + (r * Math.cos(a) + o[0]) + ',' + (r * Math.sin(a) + o[1]) + ') ' + 'rotate(' + a * 180 / Math.PI + ')';
- }
-
- 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
-
- groups.exit().remove(); // enter
-
- 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
-
- 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
-
- groups.select('circle'); // propagate bound data
- // Draw touch targets..
-
- var fillClass = context.getDebug('target') ? 'pink ' : 'nocolor ';
- groups = touchLayer.selectAll('g.turn').data(turns, function (d) {
- return d.key;
- }); // exit
-
- groups.exit().remove(); // enter
-
- 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
-
- groups = groups.merge(groupsEnter).attr('transform', turnTransform);
- groups.select('rect'); // propagate bound data
-
- groups.select('circle'); // propagate bound data
-
- return this;
- }
-
- 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 _currHoverTarget;
-
- var _currPersistent = {};
- var _currHover = {};
- var _prevHover = {};
- var _currSelected = {};
- var _prevSelected = {};
- var _radii = {};
-
- 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 fastEntityKey(d) {
- var mode = context.mode();
- var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
- return isMoving ? d.id : osmEntity.key(d);
- }
-
- 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 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 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
-
- if (entity.id !== activeID && entity.isEndpoint(graph) && !entity.isConnected(graph)) {
- r += 1.5;
- }
-
- if (klass === 'shadow') {
- // remember this value, so we don't need to
- _radii[entity.id] = r; // recompute it when we draw the touch targets
- }
-
- select(this).attr('r', r).attr('visibility', i && klass === 'fill' ? 'hidden' : null);
- });
- });
- }
-
- 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.
-
- enter.filter(function (d) {
- return d.hasInterestingTags();
- }).append('circle').attr('class', 'fill'); // update
-
- 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
-
- iconUse.exit().remove(); // enter
-
- 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
-
- var dgroups = groups.selectAll('.viewfieldgroup').data(function data(d) {
- return zoom >= 18 && getDirections(d) ? [d] : [];
- }, fastEntityKey); // exit
-
- dgroups.exit().remove(); // enter/update
-
- 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
-
- 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 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
-
- 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
-
- 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
-
-
- function renderAsVertex(entity, graph, wireframe, zoom) {
- var geometry = entity.geometry(graph);
- return geometry === 'vertex' || geometry === 'point' && (wireframe || zoom >= 18 && entity.directions(graph, projection).length);
- }
-
- 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);
- }
-
- 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);
-
- if (!context.features().isHiddenFeature(entity, graph, geometry)) {
- var i;
-
- if (entity.type === 'way') {
- for (i = 0; i < entity.nodes.length; i++) {
- var child = graph.hasEntity(entity.nodes[i]);
-
- 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 (member) {
- addChildVertices(member);
- }
- }
- } else if (renderAsVertex(entity, graph, wireframe, zoom)) {
- results[entity.id] = entity;
- }
- }
- }
-
- ids.forEach(function (id) {
- var entity = graph.hasEntity(id);
- if (!entity) return;
-
- 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;
- }
-
- 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');
-
- if (fullRedraw) {
- _currPersistent = {};
- _radii = {};
- } // Collect important vertices from the `entities` list..
- // (during a partial redraw, it will not contain everything)
-
-
- for (var i = 0; i < entities.length; i++) {
- var entity = entities[i];
- var geometry = entity.geometry(graph);
- var keep = false; // a point that looks like a vertex..
-
- 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 (!keep && !fullRedraw) {
- delete _currPersistent[entity.id];
- }
- } // 3 sets of vertices to consider:
-
-
- 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 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 filterRendered = function filterRendered(d) {
- return d.id in _currPersistent || d.id in _currSelected || d.id in _currHover || filter(d);
- };
-
- drawLayer.call(draw, graph, currentVisible(all), sets, filterRendered); // Draw touch targets..
- // When drawing, render all targets (not just those affected by a partial redraw)
-
- var filterTouch = function filterTouch(d) {
- return isMoving ? true : filterRendered(d);
- };
-
- touchLayer.call(drawTargets, graph, currentVisible(all), filterTouch);
-
- 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..
-
-
- drawVertices.drawSelected = function (selection, graph, extent) {
- var wireframe = context.surface().classed('fill-wireframe');
- var zoom = geoScaleToZoom(projection.scale());
- _prevSelected = _currSelected || {};
-
- if (context.map().isInWideSelection()) {
- _currSelected = {};
- context.selectedIDs().forEach(function (id) {
- var entity = graph.hasEntity(id);
- if (!entity) return;
-
- 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 filter = function filter(d) {
- return d.id in _prevSelected;
- };
-
- drawVertices(selection, graph, Object.values(_prevSelected), filter, extent, false);
- }; // partial redraw - only update the hovered items..
-
-
- drawVertices.drawHover = function (selection, graph, target, extent) {
- if (target === _currHoverTarget) return; // continue only if something changed
-
- var wireframe = context.surface().classed('fill-wireframe');
- var zoom = geoScaleToZoom(projection.scale());
- _prevHover = _currHover || {};
- _currHoverTarget = target;
- var entity = target && target.properties && target.properties.entity;
-
- if (entity) {
- _currHover = getSiblingAndChildVertices([entity.id], graph, wireframe, zoom);
- } else {
- _currHover = {};
- } // note that drawVertices will add `_currHover` automatically if needed..
-
-
- var filter = function filter(d) {
- return d.id in _prevHover;
- };
-
- drawVertices(selection, graph, Object.values(_prevHover), filter, extent, false);
- };
-
- return drawVertices;
- }
-
- function utilBindOnce(target, type, listener, capture) {
- var typeOnce = type + '.once';
-
- function one() {
- target.on(typeOnce, null);
- listener.apply(this, arguments);
- }
-
- target.on(typeOnce, one, capture);
- return this;
- }
-
- function defaultFilter$2(d3_event) {
- return !d3_event.ctrlKey && !d3_event.button;
- }
-
- function defaultExtent$1() {
- var e = this;
-
- if (e instanceof SVGElement) {
- e = e.ownerSVGElement || e;
-
- if (e.hasAttribute('viewBox')) {
- e = e.viewBox.baseVal;
- return [[e.x, e.y], [e.x + e.width, e.y + e.height]];
- }
-
- return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];
- }
-
- return [[0, 0], [e.clientWidth, e.clientHeight]];
- }
-
- 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));
- }
-
- 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 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);
- }
-
- zoom.transform = function (collection, transform, point) {
- var selection = collection.selection ? collection.selection() : collection;
-
- 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);
- });
- }
- };
-
- 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);
- };
-
- 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);
- };
-
- 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);
- });
- };
-
- 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);
- };
-
- 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);
- }
-
- function centroid(extent) {
- return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];
- }
-
- 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);
- };
- });
- }
-
- function gesture(that, args, clean) {
- return !clean && _activeGesture || new Gesture(that, args);
- }
-
- function Gesture(that, args) {
- this.that = that;
- this.args = args;
- this.active = 0;
- this.extent = extent.apply(that, args);
- }
-
- Gesture.prototype = {
- start: function start(d3_event) {
- if (++this.active === 1) {
- _activeGesture = this;
- dispatch$1.call('start', this, d3_event);
- }
-
- 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);
- }
-
- return this;
- }
- };
-
- 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);
- }
-
- 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);
- }
-
- 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));
-
- function wheelidled() {
- g.wheel = null;
- g.end(d3_event);
- }
- }
-
- var _downPointerIDs = new Set();
-
- var _pointerLocGetter;
-
- function pointerdown(d3_event) {
- _downPointerIDs.add(d3_event.pointerId);
-
- if (!filter.apply(this, arguments)) return;
- var g = gesture(this, arguments, _downPointerIDs.size === 1);
- var started;
- d3_event.stopImmediatePropagation();
- _pointerLocGetter = utilFastMouse(this);
-
- var loc = _pointerLocGetter(d3_event);
-
- var p = [loc, _transform.invert(loc), d3_event.pointerId];
-
- if (!g.pointer0) {
- g.pointer0 = p;
- started = true;
- } else if (!g.pointer1 && g.pointer0[2] !== p[2]) {
- g.pointer1 = p;
- }
-
- if (started) {
- interrupt(this);
- g.start(d3_event);
- }
- }
-
- 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;
-
- 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;
- }
-
- d3_event.preventDefault();
- d3_event.stopImmediatePropagation();
-
- var loc = _pointerLocGetter(d3_event);
-
- 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;
-
- g.zoom(d3_event, 'touch', constrain(translate(t, p, l), g.extent, translateExtent));
- }
-
- function pointerup(d3_event) {
- if (!_downPointerIDs.has(d3_event.pointerId)) return;
-
- _downPointerIDs["delete"](d3_event.pointerId);
-
- 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;
-
- 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);
- }
- }
-
- zoom.wheelDelta = function (_) {
- return arguments.length ? (wheelDelta = utilFunctor(+_), zoom) : wheelDelta;
- };
-
- zoom.filter = function (_) {
- return arguments.length ? (filter = utilFunctor(!!_), zoom) : filter;
- };
-
- zoom.extent = function (_) {
- return arguments.length ? (extent = utilFunctor([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
- };
-
- zoom.scaleExtent = function (_) {
- return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];
- };
-
- 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]]];
- };
-
- zoom.constrain = function (_) {
- return arguments.length ? (constrain = _, zoom) : constrain;
- };
-
- zoom.interpolate = function (_) {
- return arguments.length ? (interpolate = _, zoom) : interpolate;
- };
-
- zoom._transform = function (_) {
- return arguments.length ? (_transform = _, zoom) : _transform;
- };
-
- return utilRebind(zoom, dispatch$1, 'on');
- }
-
- // if pointer events are supported. Falls back to default `dblclick` event.
-
- function utilDoubleUp() {
- var dispatch$1 = dispatch('doubleUp');
- var _maxTimespan = 500; // milliseconds
-
- var _maxDistance = 20; // web pixels; be somewhat generous to account for touch devices
-
- var _pointer; // object representing the pointer that could trigger double up
-
-
- 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;
- }
-
- 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
-
- 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;
- }
- }
-
- 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;
-
- 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
-
-
- _pointer = undefined;
- }
- }
-
- 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));
- });
- }
- }
-
- 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;
-
- var _transformStart = projection.transform();
-
- var _transformLast;
-
- var _isTransformed = false;
- var _minzoom = 0;
-
- var _getMouseCoords;
-
- var _lastPointerEvent;
-
- var _lastWithinEditableZoom; // whether a pointerdown event started the zoom
-
-
- var _pointerDown = false; // use pointer events on supported platforms; fallback to mouse events
-
- var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse'; // use pointer event interaction if supported; fallback to touch/mouse events in d3-zoom
-
-
- var _zoomerPannerFunction = 'PointerEvent' in window ? utilZoomPan : d3_zoom;
-
- 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;
- });
-
- var _doubleUpHandler = utilDoubleUp();
-
- 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 cancelPendingRedraw() {
- scheduleRedraw.cancel(); // isRedrawScheduled = false;
- // window.cancelIdleCallback(pendingRedrawCall);
- }
-
- function map(selection) {
- _selection = selection;
- context.on('change.map', immediateRedraw);
- var osm = context.connection();
-
- if (osm) {
- osm.on('change.map', immediateRedraw);
- }
-
- function didUndoOrRedo(targetTransform) {
- var mode = context.mode().id;
- if (mode !== 'browse' && mode !== 'select') return;
-
- if (targetTransform) {
- map.transformEase(targetTransform);
- }
- }
-
- 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
-
- 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
-
- 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;
-
- 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
-
- 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
-
-
- updateAreaFill();
-
- _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);
- });
-
- context.on('enter.map', function () {
- if (!map.editableDataEnabled(true
- /* skip zoom check */
- )) return; // redraw immediately any objects affected by a change in selectedIDs.
-
- var graph = context.graph();
- var selectedAndParents = {};
- context.selectedIDs().forEach(function (id) {
- var entity = graph.hasEntity(id);
-
- if (entity) {
- selectedAndParents[entity.id] = entity;
-
- if (entity.type === 'node') {
- graph.parentWays(entity).forEach(function (parent) {
- selectedAndParents[parent.id] = parent;
- });
- }
- }
- });
- var data = Object.values(selectedAndParents);
-
- var filter = function filter(d) {
- return d.id in selectedAndParents;
- };
-
- 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
-
- scheduleRedraw();
- });
- map.dimensions(utilGetDimensions(selection));
- }
-
- 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;
- }
- }
-
- if (hasOrphan) {
- var event = window.CustomEvent;
-
- 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.
-
-
- event.view = window;
- window.dispatchEvent(event);
- }
- }
-
- return d3_event.button !== 2; // ignore right clicks
- }
-
- 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
-
- applyFeatureLayerFilters = false;
- } else if (difference) {
- var complete = difference.complete(map.extent());
- data = Object.values(complete).filter(Boolean);
- set = new Set(Object.keys(complete));
-
- filter = function filter(d) {
- return set.has(d.id);
- };
-
- 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;
- }));
-
- filter = function filter(d) {
- return set.has(d.id);
- };
- } else {
- data = all;
- fullRedraw = true;
- filter = utilFunctor(true);
- }
- }
-
- if (applyFeatureLayerFilters) {
- data = features.filter(data, graph);
- } 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());
- }
-
- 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
- });
- }
-
- 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 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));
- }
-
- dispatch$1.call('drawn', this, {
- full: true
- });
- }
-
- 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
-
- e2._rotation = e.rotation; // preserve the original rotation
-
- _selection.node().dispatchEvent(e2);
- }
-
- 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 (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
-
- 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
-
- if (detected.os !== 'mac') {
- dY *= 5;
- } // 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 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
-
- 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 (x2 !== x || y2 !== y || k2 !== k) {
- x = x2;
- y = y2;
- k = k2;
- eventTransform = identity$2.translate(x2, y2).scale(k2);
-
- if (_zoomerPanner._transform) {
- // utilZoomPan interface
- _zoomerPanner._transform(eventTransform);
- } else {
- // d3_zoom interface
- _selection.node().__zoom = eventTransform;
- }
- }
- }
-
- if (_transformStart.x === x && _transformStart.y === y && _transformStart.k === k) {
- return; // no change
- }
-
- var withinEditableZoom = map.withinEditableZoom();
-
- 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);
- }
-
- _lastWithinEditableZoom = withinEditableZoom;
- }
-
- 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;
- }
-
- projection.transform(eventTransform);
- var scale = k / _transformStart.k;
- var tX = (x / scale - _transformStart.x) * scale;
- var tY = (y / scale - _transformStart.y) * scale;
-
- if (context.inIntro()) {
- curtainProjection.transform({
- x: x - tX,
- y: y - tY,
- k: k
- });
- }
-
- if (source) {
- _lastPointerEvent = event;
- }
-
- _isTransformed = true;
- _transformLast = eventTransform;
- utilSetTransform(supersurface, tX, tY, scale);
- scheduleRedraw();
- dispatch$1.call('move', this, map);
-
- function isInteger(val) {
- return typeof val === 'number' && isFinite(val) && Math.floor(val) === val;
- }
- }
-
- function resetTransform() {
- if (!_isTransformed) return false;
- utilSetTransform(supersurface, 0, 0);
- _isTransformed = false;
-
- if (context.inIntro()) {
- curtainProjection.transform(projection.transform());
- }
-
- return true;
- }
-
- 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$2().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 immediateRedraw = function immediateRedraw(difference, extent) {
- if (!difference && !extent) cancelPendingRedraw();
- redraw(difference, extent);
- };
-
- map.lastPointerEvent = function () {
- return _lastPointerEvent;
- };
-
- map.mouse = function (d3_event) {
- var event = _lastPointerEvent || d3_event;
-
- if (event) {
- var s;
-
- while (s = event.sourceEvent) {
- event = s;
- }
-
- return _getMouseCoords(event);
- }
-
- return null;
- }; // returns Lng/Lat
-
-
- map.mouseCoordinates = function () {
- var coord = map.mouse() || pxCenter();
- return projection.invert(coord);
- };
-
- map.dblclickZoomEnable = function (val) {
- if (!arguments.length) return _dblClickZoomEnabled;
- _dblClickZoomEnabled = val;
- return map;
- };
-
- map.redrawEnable = function (val) {
- if (!arguments.length) return _redrawEnabled;
- _redrawEnabled = val;
- return map;
- };
-
- map.isTransformed = function () {
- return _isTransformed;
- };
-
- 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 (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;
-
- _selection.call(_zoomerPanner.transform, _transformStart);
- }
-
- return true;
- }
-
- 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(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);
- }
-
- map.pan = function (delta, duration) {
- var t = projection.translate();
- var k = projection.scale();
- t[0] += delta[0];
- t[1] += delta[1];
-
- 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);
-
- dispatch$1.call('move', this, map);
- immediateRedraw();
- }
-
- 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;
- };
-
- function zoomIn(delta) {
- setCenterZoom(map.center(), ~~map.zoom() + delta, 250, true);
- }
-
- function zoomOut(delta) {
- setCenterZoom(map.center(), ~~map.zoom() - delta, 250, true);
- }
-
- map.zoomIn = function () {
- zoomIn(1);
- };
-
- map.zoomInFurther = function () {
- zoomIn(4);
- };
-
- map.canZoomIn = function () {
- return map.zoom() < maxZoom;
- };
-
- map.zoomOut = function () {
- zoomOut(1);
- };
-
- map.zoomOutFurther = function () {
- zoomOut(4);
- };
-
- map.canZoomOut = function () {
- return map.zoom() > minZoom;
- };
-
- map.center = function (loc2) {
- if (!arguments.length) {
- return projection.invert(pxCenter());
- }
-
- if (setCenterZoom(loc2, map.zoom())) {
- dispatch$1.call('move', this, map);
- }
-
- scheduleRedraw();
- return map;
- };
-
- 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);
- };
-
- map.unobscuredOffsetPx = function () {
- var openPane = context.container().select('.map-panes .map-pane.shown');
-
- if (!openPane.empty()) {
- return [openPane.node().offsetWidth / 2, 0];
- }
-
- return [0, 0];
- };
-
- map.zoom = function (z2) {
- if (!arguments.length) {
- return Math.max(geoScaleToZoom(projection.scale(), TILESIZE), 0);
- }
-
- if (z2 < _minzoom) {
- surface.interrupt();
- dispatch$1.call('hitMinZoom', this, map);
- z2 = context.minEditableZoom();
- }
-
- if (setCenterZoom(map.center(), z2)) {
- dispatch$1.call('move', this, map);
- }
-
- scheduleRedraw();
- return map;
- };
-
- map.centerZoom = function (loc2, z2) {
- if (setCenterZoom(loc2, z2)) {
- dispatch$1.call('move', this, map);
- }
-
- 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);
- };
-
- 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;
- };
-
- map.centerZoomEase = function (loc2, z2, duration) {
- duration = duration || 250;
- setCenterZoom(loc2, z2, duration, false);
- return map;
- };
-
- map.transformEase = function (t2, duration) {
- duration = duration || 250;
- setTransform(t2, duration, false
- /* don't force */
- );
- return map;
- };
-
- map.zoomToEase = function (obj, duration) {
- var extent;
-
- if (Array.isArray(obj)) {
- obj.forEach(function (entity) {
- var entityExtent = entity.extent(context.graph());
-
- if (!extent) {
- extent = entityExtent;
- } else {
- extent = extent.extend(entityExtent);
- }
- });
- } 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;
- };
-
- map.cancelEase = function () {
- _selection.interrupt();
-
- return map;
- };
-
- 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));
- }
- };
-
- 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 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;
- }
-
- 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);
- };
-
- map.withinEditableZoom = function () {
- return map.zoom() >= context.minEditableZoom();
- };
-
- map.isInWideSelection = function () {
- return !map.withinEditableZoom() && context.selectedIDs().length;
- };
-
- map.editableDataEnabled = function (skipZoomCheck) {
- var layer = context.layers().layer('osm');
- if (!layer || !layer.enabled()) return false;
- return skipZoomCheck || map.withinEditableZoom();
- };
-
- map.notesEditable = function () {
- var layer = context.layers().layer('notes');
- if (!layer || !layer.enabled()) return false;
- return map.withinEditableZoom();
- };
-
- map.minzoom = function (val) {
- if (!arguments.length) return _minzoom;
- _minzoom = val;
- return map;
- };
-
- map.toggleHighlightEdited = function () {
- surface.classed('highlight-edited', !surface.classed('highlight-edited'));
- map.pan([0, 0]); // trigger a redraw
-
- dispatch$1.call('changeHighlighting', this);
- };
-
- map.areaFillOptions = ['wireframe', 'partial', 'full'];
-
- map.activeAreaFill = function (val) {
- if (!arguments.length) return corePreferences('area-fill') || 'partial';
- corePreferences('area-fill', val);
-
- if (val !== 'wireframe') {
- corePreferences('area-fill-toggle', val);
- }
-
- updateAreaFill();
- map.pan([0, 0]); // trigger a redraw
-
- dispatch$1.call('changeAreaFill', this);
- return map;
- };
-
- map.toggleWireframe = function () {
- var activeFill = map.activeAreaFill();
-
- if (activeFill === 'wireframe') {
- activeFill = corePreferences('area-fill-toggle') || 'partial';
- } else {
- activeFill = 'wireframe';
- }
-
- map.activeAreaFill(activeFill);
- };
-
- function updateAreaFill() {
- var activeFill = map.activeAreaFill();
- map.areaFillOptions.forEach(function (opt) {
- surface.classed('fill-' + opt, Boolean(opt === activeFill));
- });
- }
-
- map.layers = function () {
- return drawLayers;
- };
-
- map.doubleUpHandler = function () {
- return _doubleUpHandler;
- };
-
- return utilRebind(map, dispatch$1, 'on');
- }
-
- function rendererPhotos(context) {
- var dispatch$1 = dispatch('change');
- var _layerIDs = ['streetside', 'mapillary', 'mapillary-map-features', 'mapillary-signs', 'openstreetcam'];
- var _allPhotoTypes = ['flat', 'panoramic'];
-
- var _shownPhotoTypes = _allPhotoTypes.slice(); // shallow copy
-
-
- var _dateFilters = ['fromDate', 'toDate'];
-
- var _fromDate;
-
- var _toDate;
-
- var _usernames;
-
- function photos() {}
-
- 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 {
- delete hash.photo_overlay;
- }
-
- window.location.replace('#' + utilQsString(hash, true));
- }
-
- photos.overlayLayerIDs = function () {
- return _layerIDs;
- };
-
- photos.allPhotoTypes = function () {
- return _allPhotoTypes;
- };
-
- photos.dateFilters = function () {
- return _dateFilters;
- };
-
- photos.dateFilterValue = function (val) {
- return val === _dateFilters[0] ? _fromDate : _toDate;
- };
-
- photos.setDateFilter = function (type, val, updateUrl) {
- // validate the date
- var date = val && new Date(val);
-
- 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 (type === _dateFilters[1]) {
- _toDate = val;
-
- if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
- _fromDate = _toDate;
- }
- }
-
- dispatch$1.call('change', this);
-
- if (updateUrl) {
- var rangeString;
-
- if (_fromDate || _toDate) {
- rangeString = (_fromDate || '') + '_' + (_toDate || '');
- }
-
- setUrlFilterValue('photo_dates', rangeString);
- }
- };
-
- photos.setUsernameFilter = function (val, updateUrl) {
- if (val && typeof val === 'string') val = val.replace(/;/g, ',').split(',');
-
- if (val) {
- val = val.map(function (d) {
- return d.trim();
- }).filter(Boolean);
-
- if (!val.length) {
- val = null;
- }
- }
-
- _usernames = val;
- dispatch$1.call('change', this);
-
- if (updateUrl) {
- var hashString;
-
- if (_usernames) {
- hashString = _usernames.join(',');
- }
-
- setUrlFilterValue('photo_username', hashString);
- }
- };
-
- function setUrlFilterValue(property, val) {
- if (!window.mocha) {
- var hash = utilStringQs(window.location.hash);
-
- if (val) {
- if (hash[property] === val) return;
- hash[property] = val;
- } else {
- if (!(property in hash)) return;
- delete hash[property];
- }
-
- window.location.replace('#' + utilQsString(hash, true));
- }
- }
-
- function showsLayer(id) {
- var layer = context.layers().layer(id);
- return layer && layer.supported() && layer.enabled();
- }
-
- photos.shouldFilterByDate = function () {
- return showsLayer('mapillary') || showsLayer('openstreetcam') || showsLayer('streetside');
- };
-
- photos.shouldFilterByPhotoType = function () {
- return showsLayer('mapillary') || showsLayer('streetside') && showsLayer('openstreetcam');
- };
-
- photos.shouldFilterByUsername = function () {
- return showsLayer('mapillary') || showsLayer('openstreetcam') || showsLayer('streetside');
- };
-
- photos.showsPhotoType = function (val) {
- if (!photos.shouldFilterByPhotoType()) return true;
- return _shownPhotoTypes.indexOf(val) !== -1;
- };
-
- photos.showsFlat = function () {
- return photos.showsPhotoType('flat');
- };
-
- photos.showsPanoramic = function () {
- return photos.showsPhotoType('panoramic');
- };
-
- photos.fromDate = function () {
- return _fromDate;
- };
-
- photos.toDate = function () {
- return _toDate;
- };
-
- photos.togglePhotoType = function (val) {
- var index = _shownPhotoTypes.indexOf(val);
-
- if (index !== -1) {
- _shownPhotoTypes.splice(index, 1);
- } else {
- _shownPhotoTypes.push(val);
- }
-
- dispatch$1.call('change', this);
- return photos;
- };
-
- photos.usernames = function () {
- return _usernames;
- };
-
- photos.init = function () {
- var hash = utilStringQs(window.location.hash);
-
- 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);
- }
-
- if (hash.photo_username) {
- this.setUsernameFilter(hash.photo_username, false);
- }
-
- 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);
- });
- }
-
- 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);
-
- if (results && results.length >= 3) {
- var serviceId = results[1];
- var photoKey = results[2];
- var service = services[serviceId];
-
- 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 (!service.cachedImage(photoKey)) return;
- service.on('loadedImages.rendererPhotos', null);
- service.ensureViewerLoaded(context).then(function () {
- service.selectImage(context, photoKey).showViewer(context);
- });
- });
- }
- }
- }
-
- 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;
-
- if (!osm.authenticated()) {
- selection.selectAll('.userLink, .logoutLink').classed('hide', true);
- return;
- }
-
- 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 (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
-
-
- 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();
- });
- });
- }
-
- 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);
- }
- };
- }
-
- function uiAttribution(context) {
- var _selection = select(null);
-
- 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]);
-
- if (d.terms_html) {
- attribution.html(d.terms_html);
- return;
- }
-
- if (d.terms_url) {
- attribution = attribution.append('a').attr('href', d.terms_url).attr('target', '_blank');
- }
-
- 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()
- });
-
- if (d.icon && !d.overlay) {
- attribution.append('img').attr('class', 'source-image').attr('src', d.icon);
- }
-
- 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();
-
- _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();
- };
- }
-
- function uiContributors(context) {
- var osm = context.connection(),
- debouncedUpdate = debounce(function () {
- update();
- }, 1000),
- limit = 4,
- hidden = false,
- wrap = select(null);
-
- 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);
-
- 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 (!u.length) {
- hidden = true;
- wrap.transition().style('opacity', 0);
- } else if (hidden) {
- wrap.transition().style('opacity', 1);
- }
- }
-
- return function (selection) {
- if (!osm) return;
- wrap = selection;
- update();
- osm.on('loaded.contributors', debouncedUpdate);
- context.map().on('move.contributors', debouncedUpdate);
- };
- }
-
- var _popoverID = 0;
- function uiPopover(klass) {
- var _id = _popoverID++;
-
- var _anchorSelection = select(null);
-
- var popover = function popover(selection) {
- _anchorSelection = selection;
- selection.each(setup);
- };
-
- var _animation = utilFunctor(false);
-
- var _placement = utilFunctor('top'); // top, bottom, left, right
-
-
- var _alignment = utilFunctor('center'); // leading, center, trailing
-
-
- var _scrollContainer = utilFunctor(select(null));
-
- var _content;
-
- var _displayType = utilFunctor('');
-
- var _hasArrow = utilFunctor(true); // use pointer events on supported platforms; fallback to mouse events
-
-
- var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
-
- popover.displayType = function (val) {
- if (arguments.length) {
- _displayType = utilFunctor(val);
- return popover;
- } else {
- return _displayType;
- }
- };
-
- popover.hasArrow = function (val) {
- if (arguments.length) {
- _hasArrow = utilFunctor(val);
- return popover;
- } else {
- return _hasArrow;
- }
- };
-
- 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;
- }
- };
-
- popover.scrollContainer = function (val) {
- if (arguments.length) {
- _scrollContainer = utilFunctor(val);
- return popover;
- } else {
- return _scrollContainer;
- }
- };
-
- popover.content = function (val) {
- if (arguments.length) {
- _content = val;
- return popover;
- } else {
- return _content;
- }
- };
-
- popover.isShown = function () {
- var popoverSelection = _anchorSelection.select('.popover-' + _id);
-
- return !popoverSelection.empty() && popoverSelection.classed('in');
- };
-
- popover.show = function () {
- _anchorSelection.each(show);
- };
-
- popover.updateContent = function () {
- _anchorSelection.each(updateContent);
- };
-
- popover.hide = function () {
- _anchorSelection.each(hide);
- };
-
- popover.toggle = function () {
- _anchorSelection.each(toggle);
- };
-
- 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();
- };
-
- popover.destroyAny = function (selection) {
- selection.call(popover.destroy, '.popover');
- };
-
- function setup() {
- var anchor = select(this);
-
- 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);
-
- if (animate) {
- popoverSelection.classed('fade', true);
- }
-
- var display = _displayType.apply(this, arguments);
-
- if (display === 'hover') {
- var _lastNonMouseEnterTime;
-
- 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;
- } 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
-
-
- 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);
- });
- });
- }
- }
-
- function show() {
- var anchor = select(this);
- var popoverSelection = anchor.selectAll('.popover-' + _id);
-
- if (popoverSelection.empty()) {
- // popover was removed somehow, put it back
- anchor.call(popover.destroy);
- anchor.each(setup);
- popoverSelection = anchor.selectAll('.popover-' + _id);
- }
-
- popoverSelection.classed('in', true);
-
- var displayType = _displayType.apply(this, arguments);
-
- if (displayType === 'clickFocus') {
- anchor.classed('active', true);
- popoverSelection.node().focus();
- }
-
- anchor.each(updateContent);
- }
-
- function updateContent() {
- var anchor = select(this);
-
- if (_content) {
- anchor.selectAll('.popover-' + _id + ' > .popover-inner').call(_content.apply(this, arguments));
- }
-
- 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
-
- updatePosition.apply(this, arguments);
- updatePosition.apply(this, arguments);
- }
-
- function updatePosition() {
- var anchor = select(this);
- var popoverSelection = anchor.selectAll('.popover-' + _id);
-
- var scrollContainer = _scrollContainer && _scrollContainer.apply(this, arguments);
-
- 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);
-
- popoverSelection.classed('left', false).classed('right', false).classed('top', false).classed('bottom', false).classed(placement, true);
-
- var alignment = _alignment.apply(this, arguments);
-
- var alignFactor = 0.5;
-
- if (alignment === 'leading') {
- alignFactor = 0;
- } else if (alignment === 'trailing') {
- alignFactor = 1;
- }
-
- 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;
-
- case 'bottom':
- position = {
- x: anchorFrame.x + (anchorFrame.w - popoverFrame.w) * alignFactor,
- y: anchorFrame.y + anchorFrame.h
- };
- break;
-
- case 'left':
- position = {
- x: anchorFrame.x - popoverFrame.w,
- y: anchorFrame.y + (anchorFrame.h - popoverFrame.h) * alignFactor
- };
- break;
-
- case 'right':
- position = {
- x: anchorFrame.x + anchorFrame.w,
- y: anchorFrame.y + (anchorFrame.h - popoverFrame.h) * alignFactor
- };
- break;
- }
-
- 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;
- }
-
- var arrow = anchor.selectAll('.popover-' + _id + ' > .popover-arrow'); // keep the arrow centered on the button, or as close as possible
-
- var arrowPosX = Math.min(Math.max(popoverFrame.w / 2 - (position.x - initialPosX), 10), popoverFrame.w - 10);
- arrow.style('left', ~~arrowPosX + 'px');
- }
-
- popoverSelection.style('left', ~~position.x + 'px').style('top', ~~position.y + 'px');
- } else {
- popoverSelection.style('left', null).style('top', null);
- }
-
- function getFrame(node) {
- var positionStyle = select(node).style('position');
-
- 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
- };
- }
- }
- }
-
- function hide() {
- var anchor = select(this);
-
- if (_displayType.apply(this, arguments) === 'clickFocus') {
- anchor.classed('active', false);
- }
-
- anchor.selectAll('.popover-' + _id).classed('in', false);
- }
-
- function toggle() {
- if (select(this).select('.popover-' + _id).classed('in')) {
- hide.apply(this, arguments);
- } else {
- show.apply(this, arguments);
- }
- }
-
- return popover;
- }
-
- function uiTooltip(klass) {
- var tooltip = uiPopover((klass || '') + ' tooltip').displayType('hover');
-
- var _title = function _title() {
- var title = this.getAttribute('data-original-title');
-
- if (title) {
- return title;
- } else {
- title = this.getAttribute('title');
- this.removeAttribute('title');
- this.setAttribute('data-original-title', title);
- }
-
- return title;
- };
-
- var _heading = utilFunctor(null);
-
- var _keys = utilFunctor(null);
-
- tooltip.title = function (val) {
- if (!arguments.length) return _title;
- _title = utilFunctor(val);
- return tooltip;
- };
-
- tooltip.heading = function (val) {
- if (!arguments.length) return _heading;
- _heading = utilFunctor(val);
- return tooltip;
- };
-
- tooltip.keys = function (val) {
- if (!arguments.length) return _keys;
- _keys = utilFunctor(val);
- return tooltip;
- };
-
- 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;
- }
-
- 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
-
- var _vpSideMargin = 35; // viewport side margin
-
- var _menuTop = false;
-
- var _menuHeight;
-
- var _menuWidth; // hardcode these values to make menu positioning easier
-
-
- var _verticalPadding = 4; // see also `.edit-menu .tooltip` CSS; include margin
-
- var _tooltipWidth = 210; // offset the menu slightly from the target location
-
- var _menuSideMargin = 10;
- var _tooltips = [];
-
- var editMenu = function editMenu(selection) {
- var isTouchMenu = _triggerType.includes('touch') || _triggerType.includes('pen');
-
- var ops = _operations.filter(function (op) {
- return !isTouchMenu || !op.mouseOnly;
- });
-
- 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
-
- _menuTop = isTouchMenu; // Show labels for touch input since there aren't hover tooltips
-
- 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;
- }
-
- _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]]);
-
- _tooltips.push(tooltip);
-
- select(this).call(tooltip).append('div').attr('class', 'icon-wrap').call(svgIcon('#iD-operation-' + d.id, 'operation'));
- });
-
- 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;
- }
-
- function click(d3_event, operation) {
- d3_event.stopPropagation();
-
- 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)();
- }
-
- operation();
- editMenu.close();
- }
-
- lastPointerUpType = null;
- }
-
- dispatch$1.call('toggled', this, true);
- };
-
- function updatePosition() {
- if (!_menu || _menu.empty()) return;
- var anchorLoc = context.projection(_anchorLocLonLat);
- var viewport = context.surfaceRect();
-
- 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;
- }
-
- 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 origin = geoVecAdd(anchorLoc, offset);
-
- _menu.style('left', origin[0] + 'px').style('top', origin[1] + 'px');
-
- var tooltipSide = tooltipPosition(viewport, menuLeft);
-
- _tooltips.forEach(function (tooltip) {
- tooltip.placement(tooltipSide);
- });
-
- 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 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 '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
-
-
- return 'left';
- }
- }
- }
-
- editMenu.close = function () {
- context.map().on('move.edit-menu', null).on('drawn.edit-menu', null);
-
- _menu.remove();
-
- _tooltips = [];
- dispatch$1.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$1, '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);
- });
- };
- }
-
- function uiFlash(context) {
- var _flashTimer;
-
- var _duration = 2000;
- var _iconName = '#iD-icon-no';
- var _iconClass = 'disabled';
- var _label = '';
-
- function flash() {
- if (_flashTimer) {
- _flashTimer.stop();
- }
-
- 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
-
- 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;
- }
-
- flash.duration = function (_) {
- if (!arguments.length) return _duration;
- _duration = _;
- return flash;
- };
-
- flash.label = function (_) {
- if (!arguments.length) return _label;
- _label = _;
- return flash;
- };
-
- flash.iconName = function (_) {
- if (!arguments.length) return _iconName;
- _iconName = _;
- return flash;
- };
-
- flash.iconClass = function (_) {
- if (!arguments.length) return _iconClass;
- _iconClass = _;
- return flash;
- };
-
- return flash;
- }
-
- 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;
- }
- }
-
- 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;
- }
- }
-
- function isFullScreen() {
- return document.fullscreenElement || document.mozFullScreenElement || document.webkitFullscreenElement || document.msFullscreenElement;
- }
-
- function isSupported() {
- return !!getFullScreenFn();
- }
-
- function fullScreen(d3_event) {
- d3_event.preventDefault();
-
- if (!isFullScreen()) {
- // button.classed('active', true);
- getFullScreenFn().apply(element);
- } else {
- // button.classed('active', false);
- getExitFullScreenFn().apply(document);
- }
- }
-
- 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);
- };
- }
-
- function uiGeolocate(context) {
- var _geolocationOptions = {
- // prioritize speed and power usage over precision
- enableHighAccuracy: false,
- // don't hang indefinitely getting the location
- timeout: 6000 // 6sec
-
- };
-
- var _locating = uiLoading(context).message(_t.html('geolocate.locating')).blocking(true);
-
- var _layer = context.layers().layer('geolocate');
-
- var _position;
-
- var _extent;
-
- var _timeoutID;
-
- var _button = select(null);
-
- function click() {
- if (context.inIntro()) return;
-
- 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
-
- navigator.geolocation.getCurrentPosition(success, error, _geolocationOptions);
- } else {
- _locating.close();
-
- _layer.enabled(null, false);
-
- updateButtonState();
- }
- }
-
- function zoomTo() {
- context.enter(modeBrowse(context));
- var map = context.map();
-
- _layer.enabled(_position, true);
-
- updateButtonState();
- map.centerZoomEase(_extent.center(), Math.min(20, map.extentZoom(_extent)));
- }
-
- 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')();
- }
-
- finish();
- }
-
- function finish() {
- _locating.close(); // unblock ui
-
-
- if (_timeoutID) {
- clearTimeout(_timeoutID);
- }
-
- _timeoutID = undefined;
- }
-
- function updateButtonState() {
- _button.classed('active', _layer.enabled());
- }
-
- 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);
- };
- }
-
- function uiPanelBackground(context) {
- var background = context.background();
- var _currSourceName = null;
- var _metadata = {};
- var _metadataKeys = ['zoom', 'vintage', 'source', 'description', 'resolution', 'accuracy'];
-
- var debouncedRedraw = debounce(redraw, 250);
-
- function redraw(selection) {
- var source = background.baseLayerSource();
- if (!source) return;
- var isDG = source.id.match(/^DigitalGlobe/i) !== null;
- var sourceLabel = source.label();
-
- if (_currSourceName !== sourceLabel) {
- _currSourceName = sourceLabel;
- _metadata = {};
- }
-
- selection.html('');
- var list = selection.append('ul').attr('class', 'background-info');
- list.append('li').html(_currSourceName);
-
- _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
-
-
- ['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);
- }
- }
- });
- }
-
- var debouncedGetMetadata = debounce(getMetadata, 250);
-
- function getMetadata(selection) {
- var tile = context.container().select('.layer-background img.tile-center'); // tile near viewport center
-
- 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
-
- _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 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
-
- _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);
- });
- });
- }
-
- 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);
- });
- };
-
- panel.off = function () {
- context.map().on('drawn.info-background', null).on('move.info-background', null);
- };
-
- 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;
-
- 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);
- }
-
- function displayUser(selection, userName) {
- if (!userName) {
- selection.append('span').html(_t.html('info_panels.history.unknown'));
- return;
- }
-
- selection.append('span').attr('class', 'user-name').html(userName);
- var links = selection.append('div').attr('class', 'links');
-
- if (osm) {
- links.append('a').attr('class', 'user-osm-link').attr('href', osm.userURL(userName)).attr('target', '_blank').html('OSM');
- }
-
- links.append('a').attr('class', 'user-hdyc-link').attr('href', 'https://hdyc.neis-one.org/?' + userName).attr('target', '_blank').attr('tabindex', -1).html('HDYC');
- }
-
- function displayChangeset(selection, changeset) {
- if (!changeset) {
- selection.append('span').html(_t.html('info_panels.history.unknown'));
- return;
- }
-
- selection.append('span').attr('class', 'changeset-id').html(changeset);
- var links = selection.append('div').attr('class', 'links');
-
- if (osm) {
- links.append('a').attr('class', 'changeset-osm-link').attr('href', osm.changesetURL(changeset)).attr('target', '_blank').html('OSM');
- }
-
- 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');
- }
-
- function redraw(selection) {
- var selectedNoteID = context.selectedNoteID();
- osm = context.connection();
- var selected, note, entity;
-
- if (selectedNoteID && osm) {
- // selected 1 note
- selected = [_t('note.note') + ' ' + selectedNoteID];
- note = osm.getNote(selectedNoteID);
- } else {
- // selected 1..n entities
- selected = context.selectedIDs().filter(function (e) {
- return context.hasEntity(e);
- });
-
- if (selected.length) {
- entity = context.entity(selected[0]);
- }
- }
-
- 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;
-
- if (entity) {
- selection.call(redrawEntity, entity);
- } else if (note) {
- selection.call(redrawNote, note);
- }
- }
-
- function redrawNote(selection, note) {
- if (!note || note.isNew()) {
- selection.append('div').html(_t.html('info_panels.history.note_no_history'));
- return;
- }
-
- var list = selection.append('ul');
- list.append('li').html(_t.html('info_panels.history.note_comments') + ':').append('span').html(note.comments.length);
-
- 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);
- }
-
- 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'));
- }
- }
-
- function redrawEntity(selection, entity) {
- if (!entity || entity.isNew()) {
- selection.append('div').html(_t.html('info_panels.history.no_history'));
- return;
- }
-
- var links = selection.append('div').attr('class', 'links');
-
- 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');
- }
-
- 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 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);
- });
- };
-
- panel.off = function () {
- context.map().on('drawn.info-history', null);
- context.on('enter.info-history', null);
- };
-
- panel.id = 'history';
- panel.label = _t.html('info_panels.history.title');
- panel.key = _t('info_panels.history.key');
- return panel;
- }
-
- 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 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';
- }
-
- if (d > 1000 && d < 10000000) {
- // 0.1 - 1000 hectares
- d2 = d / 10000;
- unit2 = 'hectares';
- }
- }
-
- area = _t('units.' + unit1, {
- quantity: d1.toLocaleString(locale, {
- maximumSignificantDigits: 4
- })
- });
-
- if (d2) {
- return _t('units.area_pair', {
- area1: area,
- area2: _t('units.' + unit2, {
- quantity: d2.toLocaleString(locale, {
- maximumSignificantDigits: 2
- })
- })
- });
- } else {
- return area;
- }
- }
-
- function wrap$2(x, min, max) {
- var d = max - min;
- return ((x - min) % d + d) % d + min;
- }
-
- function clamp$1(x, min, max) {
- return Math.max(min, Math.min(x, max));
- }
-
- 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;
-
- 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)
- });
- }
-
- 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
- */
-
-
- 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)
- });
- }
-
- function uiPanelLocation(context) {
- var currLocation = '';
-
- function redraw(selection) {
- selection.html('');
- var list = selection.append('ul'); // Mouse coordinates
-
- var coord = context.map().mouseCoordinates();
-
- if (coord.some(isNaN)) {
- coord = context.map().center();
- }
-
- list.append('li').html(dmsCoordinatePair(coord)).append('li').html(decimalCoordinatePair(coord)); // Location Info
-
- selection.append('div').attr('class', 'location-info').html(currLocation || ' ');
- debouncedGetLocation(selection, coord);
- }
-
- 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 panel = function panel(selection) {
- selection.call(redraw);
- context.surface().on(('PointerEvent' in window ? 'pointer' : 'mouse') + 'move.info-location', function () {
- selection.call(redraw);
- });
- };
-
- panel.off = function () {
- context.surface().on('.info-location', null);
- };
-
- 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;
- }
-
- function steradiansToSqmeters(r) {
- // http://gis.stackexchange.com/a/124857/40446
- return r / (4 * Math.PI) * 510065621724000;
- }
-
- function toLineString(feature) {
- if (feature.type === 'LineString') return feature;
- var result = {
- type: 'LineString',
- coordinates: []
- };
-
- if (feature.type === 'Polygon') {
- result.coordinates = feature.coordinates[0];
- } else if (feature.type === 'MultiPolygon') {
- result.coordinates = feature.coordinates[0][0];
- }
-
- return result;
- }
-
- var _isImperial = !_mainLocalizer.usesMetric();
-
- 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 (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 (selected.length) {
- var extent = geoExtent();
-
- for (var i in selected) {
- var entity = selected[i];
-
- extent._extend(entity.extent(graph));
-
- geometry = entity.geometry(graph);
-
- 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
-
- centroid = d3_geoCentroid(geojsonRewind(Object.assign({}, feature), true));
-
- if (closed) {
- area += steradiansToSqmeters(entity.area(graph));
- }
- }
- }
-
- if (selected.length > 1) {
- geometry = null;
- closed = null;
- centroid = null;
- }
-
- if (selected.length === 2 && selected[0].type === 'node' && selected[1].type === 'node') {
- distance = geoSphericalDistance(selected[0].loc, selected[1].loc);
- }
-
- 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();
- }
- }
- }
-
- selection.html('');
-
- if (heading) {
- selection.append('h4').attr('class', 'measurement-heading').html(heading);
- }
-
- var list = selection.append('ul');
- var coordItem;
-
- 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 (totalNodeCount) {
- list.append('li').html(_t.html('info_panels.measurement.node_count') + ':').append('span').html(totalNodeCount.toLocaleString(localeCode));
- }
-
- if (area) {
- list.append('li').html(_t.html('info_panels.measurement.area') + ':').append('span').html(displayArea(area, _isImperial));
- }
-
- if (length) {
- list.append('li').html(_t.html('info_panels.measurement.' + (closed ? 'perimeter' : 'length')) + ':').append('span').html(displayLength(length, _isImperial));
- }
-
- if (typeof distance === 'number') {
- list.append('li').html(_t.html('info_panels.measurement.distance') + ':').append('span').html(displayLength(distance, _isImperial));
- }
-
- 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 (centroid) {
- coordItem = list.append('li').html(_t.html('info_panels.measurement.centroid') + ':');
- coordItem.append('span').html(dmsCoordinatePair(centroid));
- coordItem.append('span').html(decimalCoordinatePair(centroid));
- }
-
- 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));
- }
-
- 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);
- });
- }
- }
-
- 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);
- });
- };
-
- panel.off = function () {
- context.map().on('drawn.info-measurement', null);
- context.on('enter.info-measurement', null);
- };
-
- panel.id = 'measurement';
- panel.label = _t.html('info_panels.measurement.title');
- panel.key = _t('info_panels.measurement.key');
- return panel;
- }
-
- var uiInfoPanels = {
- background: uiPanelBackground,
- history: uiPanelHistory,
- location: uiPanelLocation,
- measurement: uiPanelMeasurement
- };
-
- function uiInfo(context) {
- var ids = Object.keys(uiInfoPanels);
- var wasActive = ['measurement'];
- var panels = {};
- var active = {}; // create panels
-
- ids.forEach(function (k) {
- if (!panels[k]) {
- panels[k] = uiInfoPanels[k](context);
- active[k] = false;
- }
- });
-
- 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
-
- infoPanels.selectAll('.panel-content').each(function (d) {
- select(this).call(panels[d]);
- });
- }
-
- info.toggle = function (which) {
- var activeids = ids.filter(function (k) {
- return active[k];
- });
-
- if (which) {
- // toggle one
- active[which] = !active[which];
-
- if (activeids.length === 1 && activeids[0] === which) {
- // none active anymore
- wasActive = [which];
- }
-
- 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;
- });
- }
- }
-
- 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);
- });
- });
- }
-
- return info;
- }
-
- 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;
-
- 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;
- }
-
- 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 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;
-
- if (replacements) {
- reps = Object.assign(replacements, helpStringReplacements);
- } else {
- reps = helpStringReplacements;
- }
-
- return _t.html(id, reps) // use keyboard key styling for shortcuts
- .replace(/\`(.*?)\`/g, '<kbd>$1</kbd>');
- }
-
- 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
-
-
- 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
- }
- }
-
- function localize(obj) {
- var key; // Assign name if entity has one..
-
- var name = obj.tags && obj.tags.name;
-
- 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..
-
-
- var street = obj.tags && obj.tags['addr:street'];
-
- 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..
-
- 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
- });
-
- if (str) {
- if (str.match(/^<.*>$/) !== null) {
- delete obj.tags[tag];
- } else {
- obj.tags[tag] = str;
- }
- }
- });
- }
-
- 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
-
- var lowerBound = Math.cos((90 - threshold) * Math.PI / 180); // near right
-
- var upperBound = Math.cos(threshold * Math.PI / 180); // near straight
-
- 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 (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;
- }
-
- function uiCurtain(containerNode) {
- var surface = select(null),
- tooltip = select(null),
- darkness = select(null);
-
- 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();
-
- function resize() {
- surface.attr('width', containerNode.clientWidth).attr('height', containerNode.clientHeight);
- curtain.cut(darkness.datum());
- }
- }
- /**
- * 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
- */
-
-
- curtain.reveal = function (box, html, options) {
- options = options || {};
-
- 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;
- }
-
- 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;
- }
-
- var tooltipBox;
-
- if (options.tooltipBox) {
- tooltipBox = options.tooltipBox;
-
- if (typeof tooltipBox === 'string') {
- tooltipBox = select(tooltipBox).node();
- }
-
- if (tooltipBox && tooltipBox.getBoundingClientRect) {
- tooltipBox = copyBox(tooltipBox.getBoundingClientRect());
- }
- } else {
- tooltipBox = box;
- }
-
- 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..
-
-
- html = html.replace(/\*\*(.*?)\*\*/g, '<span class="instruction">$1</span>');
- }
-
- html = html.replace(/\*(.*?)\*/g, '<em>$1</em>'); // emphasis
-
- html = html.replace(/\{br\}/g, '<br/><br/>'); // linebreak
-
- if (options.buttonText && options.buttonCallback) {
- html += '<div class="button-section">' + '<button href="#" class="button action">' + options.buttonText + '</button></div>';
- }
-
- 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();
- });
- }
-
- 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.
-
- if (options.tooltipClass === 'intro-mouse') {
- tip.height += 80;
- } // trim box dimensions to just the portion that fits in the container..
-
-
- if (tooltipBox.top + tooltipBox.height > h) {
- tooltipBox.height -= tooltipBox.top + tooltipBox.height - h;
- }
-
- if (tooltipBox.left + tooltipBox.width > w) {
- tooltipBox.width -= tooltipBox.left + tooltipBox.width - w;
- } // determine tooltip placement..
-
-
- 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 (_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.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;
-
- 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;
- }
- }
-
- tooltip.selectAll('.popover-inner').style('top', shiftY + 'px');
- } else {
- tooltip.classed('in', false).call(uiToggle(false));
- }
-
- curtain.cut(box, options.duration);
- return tooltip;
- };
-
- 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);
- }
-
- 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';
- });
- };
-
- 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 copyBox(src) {
- return {
- top: src.top,
- right: src.right,
- bottom: src.bottom,
- left: src.left,
- width: src.width,
- height: src.height
- };
- }
-
- return curtain;
- }
-
- function uiIntroWelcome(context, reveal) {
- var dispatch$1 = dispatch('done');
- var chapter = {
- title: 'intro.welcome.title'
- };
-
- 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
- });
- }
-
- function words() {
- reveal('.intro-nav-wrap .chapter-welcome', helpHtml('intro.welcome.words'), {
- buttonText: _t.html('intro.ok'),
- buttonCallback: chapters
- });
- }
-
- function chapters() {
- dispatch$1.call('done');
- reveal('.intro-nav-wrap .chapter-navigation', helpHtml('intro.welcome.chapters', {
- next: _t('intro.navigation.title')
- }));
- }
-
- chapter.enter = function () {
- welcome();
- };
-
- chapter.exit = function () {
- context.container().select('.curtain-tooltip.intro-mouse').selectAll('.counter').remove();
- };
-
- chapter.restart = function () {
- chapter.exit();
- chapter.enter();
- };
-
- return utilRebind(chapter, dispatch$1, 'on');
- }
-
- 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'
- };
-
- function timeout(f, t) {
- timeouts.push(window.setTimeout(f, t));
- }
-
- function eventCancel(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- }
-
- function isTownHallSelected() {
- var ids = context.selectedIDs();
- return ids.length === 1 && ids[0] === hallId;
- }
-
- function dragMap() {
- context.enter(modeBrowse(context));
- context.history().reset('initial');
- var msec = transitionTime(townHall, context.map().center());
-
- if (msec) {
- reveal(null, null, {
- duration: 0
- });
- }
-
- 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 (centerStart[0] !== centerNow[0] || centerStart[1] !== centerNow[1]) {
- context.map().on('move.intro', null);
- timeout(function () {
- continueTo(zoomMap);
- }, 3000);
- }
- });
- }, msec + 100);
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- nextStep();
- }
- }
-
- 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);
- }
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- nextStep();
- }
- }
-
- function features() {
- var onClick = function onClick() {
- continueTo(pointsLinesAreas);
- };
-
- 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();
- }
- }
-
- function pointsLinesAreas() {
- var onClick = function onClick() {
- continueTo(nodesWays);
- };
-
- 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
- });
- });
-
- function continueTo(nextStep) {
- context.map().on('drawn.intro', null);
- nextStep();
- }
- }
-
- function nodesWays() {
- var onClick = function onClick() {
- continueTo(clickTownHall);
- };
-
- 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
- });
- });
-
- function continueTo(nextStep) {
- context.map().on('drawn.intro', null);
- nextStep();
- }
- }
-
- 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
-
- context.history().on('change.intro', function () {
- if (!context.hasEntity(hallId)) {
- continueTo(clickTownHall);
- }
- });
-
- function continueTo(nextStep) {
- context.on('enter.intro', null);
- context.map().on('move.intro drawn.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- function selectedTownHall() {
- if (!isTownHallSelected()) return clickTownHall();
- var entity = context.hasEntity(hallId);
- if (!entity) return clickTownHall();
- var box = pointBox(entity.loc, context);
-
- var onClick = function onClick() {
- continueTo(editorTownHall);
- };
-
- 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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- function editorTownHall() {
- if (!isTownHallSelected()) return clickTownHall(); // disallow scrolling
-
- context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-
- var onClick = function onClick() {
- continueTo(presetTownHall);
- };
-
- 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 continueTo(nextStep) {
- context.on('exit.intro', null);
- context.history().on('change.intro', null);
- context.container().select('.inspector-wrap').on('wheel.intro', null);
- nextStep();
- }
- }
-
- function presetTownHall() {
- if (!isTownHallSelected()) return clickTownHall(); // reset pane, in case user happened to change it..
-
- context.container().select('.inspector-wrap .panewrap').style('right', '0%'); // disallow scrolling
-
- context.container().select('.inspector-wrap').on('wheel.intro', eventCancel); // preset match, in case the user happened to change it.
-
- var entity = context.entity(context.selectedIDs()[0]);
- var preset = _mainPresetIndex.match(entity, context.graph());
-
- 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);
- }
- });
-
- 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 fieldsTownHall() {
- if (!isTownHallSelected()) return clickTownHall(); // reset pane, in case user happened to change it..
-
- context.container().select('.inspector-wrap .panewrap').style('right', '0%'); // disallow scrolling
-
- context.container().select('.inspector-wrap').on('wheel.intro', eventCancel);
-
- var onClick = function onClick() {
- continueTo(closeTownHall);
- };
-
- 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 continueTo(nextStep) {
- context.on('exit.intro', null);
- context.history().on('change.intro', null);
- context.container().select('.inspector-wrap').on('wheel.intro', null);
- nextStep();
- }
- }
-
- 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
- });
- });
-
- 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 msec = transitionTime(springStreet, context.map().center());
-
- if (msec) {
- reveal(null, null, {
- duration: 0
- });
- }
-
- context.map().centerZoomEase(springStreet, 19, msec); // ..and user can see it
-
- 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);
- }
-
- function checkSearchResult() {
- var first = context.container().select('.feature-list-item:nth-child(0n+2)'); // skip "No Results" item
-
- 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);
- }
-
- function continueTo(nextStep) {
- context.on('exit.intro', null);
- context.container().select('.search-header input').on('keydown.intro', null).on('keyup.intro', null);
- nextStep();
- }
- }
-
- function selectedStreet() {
- if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
- return searchStreet();
- }
-
- var onClick = function onClick() {
- continueTo(editorStreet);
- };
-
- 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.
-
- context.on('enter.intro', function (mode) {
- if (!context.hasEntity(springStreetId)) {
- return continueTo(searchStreet);
- }
-
- var ids = context.selectedIDs();
-
- 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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- 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
- });
- });
-
- function continueTo(nextStep) {
- context.on('exit.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- 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');
- }
- });
- }
-
- chapter.enter = function () {
- dragMap();
- };
-
- 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);
- };
-
- chapter.restart = function () {
- chapter.exit();
- chapter.enter();
- };
-
- return utilRebind(chapter, dispatch$1, 'on');
- }
-
- 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'
- };
-
- function timeout(f, t) {
- timeouts.push(window.setTimeout(f, t));
- }
-
- function eventCancel(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- }
-
- function addPoint() {
- context.enter(modeBrowse(context));
- context.history().reset('initial');
- var msec = transitionTime(intersection, context.map().center());
-
- if (msec) {
- reveal(null, null, {
- duration: 0
- });
- }
-
- 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);
-
- function continueTo(nextStep) {
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function placePoint() {
- if (context.mode().id !== 'add-point') {
- return chapter.restart();
- }
-
- 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();
- }
- }
-
- 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();
-
- if (mode.id !== 'select' || !ids.length || ids[0] !== _pointID) {
- // keep the user's point selected..
- context.enter(modeSelect(context, [_pointID])); // 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.history().on('change.intro', null);
- }
- });
-
- function checkPresetSearch() {
- var first = context.container().select('.preset-list-item:first-child');
-
- 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);
- });
- }
- }
-
- 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();
- }
-
- 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 continueTo(nextStep) {
- context.on('exit.intro', null);
- nextStep();
- }
- }
-
- function addName() {
- if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
- return addPoint();
- } // reset pane, in case user happened to change it..
-
-
- 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);
-
- 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 continueTo(nextStep) {
- context.on('exit.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- 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 continueTo(nextStep) {
- context.on('exit.intro', null);
- nextStep();
- }
- }
-
- 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..
-
- 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
- });
- }
-
- 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..
-
- 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();
- }
- }
-
- function updatePoint() {
- if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
- return continueTo(reselectPoint);
- } // reset pane, in case user happened to untag the point..
-
-
- 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 updateCloseEditor() {
- if (context.mode().id !== 'select' || !_pointID || !context.hasEntity(_pointID)) {
- return continueTo(reselectPoint);
- } // reset pane, in case user happened to change it..
-
-
- 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 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
-
- 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 continueTo(nextStep) {
- context.on('enter.intro', null);
- context.map().on('move.intro', null);
- nextStep();
- }
- }
-
- function enterDelete() {
- if (!_pointID) return chapter.restart();
- var entity = context.hasEntity(_pointID);
- if (!entity) return chapter.restart();
- var node = selectMenuItem(context, 'delete').node();
-
- if (!node) {
- return continueTo(rightClickPoint);
- }
-
- 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
-
- 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();
- }
- }
-
- function undo() {
- context.history().on('change.intro', function () {
- continueTo(play);
- });
- reveal('.top-toolbar button.undo-button', helpHtml('intro.points.undo'));
-
- function continueTo(nextStep) {
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- 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');
- }
- });
- }
-
- chapter.enter = function () {
- addPoint();
- };
-
- 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();
- };
-
- return utilRebind(chapter, dispatch$1, 'on');
- }
-
- 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 = [];
-
- var _areaID;
-
- var chapter = {
- title: 'intro.areas.title'
- };
-
- function timeout(f, t) {
- timeouts.push(window.setTimeout(f, t));
- }
-
- function eventCancel(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- }
-
- 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);
- }
-
- function addArea() {
- context.enter(modeBrowse(context));
- context.history().reset('initial');
- _areaID = null;
- var msec = transitionTime(playground, context.map().center());
-
- if (msec) {
- reveal(null, null, {
- duration: 0
- });
- }
-
- 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);
-
- function continueTo(nextStep) {
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function startPlayground() {
- if (context.mode().id !== 'add-area') {
- return chapter.restart();
- }
-
- _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
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function continuePlayground() {
- if (context.mode().id !== 'draw-area') {
- return chapter.restart();
- }
-
- _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
-
- context.on('enter.intro', function (mode) {
- if (mode.id === 'draw-area') {
- var entity = context.hasEntity(context.selectedIDs()[0]);
-
- 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();
- }
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- 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
-
- 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();
- }
- }
-
- function searchPresets() {
- if (!_areaID || !context.hasEntity(_areaID)) {
- return addArea();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
- context.enter(modeSelect(context, [_areaID]));
- } // 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%');
- 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..
-
- context.on('enter.intro', function (mode) {
- if (!_areaID || !context.hasEntity(_areaID)) {
- return continueTo(addArea);
- }
-
- var ids = context.selectedIDs();
-
- 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..
-
- 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.areas.search_playground', {
- preset: playgroundPreset.name()
- }));
- context.history().on('change.intro', null);
- }
- });
-
- 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);
- });
- }
- }
-
- 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();
- }
- }
-
- function clickAddField() {
- if (!_areaID || !context.hasEntity(_areaID)) {
- return addArea();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
- return searchPresets();
- }
-
- if (!context.container().select('.form-field-description').empty()) {
- return continueTo(describePlayground);
- } // 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', '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);
-
- if (entity.tags.description) {
- return continueTo(play);
- } // scroll "Add field" into view
-
-
- var box = context.container().select('.more-fields').node().getBoundingClientRect();
-
- 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);
- };
- });
- }
-
- 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
-
- context.on('exit.intro', function () {
- return continueTo(searchPresets);
- });
-
- 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();
- }
- }
-
- function chooseDescriptionField() {
- if (!_areaID || !context.hasEntity(_areaID)) {
- return addArea();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
- return searchPresets();
- }
-
- if (!context.container().select('.form-field-description').empty()) {
- return continueTo(describePlayground);
- } // Make sure combobox is ready..
-
-
- if (context.container().select('div.combobox').empty()) {
- return continueTo(clickAddField);
- } // Watch for the combobox to go away..
-
-
- 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();
- }
- }
-
- function describePlayground() {
- if (!_areaID || !context.hasEntity(_areaID)) {
- return addArea();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
- return searchPresets();
- } // reset pane, in case user happened to change it..
-
-
- context.container().select('.inspector-wrap .panewrap').style('right', '0%');
-
- if (context.container().select('.form-field-description').empty()) {
- return continueTo(retryChooseDescription);
- }
-
- 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 retryChooseDescription() {
- if (!_areaID || !context.hasEntity(_areaID)) {
- return addArea();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _areaID) {
- return searchPresets();
- } // reset pane, in case user happened to change it..
-
-
- 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);
- });
-
- function continueTo(nextStep) {
- context.on('exit.intro', null);
- nextStep();
- }
- }
-
- 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');
- }
- });
- }
-
- chapter.enter = function () {
- addArea();
- };
-
- 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();
- };
-
- return utilRebind(chapter, dispatch$1, '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 timeout(f, t) {
- timeouts.push(window.setTimeout(f, t));
- }
-
- function eventCancel(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- }
-
- function addLine() {
- context.enter(modeBrowse(context));
- context.history().reset('initial');
- var msec = transitionTime(tulipRoadStart, context.map().center());
-
- if (msec) {
- reveal(null, null, {
- duration: 0
- });
- }
-
- 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
- });
- });
- 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();
- }
- }
-
- 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..
-
- 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();
- }
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.history().on('change.intro', null);
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function isLineConnected() {
- var entity = _tulipRoadID && context.hasEntity(_tulipRoadID);
-
- 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 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();
-
- var entity = _tulipRoadID && context.hasEntity(_tulipRoadID);
-
- 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();
- }
- }
-
- 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 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 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 continueTo(nextStep) {
- context.container().select('.preset-list-button').on('click.intro', null);
- context.on('exit.intro', null);
- nextStep();
- }
- } // selected wrong road type
-
-
- 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 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);
- });
- 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 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 continueTo(nextStep) {
- nextStep();
- }
- }
-
- function updateLine() {
- context.history().reset('doneAddLine');
-
- if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
- return chapter.restart();
- }
-
- var msec = transitionTime(woodRoadDragMidpoint, context.map().center());
-
- 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);
-
- var advance = function advance() {
- continueTo(addNode);
- };
-
- 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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- nextStep();
- }
- }
-
- function addNode() {
- context.history().reset('doneAddLine');
-
- if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
- return chapter.restart();
- }
-
- 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 (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();
- }
- }
-
- function startDragEndpoint() {
- 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);
- 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);
- }
-
- 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 (geoSphericalDistance(entity.loc, woodRoadDragEndpoint) <= 4) {
- continueTo(finishDragEndpoint);
- }
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- nextStep();
- }
- }
-
- function finishDragEndpoint() {
- 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);
- 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 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 (geoSphericalDistance(entity.loc, woodRoadDragEndpoint) > 4) {
- continueTo(startDragEndpoint);
- }
- });
- context.on('enter.intro', function () {
- continueTo(startDragMidpoint);
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function startDragMidpoint() {
- if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
- return continueTo(updateLine);
- }
-
- if (context.selectedIDs().indexOf(woodRoadID) === -1) {
- context.enter(modeSelect(context, [woodRoadID]));
- }
-
- 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);
- }
-
- 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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.history().on('change.intro', null);
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function continueDragMidpoint() {
- 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;
-
- var advance = function advance() {
- context.history().checkpoint('doneUpdateLine');
- continueTo(deleteLines);
- };
-
- 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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- nextStep();
- }
- }
-
- function deleteLines() {
- context.history().reset('doneUpdateLine');
- context.enter(modeBrowse(context));
-
- if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
- return chapter.restart();
- }
-
- var msec = transitionTime(deleteLinesLoc, context.map().center());
-
- if (msec) {
- reveal(null, null, {
- duration: 0
- });
- }
-
- 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 advance = function advance() {
- continueTo(rightClickIntersection);
- };
-
- 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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- 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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- function splitIntersection() {
- if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
- return continueTo(deleteLines);
- }
-
- var node = selectMenuItem(context, 'split').node();
-
- if (!node) {
- return continueTo(rightClickIntersection);
- }
-
- 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 (!wasChanged && !node) {
- return continueTo(rightClickIntersection);
- }
-
- 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)
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- function retrySplit() {
- context.enter(modeBrowse(context));
- context.map().centerZoomEase(eleventhAvenueEnd, 18, 500);
-
- var advance = function advance() {
- continueTo(rightClickIntersection);
- };
-
- 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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- nextStep();
- }
- }
-
- function didSplit() {
- if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
- return continueTo(rightClickIntersection);
- }
-
- 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
-
- 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);
- }
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- function multiSelect() {
- if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
- return continueTo(rightClickIntersection);
- }
-
- var ids = context.selectedIDs();
- var hasWashington = ids.indexOf(_washingtonSegmentID) !== -1;
- var hasTwelfth = ids.indexOf(twelfthAvenueID) !== -1;
-
- if (hasWashington && hasTwelfth) {
- return continueTo(multiRightClick);
- } else if (!hasWashington && !hasTwelfth) {
- return continueTo(didSplit);
- }
-
- context.map().centerZoomEase(twelfthAvenue, 18, 500);
- timeout(function () {
- var selected, other, padding, box;
-
- 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;
- }
-
- 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;
- }
-
- 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);
- }
- });
- }, 600);
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- 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();
-
- 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 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 multiDelete() {
- if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
- return continueTo(rightClickIntersection);
- }
-
- 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);
- }
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('exit.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- 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);
- }
- });
-
- function continueTo(nextStep) {
- nextStep();
- }
- }
-
- 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');
- }
- });
- }
-
- 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();
- };
-
- return utilRebind(chapter, dispatch$1, 'on');
- }
-
- 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 timeout(f, t) {
- timeouts.push(window.setTimeout(f, t));
- }
-
- function eventCancel(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- }
-
- 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 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);
- }
-
- function addHouse() {
- context.enter(modeBrowse(context));
- context.history().reset('initial');
- _houseID = null;
- var msec = transitionTime(house, context.map().center());
-
- if (msec) {
- reveal(null, null, {
- duration: 0
- });
- }
-
- 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);
-
- function continueTo(nextStep) {
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function startHouse() {
- if (context.mode().id !== 'add-area') {
- return continueTo(addHouse);
- }
-
- _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
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function continueHouse() {
- if (context.mode().id !== 'draw-area') {
- return continueTo(addHouse);
- }
-
- _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);
- });
-
- if (isMostlySquare(points)) {
- _houseID = way.id;
- return continueTo(chooseCategoryBuilding);
- } else {
- return continueTo(retryHouse);
- }
- } else {
- return chapter.restart();
- }
- });
-
- 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);
- };
-
- 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
- });
- });
-
- function continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- nextStep();
- }
- }
-
- function chooseCategoryBuilding() {
- if (!_houseID || !context.hasEntity(_houseID)) {
- return addHouse();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
- context.enter(modeSelect(context, [_houseID]));
- } // 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%');
- 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);
- }
-
- var ids = context.selectedIDs();
-
- if (mode.id !== 'select' || !ids.length || ids[0] !== _houseID) {
- return continueTo(chooseCategoryBuilding);
- }
- });
-
- 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 choosePresetHouse() {
- if (!_houseID || !context.hasEntity(_houseID)) {
- return addHouse();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
- context.enter(modeSelect(context, [_houseID]));
- } // 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%');
- 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..
-
- context.on('enter.intro', function (mode) {
- if (!_houseID || !context.hasEntity(_houseID)) {
- return continueTo(addHouse);
- }
-
- var ids = context.selectedIDs();
-
- if (mode.id !== 'select' || !ids.length || ids[0] !== _houseID) {
- return continueTo(chooseCategoryBuilding);
- }
- });
-
- 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();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _houseID) {
- context.enter(modeSelect(context, [_houseID]));
- }
-
- 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 continueTo(nextStep) {
- context.on('exit.intro', null);
- nextStep();
- }
- }
-
- function rightClickHouse() {
- if (!_houseID) return chapter.restart();
- context.enter(modeBrowse(context));
- context.history().reset('hasHouse');
- var zoom = context.map().zoom();
-
- if (zoom < 20) {
- zoom = 20;
- }
-
- 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();
- }
- }
-
- function clickSquare() {
- if (!_houseID) return chapter.restart();
- var entity = context.hasEntity(_houseID);
- if (!entity) return continueTo(rightClickHouse);
- var node = selectMenuItem(context, 'orthogonalize').node();
-
- if (!node) {
- return continueTo(rightClickHouse);
- }
-
- 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();
-
- if (!wasChanged && !node) {
- return continueTo(rightClickHouse);
- }
-
- 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.
-
- timeout(function () {
- if (context.history().undoAnnotation() === _t('operations.orthogonalize.annotation.feature', {
- n: 1
- })) {
- continueTo(doneSquare);
- } else {
- continueTo(retryClickSquare);
- }
- }, 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 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();
- }
- }
-
- 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 addTank() {
- context.enter(modeBrowse(context));
- context.history().reset('doneSquare');
- _tankID = null;
- var msec = transitionTime(tank, context.map().center());
-
- if (msec) {
- reveal(null, null, {
- duration: 0
- });
- }
-
- 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 startTank() {
- if (context.mode().id !== 'add-area') {
- return continueTo(addTank);
- }
-
- _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 continueTo(nextStep) {
- context.map().on('move.intro drawn.intro', null);
- context.on('enter.intro', null);
- nextStep();
- }
- }
-
- function continueTank() {
- if (context.mode().id !== 'draw-area') {
- return continueTo(addTank);
- }
-
- _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);
- }
- });
-
- 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 ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _tankID) {
- context.enter(modeSelect(context, [_tankID]));
- } // 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%');
- 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..
-
- context.on('enter.intro', function (mode) {
- if (!_tankID || !context.hasEntity(_tankID)) {
- return continueTo(addTank);
- }
-
- var ids = context.selectedIDs();
-
- 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..
-
- 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 checkPresetSearch() {
- var first = context.container().select('.preset-list-item:first-child');
-
- 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();
- }
- }
-
- function closeEditorTank() {
- if (!_tankID || !context.hasEntity(_tankID)) {
- return addTank();
- }
-
- var ids = context.selectedIDs();
-
- if (context.mode().id !== 'select' || !ids.length || ids[0] !== _tankID) {
- context.enter(modeSelect(context, [_tankID]));
- }
-
- 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();
- }
- }
-
- 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);
-
- function continueTo(nextStep) {
- context.on('enter.intro', null);
- context.map().on('move.intro drawn.intro', null);
- context.history().on('change.intro', null);
- nextStep();
- }
- }
-
- function clickCircle() {
- if (!_tankID) return chapter.restart();
- var entity = context.hasEntity(_tankID);
- if (!entity) return continueTo(rightClickTank);
- var node = selectMenuItem(context, 'circularize').node();
-
- 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();
-
- if (!wasChanged && !node) {
- return continueTo(rightClickTank);
- }
-
- 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.
-
- 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 retryClickCircle() {
- context.enter(modeBrowse(context));
- revealTank(tank, helpHtml('intro.buildings.retry_circle'), {
- buttonText: _t.html('intro.ok'),
- buttonCallback: function buttonCallback() {
- continueTo(rightClickTank);
- }
- });
-
- function continueTo(nextStep) {
- nextStep();
- }
- }
-
- 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');
- }
- });
- }
-
- chapter.enter = function () {
- addHouse();
- };
-
- 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();
- };
-
- return utilRebind(chapter, dispatch$1, 'on');
- }
-
- function uiIntroStartEditing(context, reveal) {
- var dispatch$1 = dispatch('done', 'startEditing');
- var modalSelection = select(null);
- var chapter = {
- title: 'intro.startediting.title'
- };
-
- function showHelp() {
- reveal('.map-control.help-control', helpHtml('intro.startediting.help'), {
- buttonText: _t.html('intro.ok'),
- buttonCallback: function buttonCallback() {
- shortcuts();
- }
- });
- }
-
- function shortcuts() {
- reveal('.map-control.help-control', helpHtml('intro.startediting.shortcuts'), {
- buttonText: _t.html('intro.ok'),
- buttonCallback: function buttonCallback() {
- showSave();
- }
- });
- }
-
- function showSave() {
- context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
-
- reveal('.top-toolbar button.save', helpHtml('intro.startediting.save'), {
- buttonText: _t.html('intro.ok'),
- buttonCallback: function buttonCallback() {
- showStart();
- }
- });
- }
-
- function showStart() {
- context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
-
- 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');
- }
-
- chapter.enter = function () {
- showHelp();
- };
-
- chapter.exit = function () {
- modalSelection.remove();
- context.container().selectAll('.shaded').remove(); // in case user opened keyboard shortcuts
- };
-
- return utilRebind(chapter, dispatch$1, 'on');
- }
-
- 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 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]));
- }
- }
-
- selection.call(startIntro);
- })["catch"](function () {
- /* ignore */
- });
- }
-
- function startIntro(selection) {
- context.enter(modeBrowse(context)); // Save current map state
-
- 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)`)
-
- context.ui().sidebar.expand();
- context.container().selectAll('button.sidebar-toggle').classed('disabled', true); // Block saving
-
- context.inIntro(true); // Load semi-real data used in intro
-
- if (osm) {
- osm.toggle(false).reset();
- }
-
- context.history().reset();
- context.history().merge(Object.values(coreGraph().load(_introGraph).entities));
- context.history().checkpoint('initial'); // Setup imagery
-
- var imagery = context.background().findSource(INTRO_IMAGERY);
-
- if (imagery) {
- context.background().baseLayerSource(imagery);
- } else {
- context.background().bing();
- }
-
- overlays.forEach(function (d) {
- return context.background().toggleOverlayLayer(d);
- }); // Setup data layers (only OSM)
-
- 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..
-
- corePreferences('walkthrough_started', 'yes'); // Restore previous walkthrough progress..
-
- 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 (i < chapterFlow.length - 1) {
- var next = chapterFlow[i + 1];
- context.container().select("button.chapter-".concat(next)).classed('next', true);
- } // Store walkthrough progress..
-
-
- 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..
-
- var incomplete = utilArrayDifference(chapterFlow, progress);
-
- if (!incomplete.length) {
- corePreferences('walkthrough_completed', 'yes');
- }
-
- curtain.remove();
- navwrap.remove();
- context.container().selectAll('.main-map .layer-background').style('opacity', opacity);
- context.container().selectAll('button.sidebar-toggle').classed('disabled', false);
-
- if (osm) {
- osm.toggle(true).reset().caches(caches);
- }
-
- 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);
- }
-
- 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);
- });
- 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();
- }
-
- context.enter(modeBrowse(context));
- _currChapter = newChapter;
-
- _currChapter.enter();
-
- buttons.classed('next', false).classed('active', function (d) {
- return d.title === _currChapter.title;
- });
- }
- }
-
- return intro;
- }
-
- 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'
- });
-
- if (liveIssues.length) {
- warningsItem.count = liveIssues.length;
- shownItems.push(warningsItem);
- }
-
- if (corePreferences('validate-what') === 'all') {
- var resolvedIssues = context.validator().getResolvedIssues();
-
- if (resolvedIssues.length) {
- resolvedItem.count = resolvedIssues.length;
- shownItems.push(resolvedItem);
- }
- }
-
- var chips = selection.selectAll('.chip').data(shownItems, function (d) {
- return d.id;
- });
- 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
-
- 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 function (selection) {
- update(selection);
- context.validator().on('validated.infobox', function () {
- update(selection);
- });
- };
- }
-
- 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
-
-
- var _cMini; // center pixel of minimap
-
-
- var _tStart; // transform at start of gesture
-
-
- var _tCurr; // transform at most recent event
-
-
- var _timeoutID;
-
- function zoomStarted() {
- if (_skipEvents) return;
- _tStart = _tCurr = projection.transform();
- _gesture = null;
- }
-
- 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;
-
- if (!isZooming && !isPanning) {
- return; // no change
- } // lock in either zooming or panning, don't allow both in minimap.
-
-
- if (!_gesture) {
- _gesture = isZooming ? 'zoom' : 'pan';
- }
-
- 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;
- }
-
- 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();
- }
-
- 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();
-
- if (_isTransformed) {
- utilSetTransform(tiles, 0, 0);
- utilSetTransform(viewport, 0, 0);
- _isTransformed = false;
- }
-
- zoom.scaleExtent([geoZoomToScale(0.5), geoZoomToScale(zMain - 3)]);
- _skipEvents = true;
- wrap.call(zoom.transform, _tCurr);
- _skipEvents = false;
- }
-
- function redraw() {
- clearTimeout(_timeoutID);
- if (_isHidden) return;
- updateProjection();
- var zMini = geoScaleToZoom(projection.scale()); // setup tile container
-
- tiles = wrap.selectAll('.map-in-map-tiles').data([0]);
- tiles = tiles.enter().append('div').attr('class', 'map-in-map-tiles').merge(tiles); // redraw background
-
- 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
-
- 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));
- }
- }
-
- 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
-
- 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;
- });
- }
- }
-
- function queueRedraw() {
- clearTimeout(_timeoutID);
- _timeoutID = setTimeout(function () {
- redraw();
- }, 750);
- }
-
- 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 (_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();
- });
- }
- }
-
- 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..
-
- _dMini = [200, 150]; //utilGetDimensions(wrap);
-
- _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);
- }
-
- 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 disableTooHigh() {
- var canEdit = context.map().zoom() >= context.minEditableZoom();
- div.style('display', canEdit ? 'none' : 'block');
- }
-
- context.map().on('move.notice', debounce(disableTooHigh, 500));
- disableTooHigh();
- };
- }
-
- function uiPhotoviewer(context) {
- var dispatch$1 = dispatch('resize');
-
- var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
-
- function photoviewer(selection) {
- selection.append('button').attr('class', 'thumb-hide').on('click', function () {
- if (services.streetside) {
- services.streetside.hideViewer(context);
- }
-
- if (services.mapillary) {
- services.mapillary.hideViewer(context);
- }
-
- if (services.openstreetcam) {
- services.openstreetcam.hideViewer(context);
- }
- }).append('div').call(svgIcon('#iD-icon-close'));
-
- function preventDefault(d3_event) {
- d3_event.preventDefault();
- }
-
- 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();
-
- if (resizeOnX) {
- var maxWidth = mapSize[0];
- var newWidth = clamp(startWidth + d3_event.clientX - startX, minWidth, maxWidth);
- target.style('width', newWidth + 'px');
- }
-
- if (resizeOnY) {
- var maxHeight = mapSize[1] - 90; // preserve space at top/bottom of map
-
- var newHeight = clamp(startHeight + startY - d3_event.clientY, minHeight, maxHeight);
- target.style('height', newHeight + 'px');
- }
-
- dispatch.call(eventName, target, utilGetDimensions(target, true));
- }
-
- function clamp(num, min, max) {
- return Math.max(min, Math.min(num, max));
- }
-
- function stopResize(d3_event) {
- if (pointerId !== (d3_event.pointerId || 'mouse')) return;
- d3_event.preventDefault();
- d3_event.stopPropagation(); // remove all the listeners we added
-
- select(window).on('.' + eventName, null);
- }
-
- 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);
-
- if (_pointerPrefix === 'pointer') {
- select(window).on('pointercancel.' + eventName, stopResize, false);
- }
- };
- }
- }
-
- 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)
-
- var photoDimensions = utilGetDimensions(photoviewer, true);
-
- 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);
- }
- };
-
- return utilRebind(photoviewer, dispatch$1, 'on');
- }
-
- 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();
- };
- }
-
- function uiScale(context) {
- var projection = context.projection,
- isImperial = !_mainLocalizer.usesMetric(),
- maxLength = 180,
- tickHeight = 8;
-
- 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 (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
-
-
- 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);
- }
- }
-
- 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 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);
- }
-
- return function (selection) {
- function switchUnits() {
- isImperial = !isImperial;
- selection.call(update);
- }
-
- 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 uiShortcuts(context) {
- var detected = utilDetect();
- var _activeTab = 0;
-
- var _modalSelection;
-
- var _selection = select(null);
-
- function shortcutsModal(_modalSelection) {
- _modalSelection.select('.modal').classed('modal-shortcuts', true);
-
- 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) {
- content.call(render, data);
- })["catch"](function () {
- /* ignore */
- });
- }
-
- function render(selection, dataShortcuts) {
- 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) {
- d3_event.preventDefault();
- var i = tabs.nodes().indexOf(this);
- _activeTab = i;
- render(selection, dataShortcuts);
- });
- tabsEnter.append('span').html(function (d) {
- return _t.html(d.text);
- }); // Update
-
- 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 (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
-
-
- 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;
- });
- }
-
- 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';
- });
- }
-
- return function (selection, show) {
- _selection = selection;
-
- if (show) {
- _modalSelection = uiModal(selection);
-
- _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();
-
- _modalSelection = null;
- }
- } else {
- _modalSelection = uiModal(_selection);
-
- _modalSelection.call(shortcutsModal);
- }
- });
- }
- };
- }
-
- var pair_1 = pair;
-
- 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
-
- var matched = m[0]; // extract dimension.. m[1] = leading, m[5] = trailing
-
- var dim;
-
- if (m[1] && m[5]) {
- // if matched both..
- dim = m[1]; // keep leading
-
- matched = matched.slice(0, -1); // remove trailing dimension from match
- } else {
- dim = m[1] || m[5];
- } // if unrecognized dimension
-
-
- if (dim && dims.indexOf(dim) === -1) return null; // extract DMS
-
- 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];
- }
- }
-
- 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;
-
- 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;
- d3_event.preventDefault();
- search.node().focus();
- }
-
- 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');
-
- if (d3_event.keyCode === 13 && // ↩ Return
- q.length && items.size()) {
- click(items.datum());
- }
- }
-
- function inputevent() {
- _geocodeResults = undefined;
- drawList();
- }
-
- 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
-
-
- var idMatch = !locationMatch && q.match(/(?:^|\W)(node|way|relation|[nwr])\W?0*([1-9]\d*)(?:\W|$)/i);
-
- 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
- });
- }
-
- var allEntities = graph.entities;
- var localResults = [];
-
- 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);
-
- (_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
- };
-
- if (d.osm_type === 'way') {
- // for ways, add some fake closed nodes
- attrs.nodes = ['a', 'a']; // so that geometry area is possible
- }
-
- 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])])
- });
- }
- });
-
- 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
- });
- }
-
- return result;
- }
-
- 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'));
-
- 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'));
- }
-
- 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 mouseover(d3_event, d) {
- if (d.id === -1) return;
- utilHighlightEntities([d.id], true, context);
- }
-
- function mouseout(d3_event, d) {
- if (d.id === -1) return;
- utilHighlightEntities([d.id], false, context);
- }
-
- function click(d3_event, d) {
- d3_event.preventDefault();
-
- 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);
- }
- }
-
- function geocoderSearch() {
- services.geocoder.search(search.property('value'), function (err, resp) {
- _geocodeResults = resp || [];
- drawList();
- });
- }
- }
-
- return featureList;
- }
-
- function uiSectionEntityIssues(context) {
- var _entityIDs = [];
- var _issues = [];
-
- var _activeIssueID;
-
- 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 reloadIssues() {
- _issues = context.validator().getSharedEntityIssues(_entityIDs, {
- includeDisabledRules: true
- });
- }
-
- function makeActiveIssue(issueID) {
- _activeIssueID = issueID;
- section.selection().selectAll('.issue-container').classed('active', function (d) {
- return d.id === _activeIssueID;
- });
- }
-
- 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
-
- 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
-
- var extent = d.extent(context.graph());
-
- 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');
-
- 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);
- });
- }
- });
- 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;
- });
- containers.selectAll('.issue-message').html(function (d) {
- return d.message(context);
- }); // fixes
-
- 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)
-
- 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';
-
- if (iconName.startsWith('maki')) {
- iconName += '-15';
- }
-
- 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;
- }
-
- return null;
- });
- }
-
- section.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
-
- if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
- _entityIDs = val;
- _activeIssueID = null;
- reloadIssues();
- }
-
- return section;
- };
-
- return section;
- }
-
- function uiPresetIcon() {
- var _preset;
-
- var _geometry;
-
- var _sizeClass = 'medium';
-
- function isSmall() {
- return _sizeClass === 'small';
- }
-
- function presetIcon(selection) {
- selection.each(render);
- }
-
- 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 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 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 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);
- });
-
- 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);
- });
- }
-
- 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 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 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
-
- 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);
-
- if (drawRoute) {
- var routeType = p.tags.type === 'waterway' ? 'waterway' : p.tags.route;
- var segmentPresetIDs = routeSegments[routeType];
-
- 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.
-
-
- 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']
- };
-
- function render() {
- var p = _preset.apply(this, arguments);
-
- var geom = _geometry ? _geometry.apply(this, arguments) : null;
-
- if (geom === 'relation' && p.tags && (p.tags.type === 'route' && p.tags.route && routeSegments[p.tags.route] || p.tags.type === 'waterway')) {
- geom = 'route';
- }
-
- 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) : {};
-
- for (var k in tags) {
- if (tags[k] === '*') {
- tags[k] = 'yes';
- }
- }
-
- 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);
- }
-
- presetIcon.preset = function (val) {
- if (!arguments.length) return _preset;
- _preset = utilFunctor(val);
- return presetIcon;
- };
-
- presetIcon.geometry = function (val) {
- if (!arguments.length) return _geometry;
- _geometry = utilFunctor(val);
- return presetIcon;
- };
-
- presetIcon.sizeClass = function (val) {
- if (!arguments.length) return _sizeClass;
- _sizeClass = val;
- return presetIcon;
- };
-
- return presetIcon;
- }
-
- function uiSectionFeatureType(context) {
- var dispatch$1 = dispatch('choose');
- var _entityIDs = [];
- var _presets = [];
-
- var _tagReference;
-
- var section = uiSection('feature-type', context).label(_t.html('inspector.feature_type')).disclosureContent(renderDisclosureContent);
-
- 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
-
- 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);
- }
-
- 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;
- });
- }
-
- section.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
- return section;
- };
-
- section.presets = function (val) {
- if (!arguments.length) return _presets; // don't reload the same preset
-
- if (!utilArrayIdentical(val, _presets)) {
- _presets = val;
-
- if (_presets.length === 1) {
- _tagReference = uiTagReference(_presets[0].reference()).showing(false);
- }
- }
-
- return section;
- };
-
- function entityGeometries() {
- var counts = {};
-
- for (var i in _entityIDs) {
- var geometry = context.graph().geometry(_entityIDs[i]);
- if (!counts[geometry]) counts[geometry] = 0;
- counts[geometry] += 1;
- }
-
- return Object.keys(counts).sort(function (geom1, geom2) {
- return counts[geom2] - counts[geom1];
- });
- }
-
- return utilRebind(section, dispatch$1, 'on');
- }
-
- // It borrows some code from uiHelp
-
- function uiFieldHelp(context, fieldName) {
- var fieldHelp = {};
-
- var _inspector = select(null);
-
- var _wrap = select(null);
-
- var _body = select(null);
-
- 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
-
- 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?
-
- var hhh = depth ? Array(depth + 1).join('#') + ' ' : ''; // if so, prepend with some ##'s
-
- return all + hhh + _t.html(subkey, replacements) + '\n\n';
- }, '');
- return {
- key: helpkey,
- title: _t.html(helpkey + '.title'),
- html: marked_1(text.trim())
- };
- });
-
- function show() {
- updatePosition();
-
- _body.classed('hide', false).style('opacity', '0').transition().duration(200).style('opacity', '1');
- }
-
- function hide() {
- _body.classed('hide', true).transition().duration(200).style('opacity', '0').on('end', function () {
- _body.classed('hide', true);
- });
- }
-
- function clickHelp(index) {
- var d = docs[index];
- var tkeys = fieldHelpKeys[fieldName][index][1];
-
- _body.selectAll('.field-help-nav-item').classed('active', function (d, i) {
- return i === index;
- });
-
- var content = _body.selectAll('.field-help-content').html(d.html); // class the paragraphs so we can find and style them
-
-
- 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'));
- }
- }
-
- fieldHelp.button = function (selection) {
- if (_body.empty()) return;
- var button = selection.selectAll('.field-help-button').data([0]); // enter/update
-
- 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 (_body.classed('hide')) {
- show();
- } else {
- hide();
- }
- });
- };
-
- function updatePosition() {
- var wrap = _wrap.node();
-
- var inspector = _inspector.node();
-
- var wRect = wrap.getBoundingClientRect();
- var iRect = inspector.getBoundingClientRect();
-
- _body.style('top', wRect.top + inspector.scrollTop - iRect.top + 'px');
- }
-
- 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 fieldHelp;
- }
-
- function uiFieldCheck(field, context) {
- var dispatch$1 = dispatch('change');
- var options = field.strings && field.strings.options;
- var values = [];
- var texts = [];
-
- var _tags;
-
- var input = select(null);
- var text = select(null);
- var label = select(null);
- var reverser = select(null);
-
- var _impliedYes;
-
- var _entityIDs = [];
-
- var _value;
-
- 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')];
-
- 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 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]);
-
- 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;
- }
- }
- }
- }
-
- 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;
- }
-
- 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');
-
- if (field.type === 'onewayCheck') {
- enter.append('button').attr('class', 'reverser' + (reverserHidden() ? ' hide' : '')).append('span').attr('class', 'reverser-span');
- }
-
- label = label.merge(enter);
- input = label.selectAll('input');
- text = label.selectAll('span.value');
- input.on('click', function (d3_event) {
- d3_event.stopPropagation();
- var t = {};
-
- 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)
-
-
- if (t[field.key] === 'reversible' || t[field.key] === 'alternating') {
- t[field.key] = values[0];
- }
-
- dispatch$1.call('change', this, t);
- });
-
- 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);
- }
-
- 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);
- });
- }
- };
-
- check.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
- return check;
- };
-
- check.tags = function (tags) {
- _tags = tags;
-
- function isChecked(val) {
- return val !== 'no' && val !== '' && val !== undefined && val !== null;
- }
-
- 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();
-
- if (field.type === 'onewayCheck' && (_value === '1' || _value === '-1')) {
- _value = 'yes';
- }
-
- 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 (field.type === 'onewayCheck') {
- reverser.classed('hide', reverserHidden()).call(reverserSetText);
- }
- };
-
- check.focus = function () {
- input.node().focus();
- };
-
- return utilRebind(check, dispatch$1, 'on');
- }
-
- function uiFieldCombo(field, context) {
- var dispatch$1 = dispatch('change');
-
- var _isMulti = field.type === 'multiCombo' || field.type === 'manyCombo';
-
- var _isNetwork = field.type === 'networkCombo';
-
- var _isSemi = field.type === 'semiCombo';
-
- var _optstrings = field.strings && field.strings.options;
-
- var _optarray = field.options;
-
- var _snake_case = field.snake_case || field.snake_case === undefined;
-
- var _combobox = uiCombobox(context, 'combo-' + field.safeid).caseSensitive(field.caseSensitive).minItems(_isMulti || _isSemi ? 1 : 2);
-
- var _container = select(null);
-
- var _inputWrap = select(null);
-
- var _input = select(null);
-
- var _comboData = [];
- var _multiData = [];
- var _entityIDs = [];
-
- var _tags;
-
- var _countryCode;
-
- var _staticPlaceholder; // initialize deprecated tags array
-
-
- 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 snake(s) {
- return s.replace(/\s+/g, '_');
- }
-
- function unsnake(s) {
- return s.replace(/_+/g, ' ');
- }
-
- 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)
-
-
- function tagValue(dval) {
- dval = clean(dval || '');
-
- if (_optstrings) {
- var found = _comboData.find(function (o) {
- return o.key && clean(o.value) === dval;
- });
-
- if (found) {
- return found.key;
- }
- }
-
- 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)
-
-
- function displayValue(tval) {
- tval = tval || '';
-
- if (_optstrings) {
- var found = _comboData.find(function (o) {
- return o.key === tval && o.value;
- });
-
- if (found) {
- return found.value;
- }
- }
-
- if (field.type === 'typeCombo' && tval.toLowerCase() === 'yes') {
- return '';
- }
-
- 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 objectDifference(a, b) {
- return a.filter(function (d1) {
- return !b.some(function (d2) {
- return !d2.isMixed && d1.value === d2.value;
- });
- });
- }
-
- 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);
- }
- }
-
- function setStaticValues(callback) {
- if (!(_optstrings || _optarray)) return;
-
- 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
- };
- });
- }
-
- _combobox.data(objectDifference(_comboData, _multiData));
-
- if (callback) callback(_comboData);
- }
-
- function setTaginfoValues(q, callback) {
- var fn = _isMulti ? 'multikeys' : 'values';
- var query = (_isMulti ? field.key : '') + q;
- var hasCountryPrefix = _isNetwork && _countryCode && _countryCode.indexOf(q.toLowerCase()) === 0;
-
- if (hasCountryPrefix) {
- query = _countryCode + ':';
- }
-
- var params = {
- debounce: q !== '',
- key: field.key,
- query: query
- };
-
- if (_entityIDs.length) {
- params.geometry = context.graph().geometry(_entityIDs[0]);
- }
-
- 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 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;
- });
- _staticPlaceholder = field.placeholder() || placeholders.slice(0, 3).join(', ');
- }
-
- if (!/(…|\.\.\.)$/.test(_staticPlaceholder)) {
- _staticPlaceholder += '…';
- }
-
- var ph;
-
- if (!_isMulti && !_isSemi && _tags && Array.isArray(_tags[field.key])) {
- ph = _t('inspector.multiple_values');
- } else {
- ph = _staticPlaceholder;
- }
-
- _container.selectAll('input').attr('placeholder', ph);
- }
-
- function change() {
- var t = {};
- var val;
-
- if (_isMulti || _isSemi) {
- val = tagValue(utilGetSetValue(_input).replace(/,/g, ';')) || '';
-
- _container.classed('active', false);
-
- utilGetSetValue(_input, '');
- var vals = val.split(';').filter(Boolean);
- if (!vals.length) return;
-
- 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;
- }
-
- key = context.cleanTagKey(key);
- field.keys.push(key);
- t[key] = 'yes';
- });
- } else if (_isSemi) {
- var arr = _multiData.map(function (d) {
- return d.key;
- });
-
- 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$1.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;
- }
-
- 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);
-
- 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
-
- if (field.key === 'destination') {
- listClass += ' full-line-chips';
- }
-
- _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]);
- }
-
- _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();
- }
-
- _input.on('change', change).on('blur', change);
-
- _input.on('keydown.field', function (d3_event) {
- switch (d3_event.keyCode) {
- case 13:
- // ↩ Return
- _input.node().blur(); // blurring also enters the value
-
-
- d3_event.stopPropagation();
- break;
- }
- });
-
- if (_isMulti || _isSemi) {
- _combobox.on('accept', function () {
- _input.node().blur();
-
- _input.node().focus();
- });
-
- _input.on('focus', function () {
- _container.classed('active', true);
- });
- }
- }
-
- combo.tags = function (tags) {
- _tags = tags;
-
- if (_isMulti || _isSemi) {
- _multiData = [];
- var maxLength;
-
- if (_isMulti) {
- // Build _multiData array containing keys already set..
- for (var k in tags) {
- if (field.key && k.indexOf(field.key) !== 0 || field.keys.indexOf(k) === -1) continue;
- var v = tags[k];
- if (!v || typeof v === 'string' && v.toLowerCase() === 'no') continue;
- var suffix = field.key ? k.substring(field.key.length) : k;
-
- _multiData.push({
- key: k,
- value: displayValue(suffix),
- isMixed: Array.isArray(v)
- });
- }
-
- 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
-
- maxLength = context.maxCharsForTagKey() - utilUnicodeCharsCount(field.key);
- } else {
- maxLength = context.maxCharsForTagKey();
- }
- } 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);
-
- 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;
- }
-
- _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
-
- 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
-
-
- 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 available = objectDifference(_comboData, _multiData);
-
- _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
-
-
- var hideAdd = _optstrings && !available.length || maxLength <= 0;
-
- _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('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);
- }
-
- 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);
- }
- };
-
- 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') {
- // meaning tags are full width
- _container.selectAll('.chip').style('transform', function (d2, index2) {
- var node = select(this).node();
-
- 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;
- }
-
- 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;
- }
-
- return 'translateY(100%)';
- }
-
- return null;
- });
- } else {
- _container.selectAll('.chip').each(function (d2, index2) {
- var node = select(this).node(); // check the cursor is in the bounding box
-
- 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 (index === index2) {
- return 'translate(' + x + 'px, ' + y + 'px)';
- } // only translate tags in the same row
-
-
- if (node.offsetTop === targetIndexOffsetTop) {
- if (index2 < index && index2 >= targetIndex) {
- return 'translateX(' + draggedTagWidth + 'px)';
- } else if (index2 > index && index2 <= targetIndex) {
- return 'translateX(-' + draggedTagWidth + 'px)';
- }
- }
-
- return null;
- });
- }
- }).on('end', function () {
- if (!select(this).classed('dragging')) {
- return;
- }
-
- var index = selection.nodes().indexOf(this);
- select(this).classed('dragging', false);
-
- _container.selectAll('.chip').style('transform', null);
-
- if (typeof targetIndex === 'number') {
- var element = _multiData[index];
-
- _multiData.splice(index, 1);
-
- _multiData.splice(targetIndex, 0, element);
-
- var t = {};
-
- if (_multiData.length) {
- t[field.key] = _multiData.map(function (element) {
- return element.key;
- }).join(';');
- } else {
- t[field.key] = undefined;
- }
-
- dispatch$1.call('change', this, t);
- }
-
- dragOrigin = undefined;
- targetIndex = undefined;
- }));
- }
-
- combo.focus = function () {
- _input.node().focus();
- };
-
- combo.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
- return combo;
- };
-
- function combinedEntityExtent() {
- return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
- }
-
- return utilRebind(combo, dispatch$1, 'on');
- }
-
- function uiFieldText(field, context) {
- var dispatch$1 = dispatch('change');
- var input = select(null);
- var outlinkButton = select(null);
- var _entityIDs = [];
-
- var _tags;
-
- var _phoneFormats = {};
-
- if (field.type === 'tel') {
- _mainFileFetcher.get('phone_formats').then(function (d) {
- _phoneFormats = d;
- updatePhonePlaceholder();
- })["catch"](function () {
- /* ignore */
- });
- }
-
- 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());
-
- 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);
- }
- }
-
- function updatePhonePlaceholder() {
- if (input.empty() || !Object.keys(_phoneFormats).length) return;
- var extent = combinedEntityExtent();
- var countryCode = extent && iso1A2Code(extent.center());
-
- var format = countryCode && _phoneFormats[countryCode.toLowerCase()];
-
- if (format) input.attr('placeholder', format);
- }
-
- function validIdentifierValueForLink() {
- if (field.type === 'identifier' && field.pattern) {
- var value = utilGetSetValue(input).trim().split(';')[0];
- return value && value.match(new RegExp(field.pattern));
- }
-
- return null;
- } // clamp number to min/max
-
-
- function clamped(num) {
- if (field.minValue !== undefined) {
- num = Math.max(num, field.minValue);
- }
-
- if (field.maxValue !== undefined) {
- num = Math.min(num, field.maxValue);
- }
-
- return num;
- }
-
- 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 (!val && Array.isArray(_tags[field.key])) return;
-
- 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(';');
- }
-
- utilGetSetValue(input, val);
- }
-
- t[field.key] = val || undefined;
- dispatch$1.call('change', this, t, onInput);
- };
- }
-
- i.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
- return i;
- };
-
- 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);
- }
- };
-
- i.focus = function () {
- var node = input.node();
- if (node) node.focus();
- };
-
- function combinedEntityExtent() {
- return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
- }
-
- return utilRebind(i, dispatch$1, 'on');
- }
-
- function uiFieldAccess(field, context) {
- var dispatch$1 = dispatch('change');
- var items = select(null);
-
- var _tags;
-
- 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 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
-
- items = items.merge(enter);
- wrap.selectAll('.preset-input-access').on('change', change).on('blur', change);
- }
-
- 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);
- }
-
- access.options = function (type) {
- var options = ['no', 'permissive', 'private', 'permit', 'destination'];
-
- if (type !== 'access') {
- options.unshift('yes');
- options.push('designated');
-
- if (type === 'bicycle') {
- options.push('dismount');
- }
- }
-
- 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'
- }
- };
-
- 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 (d === 'access') {
- return 'yes';
- }
-
- if (tags.access && typeof tags.access === 'string') {
- return tags.access;
- }
-
- 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);
-
- 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();
- });
- };
-
- access.focus = function () {
- items.selectAll('.preset-input-access').node().focus();
- };
-
- return utilRebind(access, dispatch$1, 'on');
- }
-
- function uiFieldAddress(field, context) {
- var dispatch$1 = dispatch('change');
-
- var _selection = select(null);
-
- var _wrap = select(null);
-
- var addrField = _mainPresetIndex.field('address'); // needed for placeholder strings
-
- var _entityIDs = [];
-
- var _tags;
-
- var _countryCode;
-
- var _addressFormats = [{
- format: [['housenumber', 'street'], ['city', 'postcode']]
- }];
- _mainFileFetcher.get('address_formats').then(function (d) {
- _addressFormats = d;
-
- 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');
-
- function isAddressable(d) {
- return d.tags.highway && d.tags.name && d.type === 'way';
- }
- }
-
- 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');
-
- 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;
- }
-
- if (d.tags['addr:city']) return true;
- return false;
- }
- }
-
- 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');
- }
-
- function updateForCountryCode() {
- if (!_countryCode) return;
- var addressFormat;
-
- for (var i = 0; i < _addressFormats.length; i++) {
- var format = _addressFormats[i];
-
- 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
-
- break;
- }
- }
-
- 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 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();
- });
-
- 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 + '%';
- });
-
- function addDropdown(d) {
- if (dropdowns.indexOf(d.id) === -1) return; // not a dropdown
-
- 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));
- }));
- }
-
- _wrap.selectAll('input').on('blur', change()).on('change', change());
-
- _wrap.selectAll('input:not(.combobox-input)').on('input', change(true));
-
- if (_tags) updateTags(_tags);
- }
-
- 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 (extent) {
- var countryCode;
-
- if (context.inIntro()) {
- // localize the address format for the walkthrough
- countryCode = _t('intro.graph.countrycode');
- } else {
- var center = extent.center();
- countryCode = iso1A2Code(center);
- }
-
- if (countryCode) {
- _countryCode = countryCode.toLowerCase();
- updateForCountryCode();
- }
- }
- }
-
- function change(onInput) {
- return function () {
- var tags = {};
-
- _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
-
- if (Array.isArray(_tags[key]) && !value) return;
- tags[key] = value || undefined;
- });
-
- 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');
- }
-
- if (_countryCode) {
- var localkey = subfield.id + '!' + _countryCode;
- var tkey = addrField.strings.placeholders[localkey] ? localkey : subfield.id;
- return addrField.t('placeholders.' + tkey);
- }
- });
- }
-
- 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);
- }
-
- function combinedEntityExtent() {
- return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
- }
-
- 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();
-
- if (node) node.focus();
- };
-
- return utilRebind(address, dispatch$1, 'on');
- }
-
- function uiFieldCycleway(field, context) {
- var dispatch$1 = dispatch('change');
- var items = select(null);
- var wrap = select(null);
-
- var _tags;
-
- function cycleway(selection) {
- function stripcolon(s) {
- return s.replace(':', '');
- }
-
- 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);
- }
-
- function change(d3_event, key) {
- var newValue = context.cleanTagValue(utilGetSetValue(select(this))); // don't override multiple values with blank string
-
- if (!newValue && (Array.isArray(_tags.cycleway) || Array.isArray(_tags[key]))) return;
-
- if (newValue === 'none' || newValue === '') {
- newValue = undefined;
- }
-
- 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];
- }
-
- if (otherValue === 'none' || otherValue === '') {
- otherValue = undefined;
- }
-
- 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;
- }
-
- dispatch$1.call('change', this, tag);
- }
-
- cycleway.options = function () {
- return Object.keys(field.strings.options).map(function (option) {
- return {
- title: field.t('options.' + option + '.description'),
- value: option
- };
- });
- };
-
- cycleway.tags = function (tags) {
- _tags = tags; // If cycleway is set, use that instead of individual values
-
- 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 (Array.isArray(tags.cycleway)) {
- vals = vals.concat(tags.cycleway);
- }
-
- if (Array.isArray(tags[d])) {
- vals = vals.concat(tags[d]);
- }
-
- return vals.filter(Boolean).join('\n');
- }
-
- 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]);
- });
- };
-
- cycleway.focus = function () {
- var node = wrap.selectAll('input').node();
- if (node) node.focus();
- };
-
- return utilRebind(cycleway, dispatch$1, 'on');
- }
-
- function uiFieldLanes(field, context) {
- var dispatch$1 = dispatch('change');
- var LANE_WIDTH = 40;
- var LANE_HEIGHT = 200;
- var _entityIDs = [];
-
- function lanes(selection) {
- var lanesData = context.entity(_entityIDs[0]).lanes();
-
- if (!context.container().select('.inspector-wrap.inspector-hidden').empty() || !selection.node().parentNode) {
- selection.call(lanes.off);
- return;
- }
-
- 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';
- });
- }
-
- lanes.entityIDs = function (val) {
- _entityIDs = val;
- };
-
- lanes.tags = function () {};
-
- lanes.focus = function () {};
-
- lanes.off = function () {};
-
- return utilRebind(lanes, dispatch$1, 'on');
- }
- uiFieldLanes.supportsMultiselection = false;
-
- var _languagesArray = [];
- function uiFieldLocalized(field, context) {
- var dispatch$1 = dispatch('change', 'input');
- var wikipedia = services.wikipedia;
- var input = select(null);
- var localizedInputs = select(null);
-
- var _countryCode;
-
- 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.
-
-
- _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
-
- var langCombo = uiCombobox(context, 'localized-lang').fetcher(fetchLanguages).minItems(0);
- var brandCombo = uiCombobox(context, 'localized-brand').canAutocomplete(false).minItems(1);
-
- var _selection = select(null);
-
- var _multilingual = [];
-
- var _buttonTip = uiTooltip().title(_t.html('translate.translate')).placement('left');
-
- var _wikiTitles;
-
- var _entityIDs = [];
-
- 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
-
- };
-
- for (var code in dataLanguages) {
- if (replacements[code] === false) continue;
- var metaCode = code;
- if (replacements[code]) metaCode = replacements[code];
-
- _languagesArray.push({
- localName: _mainLocalizer.languageName(metaCode, {
- localOnly: true
- }),
- nativeName: dataLanguages[metaCode].nativeName,
- code: code,
- label: _mainLocalizer.languageName(metaCode)
- });
- }
- }
-
- 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;
-
- var original = context.graph().base().entities[_entityIDs[0]];
-
- var hasOriginalName = original && entity.tags.name && entity.tags.name === original.tags.name; // if the name was already edited manually then allow further editing
-
- if (!hasOriginalName) return false; // features linked to Wikidata are likely important and should be protected
-
- if (entity.tags.wikidata) return true; // assume the name has already been confirmed if its source has been researched
-
- 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 isSuggestion && !showsBrand;
- });
-
- field.locked(isLocked);
- } // update _multilingual, maintaining the existing order
-
-
- function calcMultilingual(tags) {
- var existingLangsOrdered = _multilingual.map(function (item) {
- return item.lang;
- });
-
- var existingLangs = new Set(existingLangsOrdered.filter(Boolean));
-
- for (var k in tags) {
- var m = k.match(/^(.*):([a-zA-Z_-]+)$/);
-
- if (m && m[1] === field.key && m[2]) {
- var item = {
- lang: m[2],
- value: tags[k]
- };
-
- if (existingLangs.has(item.lang)) {
- // update the value
- _multilingual[existingLangsOrdered.indexOf(item.lang)].value = item.value;
- existingLangs["delete"](item.lang);
- } else {
- _multilingual.push(item);
- }
- }
- }
-
- _multilingual = _multilingual.filter(function (item) {
- return !item.lang || !existingLangs.has(item.lang);
- });
- }
-
- 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);
-
- if (preset && field.id === 'name') {
- var pTag = preset.id.split('/', 2);
- var pKey = pTag[0];
- var pValue = pTag[1];
-
- 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..
-
- if (allSuggestions.length && goodSuggestions.length) {
- input.on('blur.localized', checkBrandOnBlur).call(brandCombo.fetcher(fetchBrandNames(preset, allSuggestions)).on('accept', acceptBrand).on('cancel', cancelBrand));
- }
- }
- }
-
- 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.
-
- function checkBrandOnBlur() {
- var latest = _entityIDs.length === 1 && context.hasEntity(_entityIDs[0]);
- if (!latest) return; // deleting the entity blurred the field?
-
- var preset = _mainPresetIndex.match(latest, context.graph());
- if (preset && preset.suggestion) return; // already accepted
-
- var name = utilGetSetValue(input).trim();
- var matched = allSuggestions.filter(function (s) {
- return name === s.name();
- });
-
- if (matched.length === 1) {
- acceptBrand({
- suggestion: matched[0]
- });
- } else {
- cancelBrand();
- }
- }
-
- function acceptBrand(d) {
- var entity = _entityIDs.length === 1 && context.hasEntity(_entityIDs[0]);
-
- if (!d || !entity) {
- cancelBrand();
- return;
- }
-
- var tags = entity.tags;
- var geometry = entity.geometry(context.graph());
- var removed = preset.unsetTags(tags, geometry);
-
- for (var k in tags) {
- tags[k] = removed[k]; // set removed tags to `undefined`
- }
-
- 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
- });
- }
-
- 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;
- });
- }
-
- 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;
- });
-
- 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;
- }
-
- var val = utilGetSetValue(select(this));
- 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$1.call('change', this, t, onInput);
- };
- }
- }
-
- function key(lang) {
- return field.key + ':' + lang;
- }
-
- 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;
-
- if (d.lang && d.lang !== lang) {
- tags[key(d.lang)] = undefined;
- }
-
- 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];
- }
-
- d.lang = lang;
- dispatch$1.call('change', this, tags);
- }
-
- 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 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 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();
-
- if (!d.lang || !d.value) {
- _multilingual.splice(index, 1);
-
- 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;
- });
- utilGetSetValue(entries.select('.localized-lang'), function (d) {
- return _mainLocalizer.languageName(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;
- });
- }
- }
-
- 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;
- };
-
- 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 setUnitSuggestions() {
- speedCombo.data((_isImperial ? imperialValues : metricValues).map(comboValues));
- utilGetSetValue(unitInput, _isImperial ? 'mph' : 'km/h');
- }
-
- function comboValues(d) {
- return {
- value: d.toString(),
- title: d.toString()
- };
- }
-
- 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;
-
- if (!value) {
- tag[field.key] = undefined;
- } else if (isNaN(value) || !_isImperial) {
- tag[field.key] = context.cleanTagValue(value);
- } else {
- tag[field.key] = context.cleanTagValue(value + ' mph');
- }
-
- dispatch$1.call('change', this, tag);
- }
-
- maxspeed.tags = function (tags) {
- _tags = tags;
- var value = tags[field.key];
- var isMixed = Array.isArray(value);
-
- if (!isMixed) {
- if (value && value.indexOf('mph') >= 0) {
- value = parseInt(value, 10).toString();
- _isImperial = true;
- } else if (value) {
- _isImperial = false;
- }
- }
-
- 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);
- };
-
- maxspeed.focus = function () {
- input.node().focus();
- };
-
- maxspeed.entityIDs = function (val) {
- _entityIDs = val;
- };
-
- function combinedEntityExtent() {
- return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
- }
-
- 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 typeField;
- var layerField;
- var _oldType = {};
- var _entityIDs = [];
-
- function selectedKey() {
- var node = wrap.selectAll('.form-field-input-radio label.active input');
- return !node.empty() && node.datum();
- }
-
- 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);
- }
-
- 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);
- }
-
- typeField.tags(tags);
- } else {
- typeField = null;
- }
-
- var typeItem = list.selectAll('.structure-type-item').data(typeField ? [typeField] : [], function (d) {
- return d.id;
- }); // Exit
-
- typeItem.exit().remove(); // Enter
-
- 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
-
- typeItem = typeItem.merge(typeEnter);
-
- if (typeField) {
- typeItem.selectAll('.structure-input-type-wrap').call(typeField.render);
- } // Layer
-
-
- if (layer && showLayer) {
- if (!layerField) {
- layerField = uiField(context, layer, _entityIDs, {
- wrap: false
- }).on('change', changeLayer);
- }
-
- layerField.tags(tags);
- field.keys = utilArrayUnion(field.keys, ['layer']);
- } else {
- layerField = null;
- field.keys = field.keys.filter(function (k) {
- return k !== 'layer';
- });
- }
-
- var layerItem = list.selectAll('.structure-layer-item').data(layerField ? [layerField] : []); // Exit
-
- layerItem.exit().remove(); // Enter
-
- 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
-
- layerItem = layerItem.merge(layerEnter);
-
- if (layerField) {
- layerItem.selectAll('.structure-input-layer-wrap').call(layerField.render);
- }
- }
-
- function changeType(t, onInput) {
- var key = selectedKey();
- if (!key) return;
- var val = t[key];
-
- if (val !== 'no') {
- _oldType[key] = val;
- }
-
- 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 (t.layer === undefined) {
- if (key === 'bridge' && val !== 'no') {
- t.layer = '1';
- }
-
- if (key === 'tunnel' && val !== 'no' && val !== 'building_passage') {
- t.layer = '-1';
- }
- }
- }
-
- dispatch$1.call('change', this, t, onInput);
- }
-
- function changeLayer(t, onInput) {
- if (t.layer === '0') {
- t.layer = undefined;
- }
-
- dispatch$1.call('change', this, t, onInput);
- }
-
- 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;
- }
- });
-
- 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;
- }
- }
-
- dispatch$1.call('change', this, t);
- }
-
- radio.tags = function (tags) {
- radios.property('checked', function (d) {
- if (field.key) {
- return tags[field.key] === d;
- }
-
- return !!(typeof tags[d] === 'string' && tags[d].toLowerCase() !== 'no');
- });
-
- function isMixed(d) {
- if (field.key) {
- return Array.isArray(tags[field.key]) && tags[field.key].includes(d);
- }
-
- return Array.isArray(tags[d]);
- }
-
- 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;
- });
-
- 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);
- }
- };
-
- radio.focus = function () {
- radios.node().focus();
- };
-
- radio.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
- _oldType = {};
- return radio;
- };
-
- radio.isAllowed = function () {
- return _entityIDs.length === 1;
- };
-
- return utilRebind(radio, dispatch$1, 'on');
- }
-
- function uiFieldRestrictions(field, context) {
- var dispatch$1 = dispatch('change');
- var breathe = behaviorBreathe();
- corePreferences('turn-restriction-via-way', null); // remove old key
-
- var storedViaWay = corePreferences('turn-restriction-via-way0'); // use new key #6922
-
- var storedDistance = corePreferences('turn-restriction-distance');
-
- var _maxViaWay = storedViaWay !== null ? +storedViaWay : 0;
-
- var _maxDistance = storedDistance ? +storedDistance : 30;
-
- var _initialized = false;
-
- var _parent = select(null); // the entire field
-
-
- var _container = select(null); // just the map
-
-
- var _oldTurns;
-
- var _graph;
-
- var _vertexID;
-
- var _intersection;
-
- var _fromWayID;
-
- var _lastXPos;
-
- function restrictions(selection) {
- _parent = selection; // try to reuse the intersection, but always rebuild it if the graph has changed
-
- 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.
-
-
- 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;
- }
-
- 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 containerEnter = container.enter().append('div').attr('class', 'restriction-container');
- containerEnter.append('div').attr('class', 'restriction-help'); // update
-
- _container = containerEnter.merge(container).call(renderViewer);
- var controls = wrap.selectAll('.restriction-controls').data([0]); // enter/update
-
- controls.enter().append('div').attr('class', 'restriction-controls-container').append('div').attr('class', 'restriction-controls').merge(controls).call(renderControls);
- }
-
- 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;
-
- _container.selectAll('.layer-osm .layer-turns *').remove();
-
- corePreferences('turn-restriction-distance', _maxDistance);
-
- _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
-
- selection.selectAll('.restriction-via-way-input').property('value', _maxViaWay).on('input', function () {
- var val = select(this).property('value');
- _maxViaWay = +val;
-
- _container.selectAll('.layer-osm .layer-turns *').remove();
-
- corePreferences('turn-restriction-via-way0', _maxViaWay);
-
- _parent.call(restrictions);
- });
- selection.selectAll('.restriction-via-way-text').html(displayMaxVia(_maxViaWay));
- }
-
- 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);
-
- 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
-
- var extent = geoExtent();
-
- 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
-
-
- 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));
- }
-
- var padTop = 35; // reserve top space for hint text
-
- 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);
-
- 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.
-
-
- if (_fromWayID && !vgraph.hasEntity(_fromWayID)) {
- _fromWayID = null;
- _oldTurns = null;
- }
-
- 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 (_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);
-
- function click(d3_event) {
- surface.call(breathe.off).call(breathe);
- var datum = d3_event.target.__data__;
- var entity = datum && datum.properties && datum.properties.entity;
-
- if (entity) {
- datum = entity;
- }
-
- 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);
-
- 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
-
- datumOnly.only = true; // but change this property
-
- 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.
-
- turns = _intersection.turns(_fromWayID, 2);
- extraActions = [];
- _oldTurns = [];
-
- for (i = 0; i < turns.length; i++) {
- var turn = turns[i];
- if (seen[turn.restrictionID]) continue; // avoid deleting the turn twice (#4968, #4928)
-
- if (turn.direct && turn.path[1] === datum.path[1]) {
- seen[turns[i].restrictionID] = true;
- turn.restrictionType = osmInferRestriction(vgraph, turn, projection);
-
- _oldTurns.push(turn);
-
- extraActions.push(actionUnrestrictTurn(turn));
- }
- }
-
- 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));
- }
- }
-
- _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')]);
- }
-
- 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);
- }
- }
- }
-
- function highlightPathsFrom(wayID) {
- surface.selectAll('.related').classed('related', false).classed('allow', false).classed('restrict', false).classed('only', false);
- surface.selectAll('.' + wayID).classed('related', true);
-
- if (wayID) {
- var turns = _intersection.turns(wayID, _maxViaWay);
-
- 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';
-
- if (turn.only || turns.length === 1) {
- if (turn.via.ways) {
- ids = ids.concat(turn.via.ways);
- }
- } else if (turn.to.way === wayID) {
- continue;
- }
-
- 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('');
-
- 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;
-
- if (entity) {
- datum = entity;
- }
-
- if (_fromWayID) {
- way = vgraph.entity(_fromWayID);
- surface.selectAll('.' + _fromWayID).classed('selected', true).classed('related', true);
- } // Hovering a way
-
-
- 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;
-
- 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: ''
- });
- }
-
- 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)
- }));
-
- if (datum.via.ways && datum.via.ways.length) {
- var names = [];
-
- 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);
- }
-
- 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()
- }));
- }
-
- 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);
-
- 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 displayMaxDistance(maxDist) {
- var isImperial = !_mainLocalizer.usesMetric();
- var opts;
-
- 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
- })
- };
- }
-
- return _t.html('restriction.controls.distance_up_to', opts);
- }
-
- 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 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;
- }
-
- restrictions.entityIDs = function (val) {
- _intersection = null;
- _fromWayID = null;
- _oldTurns = null;
- _vertexID = val[0];
- };
-
- restrictions.tags = function () {};
-
- restrictions.focus = function () {};
-
- 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);
- };
-
- return utilRebind(restrictions, dispatch$1, 'on');
- }
- uiFieldRestrictions.supportsMultiselection = false;
-
- function uiFieldTextarea(field, context) {
- var dispatch$1 = dispatch('change');
- var input = select(null);
-
- 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);
- }
-
- 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$1.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);
- };
-
- textarea.focus = function () {
- input.node().focus();
- };
-
- return utilRebind(textarea, dispatch$1, 'on');
- }
-
- function uiFieldWikidata(field, context) {
- var wikidata = services.wikidata;
- var dispatch$1 = dispatch('change');
-
- var _selection = select(null);
-
- var _searchInput = select(null);
-
- var _qid = null;
- var _wikidataEntity = null;
- var _wikiURL = '';
- var _entityIDs = [];
-
- var _wikipediaKey = field.keys && field.keys.find(function (key) {
- return key.includes('wikipedia');
- }),
- _hintKey = field.key === 'wikidata' ? 'name' : field.key.split(':')[0];
-
- 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();
- });
- 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 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');
- });
- }
-
- 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]);
-
- if (entity.tags[_hintKey]) {
- q = entity.tags[_hintKey];
- break;
- }
- }
- }
-
- wikidata.itemsForSearchQuery(q, function (err, data) {
- if (err) return;
-
- for (var i in data) {
- data[i].value = data[i].label + ' (' + data[i].id + ')';
- data[i].title = data[i].description;
- }
-
- if (callback) callback(data);
- });
- }
-
- function change() {
- var syncTags = {};
- syncTags[field.key] = _qid;
- dispatch$1.call('change', this, syncTags); // attempt asynchronous update of wikidata tag..
-
- 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.
-
- if (context.graph() !== initGraph) return;
- if (!entity.sitelinks) return;
- var langs = wikidata.languagesToQuery(); // use the label and description languages as fallbacks
-
- ['labels', 'descriptions'].forEach(function (key) {
- if (!entity[key]) return;
- var valueLangs = Object.keys(entity[key]);
- if (valueLangs.length === 0) return;
- var valueLang = valueLangs[0];
-
- if (langs.indexOf(valueLang) === -1) {
- langs.push(valueLang);
- }
- });
- var newWikipediaValue;
-
- if (_wikipediaKey) {
- var foundPreferred;
-
- for (var i in langs) {
- var lang = langs[i];
- var siteID = lang.replace('-', '_') + 'wiki';
-
- if (entity.sitelinks[siteID]) {
- foundPreferred = true;
- newWikipediaValue = lang + ':' + entity.sitelinks[siteID].title; // use the first match
-
- break;
- }
- }
-
- 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;
- }
- }
- }
-
- 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
-
- if (newWikipediaValue === null) {
- if (!currTags[_wikipediaKey]) return null;
- delete currTags[_wikipediaKey];
- } else {
- currTags[_wikipediaKey] = newWikipediaValue;
- }
-
- return actionChangeTags(entityID, currTags);
- }).filter(Boolean);
- if (!actions.length) return; // Coalesce the update of wikidata tag into the previous tag change
-
- 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
- });
- }
-
- function setLabelForEntity() {
- var label = '';
-
- if (_wikidataEntity) {
- label = entityPropertyForDisplay(_wikidataEntity, 'labels');
-
- if (label.length === 0) {
- label = _wikidataEntity.id.toString();
- }
- }
-
- utilGetSetValue(_searchInput, label);
- }
-
- wiki.tags = function (tags) {
- var isMixed = Array.isArray(tags[field.key]);
-
- _searchInput.attr('title', isMixed ? tags[field.key].filter(Boolean).join('\n') : null).attr('placeholder', isMixed ? _t('inspector.multiple_values') : '').classed('mixed', isMixed);
-
- _qid = typeof tags[field.key] === 'string' && tags[field.key] || '';
-
- if (!/^Q[0-9]*$/.test(_qid)) {
- // not a proper QID
- unrecognized();
- return;
- } // QID value in correct format
-
-
- _wikiURL = 'https://wikidata.org/wiki/' + _qid;
- wikidata.entityByQID(_qid, function (err, entity) {
- if (err) {
- unrecognized();
- return;
- }
-
- _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);
-
- _selection.select('.preset-wikidata-identifier').style('display', function () {
- return entity.id ? 'flex' : 'none';
- }).select('input').attr('value', entity.id);
- }); // not a proper QID
-
- function unrecognized() {
- _wikidataEntity = null;
- setLabelForEntity();
-
- _selection.select('.preset-wikidata-description').style('display', 'none');
-
- _selection.select('.preset-wikidata-identifier').style('display', 'none');
-
- _selection.select('button.wiki-link').classed('disabled', true);
-
- if (_qid && _qid !== '') {
- _wikiURL = 'https://wikidata.org/wiki/Special:Search?search=' + _qid;
- } else {
- _wikiURL = '';
- }
- }
- };
-
- 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 langs = wikidata.languagesToQuery();
-
- 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;
- }
-
- 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);
-
- var _wikiURL = '';
-
- var _entityIDs;
-
- var _tags;
-
- 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 = '';
-
- for (var i in _entityIDs) {
- var entity = context.hasEntity(_entityIDs[i]);
-
- if (entity.tags.name) {
- value = entity.tags.name;
- break;
- }
- }
- }
-
- 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);
- });
-
- 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();
- if (_wikiURL) window.open(_wikiURL, '_blank');
- });
- }
-
- function defaultLanguageInfo(skipEnglishFallback) {
- var langCode = _mainLocalizer.languageCode().toLowerCase();
-
- for (var i in _dataWikipedia) {
- var d = _dataWikipedia[i]; // default to the language of iD's current locale
-
- if (d[2] === langCode) return d;
- } // fallback to English
-
-
- return skipEnglishFallback ? ['', '', ''] : ['English', 'English', 'en'];
- }
-
- function language(skipEnglishFallback) {
- var value = utilGetSetValue(_langInput).toLowerCase();
-
- for (var i in _dataWikipedia) {
- var d = _dataWikipedia[i]; // return the language already set in the UI, if supported
-
- if (d[0].toLowerCase() === value || d[1].toLowerCase() === value || d[2] === value) return d;
- } // fallback to English
-
-
- return defaultLanguageInfo(skipEnglishFallback);
- }
-
- function changeLang() {
- utilGetSetValue(_langInput, language()[1]);
- change(true);
- }
-
- 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 syncTags = {};
-
- if (langInfo) {
- var nativeLangName = langInfo[1]; // Normalize title http://www.mediawiki.org/wiki/API:Query#Title_normalization
-
- value = decodeURIComponent(m[2]).replace(/_/g, ' ');
-
- if (m[3]) {
- 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]); // }
-
- value += '#' + anchor.replace(/_/g, ' ');
- }
-
- value = value.slice(0, 1).toUpperCase() + value.slice(1);
- utilGetSetValue(_langInput, nativeLangName);
- utilGetSetValue(_titleInput, value);
- }
-
- if (value) {
- syncTags.wikipedia = context.cleanTagValue(language()[2] + ':' + value);
- } else {
- syncTags.wikipedia = undefined;
- }
-
- dispatch$1.call('change', this, syncTags);
- if (skipWikidata || !value || !language()[2]) return; // attempt asynchronous update of wikidata tag..
-
- 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 (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
- });
- }
-
- wiki.tags = function (tags) {
- _tags = tags;
- updateForTags(tags);
- };
-
- function updateForTags(tags) {
- var value = typeof tags[field.key] === 'string' ? tags[field.key] : ''; // Expect tag format of `tagLang:tagArticleTitle`, e.g. `fr:Paris`, with
- // optional suffix of `#anchor`
-
- var m = value.match(/([^:]+):([^#]+)(?:#(.+))?/);
- var tagLang = m && m[1];
- var tagArticleTitle = m && m[2];
- var anchor = m && m[3];
-
- var tagLangInfo = tagLang && _dataWikipedia.find(function (d) {
- return tagLang === d[2];
- }); // value in correct format
-
-
- if (tagLangInfo) {
- var nativeLangName = tagLangInfo[1];
- utilGetSetValue(_langInput, nativeLangName);
- utilGetSetValue(_titleInput, tagArticleTitle + (anchor ? '#' + anchor : ''));
-
- if (anchor) {
- try {
- // Best-effort `anchorencode:` implementation
- anchor = encodeURIComponent(anchor.replace(/ /g, '_')).replace(/%/g, '.');
- } catch (e) {
- anchor = anchor.replace(/ /g, '_');
- }
- }
-
- _wikiURL = 'https://' + tagLang + '.wikipedia.org/wiki/' + tagArticleTitle.replace(/ /g, '_') + (anchor ? '#' + anchor : ''); // unrecognized value format
- } else {
- utilGetSetValue(_titleInput, value);
-
- 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 = '';
- }
- }
- }
-
- wiki.entityIDs = function (val) {
- if (!_arguments.length) return _entityIDs;
- _entityIDs = val;
- return wiki;
- };
-
- wiki.focus = function () {
- _titleInput.node().focus();
- };
-
- return utilRebind(wiki, dispatch$1, 'on');
- }
- uiFieldWikipedia.supportsMultiselection = false;
-
- 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
- };
-
- 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
-
- field.domId = utilUniqueDomId('form-field-' + field.safeid);
- var _show = options.show;
- var _state = '';
- var _tags = {};
- var _locked = false;
-
- 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
-
- if (_show && !field.impl) {
- createField();
- } // Creates the field.. This is done lazily,
- // once we know that the field will be shown.
-
-
- function createField() {
- field.impl = uiFields[field.type](field, context).on('change', function (t, onInput) {
- dispatch$1.call('change', field, t, onInput);
- });
-
- if (entityIDs) {
- field.entityIDs = entityIDs; // if this field cares about the entities, pass them along
-
- if (field.impl.entityIDs) {
- field.impl.entityIDs(entityIDs);
- }
- }
- }
-
- 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 tagsContainFieldKey() {
- return field.keys.some(function (key) {
- if (field.type === 'multiCombo') {
- for (var tagKey in _tags) {
- if (tagKey.indexOf(key) === 0) {
- return true;
- }
- }
-
- return false;
- }
-
- return _tags[key] !== undefined;
- });
- }
-
- function revert(d3_event, d) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- if (!entityIDs || _locked) return;
- dispatch$1.call('revert', d, d.keys);
- }
-
- 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$1.call('change', d, t);
- }
-
- 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 (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 (options.remove) {
- labelEnter.append('button').attr('class', 'remove-icon').attr('title', _t('icons.remove')).call(svgIcon('#iD-operation-delete'));
- }
-
- 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
-
-
- 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 (!d.impl) {
- createField();
- }
-
- var reference, help; // instantiate field help
-
- if (options.wrap && field.type === 'restrictions') {
- help = uiFieldHelp(context, 'restrictions');
- } // instantiate tag reference
-
-
- if (options.wrap && options.info) {
- var referenceKey = d.key || '';
-
- if (d.type === 'multiCombo') {
- // lookup key without the trailing ':'
- referenceKey = referenceKey.replace(/:$/, '');
- }
-
- reference = uiTagReference(d.reference || {
- key: referenceKey
- });
-
- if (_state === 'hover') {
- reference.showing(false);
- }
- }
-
- selection.call(d.impl); // add field help components
-
- if (help) {
- selection.call(help.body).select('.field-label').call(help.button);
- } // add tag reference components
-
-
- if (reference) {
- selection.call(reference.body).select('.field-label').call(reference.button);
- }
-
- 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
-
- 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);
- };
-
- field.state = function (val) {
- if (!arguments.length) return _state;
- _state = val;
- return field;
- };
-
- 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;
- };
-
- field.locked = function (val) {
- if (!arguments.length) return _locked;
- _locked = val;
- return field;
- };
-
- field.show = function () {
- _show = true;
-
- if (!field.impl) {
- createField();
- }
-
- if (field["default"] && field.key && _tags[field.key] !== field["default"]) {
- var t = {};
- t[field.key] = field["default"];
- dispatch$1.call('change', this, t);
- }
- }; // A shown field has a visible UI, a non-shown field is in the 'Add field' dropdown
-
-
- 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
-
-
- 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 (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;
- }
-
- if (field.notCountryCodes && field.notCountryCodes.indexOf(countryCode) !== -1) {
- return false;
- }
- }
-
- var prerequisiteTag = field.prerequisiteTag;
-
- 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;
- }
-
- return true;
- })) return false;
- }
-
- return true;
- };
-
- field.focus = function () {
- if (field.impl) {
- field.impl.focus();
- }
- };
-
- 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());
- }
-
- return utilRebind(field, dispatch$1, 'on');
- }
-
- 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();
- });
-
- 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
-
- var enter = fields.enter().append('div').attr('class', function (d) {
- return 'wrap-form-field wrap-form-field-' + d.safeid;
- }); // Update
-
- 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
-
- 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;
- }
- }
-
- 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;
- };
-
- formFields.klass = function (val) {
- if (!arguments.length) return _klass;
- _klass = val;
- return formFields;
- };
-
- return formFields;
- }
-
- 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);
-
- var _state;
-
- var _fieldsArr;
-
- 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));
- }
-
- 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$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);
- });
-
- 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));
- }
- });
- }
-
- section.presets = function (val) {
- if (!arguments.length) return _presets;
-
- if (!_presets || !val || !utilArrayIdentical(_presets, val)) {
- _presets = val;
- _fieldsArr = null;
- }
-
- return section;
- };
-
- 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.
-
- return section;
- };
-
- section.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
-
- if (!val || !_entityIDs || !utilArrayIdentical(_entityIDs, val)) {
- _entityIDs = val;
- _fieldsArr = null;
- }
-
- return section;
- };
-
- return utilRebind(section, dispatch$1, 'on');
- }
-
- 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 _entityIDs;
-
- var _maxMembers = 1000;
-
- 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 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 selectMember(d3_event, d) {
- d3_event.preventDefault(); // remove the hover-highlight styling
-
- 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.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();
- }
- }
-
- 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();
- }
- }
-
- 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);
-
- 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
-
-
- 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 (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 'translateY(-100%)';
- } else if (index2 < index && d3_event.y < node.offsetTop + node.offsetHeight) {
- if (targetIndex === null || index2 < targetIndex) {
- targetIndex = index2;
- }
-
- return 'translateY(100%)';
- }
-
- 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);
-
- 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();
- }
- }));
-
- 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 = [];
-
- 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);
- }
-
- 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;
-
- 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);
- }));
- }
-
- function unbind() {
- var row = select(this);
- row.selectAll('input.member-role').call(uiCombobox.off, context);
- }
- }
-
- section.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
- return section;
- };
-
- return section;
- }
-
- 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;
- });
-
- for (var i in memberIndexes) {
- graph = actionDeleteMember(relationId, memberIndexes[i])(graph);
- }
-
- return graph;
- };
- }
-
- 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 = [];
-
- var _showBlank;
-
- var _maxMemberships = 1000;
-
- function getSharedParentRelations() {
- var parents = [];
-
- for (var i = 0; i < _entityIDs.length; i++) {
- var entity = context.graph().hasEntity(_entityIDs[i]);
- if (!entity) continue;
-
- if (i === 0) {
- parents = context.graph().parentRelations(entity);
- } else {
- parents = utilArrayIntersection(parents, context.graph().parentRelations(entity));
- }
-
- if (!parents.length) break;
- }
-
- return parents;
- }
-
- 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)
- };
-
- for (index = 0; index < relation.members.length; index++) {
- member = relation.members[index];
-
- if (_entityIDs.indexOf(member.id) !== -1) {
- indexedMember = Object.assign({}, member, {
- index: index
- });
- membership.members.push(indexedMember);
- membership.hash += ',' + index.toString();
-
- 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)
- };
- }
- }
- }
-
- 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;
- }
-
- function selectRelation(d3_event, d) {
- d3_event.preventDefault(); // remove the hover-highlight styling
-
- utilHighlightEntities([d.relation.id], false, context);
- context.enter(modeSelect(context, [d.relation.id]));
- }
-
- 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);
- }
-
- 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;
- });
-
- 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 addMembership(d, role) {
- this.blur(); // avoid keeping focus on the button
-
- _showBlank = false;
-
- 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);
- }
-
- return graph;
- };
- }
-
- 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`
-
- context.enter(modeSelect(context, [relation.id]).newFeature(true));
- }
- }
-
- 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
-
- 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();
- }
-
- 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;
- }
-
- 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
-
- 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;
- });
- });
- }
-
- 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 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
-
- 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);
-
- if (taginfo) {
- wrapEnter.each(bindTypeahead);
- }
-
- 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();
- });
- 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
-
- 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
-
- 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();
- });
-
- function acceptEntity(d) {
- if (!d) {
- cancelEntity();
- return;
- } // remove hover-higlighting
-
-
- 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.surface().selectAll('.highlighted').classed('highlighted', false);
- }
-
- 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 = [];
-
- 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);
- }
-
- 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);
- }
- }
-
- section.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
- _showBlank = false;
- return section;
- };
-
- 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();
- }
- });
-
- section.entityIDs = function (val) {
- if (!arguments.length) return _selectedIDs;
- _selectedIDs = val;
- return section;
- };
-
- function selectEntity(d3_event, entity) {
- context.enter(modeSelect(context, [entity.id]));
- }
-
- function deselectEntity(d3_event, entity) {
- d3_event.stopPropagation();
-
- var selectedIDs = _selectedIDs.slice();
-
- var index = selectedIDs.indexOf(entity.id);
-
- if (index > -1) {
- selectedIDs.splice(index, 1);
- context.enter(modeSelect(context, selectedIDs));
- }
- }
-
- function renderDisclosureContent(selection) {
- var list = selection.selectAll('.feature-list').data([0]);
- list = list.enter().append('div').attr('class', 'feature-list').merge(list);
-
- 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 enter = items.enter().append('button').attr('class', 'feature-list-item').on('click', selectEntity);
- enter.each(function (d) {
- select(this).on('mouseover', function () {
- utilHighlightEntities([d.id], true, context);
- });
- select(this).on('mouseout', function () {
- utilHighlightEntities([d.id], false, context);
- });
- });
- var label = enter.append('div').attr('class', 'label');
- enter.append('button').attr('class', 'close').attr('title', _t('icons.deselect')).on('click', deselectEntity).call(svgIcon('#iD-icon-close'));
- 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'); // Update
-
- 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;
- }
-
- function uiEntityEditor(context) {
- var dispatch$1 = dispatch('choose');
- var _state = 'select';
- var _coalesceChanges = false;
- var _modified = false;
-
- var _base;
-
- var _entityIDs;
-
- var _activePresets = [];
-
- var _newFeature;
-
- var _sections;
-
- function entityEditor(selection) {
- var combinedTags = utilCombinedTags(_entityIDs, context.graph()); // Header
-
- var header = selection.selectAll('.header').data([0]); // Enter
-
- 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
-
- var body = selection.selectAll('.inspector-body').data([0]); // Enter
-
- var bodyEnter = body.enter().append('div').attr('class', 'entity-editor inspector-body sep-top'); // Update
-
- 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)];
- }
-
- _sections.forEach(function (section) {
- if (section.entityIDs) {
- section.entityIDs(_entityIDs);
- }
-
- if (section.presets) {
- section.presets(_activePresets);
- }
-
- if (section.tags) {
- section.tags(combinedTags);
- }
-
- if (section.state) {
- section.state(_state);
- }
-
- body.call(section.render);
- });
-
- context.history().on('change.entity-editor', historyChanged);
-
- 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.
-
-
- function changeTags(entityIDs, changed, onInput) {
- var actions = [];
-
- for (var i in entityIDs) {
- var entityID = entityIDs[i];
- 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 (v !== undefined || tags.hasOwnProperty(k)) {
- tags[k] = v;
- }
- }
-
- if (!onInput) {
- tags = utilCleanTags(tags);
- }
-
- 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;
- };
-
- var annotation = _t('operations.change_tags.annotation');
-
- 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 revertTags(keys) {
- var actions = [];
-
- 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;
- }
-
- 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 (v !== undefined || tags.hasOwnProperty(k)) {
- tags[k] = v;
- }
- }
-
- tags = utilCleanTags(tags);
-
- 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;
- };
-
- var annotation = _t('operations.change_tags.annotation');
-
- if (_coalesceChanges) {
- context.overwrite(combinedAction, annotation);
- } else {
- context.perform(combinedAction, annotation);
- _coalesceChanges = false;
- }
- }
-
- context.validator().validate();
- }
-
- entityEditor.modified = function (val) {
- if (!arguments.length) return _modified;
- _modified = val;
- return entityEditor;
- };
-
- entityEditor.state = function (val) {
- if (!arguments.length) return _state;
- _state = val;
- return entityEditor;
- };
-
- entityEditor.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- if (val && _entityIDs && utilArrayIdentical(_entityIDs, val)) return entityEditor; // exit early if no change
-
- _entityIDs = val;
- _base = context.graph();
- _coalesceChanges = false;
- loadActivePresets(true);
- return entityEditor.modified(false);
- };
-
- entityEditor.newFeature = function (val) {
- if (!arguments.length) return _newFeature;
- _newFeature = val;
- return entityEditor;
- };
-
- 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;
- }
-
- var matches = Object.keys(counts).sort(function (p1, p2) {
- return counts[p2] - counts[p1];
- }).map(function (pID) {
- return _mainPresetIndex.item(pID);
- });
-
- 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;
- }
-
- entityEditor.presets(matches);
- }
-
- entityEditor.presets = function (val) {
- if (!arguments.length) return _activePresets; // don't reload the same preset
-
- if (!utilArrayIdentical(val, _activePresets)) {
- _activePresets = val;
- }
-
- return entityEditor;
- };
-
- return utilRebind(entityEditor, dispatch$1, 'on');
- }
-
- function uiPresetList(context) {
- var dispatch$1 = dispatch('cancel', 'choose');
-
- var _entityIDs;
-
- var _currentPresets;
-
- var _autofocus = false;
-
- 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'));
-
- 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 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 buttons = list.selectAll('.preset-list-button');
- if (!buttons.empty()) buttons.nodes()[0].focus();
- }
- }
-
- function keypress(d3_event) {
- // enter
- var value = search.property('value');
-
- if (d3_event.keyCode === 13 && // ↩ Return
- value.length) {
- list.selectAll('.preset-list-item:first-child').each(function (d) {
- d.choose.call(this);
- });
- }
- }
-
- function inputevent() {
- var value = search.property('value');
- list.classed('filtered', value.length);
- var extent = combinedEntityExtent();
- var results, messageText;
-
- 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');
- }
-
- 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()));
- context.features().on('change.preset-list', updateForFeatureHiddenState);
- }
-
- function drawList(list, presets) {
- presets = presets.matchAllGeometry(entityGeometries());
- var collection = presets.collection.reduce(function (collection, preset) {
- if (!preset) return collection;
-
- 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 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
-
- if (d3_event.keyCode === utilKeybinding.keyCodes['↓']) {
- d3_event.preventDefault();
- d3_event.stopPropagation(); // the next item in the list at the same level
-
- var nextItem = select(item.node().nextElementSibling); // if there is no next item in this list
-
- 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
-
- } else if (select(this).classed('expanded')) {
- // select the first subitem instead
- nextItem = item.select('.subgrid .preset-list-item:first-child');
- }
-
- 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
-
- var previousItem = select(item.node().previousElementSibling); // if there is no previous item in this list
-
- 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
-
- } 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
-
- } 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 (!parentItem.empty()) {
- parentItem.select('.preset-list-button').node().focus();
- } // arrow right, choose this item
-
- } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === 'rtl' ? '←' : '→']) {
- d3_event.preventDefault();
- d3_event.stopPropagation();
- item.datum().choose.call(select(this).node());
- }
- }
-
- function CategoryItem(preset) {
- var box,
- sublist,
- shown = false;
-
- function item(selection) {
- var wrap = selection.append('div').attr('class', 'preset-list-button-wrap category');
-
- 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();
- }
-
- 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 (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() + '…';
- });
- 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');
- }
-
- 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');
- }
- };
-
- item.preset = preset;
- return item;
- }
-
- 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;
-
- 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);
- }
-
- return graph;
- }, _t('operations.change_tags.annotation'));
- context.validator().validate(); // rerun validation
-
- dispatch$1.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 updateForFeatureHiddenState() {
- if (!_entityIDs.every(context.hasEntity)) return;
- var geometries = entityGeometries();
- var button = context.container().selectAll('.preset-list .preset-list-button'); // remove existing tooltips
-
- 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 isHiddenPreset = !context.inIntro() && !!hiddenPresetFeaturesId && (_currentPresets.length !== 1 || item.preset !== _currentPresets[0]);
- select(this).classed('disabled', isHiddenPreset);
-
- 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'));
- }
- });
- }
-
- presetList.autofocus = function (val) {
- if (!arguments.length) return _autofocus;
- _autofocus = val;
- return presetList;
- };
-
- presetList.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
-
- if (_entityIDs && _entityIDs.length) {
- var presets = _entityIDs.map(function (entityID) {
- return _mainPresetIndex.match(context.entity(entityID), context.graph());
- });
-
- presetList.presets(presets);
- }
-
- return presetList;
- };
-
- presetList.presets = function (val) {
- if (!arguments.length) return _currentPresets;
- _currentPresets = val;
- return presetList;
- };
-
- 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';
- }
-
- if (!counts[geometry]) counts[geometry] = 0;
- counts[geometry] += 1;
- }
-
- return Object.keys(counts).sort(function (geom1, geom2) {
- return counts[geom2] - counts[geom1];
- });
- }
-
- function combinedEntityExtent() {
- return _entityIDs.reduce(function (extent, entityID) {
- var entity = context.graph().entity(entityID);
- return extent.extend(entity.extent(context.graph()));
- }, geoExtent());
- }
-
- return utilRebind(presetList, dispatch$1, 'on');
- }
-
- 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 _entityIDs;
-
- 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');
-
- function shouldDefaultToPresetList() {
- // always show the inspector on hover
- if (_state !== 'select') return false; // can only change preset on single selection
-
- 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
-
- if (entity.hasNonGeometryTags()) return false; // prompt to select preset if feature is new and untagged
-
- if (_newFeature) return true; // all existing features except vertices should default to inspector
-
- 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 (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
-
- return true;
- }
-
- 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 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])));
- }
-
- inspector.showList = function (presets) {
- presetPane.classed('hide', false);
- wrap.transition().styleTween('right', function () {
- return interpolate('0%', '-100%');
- }).on('end', function () {
- editorPane.classed('hide', true);
- });
-
- if (presets) {
- presetList.presets(presets);
- }
-
- presetPane.call(presetList.autofocus(true));
- };
-
- 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 (preset) {
- entityEditor.presets([preset]);
- }
-
- editorPane.call(entityEditor);
- }
- };
-
- 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;
- };
-
- inspector.entityIDs = function (val) {
- if (!arguments.length) return _entityIDs;
- _entityIDs = val;
- return inspector;
- };
-
- inspector.newFeature = function (val) {
- if (!arguments.length) return _newFeature;
- _newFeature = val;
- return inspector;
- };
-
- return inspector;
- }
-
- 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;
-
- 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';
-
- function sidebar(selection) {
- var container = context.container();
- var minWidth = 240;
- var sidebarWidth;
- var containerWidth;
- var dragOffset; // Set the initial width constraints
-
- 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;
-
- 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
-
- 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);
- }
-
- 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);
-
- 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]);
- }
- }
- }
-
- 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 featureListWrap = selection.append('div').attr('class', 'feature-list-pane').call(uiFeatureList(context));
- var inspectorWrap = selection.append('div').attr('class', 'inspector-hidden inspector-wrap');
-
- var hoverModeSelect = function hoverModeSelect(targets) {
- context.container().selectAll('.feature-list-item').classed('hover', false);
-
- if (context.selectedIDs().length > 1 && targets && targets.length) {
- var elements = context.container().selectAll('.feature-list-item').filter(function (node) {
- return targets.indexOf(node) !== -1;
- });
-
- if (!elements.empty()) {
- elements.classed('hover', true);
- }
- }
- };
-
- sidebar.hoverModeSelect = throttle(hoverModeSelect, 200);
-
- function hover(targets) {
- var datum = targets && targets.length && targets[0];
-
- 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;
-
- if (osm) {
- datum = osm.getNote(datum.id); // marker may contain stale data - get latest
- }
-
- 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
-
-
- 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 (!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();
- }
- }
-
- sidebar.hover = throttle(hover, 200);
-
- 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();
-
- if (ids && ids.length) {
- var entity = ids.length === 1 && context.entity(ids[0]);
-
- 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
-
- inspector.state('select').entityIDs(ids).newFeature(newFeature);
- inspectorWrap.call(inspector);
- } else {
- inspector.state('hide');
- }
- };
-
- sidebar.showPresetList = function () {
- inspector.showList();
- };
-
- 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;
- };
-
- sidebar.expand = function (moveMap) {
- if (selection.classed('collapsed')) {
- sidebar.toggle(moveMap);
- }
- };
-
- 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
-
- selection.style('width', sidebarWidth + 'px');
- var startMargin, endMargin, lastMargin;
-
- if (isCollapsing) {
- startMargin = lastMargin = 0;
- endMargin = -sidebarWidth;
- } else {
- startMargin = lastMargin = -sidebarWidth;
- endMargin = 0;
- }
-
- if (!isCollapsing) {
- // unhide the sidebar's content before it transitions onscreen
- selection.classed('collapsed', isCollapsing);
- }
-
- 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 %
-
-
- 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
-
-
- resizer.on('dblclick', function (d3_event) {
- d3_event.preventDefault();
-
- if (d3_event.sourceEvent) {
- d3_event.sourceEvent.preventDefault();
- }
-
- sidebar.toggle();
- }); // ensure hover sidebar is closed when zooming out beyond editable zoom
-
- context.map().on('crossEditableZoom.sidebar', function (within) {
- if (!within && !selection.select('.inspector-hover').empty()) {
- hover([]);
- }
- });
- }
-
- sidebar.showPresetList = function () {};
-
- sidebar.hover = function () {};
-
- sidebar.hover.cancel = function () {};
-
- sidebar.intersects = function () {};
-
- sidebar.select = function () {};
-
- sidebar.show = function () {};
-
- sidebar.hide = function () {};
-
- sidebar.expand = function () {};
-
- sidebar.collapse = function () {};
-
- sidebar.toggle = function () {};
-
- return sidebar;
- }
-
- function uiSourceSwitch(context) {
- var keys;
-
- 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
-
- context.flush(); // remove stored data
-
- 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)
- }
-
- var sourceSwitch = function sourceSwitch(selection) {
- selection.append('a').attr('href', '#').html(_t.html('source_switch.live')).attr('class', 'live chip').on('click', click);
- };
-
- sourceSwitch.keys = function (_) {
- if (!arguments.length) return keys;
- keys = _;
- return sourceSwitch;
- };
-
- return sourceSwitch;
- }
-
- 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 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.
-
- var updateMessage = '';
- var sawPrivacyVersion = corePreferences('sawPrivacyVersion');
- var showSplash = !corePreferences('sawSplash');
-
- if (sawPrivacyVersion !== context.privacyVersion) {
- updateMessage = _t('splash.privacy_update');
- showSplash = true;
- }
-
- 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 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
-
-
- 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'));
- }
-
- selection.attr('class', 'api-status ' + (err ? 'error' : apiStatus));
- }
-
- 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 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;
-
- mode.enter = function () {
- context.install(behavior);
- };
-
- mode.exit = function () {
- context.uninstall(behavior);
- };
-
- mode.selectedIDs = function () {
- return [wayID];
- };
-
- mode.activeID = function () {
- return behavior && behavior.activeID() || [];
- };
-
- return mode;
- }
-
- 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');
-
- 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 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));
- }
-
- 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));
- }
-
- mode.enter = function () {
- context.install(behavior);
- };
-
- mode.exit = function () {
- context.uninstall(behavior);
- };
-
- return mode;
- }
-
- 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');
-
- 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));
- }
-
- 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));
- }
-
- mode.enter = function () {
- context.install(behavior);
- };
-
- mode.exit = function () {
- context.uninstall(behavior);
- };
-
- return mode;
- }
-
- 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);
- }
-
- 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 enterSelectMode(node) {
- context.enter(modeSelect(context, [node.id]).newFeature(true));
- }
-
- function addNode(node) {
- if (Object.keys(defaultTags).length === 0) {
- enterSelectMode(node);
- return;
- }
-
- var tags = Object.assign({}, node.tags); // shallow copy
-
- for (var key in defaultTags) {
- tags[key] = defaultTags[key];
- }
-
- 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);
- };
-
- mode.exit = function () {
- context.uninstall(behavior);
- };
-
- return mode;
- }
-
- 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 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)
-
- context.map().pan([0, 0]);
- context.selectedNoteID(note.id).enter(modeSelectNote(context, note.id).newFeature(true));
- }
-
- function cancel() {
- context.enter(modeBrowse(context));
- }
-
- mode.enter = function () {
- context.install(behavior);
- };
-
- mode.exit = function () {
- context.uninstall(behavior);
- };
-
- return mode;
- }
-
- function uiConflicts(context) {
- var dispatch$1 = dispatch('cancel', 'save');
- var keybinding = utilKeybinding('conflicts');
-
- var _origChanges;
-
- var _conflictList;
-
- var _shownConflictIndex;
-
- function keybindingOn() {
- select(document).call(keybinding.on('⎋', cancel, true));
- }
-
- function keybindingOff() {
- select(document).call(keybinding.unbind);
- }
-
- function tryAgain() {
- keybindingOff();
- dispatch$1.call('save');
- }
-
- 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
-
- 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');
-
- 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);
- });
- }
-
- 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);
- }
-
- 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 (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);
- }
-
- 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
-
- 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 (ul.__data__.chosen === d.id) {
- choose(null, ul, d);
- }
- });
- }
-
- 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);
- }
-
- function zoomToEntity(id, extent) {
- context.surface().selectAll('.hover').classed('hover', false);
- var entity = context.graph().hasEntity(id);
-
- if (entity) {
- if (extent) {
- context.map().trimmedExtent(extent);
- } else {
- context.map().zoomToEase(entity);
- }
-
- 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)
- // ]
- // }
-
-
- conflicts.conflictList = function (_) {
- if (!arguments.length) return _conflictList;
- _conflictList = _;
- return conflicts;
- };
-
- conflicts.origChanges = function (_) {
- if (!arguments.length) return _origChanges;
- _origChanges = _;
- return conflicts;
- };
-
- conflicts.shownEntityIds = function () {
- if (_conflictList && typeof _shownConflictIndex === 'number') {
- return [_conflictList[_shownConflictIndex].id];
- }
-
- return [];
- };
-
- return utilRebind(conflicts, dispatch$1, 'on');
- }
-
- 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;
- };
-
- return modalSelection;
- }
-
- function uiChangesetEditor(context) {
- var dispatch$1 = dispatch('change');
- var formFields = uiFormFields(context);
- var commentCombo = uiCombobox(context, 'comment').caseSensitive(true);
-
- var _fieldsArr;
-
- var _tags;
-
- var _changesetID;
-
- function changesetEditor(selection) {
- render(selection);
- }
-
- 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
- })];
-
- _fieldsArr.forEach(function (field) {
- field.on('change', function (t, onInput) {
- dispatch$1.call('change', field, undefined, t, onInput);
- });
- });
- }
-
- _fieldsArr.forEach(function (field) {
- field.tags(_tags);
- });
-
- selection.call(formFields.fieldsArr(_fieldsArr));
-
- if (initial) {
- var commentField = selection.select('.form-field-comment textarea');
- var commentNode = commentField.node();
-
- 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
-
-
- utilTriggerEvent(commentField, 'blur');
- var osm = context.connection();
-
- 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 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);
- }
-
- changesetEditor.tags = function (_) {
- if (!arguments.length) return _tags;
- _tags = _; // Don't reset _fieldsArr here.
-
- return changesetEditor;
- };
-
- changesetEditor.changesetID = function (_) {
- if (!arguments.length) return _changesetID;
- if (_changesetID === _) return changesetEditor;
- _changesetID = _;
- _fieldsArr = null;
- return changesetEditor;
- };
-
- return utilRebind(changesetEditor, dispatch$1, 'on');
- }
-
- 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);
-
- 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 (name !== '') {
- string += ':';
- }
-
- 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
-
- 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 (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);
- });
- }
-
- 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);
- }
- }
-
- function mouseout() {
- context.surface().selectAll('.hover').classed('hover', false);
- }
-
- 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 uiCommitWarnings(context) {
- function commitWarnings(selection) {
- var issuesBySeverity = context.validator().getIssuesBySeverity({
- what: 'edited',
- where: 'all',
- includeDisabledRules: true
- });
-
- 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;
- }
-
- 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');
-
- 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 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 (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());
- }
-
- if (context.defaultChangesetSource()) {
- corePreferences('source', context.defaultChangesetSource());
- corePreferences('commentDate', Date.now());
- }
-
- if (context.defaultChangesetHashtags()) {
- corePreferences('hashtags', context.defaultChangesetHashtags());
- corePreferences('commentDate', Date.now());
- }
-
- 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;
- }
-
- var source = corePreferences('source');
-
- if (source) {
- tags.source = source;
- }
-
- var photoOverlaysUsed = context.history().photoOverlaysUsed();
-
- 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
-
-
- photoOverlaysUsed.forEach(function (photoOverlay) {
- if (sources.indexOf(photoOverlay) === -1) {
- sources.push(photoOverlay);
- }
- });
- tags.source = context.cleanTagValue(sources.join(';'));
- }
-
- 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 loadDerivedChangesetTags() {
- var osm = context.connection();
- if (!osm) return;
- var tags = Object.assign({}, context.changeset.tags); // shallow copy
- // assign tags for imagery used
-
- var imageryUsed = context.cleanTagValue(context.history().imageryUsed().join(';'));
- tags.imagery_used = imageryUsed || 'None'; // assign tags for closed issues and notes
-
- var osmClosed = osm.getClosedIDs();
- var itemType;
-
- if (osmClosed.length) {
- tags['closed:note'] = context.cleanTagValue(osmClosed.join(';'));
- }
-
- if (services.keepRight) {
- var krClosed = services.keepRight.getClosedIDs();
-
- if (krClosed.length) {
- tags['closed:keepright'] = context.cleanTagValue(krClosed.join(';'));
- }
- }
-
- if (services.improveOSM) {
- var iOsmClosed = services.improveOSM.getClosedCounts();
-
- for (itemType in iOsmClosed) {
- tags['closed:improveosm:' + itemType] = context.cleanTagValue(iOsmClosed[itemType].toString());
- }
- }
-
- if (services.osmose) {
- var osmoseClosed = services.osmose.getClosedCounts();
-
- for (itemType in osmoseClosed) {
- tags['closed:osmose:' + itemType] = context.cleanTagValue(osmoseClosed[itemType].toString());
- }
- } // remove existing issue counts
-
-
- for (var key in tags) {
- if (key.match(/(^warnings:)|(^resolved:)/)) {
- delete tags[key];
- }
- }
-
- function addIssueCounts(issues, prefix) {
- var issuesByType = utilArrayGroupBy(issues, 'type');
-
- for (var issueType in issuesByType) {
- var issuesOfType = issuesByType[issueType];
-
- if (issuesOfType[0].subtype) {
- var issuesBySubtype = utilArrayGroupBy(issuesOfType, 'subtype');
-
- 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
-
-
- 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
- });
- }
-
- 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
-
- 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
-
- 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]);
-
- 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
-
- osm.userDetails(function (err, user) {
- if (err) return;
- if (_userDetails === user) return; // no change
-
- _userDetails = user;
- 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');
- }
-
- 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
-
- 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
-
- requestReview = requestReview.merge(requestReviewEnter);
- var requestReviewInput = requestReview.selectAll('input').property('checked', isReviewRequested(context.changeset.tags)).on('change', toggleRequestReview); // Buttons
-
- var buttonSection = saveSection.selectAll('.buttons').data([0]); // enter
-
- 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
-
- 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
-
- for (var key in context.changeset.tags) {
- // remove any empty keys before upload
- if (!key) delete context.changeset.tags[key];
- }
-
- context.uploader().save(context.changeset);
- }
- }); // remove any existing tooltip
-
- uiTooltip().destroyAny(buttonSection.selectAll('.save-button'));
-
- if (uploadBlockerTooltipText) {
- buttonSection.selectAll('.save-button').call(uiTooltip().title(uploadBlockerTooltipText).placement('top'));
- } // Raw Tag Editor
-
-
- 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);
-
- 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);
- }
- }
-
- function getUploadBlockerMessage() {
- var errors = context.validator().getIssuesBySeverity({
- what: 'edited',
- where: 'all'
- }).error;
-
- 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');
- }
- }
-
- return null;
- }
-
- function changeTags(_, changed, onInput) {
- if (changed.hasOwnProperty('comment')) {
- if (changed.comment === undefined) {
- changed.comment = '';
- }
-
- if (!onInput) {
- corePreferences('comment', changed.comment);
- corePreferences('commentDate', Date.now());
- }
- }
-
- 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`
-
-
- updateChangeset(changed, onInput);
-
- if (_selection) {
- _selection.call(render);
- }
- }
-
- function findHashtags(tags, commentOnly) {
- var detectedHashtags = commentHashtags();
-
- if (detectedHashtags.length) {
- // always remove stored hashtags if there are hashtags in the comment - #4304
- corePreferences('hashtags', null);
- }
-
- if (!detectedHashtags.length || !commentOnly) {
- detectedHashtags = detectedHashtags.concat(hashtagHashtags());
- }
-
- 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;
- }
-
- 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
-
- return matches || [];
- }
- }
-
- 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
-
- 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);
- }
- });
-
- 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
-
-
- if (_userDetails && _userDetails.changesets_count !== undefined) {
- var changesetsCount = parseInt(_userDetails.changesets_count, 10) + 1; // #4283
-
- tags.changesets_count = String(changesetsCount); // first 100 edits - new user
-
- if (changesetsCount <= 100) {
- var s;
- s = corePreferences('walkthrough_completed');
-
- if (s) {
- tags['ideditor:walkthrough_completed'] = s;
- }
-
- s = corePreferences('walkthrough_progress');
-
- if (s) {
- tags['ideditor:walkthrough_progress'] = s;
- }
-
- s = corePreferences('walkthrough_started');
-
- if (s) {
- tags['ideditor:walkthrough_started'] = s;
- }
- }
- } else {
- delete tags.changesets_count;
- }
-
- if (!fastDeepEqual(context.changeset.tags, tags)) {
- context.changeset = context.changeset.update({
- tags: tags
- });
- }
- }
-
- commit.reset = function () {
- context.changeset = null;
- };
-
- return utilRebind(commit, dispatch$1, 'on');
- }
-
- var globalIsFinite = global_1.isFinite;
-
- // `Number.isFinite` method
- // https://tc39.github.io/ecma262/#sec-number.isfinite
- var numberIsFinite = Number.isFinite || function isFinite(it) {
- return typeof it == 'number' && globalIsFinite(it);
- };
-
- // `Number.isFinite` method
- // https://tc39.github.io/ecma262/#sec-number.isfinite
- _export({ target: 'Number', stat: true }, { isFinite: numberIsFinite });
-
- 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
- };
-
- var geometry_1 = geometry;
- var ring = ringArea;
-
- function geometry(_) {
- var area = 0,
- i;
-
- switch (_.type) {
- case 'Polygon':
- return polygonArea(_.coordinates);
-
- case 'MultiPolygon':
- for (i = 0; i < _.coordinates.length; i++) {
- area += polygonArea(_.coordinates[i]);
- }
-
- return area;
-
- case 'Point':
- case 'MultiPoint':
- case 'LineString':
- case 'MultiLineString':
- return 0;
-
- case 'GeometryCollection':
- for (i = 0; i < _.geometries.length; i++) {
- area += geometry(_.geometries[i]);
- }
-
- return area;
- }
- }
-
- function polygonArea(coords) {
- var area = 0;
-
- 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]));
- }
- }
-
- 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 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;
- }
-
- 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;
- }
-
- return area;
- }
-
- function rad(_) {
- return _ * Math.PI / 180;
- }
-
- var geojsonArea = {
- geometry: geometry_1,
- ring: ring
- };
-
- function toRadians(angleInDegrees) {
- return angleInDegrees * Math.PI / 180;
- }
-
- function toDegrees(angleInRadians) {
- return angleInRadians * 180 / Math.PI;
- }
-
- function offset(c1, distance, bearing) {
- var lat1 = toRadians(c1[1]);
- var lon1 = toRadians(c1[0]);
- var dByR = distance / 6378137; // distance divided by 6378137 (radius of the earth) wgs84
-
- 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)];
- }
-
- function validateCenter(center) {
- var validCenterLengths = [2, 3];
-
- if (!Array.isArray(center) || !validCenterLengths.includes(center.length)) {
- throw new Error("ERROR! Center has to be an array of length two or three");
- }
-
- 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)));
- }
-
- if (lng > 180 || lng < -180) {
- throw new Error("ERROR! Longitude has to be between -180 and 180 but was ".concat(lng));
- }
-
- if (lat > 90 || lat < -90) {
- throw new Error("ERROR! Latitude has to be between -90 and 90 but was ".concat(lat));
- }
- }
-
- function validateRadius(radius) {
- if (typeof radius !== "number") {
- throw new Error("ERROR! Radius has to be a positive number but was: ".concat(_typeof(radius)));
- }
-
- if (radius <= 0) {
- throw new Error("ERROR! Radius has to be a positive number but was: ".concat(radius));
- }
- }
-
- function validateNumberOfSegments(numberOfSegments) {
- if (typeof numberOfSegments !== "number" && numberOfSegments !== undefined) {
- throw new Error("ERROR! Number of segments has to be a number but was: ".concat(_typeof(numberOfSegments)));
- }
-
- if (numberOfSegments < 3) {
- throw new Error("ERROR! Number of segments has to be at least 3 but was: ".concat(numberOfSegments));
- }
- }
-
- function validateInput(_ref) {
- var center = _ref.center,
- radius = _ref.radius,
- numberOfSegments = _ref.numberOfSegments;
- validateCenter(center);
- validateRadius(radius);
- validateNumberOfSegments(numberOfSegments);
- }
-
- var circleToPolygon = function circleToPolygon(center, radius, numberOfSegments) {
- var n = numberOfSegments ? numberOfSegments : 32; // validateInput() throws error on invalid input and do nothing on valid input
-
- validateInput({
- center: center,
- radius: radius,
- numberOfSegments: numberOfSegments
- });
- var coordinates = [];
-
- for (var i = 0; i < n; ++i) {
- coordinates.push(offset(center, radius, 2 * Math.PI * -i / n));
- }
-
- coordinates.push(coordinates[0]);
- return {
- type: "Polygon",
- coordinates: [coordinates]
- };
- };
-
- // `Number.EPSILON` constant
- // https://tc39.github.io/ecma262/#sec-number.epsilon
- _export({ target: 'Number', stat: true }, {
- EPSILON: Math.pow(2, -52)
- });
-
- /**
- * splaytree v3.0.1
- * Fast Splay tree for Node and browser
- *
- * @author Alexander Milevski <info@w8r.name>
- * @license MIT
- * @preserve
- */
- var Node$1 = function Node(key, data) {
- _classCallCheck(this, Node);
-
- this.next = null;
- this.key = key;
- this.data = data;
- this.left = null;
- this.right = null;
- };
- /* follows "An implementation of top-down splaying"
- * by D. Sleator <sleator@cs.cmu.edu> March 1992
- */
-
-
- 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.
- */
-
-
- function splay(i, t, comparator) {
- var N = new Node$1(null, null);
- var l = N;
- var r = N;
-
- 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) {
-
- if (comparator(i, t.left.key) < 0) {
- var y = t.left;
- /* rotate right */
-
- t.left = y.right;
- y.right = t;
- t = y;
- if (t.left === null) break;
- }
-
- 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 */
-
- t.right = _y.left;
- _y.left = t;
- t = _y;
- if (t.right === null) break;
- }
-
- l.right = t;
- /* link left */
-
- l = t;
- t = t.right;
- } else break;
- }
- /* assemble */
-
-
- l.right = t.left;
- r.left = t.right;
- t.left = N.right;
- t.right = N.left;
- return t;
- }
-
- function _insert(i, data, t, comparator) {
- var node = new Node$1(i, data);
-
- if (t === null) {
- node.left = node.right = null;
- return node;
- }
-
- t = splay(i, t, comparator);
- var cmp = comparator(i, t.key);
-
- 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 node;
- }
-
- function _split(key, v, comparator) {
- var left = null;
- var right = null;
-
- if (v) {
- v = splay(key, v, comparator);
- var cmp = comparator(v.key, key);
-
- 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 {
- 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("".concat(prefix).concat(isTail ? '└── ' : '├── ').concat(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);
- }
- }
-
- var Tree = /*#__PURE__*/function () {
- function Tree() {
- var comparator = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : DEFAULT_COMPARE$1;
-
- _classCallCheck(this, Tree);
-
- this._root = null;
- this._size = 0;
- this._comparator = comparator;
- }
- /**
- * Inserts a key, allows duplicates
- */
-
-
- _createClass(Tree, [{
- key: "insert",
- value: function insert(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
- */
-
- }, {
- key: "add",
- value: function add(key, data) {
- var node = new Node$1(key, data);
-
- if (this._root === null) {
- node.left = node.right = null;
- this._size++;
- this._root = node;
- }
-
- 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;
- }
-
- this._size++;
- this._root = node;
- }
- return this._root;
- }
- /**
- * @param {Key} key
- * @return {Node|null}
- */
-
- }, {
- key: "remove",
- value: function remove(key) {
- this._root = this._remove(key, this._root, this._comparator);
- }
- /**
- * Deletes i from the tree if it's there
- */
-
- }, {
- key: "_remove",
- value: function _remove(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;
- }
-
- this._size--;
- return x;
- }
-
- return t;
- /* It wasn't there */
- }
- /**
- * Removes and returns the node with smallest key
- */
-
- }, {
- key: "pop",
- value: function pop() {
- var node = this._root;
-
- if (node) {
- while (node.left) {
- node = node.left;
- }
-
- 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
- */
-
- }, {
- key: "findStatic",
- value: function findStatic(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;
- }
-
- return null;
- }
- }, {
- key: "find",
- value: function find(key) {
- if (this._root) {
- this._root = splay(key, this._root, this._comparator);
- if (this._comparator(key, this._root.key) !== 0) return null;
- }
-
- return this._root;
- }
- }, {
- key: "contains",
- value: function contains(key) {
- var current = this._root;
- var compare = this._comparator;
-
- 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 false;
- }
- }, {
- key: "forEach",
- value: function forEach(visitor, ctx) {
- var current = this._root;
- var Q = [];
- /* Initialize stack s */
-
- var done = false;
-
- 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;
- }
- }
-
- return this;
- }
- /**
- * Walk key range from `low` to `high`. Stops if `fn` returns a value.
- */
-
- }, {
- key: "range",
- value: function range(low, high, fn, ctx) {
- var Q = [];
- var compare = this._comparator;
- var node = this._root;
- var cmp;
-
- while (Q.length !== 0 || node) {
- if (node) {
- Q.push(node);
- node = node.left;
- } else {
- node = Q.pop();
- cmp = compare(node.key, high);
-
- if (cmp > 0) {
- break;
- } else if (compare(node.key, low) >= 0) {
- if (fn.call(ctx, node)) return this; // stop if smth is returned
- }
-
- node = node.right;
- }
- }
-
- return this;
- }
- /**
- * Returns array of keys
- */
-
- }, {
- key: "keys",
- value: function keys() {
- var keys = [];
- this.forEach(function (_ref) {
- var key = _ref.key;
- return keys.push(key);
- });
- return keys;
- }
- /**
- * Returns array of all the data in the nodes
- */
-
- }, {
- key: "values",
- value: function values() {
- var values = [];
- this.forEach(function (_ref2) {
- var data = _ref2.data;
- return values.push(data);
- });
- return values;
- }
- }, {
- key: "min",
- value: function min() {
- if (this._root) return this.minNode(this._root).key;
- return null;
- }
- }, {
- key: "max",
- value: function max() {
- if (this._root) return this.maxNode(this._root).key;
- return null;
- }
- }, {
- key: "minNode",
- value: function minNode() {
- var t = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : this._root;
- if (t) while (t.left) {
- t = t.left;
- }
- return t;
- }
- }, {
- key: "maxNode",
- value: function maxNode() {
- var t = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : this._root;
- if (t) while (t.right) {
- t = t.right;
- }
- return t;
- }
- /**
- * Returns node at given index
- */
-
- }, {
- key: "at",
- value: function at(index) {
- var current = this._root;
- var done = false;
- var i = 0;
- var Q = [];
-
- 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;
- }
- }
-
- return null;
- }
- }, {
- key: "next",
- value: function next(d) {
- var root = this._root;
- var successor = null;
-
- if (d.right) {
- successor = d.right;
-
- while (successor.left) {
- successor = successor.left;
- }
-
- return successor;
- }
-
- var comparator = this._comparator;
-
- 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;
- }
- }, {
- key: "prev",
- value: function prev(d) {
- var root = this._root;
- var predecessor = null;
-
- if (d.left !== null) {
- predecessor = d.left;
-
- while (predecessor.right) {
- predecessor = predecessor.right;
- }
-
- return predecessor;
- }
-
- var comparator = this._comparator;
-
- 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;
- }
- }, {
- key: "clear",
- value: function clear() {
- this._root = null;
- this._size = 0;
- return this;
- }
- }, {
- key: "toList",
- value: function toList() {
- return _toList(this._root);
- }
- /**
- * Bulk-load items. Both array have to be same size
- */
-
- }, {
- key: "load",
- value: function load(keys) {
- var values = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : [];
- var presort = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false;
- var size = keys.length;
- var comparator = this._comparator; // sort if needed
-
- if (presort) sort$1(keys, values, 0, size - 1, comparator);
-
- 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);
- }
-
- return this;
- }
- }, {
- key: "isEmpty",
- value: function isEmpty() {
- return this._root === null;
- }
- }, {
- key: "toString",
- value: function toString() {
- var printNode = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : function (n) {
- return String(n.key);
- };
- var out = [];
- printRow(this._root, '', true, function (v) {
- return out.push(v);
- }, printNode);
- return out.join('');
- }
- }, {
- key: "update",
- value: function update(key, newKey, newData) {
- var comparator = this._comparator;
-
- var _split2 = _split(key, this._root, comparator),
- left = _split2.left,
- right = _split2.right;
-
- if (comparator(key, newKey) < 0) {
- right = _insert(newKey, newData, right, comparator);
- } else {
- left = _insert(newKey, newData, left, comparator);
- }
-
- this._root = merge$4(left, right, comparator);
- }
- }, {
- key: "split",
- value: function split(key) {
- return _split(key, this._root, this._comparator);
- }
- }, {
- key: "size",
- get: function get() {
- return this._size;
- }
- }, {
- key: "root",
- get: function get() {
- return this._root;
- }
- }]);
-
- return Tree;
- }();
-
- function loadRecursive$1(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 = new Node$1(key, data);
- node.left = loadRecursive$1(keys, values, start, middle);
- node.right = loadRecursive$1(keys, values, middle + 1, end);
- return node;
- }
-
- return null;
- }
-
- 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]);
- }
-
- p.next = null;
- return head.next;
- }
-
- 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;
- }
- }
-
- p.next = null; // that'll work even if the tree was empty
-
- return head.next;
- }
-
- function sortedListToBST(list, start, end) {
- var size = end - start;
-
- 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 null;
- }
-
- function mergeLists(l1, l2, compare) {
- var head = new Node$1(null, null); // dummy
-
- 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;
- }
-
- if (p1 !== null) {
- p.next = p1;
- } else if (p2 !== null) {
- p.next = p2;
- }
-
- return head.next;
- }
-
- 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;
-
- 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;
- }
-
- sort$1(keys, values, left, j, compare);
- sort$1(keys, values, j + 1, right, compare);
- }
-
- function _classCallCheck$1(instance, Constructor) {
- if (!(instance instanceof Constructor)) {
- throw new TypeError("Cannot call a class as a function");
- }
- }
-
- 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 _createClass$1(Constructor, protoProps, staticProps) {
- if (protoProps) _defineProperties$1(Constructor.prototype, protoProps);
- if (staticProps) _defineProperties$1(Constructor, staticProps);
- return Constructor;
- }
- /**
- * 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
-
- 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
-
- 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
- */
-
-
- var epsilon$2 = Number.EPSILON; // IE Polyfill
-
- if (epsilon$2 === undefined) epsilon$2 = Math.pow(2, -52);
- var EPSILON_SQ = epsilon$2 * epsilon$2;
- /* FLP comparator */
-
- 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
-
-
- var ab = a - b;
-
- if (ab * ab < EPSILON_SQ * a * b) {
- return 0;
- } // normal comparison
-
-
- 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.
- */
-
-
- var PtRounder = /*#__PURE__*/function () {
- function PtRounder() {
- _classCallCheck$1(this, PtRounder);
-
- this.reset();
- }
-
- _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)
- };
- }
- }]);
-
- return PtRounder;
- }();
-
- var CoordRounder = /*#__PURE__*/function () {
- function CoordRounder() {
- _classCallCheck$1(this, CoordRounder);
-
- this.tree = new Tree(); // preseed with 0 so we don't end up with values < Number.EPSILON
-
- 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).
-
-
- _createClass$1(CoordRounder, [{
- key: "round",
- value: function round(coord) {
- var node = this.tree.add(coord);
- var prevNode = this.tree.prev(node);
-
- if (prevNode !== null && cmp(node.key, prevNode.key) === 0) {
- this.tree.remove(coord);
- return prevNode.key;
- }
-
- var nextNode = this.tree.next(node);
-
- if (nextNode !== null && cmp(node.key, nextNode.key) === 0) {
- this.tree.remove(coord);
- return nextNode.key;
- }
-
- return coord;
- }
- }]);
-
- return CoordRounder;
- }(); // singleton available by import
-
-
- var rounder = new PtRounder();
- /* Cross Product of two vectors with first point at origin */
-
- 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 */
-
-
- 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 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);
- };
-
- var length = function length(v) {
- return Math.sqrt(dotProduct$1(v, v));
- };
- /* Get the sine of the angle from pShared -> pAngle to pShaed -> pBase */
-
-
- 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 */
-
-
- 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. */
-
-
- 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. */
-
-
- 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. */
-
-
- 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
-
- 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
- };
- };
-
- 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 (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
-
- return Segment.compare(a.segment, b.segment);
- } // for ordering points in sweep line order
-
- }, {
- 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)
-
- }]);
-
- function SweepEvent(point, isLeft) {
- _classCallCheck$1(this, SweepEvent);
-
- 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');
- }
-
- var otherEvents = other.point.events;
-
- 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. */
-
- }, {
- 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;
-
- 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, iMax = this.point.events.length; i < iMax; i++) {
- var evt = this.point.events[i];
-
- if (evt !== this && !evt.segment.ringOut && evt.segment.isInResult()) {
- events.push(evt);
- }
- }
-
- 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;
-
- var cache = new Map();
-
- 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)
- });
- };
-
- 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;
-
- var _cache$get2 = cache.get(b),
- bsine = _cache$get2.sine,
- bcosine = _cache$get2.cosine; // both on or above x-axis
-
-
- if (asine >= 0 && bsine >= 0) {
- if (acosine < bcosine) return 1;
- if (acosine > bcosine) return -1;
- return 0;
- } // both below x-axis
-
-
- if (asine < 0 && bsine < 0) {
- if (acosine < bcosine) return -1;
- if (acosine > bcosine) return 1;
- return 0;
- } // one above x-axis, one below
-
-
- if (bsine < asine) return -1;
- if (bsine > asine) return 1;
- return 0;
- };
- }
- }]);
-
- return SweepEvent;
- }(); // segments and sweep events when all else is identical
-
-
- var segmentId = 0;
-
- var Segment = /*#__PURE__*/function () {
- _createClass$1(Segment, null, [{
- key: "compare",
-
- /* 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
-
- 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 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?
-
-
- 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 bCmpALeft = b.comparePoint(a.leftSE.point);
- if (bCmpALeft !== 0) return bCmpALeft; // is the B right endpoint colinear to segment A?
-
- 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?)
-
- 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 (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?
-
- if (arx < brx) {
- var _bCmpARight = b.comparePoint(a.rightSE.point);
-
- if (_bCmpARight !== 0) return _bCmpARight;
- } // is the B right endpoint more left-more?
-
-
- if (arx > brx) {
- var _aCmpBRight = a.comparePoint(b.rightSE.point);
-
- if (_aCmpBRight < 0) return 1;
- if (_aCmpBRight > 0) return -1;
- }
-
- 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 (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 (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
-
- return 0;
- }
- /* Warning: a reference to ringWindings input will be stored,
- * and possibly will be later modified */
-
- }]);
-
- function Segment(leftSE, rightSE, rings, windings) {
- _classCallCheck$1(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
- }
-
- _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
- }
- };
- }
- /* 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
- };
- }
- }, {
- 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)
- */
-
- }, {
- 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.
-
-
- 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.
- */
-
- }, {
- 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
-
- 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
-
- return null;
- } // does this left endpoint matches (other doesn't)
-
-
- 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
-
-
- return tlp;
- } // does other left endpoint matches (this doesn't)
-
-
- 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 olp;
- } // trivial intersection on right endpoints
-
-
- if (touchesThisRSE && touchesOtherRSE) return null; // t-intersections on just one right endpoint
-
- 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 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
-
- if (pt === null) return null; // is the intersection found between the lines not on the segments?
-
- 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
- */
-
- }, {
- 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();
- }
-
- 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
-
-
- if (alreadyLinked) {
- newLeftSE.checkForConsuming();
- newRightSE.checkForConsuming();
- }
-
- 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;
- }
-
- while (consumee.consumedBy) {
- consumee = consumee.consumedBy;
- }
-
- 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 (cmp > 0) {
- var tmp = consumer;
- consumer = consumee;
- consumee = tmp;
- } // make sure a segment doesn't consume it's prev
-
-
- 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);
-
- if (index === -1) {
- consumer.rings.push(ring);
- consumer.windings.push(winding);
- } else consumer.windings[index] += winding;
- }
-
- consumee.rings = null;
- consumee.windings = null;
- consumee.consumedBy = consumer; // mark sweep events consumed as to maintain ordering in sweep event queue
-
- consumee.leftSE.consumedBy = consumer.leftSE;
- consumee.rightSE.consumedBy = consumer.rightSE;
- }
- /* The first segment previous segment chain that is in the result */
-
- }, {
- 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
-
- 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);
-
- if (index === -1) {
- ringsAfter.push(ring);
- windingsAfter.push(winding);
- } else windingsAfter[index] += winding;
- } // calcualte polysAfter
-
-
- var polysAfter = [];
- var polysExclude = [];
-
- for (var _i = 0, _iMax = ringsAfter.length; _i < _iMax; _i++) {
- if (windingsAfter[_i] === 0) continue; // non-zero rule
-
- 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);
-
- var _index = polysAfter.indexOf(_ring.poly);
-
- if (_index !== -1) polysAfter.splice(_index, 1);
- }
- } // calculate multiPolysAfter
-
-
- for (var _i2 = 0, _iMax2 = polysAfter.length; _i2 < _iMax2; _i2++) {
- var mp = polysAfter[_i2].multiPoly;
- if (mpsAfter.indexOf(mp) === -1) mpsAfter.push(mp);
- }
-
- return this._afterState;
- }
- /* 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;
-
- if (mpsBefore.length < mpsAfter.length) {
- least = mpsBefore.length;
- most = mpsAfter.length;
- } else {
- least = mpsAfter.length;
- most = mpsBefore.length;
- }
-
- 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));
- }
-
- return this._isInResult;
- }
- }], [{
- 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, "]"));
-
- var leftSE = new SweepEvent$1(leftPt, true);
- var rightSE = new SweepEvent$1(rightPt, false);
- return new Segment(leftSE, rightSE, [ring], [winding]);
- }
- }]);
-
- return Segment;
- }();
-
- var RingIn = /*#__PURE__*/function () {
- function RingIn(geomRing, poly, isExterior) {
- _classCallCheck$1(this, RingIn);
-
- if (!Array.isArray(geomRing) || geomRing.length === 0) {
- throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
- }
-
- this.poly = poly;
- this.isExterior = isExterior;
- this.segments = [];
-
- if (typeof geomRing[0][0] !== 'number' || typeof geomRing[0][1] !== 'number') {
- throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
- }
-
- 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 (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
-
- 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 (firstPoint.x !== prevPoint.x || firstPoint.y !== prevPoint.y) {
- this.segments.push(Segment.fromRing(prevPoint, firstPoint, this));
- }
- }
-
- _createClass$1(RingIn, [{
- key: "getSweepEvents",
- value: function getSweepEvents() {
- var sweepEvents = [];
-
- 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;
- }
- }]);
-
- return RingIn;
- }();
-
- var PolyIn = /*#__PURE__*/function () {
- function PolyIn(geomPoly, multiPoly) {
- _classCallCheck$1(this, PolyIn);
-
- if (!Array.isArray(geomPoly)) {
- throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
- }
-
- this.exteriorRing = new RingIn(geomPoly[0], this, true); // copy by value
-
- 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
- }
- };
- 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;
- }
-
- _createClass$1(PolyIn, [{
- key: "getSweepEvents",
- value: function getSweepEvents() {
- var sweepEvents = this.exteriorRing.getSweepEvents();
-
- for (var i = 0, iMax = this.interiorRings.length; i < iMax; i++) {
- var ringSweepEvents = this.interiorRings[i].getSweepEvents();
-
- for (var j = 0, jMax = ringSweepEvents.length; j < jMax; j++) {
- sweepEvents.push(ringSweepEvents[j]);
- }
- }
-
- return sweepEvents;
- }
- }]);
-
- return PolyIn;
- }();
-
- var MultiPolyIn = /*#__PURE__*/function () {
- function MultiPolyIn(geom, isSubject) {
- _classCallCheck$1(this, MultiPolyIn);
-
- if (!Array.isArray(geom)) {
- throw new Error('Input geometry is not a valid Polygon or MultiPolygon');
- }
-
- 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.
- }
-
- this.polys = [];
- this.bbox = {
- ll: {
- x: Number.POSITIVE_INFINITY,
- y: Number.POSITIVE_INFINITY
- },
- ur: {
- x: Number.NEGATIVE_INFINITY,
- y: Number.NEGATIVE_INFINITY
- }
- };
-
- 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;
- }
-
- _createClass$1(MultiPolyIn, [{
- key: "getSweepEvents",
- value: function getSweepEvents() {
- var sweepEvents = [];
-
- 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 sweepEvents;
- }
- }]);
-
- return MultiPolyIn;
- }();
-
- var RingOut = /*#__PURE__*/function () {
- _createClass$1(RingOut, null, [{
- key: "factory",
-
- /* 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 */
-
- while (true) {
- prevEvent = event;
- event = nextEvent;
- events.push(event);
- /* Is the ring complete? */
-
- if (event.point === startingPoint) 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 (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 (availableLEs.length === 1) {
- nextEvent = availableLEs[0].otherSE;
- break;
- }
- /* We must have an intersection. Check for a completed loop */
-
-
- var indexLE = null;
-
- 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 (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;
- }
- }
-
- ringsOut.push(new RingOut(events));
- }
-
- return ringsOut;
- }
- }]);
-
- function RingOut(events) {
- _classCallCheck$1(this, RingOut);
-
- this.events = events;
-
- for (var i = 0, iMax = events.length; i < iMax; i++) {
- events[i].segment.ringOut = this;
- }
-
- 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];
-
- 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 (points.length === 1) return null; // check if the starting point is necessary
-
- 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 = [];
-
- 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;
- }
-
- return this._isExteriorRing;
- }
- }, {
- key: "enclosingRing",
- value: function enclosingRing() {
- if (this._enclosingRing === undefined) {
- this._enclosingRing = this._calcEnclosingRing();
- }
-
- 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];
-
- 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;
-
- 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 (!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
-
- 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
-
-
- prevSeg = prevPrevSeg.prevInResult();
- prevPrevSeg = prevSeg ? prevSeg.prevInResult() : null;
- }
- }
- }]);
-
- return RingOut;
- }();
-
- var PolyOut = /*#__PURE__*/function () {
- function PolyOut(exteriorRing) {
- _classCallCheck$1(this, PolyOut);
-
- this.exteriorRing = exteriorRing;
- exteriorRing.poly = this;
- this.interiorRings = [];
- }
-
- _createClass$1(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 (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);
- }
-
- return geom;
- }
- }]);
-
- return PolyOut;
- }();
-
- var MultiPolyOut = /*#__PURE__*/function () {
- function MultiPolyOut(rings) {
- _classCallCheck$1(this, MultiPolyOut);
-
- this.rings = rings;
- this.polys = this._composePolys(rings);
- }
-
- _createClass$1(MultiPolyOut, [{
- key: "getGeom",
- value: function getGeom() {
- var geom = [];
-
- 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 (polyGeom === null) continue;
- geom.push(polyGeom);
- }
-
- 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);
- }
- }
-
- return polys;
- }
- }]);
-
- 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 SweepLine = /*#__PURE__*/function () {
- function SweepLine(queue) {
- var comparator = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : Segment.compare;
-
- _classCallCheck$1(this, SweepLine);
-
- this.queue = queue;
- this.tree = new Tree(comparator);
- this.segments = [];
- }
-
- _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 (event.consumedBy) {
- if (event.isLeft) this.queue.remove(event.otherSE);else this.tree.remove(segment);
- return newEvents;
- }
-
- 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
-
- 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
-
-
- while (nextSeg === undefined) {
- nextNode = this.tree.next(nextNode);
- if (nextNode === null) nextSeg = null;else if (nextNode.key.consumedBy === undefined) nextSeg = nextNode.key;
- }
-
- if (event.isLeft) {
- // Check for intersections against the previous segment in the sweep line
- var prevMySplitter = null;
-
- if (prevSeg) {
- var prevInter = prevSeg.getIntersection(segment);
-
- if (prevInter !== null) {
- if (!segment.isAnEndpoint(prevInter)) prevMySplitter = prevInter;
-
- 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
-
-
- var nextMySplitter = null;
-
- if (nextSeg) {
- var nextInter = nextSeg.getIntersection(segment);
-
- if (nextInter !== null) {
- if (!segment.isAnEndpoint(nextInter)) nextMySplitter = nextInter;
-
- if (!nextSeg.isAnEndpoint(nextInter)) {
- var _newEventsFromSplit = this._splitSafely(nextSeg, nextInter);
-
- 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().
-
-
- 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
-
- this.queue.remove(segment.rightSE);
- newEvents.push(segment.rightSE);
-
- var _newEventsFromSplit2 = segment.split(mySplitter);
-
- 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);
- } 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 (inter !== null) {
- if (!prevSeg.isAnEndpoint(inter)) {
- var _newEventsFromSplit3 = this._splitSafely(prevSeg, inter);
-
- for (var _i3 = 0, _iMax3 = _newEventsFromSplit3.length; _i3 < _iMax3; _i3++) {
- newEvents.push(_newEventsFromSplit3[_i3]);
- }
- }
-
- if (!nextSeg.isAnEndpoint(inter)) {
- var _newEventsFromSplit4 = this._splitSafely(nextSeg, inter);
-
- for (var _i4 = 0, _iMax4 = _newEventsFromSplit4.length; _i4 < _iMax4; _i4++) {
- newEvents.push(_newEventsFromSplit4[_i4]);
- }
- }
- }
- }
-
- this.tree.remove(segment);
- }
-
- return newEvents;
- }
- /* 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
-
- 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$1(this, Operation);
- }
-
- _createClass$1(Operation, [{
- key: "run",
- value: function run(type, geom, moreGeoms) {
- operation.type = type;
- rounder.reset();
- /* Convert inputs to MultiPoly objects */
-
- var multipolys = [new MultiPolyIn(geom, true)];
-
- for (var i = 0, iMax = moreGeoms.length; i < iMax; i++) {
- multipolys.push(new MultiPolyIn(moreGeoms[i], false));
- }
-
- 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. */
-
- if (operation.type === 'difference') {
- // in place removal
- var subject = multipolys[0];
- var _i = 1;
-
- while (_i < multipolys.length) {
- if (getBboxOverlap(multipolys[_i].bbox, subject.bbox) !== null) _i++;else multipolys.splice(_i, 1);
- }
- }
- /* 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 (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];
-
- 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 */
-
-
- var queue = new Tree(SweepEvent$1.compare);
-
- for (var _i3 = 0, _iMax2 = multipolys.length; _i3 < _iMax2; _i3++) {
- var sweepEvents = multipolys[_i3].getSweepEvents();
-
- for (var _j = 0, _jMax = sweepEvents.length; _j < _jMax; _j++) {
- queue.insert(sweepEvents[_j]);
-
- 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 */
-
-
- 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 (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.');
- }
-
- var newEvents = sweepLine.process(evt);
-
- for (var _i4 = 0, _iMax3 = newEvents.length; _i4 < _iMax3; _i4++) {
- var _evt = newEvents[_i4];
- if (_evt.consumedBy === undefined) queue.insert(_evt);
- }
-
- 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 */
-
- var ringsOut = RingOut.factory(sweepLine.segments);
- var result = new MultiPolyOut(ringsOut);
- return result.getGeom();
- }
- }]);
-
- return Operation;
- }(); // singleton available by import
-
-
- 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];
- }
-
- return operation.run('union', geom, moreGeoms);
- };
-
- 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];
- }
-
- 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];
- }
-
- return operation.run('xor', geom, moreGeoms);
- };
-
- 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];
- }
-
- return operation.run('difference', subjectGeom, clippingGeoms);
- };
-
- var index$1 = {
- union: union$1,
- intersection: intersection$1$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);
- }
- });
- }
-
- function multi(l) {
- return l.map(point);
- }
-
- function poly(p) {
- return p.map(multi);
- }
-
- function multiPoly(m) {
- return m.map(poly);
- }
-
- function geometry(obj) {
- if (!obj) {
- return {};
- }
-
- switch (obj.type) {
- case "Point":
- obj.coordinates = point(obj.coordinates);
- return obj;
-
- case "LineString":
- case "MultiPoint":
- obj.coordinates = multi(obj.coordinates);
- return obj;
-
- 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 {};
- }
- }
-
- function feature(obj) {
- obj.geometry = geometry(obj.geometry);
- return obj;
- }
-
- function featureCollection(f) {
- f.features = f.features.map(feature);
- return f;
- }
-
- function geometryCollection(g) {
- g.geometries = g.geometries.map(geometry);
- return g;
- }
-
- if (!t) {
- return t;
- }
-
- switch (t.type) {
- case "Feature":
- return feature(t);
-
- case "GeometryCollection":
- return geometryCollection(t);
-
- case "FeatureCollection":
- return featureCollection(t);
-
- case "Point":
- case "LineString":
- case "Polygon":
- case "MultiPoint":
- case "MultiPolygon":
- case "MultiLineString":
- return geometry(t);
-
- default:
- return 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);
- });
- }
- }
-
- function getTreeDepth(obj) {
- var depth = 0;
-
- if (Array.isArray(obj) || isObject$4(obj)) {
- forEach(obj, function (val) {
- if (Array.isArray(val) || isObject$4(val)) {
- var tmpDepth = getTreeDepth(val);
-
- if (tmpDepth > depth) {
- depth = tmpDepth;
- }
- }
- });
- return depth + 1;
- }
-
- return depth;
- }
-
- 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();
- }
-
- var string = JSON.stringify(obj);
-
- if (string === undefined) {
- return string;
- }
-
- var length = maxLength - currentIndent.length - reserved;
- var treeDepth = getTreeDepth(obj);
-
- if (treeDepth <= maxNesting && string.length <= length) {
- var prettified = prettify(string, {
- addMargin: addMargin,
- addArrayMargin: addArrayMargin,
- addObjectMargin: addObjectMargin
- });
-
- if (prettified.length <= length) {
- return prettified;
- }
- }
-
- if (isObject$4(obj)) {
- var nextIndent = currentIndent + indent;
- var items = [];
- var delimiters;
-
- var comma = function comma(array, index) {
- return index === array.length - 1 ? 0 : 1;
- };
-
- if (Array.isArray(obj)) {
- for (var index = 0; index < obj.length; index++) {
- items.push(_stringify(obj[index], nextIndent, comma(obj, index)) || 'null');
- }
-
- 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 = {
- '{': '{',
- '}': '}',
- '[': '[',
- ']': ']',
- ',': ', ',
- ':': ': '
- };
-
- 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 get$5(options, name, defaultValue) {
- return name in options ? options[name] : defaultValue;
- }
-
- var jsonStringifyPrettyCompact = stringify;
-
- var _default$3 = /*#__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(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 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
-
- if (!props.area) {
- var area = geojsonArea.geometry(feature.geometry) / 1e6; // m² to km²
-
- props.area = Number(area.toFixed(2));
- }
-
- _this._cache[id] = feature;
- });
- } // Replace CountryCoder world geometry to be a polygon covering the world.
-
-
- var world = _cloneDeep(feature('Q2'));
-
- 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²
-
- 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
- //
-
-
- _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$1 = feature(location);
-
- if (feature$1) {
- // 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$1.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
- //
-
- }, {
- 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?
-
-
- if (valid.type === 'point') {
- var RADIUS = 25000; // meters
-
- var EDGES = 10;
- var PRECISION = 3;
- var area = Math.PI * RADIUS * RADIUS / 1e6; // m² to km²
-
- var feature$1 = 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$1
- }); // a .geojson filename?
- } else if (valid.type === 'geojson') ; else if (valid.type === 'countrycoder') {
- var _feature = _cloneDeep(feature(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.
-
- 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
-
-
- if (!props.area) {
- var _area = geojsonArea.geometry(_feature.geometry) / 1e6; // m² to km²
-
-
- props.area = Number(_area.toFixed(2));
- } // ensure `id` property exists
-
-
- _feature.id = id;
- props.id = id;
- this._cache[id] = _feature;
- return Object.assign(valid, {
- feature: _feature
- });
- }
-
- 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
- //
-
- }, {
- 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 (!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
-
-
- include.sort(_sortLocations);
- var id = '+[' + include.map(function (d) {
- return d.id;
- }).join(',') + ']';
-
- if (exclude.length) {
- exclude.sort(_sortLocations);
- id += '-[' + exclude.map(function (d) {
- return d.id;
- }).join(',') + ']';
- }
-
- 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
- //
-
- }, {
- 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
-
- if (this._cache[id]) {
- return Object.assign(valid, {
- feature: this._cache[id]
- });
- }
-
- 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..
-
- if (include.length === 1 && exclude.length === 0) {
- return Object.assign(valid, {
- feature: include[0].feature
- });
- } // calculate unions
-
-
- 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
-
- var resultGeoJSON = excludeGeoJSON ? _clip(includeGeoJSON, excludeGeoJSON, 'DIFFERENCE') : includeGeoJSON;
- var area = geojsonArea.geometry(resultGeoJSON.geometry) / 1e6; // m² to km²
-
- resultGeoJSON.id = id;
- resultGeoJSON.properties = {
- id: id,
- area: Number(area.toFixed(2))
- };
- this._cache[id] = resultGeoJSON;
- return Object.assign(valid, {
- feature: resultGeoJSON
- });
- } // strict
- //
-
- }, {
- 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
-
- }, {
- 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);
- }
- }]);
-
- return _default;
- }(); // Wrap the mfogel/polygon-clipping library and return a GeoJSON feature.
-
- 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?
-
- 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
-
-
- function _locationReducer(accumulator, location) {
- /* eslint-disable no-console, no-invalid-this */
- var result;
-
- try {
- var resolved = this.resolveLocation(location);
-
- if (!resolved || !resolved.feature) {
- console.warn("Warning: Couldn't resolve location \"".concat(location, "\""));
- return accumulator;
- }
-
- result = !accumulator ? resolved.feature : _clip(accumulator, resolved.feature, 'UNION');
- } catch (e) {
- console.warn("Warning: Error resolving location \"".concat(location, "\""));
- console.warn(e);
- result = accumulator;
- }
-
- return result;
- /* eslint-enable no-console, no-invalid-this */
- }
-
- 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);
- }
-
- var _oci = null;
- function uiSuccess(context) {
- var MAXEVENTS = 2;
- var dispatch$1 = dispatch('cancel');
-
- var _changeset;
-
- var _location;
-
- ensureOSMCommunityIndex(); // start fetching the data
-
- 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$3(vals[1]);
- var ociFeatures = {};
- Object.values(ociResources).forEach(function (resource) {
- var feature = loco.resolveLocationSet(resource.locationSet).feature;
- var ociFeature = ociFeatures[feature.id];
-
- if (!ociFeature) {
- ociFeature = JSON.parse(JSON.stringify(feature)); // deep clone
-
- ociFeature.properties.resourceIDs = new Set();
- ociFeatures[feature.id] = ociFeature;
- }
-
- ociFeature.properties.resourceIDs.add(resource.id);
- });
- return _oci = {
- features: ociFeatures,
- resources: ociResources,
- query: whichPolygon_1({
- type: 'FeatureCollection',
- features: Object.values(ociFeatures)
- })
- };
- });
- } // string-to-date parsing in JavaScript is weird
-
-
- function parseEventDate(when) {
- if (!when) return;
- var raw = when.trim();
- if (!raw) return;
-
- if (!/Z$/.test(raw)) {
- // if no trailing 'Z', add one
- raw += 'Z'; // this forces date to be parsed as a UTC date
- }
-
- var parsed = new Date(raw);
- return new Date(parsed.toUTCString().substr(0, 25)); // convert to local timezone
- }
-
- 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..
-
- 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
-
- 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 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'));
- }
-
- 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.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);
-
- 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));
- }
-
- 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
-
- 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);
- }
-
- function showMore(selection) {
- var more = selection.selectAll('.community-more').data([0]);
- var moreEnter = more.enter().append('div').attr('class', 'community-more');
-
- 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
- }));
- }
- }
-
- 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
- });
- }
-
- 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;
- });
- }
-
- function linkify(url, text) {
- text = text || url;
- return "<a target=\"_blank\" href=\"".concat(url, "\">").concat(text, "</a>");
- }
- }
-
- success.changeset = function (val) {
- if (!arguments.length) return _changeset;
- _changeset = val;
- return success;
- };
-
- success.location = function (val) {
- if (!arguments.length) return _location;
- _location = val;
- return success;
- };
-
- return utilRebind(success, dispatch$1, 'on');
- }
-
- function modeSave(context) {
- var mode = {
- id: 'save'
- };
- var keybinding = utilKeybinding('modeSave');
- var commit = uiCommit(context).on('cancel', cancel);
-
- var _conflictsUi; // uiConflicts
-
-
- var _location;
-
- var _success;
-
- 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 cancel() {
- context.enter(modeBrowse(context));
- }
-
- function showProgress(num, total) {
- var modal = context.container().select('.loading-modal .modal-section');
- var progress = modal.selectAll('.progress').data([0]); // enter/update
-
- progress.enter().append('div').attr('class', 'progress').merge(progress).text(_t('save.conflict_progress', {
- num: num,
- total: total
- }));
- }
-
- 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);
- }
-
- 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 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();
- }
-
- function showSuccess(changeset) {
- commit.reset();
-
- var ui = _success.changeset(changeset).location(_location).on('cancel', function () {
- context.ui().sidebar.hide();
- });
-
- context.enter(modeBrowse(context).sidebar(ui));
- }
-
- function keybindingOn() {
- select(document).call(keybinding.on('⎋', cancel, true));
- }
-
- 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 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() : [];
- };
-
- mode.enter = function () {
- // Show sidebar
- context.ui().sidebar.expand();
-
- function done() {
- context.ui().sidebar.show(commit);
- }
-
- keybindingOn();
- context.container().selectAll('.main-content').classed('active', false).classed('inactive', true);
- var osm = context.connection();
-
- if (!osm) {
- cancel();
- return;
- }
-
- if (osm.authenticated()) {
- done();
- } else {
- osm.authenticate(function (err) {
- if (err) {
- cancel();
- } else {
- done();
- }
- });
- }
- };
-
- mode.exit = function () {
- keybindingOff();
- context.container().selectAll('.main-content').classed('active', true).classed('inactive', false);
- context.ui().sidebar.hide();
- };
-
- return mode;
- }
-
- 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'
- })];
-
- function enabled() {
- return osmEditable();
- }
-
- 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);
- }
- });
- });
-
- tool.render = function (selection) {
- var wrap = selection.append('div').attr('class', 'joined').style('display', 'flex');
-
- var debouncedUpdate = debounce(update, 500, {
- leading: true,
- trailing: true
- });
-
- 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
-
-
- buttons = buttons.merge(buttonsEnter).classed('disabled', function (d) {
- return !enabled();
- }).classed('active', function (d) {
- return context.mode() && context.mode().button === d.button;
- });
- }
- };
-
- return tool;
- }
-
- function uiToolNotes(context) {
- var tool = {
- id: 'notes',
- label: _t.html('modes.add_note.label')
- };
- var mode = modeAddNote(context);
-
- function enabled() {
- return notesEnabled() && notesEditable();
- }
-
- function notesEnabled() {
- var noteLayer = context.layers().layer('notes');
- return noteLayer && noteLayer.enabled();
- }
-
- function notesEditable() {
- var mode = context.mode();
- return context.map().notesEditable() && mode && mode.id !== 'save';
- }
-
- context.keybinding().on(mode.key, function () {
- if (!enabled()) return;
-
- if (mode.id === context.mode().id) {
- context.enter(modeBrowse(context));
- } else {
- context.enter(mode);
- }
- });
-
- 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;
- });
- }
- };
-
- 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);
- };
-
- return tool;
- }
-
- 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 isDisabled() {
- return _numChanges === 0 || isSaving();
- }
-
- function save(d3_event) {
- d3_event.preventDefault();
-
- if (!context.inIntro() && !isSaving() && history.hasChanges()) {
- context.enter(modeSave(context));
- }
- }
-
- function bgColor() {
- var step;
-
- 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;
-
- if (tooltipBehavior) {
- tooltipBehavior.title(_t.html(_numChanges > 0 ? 'save.help' : 'save.no_changes')).keys([key]);
- }
-
- if (button) {
- button.classed('disabled', isDisabled()).style('background', bgColor());
- button.select('span.count').html(_numChanges);
- }
- }
-
- 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);
-
- 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'))();
- }
-
- 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 (isSaving()) {
- button.call(tooltipBehavior.hide);
- }
- }
- });
- };
-
- tool.uninstall = function () {
- context.keybinding().off(key, true);
- context.history().on('change.save', null);
- context.on('enter.save', null);
- button = null;
- tooltipBehavior = null;
- };
-
- return tool;
- }
-
- function uiToolSidebarToggle(context) {
- var tool = {
- id: 'sidebar_toggle',
- label: _t.html('toolbar.inspect')
- };
-
- 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 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')
- }];
-
- function editable() {
- return context.mode() && context.mode().id !== 'save' && context.map().editableDataEnabled(true
- /* ignore min zoom */
- );
- }
-
- 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 (editable() && annotation) {
- d.action();
- }
-
- 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)();
- }
-
- 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
- });
-
- 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);
-
- function update() {
- buttons.classed('disabled', function (d) {
- return !editable() || !d.annotation();
- }).each(function () {
- var selection = select(this);
-
- if (!selection.select('.tooltip.in').empty()) {
- selection.call(tooltipBehavior.updateContent);
- }
- });
- }
- };
-
- 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);
- };
-
- return tool;
- }
-
- function uiTopToolbar(context) {
- var sidebarToggle = uiToolSidebarToggle(context),
- modes = uiToolOldDrawModes(context),
- notes = uiToolNotes(context),
- undoRedo = uiToolUndoRedo(context),
- save = uiToolSave(context);
-
- function notesEnabled() {
- var noteLayer = context.layers().layer('notes');
- return noteLayer && noteLayer.enabled();
- }
-
- 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;
- }
- });
-
- var debouncedUpdate = debounce(update, 500, {
- leading: true,
- trailing: true
- });
-
- context.layers().on('change.topToolbar', debouncedUpdate);
- update();
-
- function update() {
- var tools = [sidebarToggle, 'spacer', modes];
- tools.push('spacer');
-
- if (notesEnabled()) {
- tools = tools.concat([notes, 'spacer']);
- }
-
- 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;
- });
- }
- }
-
- return topToolbar;
- }
-
- 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;
- }
-
- 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 (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 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 zoomIn(d3_event) {
- if (d3_event.shiftKey) return;
- d3_event.preventDefault();
- context.map().zoomIn();
- }
-
- function zoomOut(d3_event) {
- if (d3_event.shiftKey) return;
- d3_event.preventDefault();
- context.map().zoomOut();
- }
-
- function zoomInFurther(d3_event) {
- if (d3_event.shiftKey) return;
- d3_event.preventDefault();
- context.map().zoomInFurther();
- }
-
- function zoomOutFurther(d3_event) {
- if (d3_event.shiftKey) return;
- d3_event.preventDefault();
- context.map().zoomOutFurther();
- }
-
- return function (selection) {
- var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === 'rtl' ? 'right' : 'left').title(function (d) {
- if (d.disabled()) {
- return d.disabledTitle;
- }
-
- 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)();
- }
-
- 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);
- });
-
- function updateButtonStates() {
- buttons.classed('disabled', function (d) {
- return d.disabled();
- }).each(function () {
- var selection = select(this);
-
- if (!selection.select('.tooltip.in').empty()) {
- selection.call(tooltipBehavior.updateContent);
- }
- });
- }
-
- updateButtonStates();
- context.map().on('move.uiZoom', updateButtonStates);
- };
- }
-
- function uiZoomToSelection(context) {
- function isDisabled() {
- var mode = context.mode();
- return !mode || !mode.zoomToSelected;
- }
-
- var _lastPointerUpType;
-
- function pointerup(d3_event) {
- _lastPointerUpType = d3_event.pointerType;
- }
-
- function click(d3_event) {
- d3_event.preventDefault();
-
- 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();
-
- if (mode && mode.zoomToSelected) {
- mode.zoomToSelected();
- }
- }
-
- _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');
- }
-
- 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);
- }
- }
-
- context.on('enter.uiZoomToSelection', setEnabledState);
- setEnabledState();
- };
- }
-
- function uiPane(id, context) {
- var _key;
-
- var _label = '';
- var _description = '';
- var _iconName = '';
-
- var _sections; // array of uiSection objects
-
-
- var _paneSelection = select(null);
-
- var _paneTooltip;
-
- var pane = {
- id: id
- };
-
- pane.label = function (val) {
- if (!arguments.length) return _label;
- _label = val;
- return pane;
- };
-
- pane.key = function (val) {
- if (!arguments.length) return _key;
- _key = val;
- return pane;
- };
-
- pane.description = function (val) {
- if (!arguments.length) return _description;
- _description = val;
- return pane;
- };
-
- pane.iconName = function (val) {
- if (!arguments.length) return _iconName;
- _iconName = val;
- return pane;
- };
-
- pane.sections = function (val) {
- if (!arguments.length) return _sections;
- _sections = val;
- return pane;
- };
-
- pane.selection = function () {
- return _paneSelection;
- };
-
- function hidePane() {
- context.ui().togglePanes();
- }
-
- pane.togglePane = function (d3_event) {
- if (d3_event) d3_event.preventDefault();
-
- _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]);
- }
-
- selection.append('button').on('click', pane.togglePane).call(svgIcon('#' + _iconName, 'light')).call(_paneTooltip);
- };
-
- pane.renderContent = function (selection) {
- // override to fully customize content
- if (_sections) {
- _sections.forEach(function (section) {
- selection.call(section.render);
- });
- }
- };
-
- pane.renderPane = function (selection) {
- _paneSelection = selection.append('div').attr('class', 'fillL map-pane hide ' + id + '-pane').attr('pane', id);
-
- var heading = _paneSelection.append('div').attr('class', 'pane-heading');
-
- heading.append('h2').html(_label);
- heading.append('button').on('click', hidePane).call(svgIcon('#iD-icon-close'));
-
- _paneSelection.append('div').attr('class', 'pane-content').call(pane.renderContent);
-
- if (_key) {
- context.keybinding().on(_key, pane.togglePane);
- }
- };
-
- return pane;
- }
-
- function uiSectionBackgroundDisplayOptions(context) {
- var section = uiSection('background-display-options', context).label(_t.html('background.display_options')).disclosureContent(renderDisclosureContent);
-
- var _detected = utilDetect();
-
- var _storedOpacity = corePreferences('background-opacity');
-
- var _minVal = 0;
-
- var _maxVal = _detected.cssfilters ? 3 : 1;
-
- var _sliders = _detected.cssfilters ? ['brightness', 'contrast', 'saturation', 'sharpness'] : ['brightness'];
-
- var _options = {
- brightness: _storedOpacity !== null ? +_storedOpacity : 1,
- contrast: 1,
- saturation: 1,
- sharpness: 1
- };
-
- function clamp(x, min, max) {
- return Math.max(min, Math.min(x, max));
- }
-
- function updateValue(d, val) {
- val = clamp(val, _minVal, _maxVal);
- _options[d] = val;
- context.background()[d](val);
-
- if (d === 'brightness') {
- corePreferences('background-opacity', val);
- }
-
- section.reRender();
- }
-
- 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
-
- 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');
-
- if (!val && d3_event && d3_event.target) {
- val = d3_event.target.value;
- }
-
- 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
-
- 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
-
- container = containerEnter.merge(container);
- container.selectAll('.display-option-input').property('value', function (d) {
- return _options[d];
- });
- 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 (containerEnter.size() && _options.brightness !== 1) {
- context.background().brightness(_options.brightness);
- }
- }
-
- return section;
- }
-
- function uiSettingsCustomBackground() {
- var dispatch$1 = dispatch('change');
-
- 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 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);
-
- function isSaveDisabled() {
- return null;
- } // restore the original template
-
-
- 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 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);
- }
- }
-
- return utilRebind(render, dispatch$1, 'on');
- }
-
- function uiSectionBackgroundList(context) {
- var _backgroundList = select(null);
-
- var _customSource = context.background().findSource('custom');
-
- var _settingsCustomBackground = uiSettingsCustomBackground().on('change', customChanged);
-
- var section = uiSection('background-list', context).label(_t.html('background.backgrounds')).disclosureContent(renderDisclosureContent);
-
- function previousBackgroundID() {
- return corePreferences('background-last-used-toggle');
- }
-
- 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
-
- 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'));
-
- _backgroundList.call(drawListItems, 'radio', function (d3_event, d) {
- chooseBackground(d);
- }, function (d) {
- return !d.isHidden() && !d.overlay;
- });
- }
-
- 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 (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', 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('★');
- layerList.call(updateLayerSelections);
- }
-
- function updateLayerSelections(selection) {
- function active(d) {
- return context.background().showsLayer(d);
- }
-
- selection.selectAll('li').classed('active', active).classed('switch', function (d) {
- return d.id === previousBackgroundID();
- }).call(setTooltips).selectAll('input').property('checked', active);
- }
-
- function chooseBackground(d) {
- if (d.id === 'custom' && !d.template()) {
- return editCustom();
- }
-
- var previousBackground = context.background().baseLayerSource();
- corePreferences('background-last-used-toggle', previousBackground.id);
- corePreferences('background-last-used', d.id);
- context.background().baseLayerSource(d);
- }
-
- function customChanged(d) {
- if (d && d.template) {
- _customSource.template(d.template);
-
- chooseBackground(_customSource);
- } else {
- _customSource.template('');
-
- chooseBackground(context.background().findSource('none'));
- }
- }
-
- function editCustom(d3_event) {
- d3_event.preventDefault();
- context.container().call(_settingsCustomBackground);
- }
-
- 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;
- }
-
- function uiSectionBackgroundOffset(context) {
- var section = uiSection('background-offset', context).label(_t.html('background.fix_misalignment')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
-
- var _pointerPrefix = 'PointerEvent' in window ? 'pointer' : 'mouse';
-
- var _directions = [['top', [0, -0.5]], ['left', [-0.5, 0]], ['right', [0.5, 0]], ['bottom', [0, 0.5]]];
-
- 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;
- });
- }
-
- function resetOffset() {
- context.background().offset([0, 0]);
- updateValue();
- }
-
- function nudge(d) {
- context.background().nudge(d, context.map().zoom());
- updateValue();
- }
-
- 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 (d.length !== 2 || !d[0] || !d[1]) {
- input.classed('error', true);
- return;
- }
-
- context.background().offset(geoMetersToOffset(d));
- updateValue();
- }
-
- 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);
- }
-
- 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);
- }
-
- 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);
- }
- }
-
- 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();
- }
-
- context.background().on('change.backgroundOffset-update', updateValue);
- return section;
- }
-
- function uiSectionOverlayList(context) {
- var section = uiSection('overlay-list', context).label(_t.html('background.overlays')).disclosureContent(renderDisclosureContent);
-
- 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);
-
- if (description || isOverflowing) {
- item.call(uiTooltip().placement(placement).title(description || d.name()));
- }
- });
- }
-
- function updateLayerSelections(selection) {
- function active(d) {
- return context.background().showsLayer(d);
- }
-
- selection.selectAll('li').classed('active', active).call(setTooltips).selectAll('input').property('checked', active);
- }
-
- function chooseOverlay(d3_event, d) {
- d3_event.preventDefault();
- context.background().toggleOverlayLayer(d);
-
- _overlayList.call(updateLayerSelections);
-
- document.activeElement.blur();
- }
-
- 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 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 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);
-
- _overlayList.call(drawListItems, 'checkbox', chooseOverlay, function (d) {
- return !d.isHidden() && d.overlay;
- });
- }
-
- 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;
- }
-
- 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;
- }
-
- 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
-
- 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?
-
- var hhh = depth ? Array(depth + 1).join('#') + ' ' : ''; // if so, prepend with some ##'s
-
- 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');
-
- 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 (rtl) {
- nav.call(drawNext).call(drawPrevious);
- } 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'));
- }
- }
-
- 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 clickWalkthrough(d3_event) {
- d3_event.preventDefault();
- if (context.inIntro()) return;
- context.container().call(uiIntro(context));
- context.ui().togglePanes();
- }
-
- function clickShortcuts(d3_event) {
- d3_event.preventDefault();
- context.container().call(context.ui().shortcuts, true);
- }
-
- 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);
- };
-
- 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
-
-
- function reloadIssues() {
- _issues = context.validator().getIssuesBySeverity(getOptions())[severity];
- }
-
- 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);
- }
-
- 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
-
- items.exit().remove(); // Enter
-
- 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
-
- 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();
- });
- */
- }
-
- 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
-
-
- section.reRender();
- });
- }, 1000));
- return section;
- }
-
- function uiSectionValidationOptions(context) {
- var section = uiSection('issues-options', context).content(renderContent);
-
- 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 getOptions() {
- return {
- what: corePreferences('validate-what') || 'edited',
- // 'all', 'edited'
- where: corePreferences('validate-where') || 'all' // 'all', 'visible'
-
- };
- }
-
- function updateOptionValue(d3_event, d, val) {
- if (!val && d3_event && d3_event.target) {
- val = d3_event.target.value;
- }
-
- corePreferences('validate-' + d, val);
- context.validator().validate();
- }
-
- return section;
- }
-
- function uiSectionValidationRules(context) {
- var MINSQUARE = 0;
- var MAXSQUARE = 20;
- var DEFAULTSQUARE = 5; // see also unsquare_way.js
-
- var section = uiSection('issues-rules', context).disclosureContent(renderDisclosureContent).label(_t.html('issues.rules.title'));
-
- 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;
- });
-
- 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
-
- container = container.merge(containerEnter);
- container.selectAll('.issue-rules-list').call(drawListItems, _ruleKeys, 'checkbox', 'rule', toggleRule, isRuleEnabled);
- }
-
- 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');
-
- if (name === 'rule') {
- enter.call(uiTooltip().title(function (d) {
- return _t.html('issues.' + d + '.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) {
- var params = {};
-
- if (d === 'unsquare_way') {
- params.val = '<span class="square-degrees"></span>';
- }
-
- return _t.html('issues.' + d + '.title', params);
- }); // Update
-
- items = items.merge(enter);
- items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', false); // user-configurable square threshold
-
- var degStr = corePreferences('validate-square-degrees');
-
- 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
-
- degStr = degNum.toString();
- input.property('value', degStr);
- corePreferences('validate-square-degrees', degStr);
- context.validator().reloadUnsquareIssues();
- }
-
- function isRuleEnabled(d) {
- return context.validator().isRuleEnabled(d);
- }
-
- function toggleRule(d3_event, d) {
- context.validator().toggleRule(d);
- }
-
- context.validator().on('validated.uiSectionValidationRules', function () {
- window.requestIdleCallback(section.reRender);
- });
- return section;
- }
-
- function uiSectionValidationStatus(context) {
- var section = uiSection('issues-status', context).content(renderContent).shouldDisplay(function () {
- var issues = context.validator().getIssues(getOptions());
- return issues.length === 0;
- });
-
- function getOptions() {
- return {
- what: corePreferences('validate-what') || 'edited',
- where: corePreferences('validate-where') || 'all'
- };
- }
-
- 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);
- }
-
- 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
-
- resetIgnored.exit().remove(); // enter
-
- var resetIgnoredEnter = resetIgnored.enter().append('div').attr('class', 'reset-ignored section-footer');
- resetIgnoredEnter.append('a').attr('href', '#'); // update
-
- 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();
- });
- }
-
- function setNoIssuesText(selection) {
- var opts = getOptions();
-
- 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'));
- }
-
- var messageType;
-
- 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 (opts.what === 'edited' && context.history().difference().summary().length === 0) {
- messageType = 'no_edits';
- }
-
- selection.select('.box .message').html(_t.html('issues.no_issues.message.' + messageType));
- }
-
- 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 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;
- }
-
- function uiSettingsCustomData(context) {
- var dispatch$1 = dispatch('change');
-
- function render(selection) {
- var dataLayer = context.layers().layer('data'); // keep separate copies of original and current settings
-
- 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';
-
- 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 (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
-
- 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);
-
- function isSaveDisabled() {
- return null;
- } // restore the original url
-
-
- 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
-
-
- function clickSave() {
- _currSettings.url = textSection.select('.field-url').property('value').trim(); // one or the other but not both
-
- if (_currSettings.url) {
- _currSettings.fileList = null;
- }
-
- if (_currSettings.fileList) {
- _currSettings.url = '';
- }
-
- corePreferences('settings-custom-data-url', _currSettings.url);
- this.blur();
- modal.close();
- dispatch$1.call('change', this, _currSettings);
- }
- }
-
- return utilRebind(render, dispatch$1, 'on');
- }
-
- 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);
-
- 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);
- }
-
- function showsLayer(which) {
- var layer = layers.layer(which);
-
- if (layer) {
- return layer.enabled();
- }
-
- return false;
- }
-
- 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);
-
- if (!enabled && (which === 'osm' || which === 'notes')) {
- context.enter(modeBrowse(context));
- }
- }
- }
-
- function toggleLayer(which) {
- setLayer(which, !showsLayer(which));
- }
-
- 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
-
- li.merge(liEnter).classed('active', function (d) {
- return d.layer.enabled();
- }).selectAll('input').property('checked', function (d) {
- return d.layer.enabled();
- });
- }
-
- 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
-
- 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
-
-
- 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..
-
- 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
-
- li.merge(liEnter).classed('active', isVTLayerSelected).selectAll('input').property('checked', isVTLayerSelected);
-
- function isVTLayerSelected(d) {
- return dataLayer && dataLayer.template() === d.template;
- }
-
- function selectVTLayer(d3_event, d) {
- corePreferences('settings-custom-data-url', d.template);
-
- if (dataLayer) {
- dataLayer.template(d.template, d.src);
- dataLayer.enabled(true);
- }
- }
- }
-
- 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
-
- ul.exit().remove(); // Enter
-
- 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', 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
-
- 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);
- }
-
- function editCustom(d3_event) {
- d3_event.preventDefault();
- context.container().call(settingsCustomData);
- }
-
- function customChanged(d) {
- var dataLayer = layers.layer('data');
-
- if (d && d.url) {
- dataLayer.url(d.url);
- } else if (d && d.fileList) {
- dataLayer.fileList(d.fileList);
- }
- }
-
- 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'));
- }
-
- 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;
- }
-
- function uiSectionMapFeatures(context) {
- var _features = context.features().keys();
-
- 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
-
- container = container.merge(containerEnter);
- container.selectAll('.layer-feature-list').call(drawListItems, _features, 'checkbox', 'feature', clickFeature, showsFeature);
- }
-
- 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').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 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
-
- items = items.merge(enter);
- items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', autoHiddenFeature);
- }
-
- function autoHiddenFeature(d) {
- return context.features().autoHidden(d);
- }
-
- function showsFeature(d) {
- return context.features().enabled(d);
- }
-
- function clickFeature(d3_event, d) {
- context.features().toggle(d);
- }
-
- function showsLayer(id) {
- var layer = context.layers().layer(id);
- return layer && layer.enabled();
- } // add listeners
-
-
- context.features().on('change.map_features', section.reRender);
- return section;
- }
-
- function uiSectionMapStyleOptions(context) {
- var section = uiSection('fill-area', context).label(_t.html('map_data.style_options')).disclosureContent(renderDisclosureContent).expandedByDefault(false);
-
- 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');
- });
- }
-
- 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').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
-
- items = items.merge(enter);
- items.classed('active', active).selectAll('input').property('checked', active).property('indeterminate', false);
- }
-
- function isActiveFill(d) {
- return context.map().activeAreaFill() === d;
- }
-
- function toggleHighlightEdited(d3_event) {
- d3_event.preventDefault();
- context.map().toggleHighlightEdited();
- }
-
- function setFill(d3_event, d) {
- context.map().activeAreaFill(d);
- }
-
- context.map().on('changeHighlighting.ui_style, changeAreaFill.ui_style', section.reRender);
- return section;
- }
-
- 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 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();
- }
-
- function layerEnabled(d) {
- return layerSupported(d) && d.layer.enabled();
- }
-
- 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';
- }
-
- 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);
- }
-
- function drawPhotoTypeItems(selection) {
- var data = context.photos().allPhotoTypes();
-
- function typeEnabled(d) {
- return context.photos().showsPhotoType(d);
- }
-
- 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
-
- li.merge(liEnter).classed('active', typeEnabled).selectAll('input').property('checked', typeEnabled);
- }
-
- function drawDateFilter(selection) {
- var data = context.photos().dateFilters();
-
- function filterEnabled(d) {
- return context.photos().dateFilterValue(d);
- }
-
- 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) || '');
- });
- });
- 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);
- });
- li.merge(liEnter).classed('active', filterEnabled);
-
- function usernameValue() {
- var usernames = context.photos().usernames();
- if (usernames) return usernames.join('; ');
- return usernames;
- }
- }
-
- function toggleLayer(which) {
- setLayer(which, !showsLayer(which));
- }
-
- function showsLayer(which) {
- var layer = layers.layer(which);
-
- if (layer) {
- return layer.enabled();
- }
-
- return false;
- }
-
- function setLayer(which, enabled) {
- var layer = layers.layer(which);
-
- if (layer) {
- layer.enabled(enabled);
- }
- }
-
- 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();
- });
- 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');
- }
- }
-
- return section;
- }
-
- 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;
- }
-
- 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');
- });
- 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();
- });
- }
-
- if ('PointerEvent' in window) {
- select(window).on('pointerdown.ui pointerup.ui', function (d3_event) {
- var pointerType = d3_event.pointerType || 'mouse';
-
- 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.call(uiFullScreen(context));
- var map = context.map();
- map.redrawEnable(false); // don't draw until we've set zoom/lat/long
-
- 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
-
- 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
-
- 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();
-
- if (apiConnections && apiConnections.length > 1) {
- aboutList.append('li').attr('class', 'source-switch').call(uiSourceSwitch(context).keys(apiConnections));
- }
-
- 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));
-
- 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.
-
-
- ui.onResize();
- map.redrawEnable(true);
- ui.hash = behaviorHash(context);
- ui.hash();
-
- if (!ui.hash.hadHash) {
- map.centerZoom([0, 0], 2);
- } // Bind events
-
-
- window.onbeforeunload = function () {
- return context.save();
- };
-
- window.onunload = function () {
- context.history().unlock();
- };
-
- 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 previousBackground = context.background().findSource(corePreferences('background-last-used-toggle'));
-
- 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 mode = context.mode();
- if (mode && /^draw/.test(mode.id)) return;
- var layer = context.layers().layer('osm');
-
- if (layer) {
- layer.enabled(!layer.enabled());
-
- 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);
- }
-
- var osm = context.connection();
- var auth = uiLoading(context).message(_t.html('loading_auth')).blocking(true);
-
- if (osm && auth) {
- osm.on('authLoading.ui', function () {
- context.container().call(auth);
- }).on('authDone.ui', function () {
- auth.close();
- });
- }
-
- _initCounter++;
-
- if (ui.hash.startWalkthrough) {
- ui.hash.startWalkthrough = false;
- context.container().call(uiIntro(context));
- }
-
- 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);
- };
- }
- }
-
- var ui = {};
-
- var _loadPromise; // renders the iD interface into the container node
-
-
- 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.
-
-
- ui.restart = function () {
- context.keybinding().clear();
- _loadPromise = null;
- context.container().selectAll('*').remove();
- ui.ensureLoaded();
- };
-
- ui.lastPointerType = function () {
- return _lastPointerType;
- };
-
- ui.svgDefs = svgDefs(context);
- ui.flash = uiFlash(context);
- ui.sidebar = uiSidebar(context);
- ui.photoviewer = uiPhotoviewer(context);
- ui.shortcuts = uiShortcuts(context);
-
- 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.
-
- var mapDimensions = utilGetDimensions(context.container().select('.main-content'), true);
- utilGetDimensions(context.container().select('.sidebar'), true);
-
- if (withPan !== undefined) {
- map.redrawEnable(false);
- map.pan(withPan);
- map.redrawEnable(true);
- }
-
- map.dimensions(mapDimensions);
- ui.photoviewer.onMapResize(); // check if header or footer have overflowed
-
- 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];
- }
-
- 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;
-
- if (scrollWidth > clientWidth) {
- // overflow happening
- selection.classed('narrow', true);
-
- if (!_needWidth[selector]) {
- _needWidth[selector] = scrollWidth;
- }
- } else if (scrollWidth >= needed) {
- selection.classed('narrow', false);
- }
- };
-
- 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);
-
- 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);
-
- 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);
- });
- }
- };
-
- 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();
- }
-
- operations.forEach(function (operation) {
- if (operation.point) operation.point(anchorPoint);
- });
-
- _editMenu.anchorLoc(anchorPoint).triggerType(triggerType).operations(operations); // render the menu
-
-
- context.map().supersurface.call(_editMenu);
- };
-
- ui.closeEditMenu = function () {
- // remove any existing menu no matter how it was added
- context.map().supersurface.select('.edit-menu').remove();
- };
-
- var _saveLoading = select(null);
-
- 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();
-
- _saveLoading = select(null);
- });
- return ui;
- }
-
- function coreContext() {
- var _this = this;
-
- var dispatch$1 = dispatch('enter', 'exit', 'change');
- var context = utilRebind({}, dispatch$1, 'on');
-
- var _deferred = new Set();
-
- context.version = '2.19.3';
- context.privacyVersion = '20200407'; // iD will alter the hash so cache the parameters intended to setup the session
-
- context.initialHashParams = window.location.hash ? utilStringQs(window.location.hash) : {};
- context.isFirstSession = !corePreferences('sawSplash') && !corePreferences('sawPrivacyVersion');
- /* Changeset */
- // An osmChangeset object. Not loaded until needed.
-
- context.changeset = null;
- var _defaultChangesetComment = context.initialHashParams.comment;
- var _defaultChangesetSource = context.initialHashParams.source;
- var _defaultChangesetHashtags = context.initialHashParams.hashtags;
-
- context.defaultChangesetComment = function (val) {
- if (!arguments.length) return _defaultChangesetComment;
- _defaultChangesetComment = val;
- return context;
- };
-
- context.defaultChangesetSource = function (val) {
- if (!arguments.length) return _defaultChangesetSource;
- _defaultChangesetSource = val;
- return context;
- };
-
- context.defaultChangesetHashtags = function (val) {
- if (!arguments.length) return _defaultChangesetHashtags;
- _defaultChangesetHashtags = val;
- return context;
- };
- /* Document title */
-
- /* (typically shown as the label for the browser window/tab) */
- // If true, iD will update the title based on what the user is doing
-
-
- var _setsDocumentTitle = true;
-
- context.setsDocumentTitle = function (val) {
- if (!arguments.length) return _setsDocumentTitle;
- _setsDocumentTitle = val;
- return context;
- }; // The part of the title that is always the same
-
-
- var _documentTitleBase = document.title;
-
- context.documentTitleBase = function (val) {
- if (!arguments.length) return _documentTitleBase;
- _documentTitleBase = val;
- return context;
- };
- /* User interface and keybinding */
-
-
- var _ui;
-
- context.ui = function () {
- return _ui;
- };
-
- context.lastPointerType = function () {
- return _ui.lastPointerType();
- };
-
- var _keybinding = utilKeybinding('context');
-
- context.keybinding = function () {
- return _keybinding;
- };
-
- 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;
-
- var _history;
-
- var _validator;
-
- var _uploader;
-
- context.connection = function () {
- return _connection;
- };
-
- context.history = function () {
- return _history;
- };
-
- context.validator = function () {
- return _validator;
- };
-
- context.uploader = function () {
- return _uploader;
- };
- /* Connection */
-
-
- context.preauth = function (options) {
- if (_connection) {
- _connection["switch"](options);
- }
-
- return context;
- };
- /* connection options for source switcher (optional) */
-
-
- var _apiConnections;
-
- 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
-
-
- 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();
- }
- }
-
- if (typeof callback === 'function') {
- callback(err);
- }
-
- return;
- } else if (_connection && _connection.getConnectionId() !== cid) {
- if (typeof callback === 'function') {
- callback({
- message: 'Connection Switched',
- status: -1
- });
- }
-
- return;
- } else {
- _history.merge(result.data, result.extent);
-
- if (typeof callback === 'function') {
- callback(err, result);
- }
-
- return;
- }
- };
- }
-
- 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));
- }
- });
-
- _deferred.add(handle);
- };
-
- context.loadTileAtLoc = function (loc, callback) {
- var handle = window.requestIdleCallback(function () {
- _deferred["delete"](handle);
-
- if (_connection && context.editableDataEnabled()) {
- var cid = _connection.getConnectionId();
-
- _connection.loadTileAtLoc(loc, afterLoad(cid, callback));
- }
- });
-
- _deferred.add(handle);
- };
-
- context.loadEntity = function (entityID, callback) {
- if (_connection) {
- var cid = _connection.getConnectionId();
-
- _connection.loadEntity(entityID, afterLoad(cid, callback));
- }
- };
-
- context.zoomToEntity = function (entityID, zoomTo) {
- if (zoomTo !== false) {
- context.loadEntity(entityID, function (err, result) {
- if (err) return;
- var entity = result.data.find(function (e) {
- return e.id === entityID;
- });
-
- if (entity) {
- _map.zoomTo(entity);
- }
- });
- }
-
- _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]));
- });
-
- context.on('enter.zoomToEntity', function () {
- if (_mode.id !== 'browse') {
- _map.on('drawn.zoomToEntity', null);
-
- context.on('enter.zoomToEntity', null);
- }
- });
- };
-
- var _minEditableZoom = 16;
-
- context.minEditableZoom = function (val) {
- if (!arguments.length) return _minEditableZoom;
- _minEditableZoom = val;
-
- if (_connection) {
- _connection.tileZoom(val);
- }
-
- return context;
- }; // String length limits in Unicode characters, not JavaScript UTF-16 code units
-
-
- context.maxCharsForTagKey = function () {
- return 255;
- };
-
- context.maxCharsForTagValue = function () {
- return 255;
- };
-
- context.maxCharsForRelationRole = function () {
- return 255;
- };
-
- function cleanOsmString(val, maxChars) {
- // be lenient with input
- if (val === undefined || val === null) {
- val = '';
- } else {
- val = val.toString();
- } // remove whitespace
-
-
- val = val.trim(); // use the canonical form of the string
-
- if (val.normalize) val = val.normalize('NFC'); // trim to the number of allowed characters
-
- return utilUnicodeCharsTruncated(val, maxChars);
- }
-
- context.cleanTagKey = function (val) {
- return cleanOsmString(val, context.maxCharsForTagKey());
- };
-
- context.cleanTagValue = function (val) {
- return cleanOsmString(val, context.maxCharsForTagValue());
- };
-
- context.cleanRelationRole = function (val) {
- return cleanOsmString(val, context.maxCharsForRelationRole());
- };
- /* History */
-
-
- var _inIntro = 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
-
-
- context.save = function () {
- // no history save, no message onbeforeunload
- if (_inIntro || context.container().select('.modal').size()) return;
- var canSave;
-
- if (_mode && _mode.id === 'save') {
- canSave = false; // Attempt to prevent user from creating duplicate changes - see #5200
-
- if (services.osm && services.osm.isChangesetInflight()) {
- _history.clearSaved();
-
- return;
- }
- } else {
- canSave = context.selectedIDs().every(function (id) {
- var entity = context.hasEntity(id);
- return entity && !entity.isDegenerate();
- });
- }
-
- if (canSave) {
- _history.save();
- }
-
- 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).
-
-
- context.debouncedSave = debounce(context.save, 350);
-
- function withDebouncedSave(fn) {
- return function () {
- var result = fn.apply(_history, arguments);
- context.debouncedSave();
- return result;
- };
- }
- /* Graph */
-
-
- context.hasEntity = function (id) {
- return _history.graph().hasEntity(id);
- };
-
- context.entity = function (id) {
- return _history.graph().entity(id);
- };
- /* Modes */
-
-
- var _mode;
-
- context.mode = function () {
- return _mode;
- };
-
- context.enter = function (newMode) {
- if (_mode) {
- _mode.exit();
-
- dispatch$1.call('exit', _this, _mode);
- }
-
- _mode = newMode;
-
- _mode.enter();
-
- dispatch$1.call('enter', _this, _mode);
- };
-
- context.selectedIDs = function () {
- return _mode && _mode.selectedIDs && _mode.selectedIDs() || [];
- };
-
- context.activeID = function () {
- return _mode && _mode.activeID && _mode.activeID();
- };
-
- var _selectedNoteID;
-
- 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
-
-
- var _selectedErrorID;
-
- context.selectedErrorID = function (errorID) {
- if (!arguments.length) return _selectedErrorID;
- _selectedErrorID = errorID;
- return context;
- };
- /* Behaviors */
-
-
- context.install = function (behavior) {
- return context.surface().call(behavior);
- };
-
- context.uninstall = function (behavior) {
- return context.surface().call(behavior.off);
- };
- /* Copy/Paste */
-
-
- var _copyGraph;
-
- context.copyGraph = function () {
- return _copyGraph;
- };
-
- var _copyIDs = [];
-
- context.copyIDs = function (val) {
- if (!arguments.length) return _copyIDs;
- _copyIDs = val;
- _copyGraph = _history.graph();
- return context;
- };
-
- var _copyLonLat;
-
- context.copyLonLat = function (val) {
- if (!arguments.length) return _copyLonLat;
- _copyLonLat = val;
- return context;
- };
- /* Background */
-
-
- var _background;
-
- context.background = function () {
- return _background;
- };
- /* Features */
-
-
- var _features;
-
- context.features = function () {
- return _features;
- };
-
- context.hasHiddenConnections = function (id) {
- var graph = _history.graph();
-
- var entity = graph.entity(id);
- return _features.hasHiddenConnections(entity, graph);
- };
- /* Photos */
-
-
- var _photos;
-
- context.photos = function () {
- return _photos;
- };
- /* Map */
-
-
- var _map;
-
- context.map = function () {
- return _map;
- };
-
- context.layers = function () {
- return _map.layers();
- };
-
- context.surface = function () {
- return _map.surface;
- };
-
- context.editableDataEnabled = function () {
- return _map.editableDataEnabled();
- };
-
- context.surfaceRect = function () {
- return _map.surface.node().getBoundingClientRect();
- };
-
- context.editable = function () {
- // don't allow editing during save
- var mode = context.mode();
- if (!mode || mode.id === 'save') return false;
- return _map.editableDataEnabled();
- };
- /* Debug */
-
-
- 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
-
- };
-
- context.debugFlags = function () {
- return _debugFlags;
- };
-
- context.getDebug = function (flag) {
- return flag && _debugFlags[flag];
- };
-
- context.setDebug = function (flag, val) {
- if (arguments.length === 1) val = true;
- _debugFlags[flag] = val;
- dispatch$1.call('change');
- return context;
- };
- /* Container */
-
-
- var _container = select(null);
-
- context.container = function (val) {
- if (!arguments.length) return _container;
- _container = val;
-
- _container.classed('ideditor', true);
-
- return context;
- };
-
- context.containerNode = function (val) {
- if (!arguments.length) return context.container().node();
- context.container(select(val));
- return context;
- };
-
- var _embed;
-
- context.embed = function (val) {
- if (!arguments.length) return _embed;
- _embed = val;
- return context;
- };
- /* Assets */
-
-
- var _assetPath = '';
-
- context.assetPath = function (val) {
- if (!arguments.length) return _assetPath;
- _assetPath = val;
- _mainFileFetcher.assetPath(val);
- return context;
- };
-
- var _assetMap = {};
-
- context.assetMap = function (val) {
- if (!arguments.length) return _assetMap;
- _assetMap = val;
- _mainFileFetcher.assetMap(val);
- return context;
- };
-
- context.asset = function (val) {
- if (/^http(s)?:\/\//i.test(val)) return val;
- var filename = _assetPath + val;
- return _assetMap[filename] || filename;
- };
-
- context.imagePath = function (val) {
- return context.asset("img/".concat(val));
- };
- /* reset (aka flush) */
-
-
- 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;
-
- _validator.reset();
-
- _features.reset();
-
- _history.reset();
-
- _uploader.reset(); // don't leave stale state in the inspector
-
-
- context.container().select('.inspector-wrap *').remove();
- return context;
- };
- /* Projections */
-
-
- context.projection = geoRawMercator();
- context.curtainProjection = geoRawMercator();
- /* Init */
-
- 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.
-
- 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.
-
-
- function initializeDependents() {
- if (context.initialHashParams.presets) {
- _mainPresetIndex.addablePresetIDs(new Set(context.initialHashParams.presets.split(',')));
- }
-
- if (context.initialHashParams.locale) {
- _mainLocalizer.preferredLocaleCodes(context.initialHashParams.locale);
- } // kick off some async work
-
-
- _mainLocalizer.ensureLoaded();
-
- _background.ensureLoaded();
-
- _mainPresetIndex.ensureLoaded();
- Object.values(services).forEach(function (service) {
- if (service && typeof service.init === 'function') {
- service.init();
- }
- });
-
- _map.init();
-
- _validator.init();
-
- _features.init();
-
- 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 (!context.container().empty()) {
- _ui.ensureLoaded().then(function () {
- _photos.init();
- });
- }
- }
- };
-
- return context;
- }
-
- // 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,
- geoMercator: mercator,
- geoProjection: projection,
- polygonArea: d3_polygonArea,
- polygonCentroid: d3_polygonCentroid,
- select: select,
- selectAll: selectAll,
- timerFlush: timerFlush
- };
-
- var iD = /*#__PURE__*/Object.freeze({
- __proto__: null,
- debug: debug,
- d3: d3,
- actionAddEntity: actionAddEntity,
- actionAddMember: actionAddMember,
- actionAddMidpoint: actionAddMidpoint,
- actionAddVertex: actionAddVertex,
- actionChangeMember: actionChangeMember,
- actionChangePreset: actionChangePreset,
- actionChangeTags: actionChangeTags,
- actionCircularize: actionCircularize,
- actionConnect: actionConnect,
- actionCopyEntities: actionCopyEntities,
- actionDeleteMember: actionDeleteMember,
- actionDeleteMultiple: actionDeleteMultiple,
- actionDeleteNode: actionDeleteNode,
- actionDeleteRelation: actionDeleteRelation,
- actionDeleteWay: actionDeleteWay,
- actionDiscardTags: actionDiscardTags,
- actionDisconnect: actionDisconnect,
- actionExtract: actionExtract,
- actionJoin: actionJoin,
- actionMerge: actionMerge,
- actionMergeNodes: actionMergeNodes,
- actionMergePolygon: actionMergePolygon,
- actionMergeRemoteChanges: actionMergeRemoteChanges,
- actionMove: actionMove,
- actionMoveMember: actionMoveMember,
- actionMoveNode: actionMoveNode,
- actionNoop: actionNoop,
- actionOrthogonalize: actionOrthogonalize,
- actionRestrictTurn: actionRestrictTurn,
- actionReverse: actionReverse,
- actionRevert: actionRevert,
- actionRotate: actionRotate,
- actionScale: actionScale,
- actionSplit: actionSplit,
- actionStraightenNodes: actionStraightenNodes,
- actionStraightenWay: actionStraightenWay,
- actionUnrestrictTurn: actionUnrestrictTurn,
- actionReflect: actionReflect,
- actionUpgradeTags: actionUpgradeTags,
- behaviorAddWay: behaviorAddWay,
- behaviorBreathe: behaviorBreathe,
- behaviorDrag: behaviorDrag,
- behaviorDrawWay: behaviorDrawWay,
- behaviorDraw: behaviorDraw,
- behaviorEdit: behaviorEdit,
- behaviorHash: behaviorHash,
- behaviorHover: behaviorHover,
- behaviorLasso: behaviorLasso,
- behaviorOperation: behaviorOperation,
- behaviorPaste: behaviorPaste,
- behaviorSelect: behaviorSelect,
- coreContext: coreContext,
- coreFileFetcher: coreFileFetcher,
- fileFetcher: _mainFileFetcher,
- coreDifference: coreDifference,
- coreGraph: coreGraph,
- coreHistory: coreHistory,
- coreLocalizer: coreLocalizer,
- t: _t,
- localizer: _mainLocalizer,
- prefs: corePreferences,
- coreTree: coreTree,
- coreUploader: coreUploader,
- coreValidator: coreValidator,
- geoExtent: geoExtent,
- geoLatToMeters: geoLatToMeters,
- geoLonToMeters: geoLonToMeters,
- geoMetersToLat: geoMetersToLat,
- geoMetersToLon: geoMetersToLon,
- geoMetersToOffset: geoMetersToOffset,
- geoOffsetToMeters: geoOffsetToMeters,
- geoScaleToZoom: geoScaleToZoom,
- geoSphericalClosestNode: geoSphericalClosestNode,
- geoSphericalDistance: geoSphericalDistance,
- geoZoomToScale: geoZoomToScale,
- geoAngle: geoAngle,
- geoChooseEdge: geoChooseEdge,
- geoEdgeEqual: geoEdgeEqual,
- geoGetSmallestSurroundingRectangle: geoGetSmallestSurroundingRectangle,
- geoHasLineIntersections: geoHasLineIntersections,
- geoHasSelfIntersections: geoHasSelfIntersections,
- geoRotate: geoRotate,
- geoLineIntersection: geoLineIntersection,
- geoPathHasIntersections: geoPathHasIntersections,
- geoPathIntersections: geoPathIntersections,
- geoPathLength: geoPathLength,
- geoPointInPolygon: geoPointInPolygon,
- geoPolygonContainsPolygon: geoPolygonContainsPolygon,
- geoPolygonIntersectsPolygon: geoPolygonIntersectsPolygon,
- geoViewportEdge: geoViewportEdge,
- geoRawMercator: geoRawMercator,
- geoVecAdd: geoVecAdd,
- geoVecAngle: geoVecAngle,
- geoVecCross: geoVecCross,
- geoVecDot: geoVecDot,
- geoVecEqual: geoVecEqual,
- geoVecFloor: geoVecFloor,
- geoVecInterp: geoVecInterp,
- geoVecLength: geoVecLength,
- geoVecLengthSquare: geoVecLengthSquare,
- geoVecNormalize: geoVecNormalize,
- geoVecNormalizedDot: geoVecNormalizedDot,
- geoVecProject: geoVecProject,
- geoVecSubtract: geoVecSubtract,
- geoVecScale: geoVecScale,
- geoOrthoNormalizedDotProduct: geoOrthoNormalizedDotProduct,
- geoOrthoCalcScore: geoOrthoCalcScore,
- geoOrthoMaxOffsetAngle: geoOrthoMaxOffsetAngle,
- geoOrthoCanOrthogonalize: geoOrthoCanOrthogonalize,
- modeAddArea: modeAddArea,
- modeAddLine: modeAddLine,
- modeAddPoint: modeAddPoint,
- modeAddNote: modeAddNote,
- modeBrowse: modeBrowse,
- modeDragNode: modeDragNode,
- modeDragNote: modeDragNote,
- modeDrawArea: modeDrawArea,
- modeDrawLine: modeDrawLine,
- modeMove: modeMove,
- modeRotate: modeRotate,
- modeSave: modeSave,
- modeSelect: modeSelect,
- modeSelectData: modeSelectData,
- modeSelectError: modeSelectError,
- modeSelectNote: modeSelectNote,
- 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,
- osmChangeset: osmChangeset,
- osmEntity: osmEntity,
- osmNode: osmNode,
- osmNote: osmNote,
- osmRelation: osmRelation,
- osmWay: osmWay,
- QAItem: QAItem,
- osmIntersection: osmIntersection,
- osmTurn: osmTurn,
- osmInferRestriction: osmInferRestriction,
- osmLanes: osmLanes,
- osmOldMultipolygonOuterMemberOfRelation: osmOldMultipolygonOuterMemberOfRelation,
- osmIsOldMultipolygonOuterMember: osmIsOldMultipolygonOuterMember,
- osmOldMultipolygonOuterMember: osmOldMultipolygonOuterMember,
- osmJoinWays: osmJoinWays,
- get osmAreaKeys () { return osmAreaKeys; },
- osmSetAreaKeys: osmSetAreaKeys,
- osmTagSuggestingArea: osmTagSuggestingArea,
- get osmPointTags () { return osmPointTags; },
- osmSetPointTags: osmSetPointTags,
- get osmVertexTags () { return osmVertexTags; },
- osmSetVertexTags: osmSetVertexTags,
- osmNodeGeometriesForTags: osmNodeGeometriesForTags,
- osmOneWayTags: osmOneWayTags,
- osmPavedTags: osmPavedTags,
- osmIsInterestingTag: osmIsInterestingTag,
- osmRoutableHighwayTagValues: osmRoutableHighwayTagValues,
- osmFlowingWaterwayTagValues: osmFlowingWaterwayTagValues,
- osmRailwayTrackTagValues: osmRailwayTrackTagValues,
- presetCategory: presetCategory,
- presetCollection: presetCollection,
- presetField: presetField,
- presetPreset: presetPreset,
- presetManager: _mainPresetIndex,
- presetIndex: presetIndex,
- rendererBackgroundSource: rendererBackgroundSource,
- rendererBackground: rendererBackground,
- rendererFeatures: rendererFeatures,
- rendererMap: rendererMap,
- rendererPhotos: rendererPhotos,
- rendererTileLayer: rendererTileLayer,
- services: services,
- serviceKeepRight: serviceKeepRight,
- serviceImproveOSM: serviceImproveOSM,
- serviceOsmose: serviceOsmose,
- serviceMapillary: serviceMapillary,
- serviceMapRules: serviceMapRules,
- serviceNominatim: serviceNominatim,
- serviceOpenstreetcam: serviceOpenstreetcam,
- serviceOsm: serviceOsm,
- serviceOsmWikibase: serviceOsmWikibase,
- serviceStreetside: serviceStreetside,
- serviceTaginfo: serviceTaginfo,
- serviceVectorTile: serviceVectorTile,
- serviceWikidata: serviceWikidata,
- serviceWikipedia: serviceWikipedia,
- svgAreas: svgAreas,
- svgData: svgData,
- svgDebug: svgDebug,
- svgDefs: svgDefs,
- svgKeepRight: svgKeepRight,
- svgIcon: svgIcon,
- svgGeolocate: svgGeolocate,
- svgLabels: svgLabels,
- svgLayers: svgLayers,
- svgLines: svgLines,
- svgMapillaryImages: svgMapillaryImages,
- svgMapillarySigns: svgMapillarySigns,
- svgMidpoints: svgMidpoints,
- svgNotes: svgNotes,
- svgMarkerSegments: svgMarkerSegments,
- svgOpenstreetcamImages: svgOpenstreetcamImages,
- svgOsm: svgOsm,
- svgPassiveVertex: svgPassiveVertex,
- svgPath: svgPath,
- svgPointTransform: svgPointTransform,
- svgPoints: svgPoints,
- svgRelationMemberTags: svgRelationMemberTags,
- svgSegmentWay: svgSegmentWay,
- svgStreetside: svgStreetside,
- svgTagClasses: svgTagClasses,
- svgTagPattern: svgTagPattern,
- svgTouch: svgTouch,
- svgTurns: svgTurns,
- svgVertices: svgVertices,
- uiFieldDefaultCheck: uiFieldCheck,
- uiFieldOnewayCheck: uiFieldCheck,
- uiFieldCheck: uiFieldCheck,
- uiFieldManyCombo: uiFieldCombo,
- uiFieldMultiCombo: uiFieldCombo,
- uiFieldNetworkCombo: uiFieldCombo,
- uiFieldSemiCombo: uiFieldCombo,
- uiFieldTypeCombo: uiFieldCombo,
- uiFieldCombo: uiFieldCombo,
- uiFieldUrl: uiFieldText,
- uiFieldIdentifier: uiFieldText,
- uiFieldNumber: uiFieldText,
- uiFieldTel: uiFieldText,
- uiFieldEmail: uiFieldText,
- uiFieldText: uiFieldText,
- uiFieldAccess: uiFieldAccess,
- uiFieldAddress: uiFieldAddress,
- uiFieldCycleway: uiFieldCycleway,
- uiFieldLanes: uiFieldLanes,
- uiFieldLocalized: uiFieldLocalized,
- uiFieldMaxspeed: uiFieldMaxspeed,
- uiFieldStructureRadio: uiFieldRadio,
- uiFieldRadio: uiFieldRadio,
- uiFieldRestrictions: uiFieldRestrictions,
- uiFieldTextarea: uiFieldTextarea,
- uiFieldWikidata: uiFieldWikidata,
- uiFieldWikipedia: uiFieldWikipedia,
- uiFields: uiFields,
- uiIntro: uiIntro,
- uiPanelBackground: uiPanelBackground,
- uiPanelHistory: uiPanelHistory,
- uiPanelLocation: uiPanelLocation,
- uiPanelMeasurement: uiPanelMeasurement,
- uiInfoPanels: uiInfoPanels,
- uiPaneBackground: uiPaneBackground,
- uiPaneHelp: uiPaneHelp,
- uiPaneIssues: uiPaneIssues,
- uiPaneMapData: uiPaneMapData,
- uiPanePreferences: uiPanePreferences,
- uiSectionBackgroundDisplayOptions: uiSectionBackgroundDisplayOptions,
- uiSectionBackgroundList: uiSectionBackgroundList,
- uiSectionBackgroundOffset: uiSectionBackgroundOffset,
- uiSectionChanges: uiSectionChanges,
- uiSectionDataLayers: uiSectionDataLayers,
- uiSectionEntityIssues: uiSectionEntityIssues,
- uiSectionFeatureType: uiSectionFeatureType,
- uiSectionMapFeatures: uiSectionMapFeatures,
- uiSectionMapStyleOptions: uiSectionMapStyleOptions,
- uiSectionOverlayList: uiSectionOverlayList,
- uiSectionPhotoOverlays: uiSectionPhotoOverlays,
- uiSectionPresetFields: uiSectionPresetFields,
- uiSectionPrivacy: uiSectionPrivacy,
- uiSectionRawMemberEditor: uiSectionRawMemberEditor,
- uiSectionRawMembershipEditor: uiSectionRawMembershipEditor,
- uiSectionRawTagEditor: uiSectionRawTagEditor,
- uiSectionSelectionList: uiSectionSelectionList,
- uiSectionValidationIssues: uiSectionValidationIssues,
- uiSectionValidationOptions: uiSectionValidationOptions,
- uiSectionValidationRules: uiSectionValidationRules,
- uiSectionValidationStatus: uiSectionValidationStatus,
- uiSettingsCustomBackground: uiSettingsCustomBackground,
- uiSettingsCustomData: uiSettingsCustomData,
- uiInit: uiInit,
- uiAccount: uiAccount,
- uiAttribution: uiAttribution,
- uiChangesetEditor: uiChangesetEditor,
- uiCmd: uiCmd,
- uiCombobox: uiCombobox,
- uiCommit: uiCommit,
- uiCommitWarnings: uiCommitWarnings,
- uiConfirm: uiConfirm,
- uiConflicts: uiConflicts,
- uiContributors: uiContributors,
- uiCurtain: uiCurtain,
- uiDataEditor: uiDataEditor,
- uiDataHeader: uiDataHeader,
- uiDisclosure: uiDisclosure,
- uiEditMenu: uiEditMenu,
- uiEntityEditor: uiEntityEditor,
- uiFeatureInfo: uiFeatureInfo,
- uiFeatureList: uiFeatureList,
- uiField: uiField,
- uiFieldHelp: uiFieldHelp,
- uiFlash: uiFlash,
- uiFormFields: uiFormFields,
- uiFullScreen: uiFullScreen,
- uiGeolocate: uiGeolocate,
- uiImproveOsmComments: uiImproveOsmComments,
- uiImproveOsmDetails: uiImproveOsmDetails,
- uiImproveOsmEditor: uiImproveOsmEditor,
- uiImproveOsmHeader: uiImproveOsmHeader,
- uiInfo: uiInfo,
- uiInspector: uiInspector,
- uiIssuesInfo: uiIssuesInfo,
- uiKeepRightDetails: uiKeepRightDetails,
- uiKeepRightEditor: uiKeepRightEditor,
- uiKeepRightHeader: uiKeepRightHeader,
- uiLasso: uiLasso,
- uiLoading: uiLoading,
- uiMapInMap: uiMapInMap,
- uiModal: uiModal,
- uiNotice: uiNotice,
- uiNoteComments: uiNoteComments,
- uiNoteEditor: uiNoteEditor,
- uiNoteHeader: uiNoteHeader,
- uiNoteReport: uiNoteReport,
- uiPopover: uiPopover,
- uiPresetIcon: uiPresetIcon,
- uiPresetList: uiPresetList,
- uiRestore: uiRestore,
- uiScale: uiScale,
- uiSidebar: uiSidebar,
- uiSourceSwitch: uiSourceSwitch,
- uiSpinner: uiSpinner,
- uiSplash: uiSplash,
- uiStatus: uiStatus,
- uiSuccess: uiSuccess,
- uiTagReference: uiTagReference,
- uiToggle: uiToggle,
- uiTooltip: uiTooltip,
- uiVersion: uiVersion,
- uiViewOnOSM: uiViewOnOSM,
- uiViewOnKeepRight: uiViewOnKeepRight,
- uiZoom: uiZoom,
- utilAesEncrypt: utilAesEncrypt,
- utilAesDecrypt: utilAesDecrypt,
- utilArrayChunk: utilArrayChunk,
- utilArrayDifference: utilArrayDifference,
- utilArrayFlatten: utilArrayFlatten,
- utilArrayGroupBy: utilArrayGroupBy,
- utilArrayIdentical: utilArrayIdentical,
- utilArrayIntersection: utilArrayIntersection,
- utilArrayUnion: utilArrayUnion,
- utilArrayUniq: utilArrayUniq,
- utilArrayUniqBy: utilArrayUniqBy,
- utilAsyncMap: utilAsyncMap,
- utilCleanTags: utilCleanTags,
- utilCombinedTags: utilCombinedTags,
- utilDeepMemberSelector: utilDeepMemberSelector,
- utilDetect: utilDetect,
- utilDisplayName: utilDisplayName,
- utilDisplayNameForPath: utilDisplayNameForPath,
- utilDisplayType: utilDisplayType,
- utilDisplayLabel: utilDisplayLabel,
- utilEntityRoot: utilEntityRoot,
- utilEditDistance: utilEditDistance,
- utilEntitySelector: utilEntitySelector,
- utilEntityOrMemberSelector: utilEntityOrMemberSelector,
- utilEntityOrDeepMemberSelector: utilEntityOrDeepMemberSelector,
- utilFastMouse: utilFastMouse,
- utilFunctor: utilFunctor,
- utilGetAllNodes: utilGetAllNodes,
- utilGetSetValue: utilGetSetValue,
- utilHashcode: utilHashcode,
- utilHighlightEntities: utilHighlightEntities,
- utilKeybinding: utilKeybinding,
- utilNoAuto: utilNoAuto,
- utilObjectOmit: utilObjectOmit,
- utilPrefixCSSProperty: utilPrefixCSSProperty,
- utilPrefixDOMProperty: utilPrefixDOMProperty,
- utilQsString: utilQsString,
- utilRebind: utilRebind,
- utilSafeClassName: utilSafeClassName,
- utilSetTransform: utilSetTransform,
- utilSessionMutex: utilSessionMutex,
- utilStringQs: utilStringQs,
- utilTagDiff: utilTagDiff,
- utilTagText: utilTagText,
- utilTiler: utilTiler,
- utilTotalExtent: utilTotalExtent,
- utilTriggerEvent: utilTriggerEvent,
- utilUnicodeCharsCount: utilUnicodeCharsCount,
- utilUnicodeCharsTruncated: utilUnicodeCharsTruncated,
- utilUniqueDomId: utilUniqueDomId,
- utilWrap: utilWrap,
- 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
- });
-
- window.requestIdleCallback = window.requestIdleCallback || function (cb) {
- var start = Date.now();
- return window.requestAnimationFrame(function () {
- cb({
- didTimeout: false,
- timeRemaining: function timeRemaining() {
- return Math.max(0, 50 - (Date.now() - start));
- }
- });
- });
- };
-
- window.cancelIdleCallback = window.cancelIdleCallback || function (id) {
- window.cancelAnimationFrame(id);
- };
- window.iD = iD;
-
-}());
+(() => {
+ var __create = Object.create;
+ var __defProp = Object.defineProperty;
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
+ var __getOwnPropNames = Object.getOwnPropertyNames;
+ var __getProtoOf = Object.getPrototypeOf;
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
+ var __commonJS = (cb, mod) => function __require() {
+ return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
+ };
+ var __export = (target, all) => {
+ for (var name in all)
+ __defProp(target, name, { get: all[name], enumerable: true });
+ };
+ var __copyProps = (to, from, except, desc) => {
+ if (from && typeof from === "object" || typeof from === "function") {
+ for (let key of __getOwnPropNames(from))
+ if (!__hasOwnProp.call(to, key) && key !== except)
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
+ }
+ return to;
+ };
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
+ // If the importer is in node compatibility mode or this is not an ESM
+ // file that has been converted to a CommonJS file using a Babel-
+ // compatible transform (i.e. "__esModule" has not been set), then set
+ // "default" to the CommonJS "module.exports" for node compatibility.
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
+ mod
+ ));
+
+ // node_modules/diacritics/index.js
+ var require_diacritics = __commonJS({
+ "node_modules/diacritics/index.js"(exports2) {
+ exports2.remove = removeDiacritics2;
+ var replacementList = [
+ {
+ base: " ",
+ chars: "\xA0"
+ },
+ {
+ base: "0",
+ chars: "\u07C0"
+ },
+ {
+ base: "A",
+ chars: "\u24B6\uFF21\xC0\xC1\xC2\u1EA6\u1EA4\u1EAA\u1EA8\xC3\u0100\u0102\u1EB0\u1EAE\u1EB4\u1EB2\u0226\u01E0\xC4\u01DE\u1EA2\xC5\u01FA\u01CD\u0200\u0202\u1EA0\u1EAC\u1EB6\u1E00\u0104\u023A\u2C6F"
+ },
+ {
+ base: "AA",
+ chars: "\uA732"
+ },
+ {
+ base: "AE",
+ chars: "\xC6\u01FC\u01E2"
+ },
+ {
+ base: "AO",
+ chars: "\uA734"
+ },
+ {
+ base: "AU",
+ chars: "\uA736"
+ },
+ {
+ base: "AV",
+ chars: "\uA738\uA73A"
+ },
+ {
+ base: "AY",
+ chars: "\uA73C"
+ },
+ {
+ base: "B",
+ chars: "\u24B7\uFF22\u1E02\u1E04\u1E06\u0243\u0181"
+ },
+ {
+ base: "C",
+ chars: "\u24B8\uFF23\uA73E\u1E08\u0106C\u0108\u010A\u010C\xC7\u0187\u023B"
+ },
+ {
+ base: "D",
+ chars: "\u24B9\uFF24\u1E0A\u010E\u1E0C\u1E10\u1E12\u1E0E\u0110\u018A\u0189\u1D05\uA779"
+ },
+ {
+ base: "Dh",
+ chars: "\xD0"
+ },
+ {
+ base: "DZ",
+ chars: "\u01F1\u01C4"
+ },
+ {
+ base: "Dz",
+ chars: "\u01F2\u01C5"
+ },
+ {
+ base: "E",
+ chars: "\u025B\u24BA\uFF25\xC8\xC9\xCA\u1EC0\u1EBE\u1EC4\u1EC2\u1EBC\u0112\u1E14\u1E16\u0114\u0116\xCB\u1EBA\u011A\u0204\u0206\u1EB8\u1EC6\u0228\u1E1C\u0118\u1E18\u1E1A\u0190\u018E\u1D07"
+ },
+ {
+ base: "F",
+ chars: "\uA77C\u24BB\uFF26\u1E1E\u0191\uA77B"
+ },
+ {
+ base: "G",
+ chars: "\u24BC\uFF27\u01F4\u011C\u1E20\u011E\u0120\u01E6\u0122\u01E4\u0193\uA7A0\uA77D\uA77E\u0262"
+ },
+ {
+ base: "H",
+ chars: "\u24BD\uFF28\u0124\u1E22\u1E26\u021E\u1E24\u1E28\u1E2A\u0126\u2C67\u2C75\uA78D"
+ },
+ {
+ base: "I",
+ chars: "\u24BE\uFF29\xCC\xCD\xCE\u0128\u012A\u012C\u0130\xCF\u1E2E\u1EC8\u01CF\u0208\u020A\u1ECA\u012E\u1E2C\u0197"
+ },
+ {
+ base: "J",
+ chars: "\u24BF\uFF2A\u0134\u0248\u0237"
+ },
+ {
+ base: "K",
+ chars: "\u24C0\uFF2B\u1E30\u01E8\u1E32\u0136\u1E34\u0198\u2C69\uA740\uA742\uA744\uA7A2"
+ },
+ {
+ base: "L",
+ chars: "\u24C1\uFF2C\u013F\u0139\u013D\u1E36\u1E38\u013B\u1E3C\u1E3A\u0141\u023D\u2C62\u2C60\uA748\uA746\uA780"
+ },
+ {
+ base: "LJ",
+ chars: "\u01C7"
+ },
+ {
+ base: "Lj",
+ chars: "\u01C8"
+ },
+ {
+ base: "M",
+ chars: "\u24C2\uFF2D\u1E3E\u1E40\u1E42\u2C6E\u019C\u03FB"
+ },
+ {
+ base: "N",
+ chars: "\uA7A4\u0220\u24C3\uFF2E\u01F8\u0143\xD1\u1E44\u0147\u1E46\u0145\u1E4A\u1E48\u019D\uA790\u1D0E"
+ },
+ {
+ base: "NJ",
+ chars: "\u01CA"
+ },
+ {
+ base: "Nj",
+ chars: "\u01CB"
+ },
+ {
+ base: "O",
+ chars: "\u24C4\uFF2F\xD2\xD3\xD4\u1ED2\u1ED0\u1ED6\u1ED4\xD5\u1E4C\u022C\u1E4E\u014C\u1E50\u1E52\u014E\u022E\u0230\xD6\u022A\u1ECE\u0150\u01D1\u020C\u020E\u01A0\u1EDC\u1EDA\u1EE0\u1EDE\u1EE2\u1ECC\u1ED8\u01EA\u01EC\xD8\u01FE\u0186\u019F\uA74A\uA74C"
+ },
+ {
+ base: "OE",
+ chars: "\u0152"
+ },
+ {
+ base: "OI",
+ chars: "\u01A2"
+ },
+ {
+ base: "OO",
+ chars: "\uA74E"
+ },
+ {
+ base: "OU",
+ chars: "\u0222"
+ },
+ {
+ base: "P",
+ chars: "\u24C5\uFF30\u1E54\u1E56\u01A4\u2C63\uA750\uA752\uA754"
+ },
+ {
+ base: "Q",
+ chars: "\u24C6\uFF31\uA756\uA758\u024A"
+ },
+ {
+ base: "R",
+ chars: "\u24C7\uFF32\u0154\u1E58\u0158\u0210\u0212\u1E5A\u1E5C\u0156\u1E5E\u024C\u2C64\uA75A\uA7A6\uA782"
+ },
+ {
+ base: "S",
+ chars: "\u24C8\uFF33\u1E9E\u015A\u1E64\u015C\u1E60\u0160\u1E66\u1E62\u1E68\u0218\u015E\u2C7E\uA7A8\uA784"
+ },
+ {
+ base: "T",
+ chars: "\u24C9\uFF34\u1E6A\u0164\u1E6C\u021A\u0162\u1E70\u1E6E\u0166\u01AC\u01AE\u023E\uA786"
+ },
+ {
+ base: "Th",
+ chars: "\xDE"
+ },
+ {
+ base: "TZ",
+ chars: "\uA728"
+ },
+ {
+ base: "U",
+ chars: "\u24CA\uFF35\xD9\xDA\xDB\u0168\u1E78\u016A\u1E7A\u016C\xDC\u01DB\u01D7\u01D5\u01D9\u1EE6\u016E\u0170\u01D3\u0214\u0216\u01AF\u1EEA\u1EE8\u1EEE\u1EEC\u1EF0\u1EE4\u1E72\u0172\u1E76\u1E74\u0244"
+ },
+ {
+ base: "V",
+ chars: "\u24CB\uFF36\u1E7C\u1E7E\u01B2\uA75E\u0245"
+ },
+ {
+ base: "VY",
+ chars: "\uA760"
+ },
+ {
+ base: "W",
+ chars: "\u24CC\uFF37\u1E80\u1E82\u0174\u1E86\u1E84\u1E88\u2C72"
+ },
+ {
+ base: "X",
+ chars: "\u24CD\uFF38\u1E8A\u1E8C"
+ },
+ {
+ base: "Y",
+ chars: "\u24CE\uFF39\u1EF2\xDD\u0176\u1EF8\u0232\u1E8E\u0178\u1EF6\u1EF4\u01B3\u024E\u1EFE"
+ },
+ {
+ base: "Z",
+ chars: "\u24CF\uFF3A\u0179\u1E90\u017B\u017D\u1E92\u1E94\u01B5\u0224\u2C7F\u2C6B\uA762"
+ },
+ {
+ base: "a",
+ chars: "\u24D0\uFF41\u1E9A\xE0\xE1\xE2\u1EA7\u1EA5\u1EAB\u1EA9\xE3\u0101\u0103\u1EB1\u1EAF\u1EB5\u1EB3\u0227\u01E1\xE4\u01DF\u1EA3\xE5\u01FB\u01CE\u0201\u0203\u1EA1\u1EAD\u1EB7\u1E01\u0105\u2C65\u0250\u0251"
+ },
+ {
+ base: "aa",
+ chars: "\uA733"
+ },
+ {
+ base: "ae",
+ chars: "\xE6\u01FD\u01E3"
+ },
+ {
+ base: "ao",
+ chars: "\uA735"
+ },
+ {
+ base: "au",
+ chars: "\uA737"
+ },
+ {
+ base: "av",
+ chars: "\uA739\uA73B"
+ },
+ {
+ base: "ay",
+ chars: "\uA73D"
+ },
+ {
+ base: "b",
+ chars: "\u24D1\uFF42\u1E03\u1E05\u1E07\u0180\u0183\u0253\u0182"
+ },
+ {
+ base: "c",
+ chars: "\uFF43\u24D2\u0107\u0109\u010B\u010D\xE7\u1E09\u0188\u023C\uA73F\u2184"
+ },
+ {
+ base: "d",
+ chars: "\u24D3\uFF44\u1E0B\u010F\u1E0D\u1E11\u1E13\u1E0F\u0111\u018C\u0256\u0257\u018B\u13E7\u0501\uA7AA"
+ },
+ {
+ base: "dh",
+ chars: "\xF0"
+ },
+ {
+ base: "dz",
+ chars: "\u01F3\u01C6"
+ },
+ {
+ base: "e",
+ chars: "\u24D4\uFF45\xE8\xE9\xEA\u1EC1\u1EBF\u1EC5\u1EC3\u1EBD\u0113\u1E15\u1E17\u0115\u0117\xEB\u1EBB\u011B\u0205\u0207\u1EB9\u1EC7\u0229\u1E1D\u0119\u1E19\u1E1B\u0247\u01DD"
+ },
+ {
+ base: "f",
+ chars: "\u24D5\uFF46\u1E1F\u0192"
+ },
+ {
+ base: "ff",
+ chars: "\uFB00"
+ },
+ {
+ base: "fi",
+ chars: "\uFB01"
+ },
+ {
+ base: "fl",
+ chars: "\uFB02"
+ },
+ {
+ base: "ffi",
+ chars: "\uFB03"
+ },
+ {
+ base: "ffl",
+ chars: "\uFB04"
+ },
+ {
+ base: "g",
+ chars: "\u24D6\uFF47\u01F5\u011D\u1E21\u011F\u0121\u01E7\u0123\u01E5\u0260\uA7A1\uA77F\u1D79"
+ },
+ {
+ base: "h",
+ chars: "\u24D7\uFF48\u0125\u1E23\u1E27\u021F\u1E25\u1E29\u1E2B\u1E96\u0127\u2C68\u2C76\u0265"
+ },
+ {
+ base: "hv",
+ chars: "\u0195"
+ },
+ {
+ base: "i",
+ chars: "\u24D8\uFF49\xEC\xED\xEE\u0129\u012B\u012D\xEF\u1E2F\u1EC9\u01D0\u0209\u020B\u1ECB\u012F\u1E2D\u0268\u0131"
+ },
+ {
+ base: "j",
+ chars: "\u24D9\uFF4A\u0135\u01F0\u0249"
+ },
+ {
+ base: "k",
+ chars: "\u24DA\uFF4B\u1E31\u01E9\u1E33\u0137\u1E35\u0199\u2C6A\uA741\uA743\uA745\uA7A3"
+ },
+ {
+ base: "l",
+ chars: "\u24DB\uFF4C\u0140\u013A\u013E\u1E37\u1E39\u013C\u1E3D\u1E3B\u017F\u0142\u019A\u026B\u2C61\uA749\uA781\uA747\u026D"
+ },
+ {
+ base: "lj",
+ chars: "\u01C9"
+ },
+ {
+ base: "m",
+ chars: "\u24DC\uFF4D\u1E3F\u1E41\u1E43\u0271\u026F"
+ },
+ {
+ base: "n",
+ chars: "\u24DD\uFF4E\u01F9\u0144\xF1\u1E45\u0148\u1E47\u0146\u1E4B\u1E49\u019E\u0272\u0149\uA791\uA7A5\u043B\u0509"
+ },
+ {
+ base: "nj",
+ chars: "\u01CC"
+ },
+ {
+ base: "o",
+ chars: "\u24DE\uFF4F\xF2\xF3\xF4\u1ED3\u1ED1\u1ED7\u1ED5\xF5\u1E4D\u022D\u1E4F\u014D\u1E51\u1E53\u014F\u022F\u0231\xF6\u022B\u1ECF\u0151\u01D2\u020D\u020F\u01A1\u1EDD\u1EDB\u1EE1\u1EDF\u1EE3\u1ECD\u1ED9\u01EB\u01ED\xF8\u01FF\uA74B\uA74D\u0275\u0254\u1D11"
+ },
+ {
+ base: "oe",
+ chars: "\u0153"
+ },
+ {
+ base: "oi",
+ chars: "\u01A3"
+ },
+ {
+ base: "oo",
+ chars: "\uA74F"
+ },
+ {
+ base: "ou",
+ chars: "\u0223"
+ },
+ {
+ base: "p",
+ chars: "\u24DF\uFF50\u1E55\u1E57\u01A5\u1D7D\uA751\uA753\uA755\u03C1"
+ },
+ {
+ base: "q",
+ chars: "\u24E0\uFF51\u024B\uA757\uA759"
+ },
+ {
+ base: "r",
+ chars: "\u24E1\uFF52\u0155\u1E59\u0159\u0211\u0213\u1E5B\u1E5D\u0157\u1E5F\u024D\u027D\uA75B\uA7A7\uA783"
+ },
+ {
+ base: "s",
+ chars: "\u24E2\uFF53\u015B\u1E65\u015D\u1E61\u0161\u1E67\u1E63\u1E69\u0219\u015F\u023F\uA7A9\uA785\u1E9B\u0282"
+ },
+ {
+ base: "ss",
+ chars: "\xDF"
+ },
+ {
+ base: "t",
+ chars: "\u24E3\uFF54\u1E6B\u1E97\u0165\u1E6D\u021B\u0163\u1E71\u1E6F\u0167\u01AD\u0288\u2C66\uA787"
+ },
+ {
+ base: "th",
+ chars: "\xFE"
+ },
+ {
+ base: "tz",
+ chars: "\uA729"
+ },
+ {
+ base: "u",
+ chars: "\u24E4\uFF55\xF9\xFA\xFB\u0169\u1E79\u016B\u1E7B\u016D\xFC\u01DC\u01D8\u01D6\u01DA\u1EE7\u016F\u0171\u01D4\u0215\u0217\u01B0\u1EEB\u1EE9\u1EEF\u1EED\u1EF1\u1EE5\u1E73\u0173\u1E77\u1E75\u0289"
+ },
+ {
+ base: "v",
+ chars: "\u24E5\uFF56\u1E7D\u1E7F\u028B\uA75F\u028C"
+ },
+ {
+ base: "vy",
+ chars: "\uA761"
+ },
+ {
+ base: "w",
+ chars: "\u24E6\uFF57\u1E81\u1E83\u0175\u1E87\u1E85\u1E98\u1E89\u2C73"
+ },
+ {
+ base: "x",
+ chars: "\u24E7\uFF58\u1E8B\u1E8D"
+ },
+ {
+ base: "y",
+ chars: "\u24E8\uFF59\u1EF3\xFD\u0177\u1EF9\u0233\u1E8F\xFF\u1EF7\u1E99\u1EF5\u01B4\u024F\u1EFF"
+ },
+ {
+ base: "z",
+ chars: "\u24E9\uFF5A\u017A\u1E91\u017C\u017E\u1E93\u1E95\u01B6\u0225\u0240\u2C6C\uA763"
+ }
+ ];
+ var diacriticsMap = {};
+ for (i2 = 0; i2 < replacementList.length; i2 += 1) {
+ chars = replacementList[i2].chars;
+ for (j2 = 0; j2 < chars.length; j2 += 1) {
+ diacriticsMap[chars[j2]] = replacementList[i2].base;
+ }
+ }
+ var chars;
+ var j2;
+ var i2;
+ function removeDiacritics2(str2) {
+ return str2.replace(/[^\u0000-\u007e]/g, function(c) {
+ return diacriticsMap[c] || c;
+ });
+ }
+ exports2.replacementList = replacementList;
+ exports2.diacriticsMap = diacriticsMap;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/isArabic.js
+ var require_isArabic = __commonJS({
+ "node_modules/alif-toolkit/lib/isArabic.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var arabicBlocks = [
+ [1536, 1791],
+ [1872, 1919],
+ [2208, 2303],
+ [64336, 65023],
+ [65136, 65279],
+ [69216, 69247],
+ [126064, 126143],
+ [126464, 126719]
+ // Mathematical Alphabetic symbols https://www.unicode.org/charts/PDF/U1EE00.pdf
+ ];
+ function isArabic(char) {
+ if (char.length > 1) {
+ throw new Error("isArabic works on only one-character strings");
+ }
+ let code = char.charCodeAt(0);
+ for (let i2 = 0; i2 < arabicBlocks.length; i2++) {
+ let block2 = arabicBlocks[i2];
+ if (code >= block2[0] && code <= block2[1]) {
+ return true;
+ }
+ }
+ return false;
+ }
+ exports2.isArabic = isArabic;
+ function isMath(char) {
+ if (char.length > 2) {
+ throw new Error("isMath works on only one-character strings");
+ }
+ let code = char.charCodeAt(0);
+ return code >= 1632 && code <= 1644 || code >= 1776 && code <= 1785;
+ }
+ exports2.isMath = isMath;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/unicode-arabic.js
+ var require_unicode_arabic = __commonJS({
+ "node_modules/alif-toolkit/lib/unicode-arabic.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__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"
+ },
+ "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"
+ },
+ "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"
+ }
+ },
+ "rohingya yeh": {
+ "normal": [
+ "\u08AC"
+ ]
+ },
+ "low alef": {
+ "normal": [
+ "\u08AD"
+ ]
+ },
+ "straight waw": {
+ "normal": [
+ "\u08B1"
+ ]
+ },
+ "african feh": {
+ "normal": [
+ "\u08BB"
+ ]
+ },
+ "african qaf": {
+ "normal": [
+ "\u08BC"
+ ]
+ },
+ "african noon": {
+ "normal": [
+ "\u08BD"
+ ]
+ }
+ };
+ exports2.default = arabicReference;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/unicode-ligatures.js
+ var require_unicode_ligatures = __commonJS({
+ "node_modules/alif-toolkit/lib/unicode-ligatures.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__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"
+ }
+ };
+ exports2.default = ligatureReference;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/reference.js
+ var require_reference = __commonJS({
+ "node_modules/alif-toolkit/lib/reference.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var unicode_arabic_1 = require_unicode_arabic();
+ var unicode_ligatures_1 = require_unicode_ligatures();
+ var letterList = Object.keys(unicode_arabic_1.default);
+ exports2.letterList = letterList;
+ var ligatureList = Object.keys(unicode_ligatures_1.default);
+ exports2.ligatureList = ligatureList;
+ var ligatureWordList = Object.keys(unicode_ligatures_1.default.words);
+ exports2.ligatureWordList = ligatureWordList;
+ var lams = "\u0644\u06B5\u06B6\u06B7\u06B8";
+ exports2.lams = lams;
+ var alefs = "\u0627\u0622\u0623\u0625\u0671\u0672\u0673\u0675\u0773\u0774";
+ exports2.alefs = alefs;
+ var tashkeel = "\u0605\u0640\u0670\u0674\u06DF\u06E7\u06E8";
+ exports2.tashkeel = tashkeel;
+ function addToTashkeel(start2, finish) {
+ for (var i2 = start2; i2 <= finish; i2++) {
+ exports2.tashkeel = tashkeel += String.fromCharCode(i2);
+ }
+ }
+ addToTashkeel(1552, 1562);
+ addToTashkeel(1611, 1631);
+ addToTashkeel(1750, 1756);
+ addToTashkeel(1760, 1764);
+ addToTashkeel(1770, 1773);
+ addToTashkeel(2259, 2273);
+ addToTashkeel(2275, 2303);
+ addToTashkeel(65136, 65151);
+ var lineBreakers = "\u0627\u0629\u0648\u06C0\u06CF\u06FD\u06FE\u076B\u076C\u0771\u0773\u0774\u0778\u0779\u08E2\u08B1\u08B2\u08B9";
+ exports2.lineBreakers = lineBreakers;
+ function addToLineBreakers(start2, finish) {
+ for (var i2 = start2; i2 <= finish; i2++) {
+ exports2.lineBreakers = lineBreakers += String.fromCharCode(i2);
+ }
+ }
+ addToLineBreakers(1536, 1567);
+ addToLineBreakers(1569, 1573);
+ addToLineBreakers(1583, 1586);
+ addToLineBreakers(1632, 1645);
+ addToLineBreakers(1649, 1655);
+ addToLineBreakers(1672, 1689);
+ addToLineBreakers(1731, 1739);
+ addToLineBreakers(1746, 1785);
+ addToLineBreakers(1881, 1883);
+ addToLineBreakers(2218, 2222);
+ addToLineBreakers(64336, 65021);
+ addToLineBreakers(65152, 65276);
+ addToLineBreakers(69216, 69247);
+ addToLineBreakers(126064, 126143);
+ addToLineBreakers(126464, 126719);
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/GlyphSplitter.js
+ var require_GlyphSplitter = __commonJS({
+ "node_modules/alif-toolkit/lib/GlyphSplitter.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var isArabic_1 = require_isArabic();
+ var reference_1 = require_reference();
+ function GlyphSplitter(word) {
+ let letters = [];
+ let lastLetter = "";
+ word.split("").forEach((letter) => {
+ if (isArabic_1.isArabic(letter)) {
+ if (reference_1.tashkeel.indexOf(letter) > -1) {
+ letters[letters.length - 1] += letter;
+ } else if (lastLetter.length && (reference_1.lams.indexOf(lastLetter) === 0 && reference_1.alefs.indexOf(letter) > -1 || reference_1.lams.indexOf(lastLetter) > 0 && reference_1.alefs.indexOf(letter) === 0)) {
+ letters[letters.length - 1] += letter;
+ } else {
+ letters.push(letter);
+ }
+ } else {
+ letters.push(letter);
+ }
+ if (reference_1.tashkeel.indexOf(letter) === -1) {
+ lastLetter = letter;
+ }
+ });
+ return letters;
+ }
+ exports2.GlyphSplitter = GlyphSplitter;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/BaselineSplitter.js
+ var require_BaselineSplitter = __commonJS({
+ "node_modules/alif-toolkit/lib/BaselineSplitter.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var isArabic_1 = require_isArabic();
+ var reference_1 = require_reference();
+ function BaselineSplitter(word) {
+ let letters = [];
+ let lastLetter = "";
+ word.split("").forEach((letter) => {
+ if (isArabic_1.isArabic(letter) && isArabic_1.isArabic(lastLetter)) {
+ if (lastLetter.length && reference_1.tashkeel.indexOf(letter) > -1) {
+ letters[letters.length - 1] += letter;
+ } else if (reference_1.lineBreakers.indexOf(lastLetter) > -1) {
+ letters.push(letter);
+ } else {
+ letters[letters.length - 1] += letter;
+ }
+ } else {
+ letters.push(letter);
+ }
+ if (reference_1.tashkeel.indexOf(letter) === -1) {
+ lastLetter = letter;
+ }
+ });
+ return letters;
+ }
+ exports2.BaselineSplitter = BaselineSplitter;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/Normalization.js
+ var require_Normalization = __commonJS({
+ "node_modules/alif-toolkit/lib/Normalization.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var unicode_arabic_1 = require_unicode_arabic();
+ var unicode_ligatures_1 = require_unicode_ligatures();
+ var isArabic_1 = require_isArabic();
+ var reference_1 = require_reference();
+ function Normal(word, breakPresentationForm) {
+ if (typeof breakPresentationForm === "undefined") {
+ breakPresentationForm = true;
+ }
+ let returnable = "";
+ word.split("").forEach((letter) => {
+ if (!isArabic_1.isArabic(letter)) {
+ returnable += letter;
+ return;
+ }
+ for (let w = 0; w < reference_1.letterList.length; w++) {
+ let letterForms = unicode_arabic_1.default[reference_1.letterList[w]];
+ let versions = Object.keys(letterForms);
+ for (let v = 0; v < versions.length; v++) {
+ let localVersion = letterForms[versions[v]];
+ if (typeof localVersion === "object" && typeof localVersion.indexOf === "undefined") {
+ let embeddedForms = Object.keys(localVersion);
+ for (let ef = 0; ef < embeddedForms.length; ef++) {
+ let form = localVersion[embeddedForms[ef]];
+ if (form === letter || typeof form === "object" && form.indexOf && form.indexOf(letter) > -1) {
+ if (form === letter) {
+ if (breakPresentationForm && localVersion["normal"] && ["isolated", "initial", "medial", "final"].indexOf(embeddedForms[ef]) > -1) {
+ if (typeof localVersion["normal"] === "object") {
+ returnable += localVersion["normal"][0];
+ } else {
+ returnable += localVersion["normal"];
+ }
+ return;
+ }
+ returnable += letter;
+ return;
+ } else if (typeof form === "object" && form.indexOf && form.indexOf(letter) > -1) {
+ returnable += form[0];
+ return;
+ }
+ }
+ }
+ } else if (localVersion === letter) {
+ if (breakPresentationForm && letterForms["normal"] && ["isolated", "initial", "medial", "final"].indexOf(versions[v]) > -1) {
+ if (typeof letterForms["normal"] === "object") {
+ returnable += letterForms["normal"][0];
+ } else {
+ returnable += letterForms["normal"];
+ }
+ return;
+ }
+ returnable += letter;
+ return;
+ } else if (typeof localVersion === "object" && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
+ returnable += localVersion[0];
+ return;
+ }
+ }
+ }
+ for (let v2 = 0; v2 < reference_1.ligatureList.length; v2++) {
+ let normalForm = reference_1.ligatureList[v2];
+ if (normalForm !== "words") {
+ let ligForms = Object.keys(unicode_ligatures_1.default[normalForm]);
+ for (let f2 = 0; f2 < ligForms.length; f2++) {
+ if (unicode_ligatures_1.default[normalForm][ligForms[f2]] === letter) {
+ returnable += normalForm;
+ return;
+ }
+ }
+ }
+ }
+ for (let v3 = 0; v3 < reference_1.ligatureWordList.length; v3++) {
+ let normalForm = reference_1.ligatureWordList[v3];
+ if (unicode_ligatures_1.default.words[normalForm] === letter) {
+ returnable += normalForm;
+ return;
+ }
+ }
+ returnable += letter;
+ });
+ return returnable;
+ }
+ exports2.Normal = Normal;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/CharShaper.js
+ var require_CharShaper = __commonJS({
+ "node_modules/alif-toolkit/lib/CharShaper.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var unicode_arabic_1 = require_unicode_arabic();
+ var isArabic_1 = require_isArabic();
+ var reference_1 = require_reference();
+ function CharShaper(letter, form) {
+ if (!isArabic_1.isArabic(letter)) {
+ throw new Error("Not Arabic");
+ }
+ if (letter === "\u0621") {
+ return "\u0621";
+ }
+ for (let w = 0; w < reference_1.letterList.length; w++) {
+ let letterForms = unicode_arabic_1.default[reference_1.letterList[w]];
+ let versions = Object.keys(letterForms);
+ for (let v = 0; v < versions.length; v++) {
+ let 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") {
+ let embeddedVersions = Object.keys(localVersion);
+ for (let 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];
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ exports2.CharShaper = CharShaper;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/WordShaper.js
+ var require_WordShaper = __commonJS({
+ "node_modules/alif-toolkit/lib/WordShaper.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var isArabic_1 = require_isArabic();
+ var reference_1 = require_reference();
+ var CharShaper_1 = require_CharShaper();
+ var unicode_ligatures_1 = require_unicode_ligatures();
+ function WordShaper2(word) {
+ let state = "initial";
+ let output = "";
+ for (let w = 0; w < word.length; w++) {
+ let nextLetter = " ";
+ for (let nxw = w + 1; nxw < word.length; nxw++) {
+ if (!isArabic_1.isArabic(word[nxw])) {
+ break;
+ }
+ if (reference_1.tashkeel.indexOf(word[nxw]) === -1) {
+ nextLetter = word[nxw];
+ break;
+ }
+ }
+ if (!isArabic_1.isArabic(word[w]) || isArabic_1.isMath(word[w])) {
+ output += word[w];
+ state = "initial";
+ } else if (reference_1.tashkeel.indexOf(word[w]) > -1) {
+ output += word[w];
+ } else if (nextLetter === " " || reference_1.lineBreakers.indexOf(word[w]) > -1) {
+ output += CharShaper_1.CharShaper(word[w], state === "initial" ? "isolated" : "final");
+ state = "initial";
+ } else if (reference_1.lams.indexOf(word[w]) > -1 && reference_1.alefs.indexOf(nextLetter) > -1) {
+ output += unicode_ligatures_1.default[word[w] + nextLetter][state === "initial" ? "isolated" : "final"];
+ while (word[w] !== nextLetter) {
+ w++;
+ }
+ state = "initial";
+ } else {
+ output += CharShaper_1.CharShaper(word[w], state);
+ state = "medial";
+ }
+ }
+ return output;
+ }
+ exports2.WordShaper = WordShaper2;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/ParentLetter.js
+ var require_ParentLetter = __commonJS({
+ "node_modules/alif-toolkit/lib/ParentLetter.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var unicode_arabic_1 = require_unicode_arabic();
+ var isArabic_1 = require_isArabic();
+ var reference_1 = require_reference();
+ function ParentLetter(letter) {
+ if (!isArabic_1.isArabic(letter)) {
+ throw new Error("Not an Arabic letter");
+ }
+ for (let w = 0; w < reference_1.letterList.length; w++) {
+ let letterForms = unicode_arabic_1.default[reference_1.letterList[w]];
+ let versions = Object.keys(letterForms);
+ for (let v = 0; v < versions.length; v++) {
+ let localVersion = letterForms[versions[v]];
+ if (typeof localVersion === "object" && typeof localVersion.indexOf === "undefined") {
+ let embeddedForms = Object.keys(localVersion);
+ for (let ef = 0; ef < embeddedForms.length; ef++) {
+ let form = localVersion[embeddedForms[ef]];
+ if (form === letter || typeof form === "object" && form.indexOf && form.indexOf(letter) > -1) {
+ return localVersion;
+ }
+ }
+ } else if (localVersion === letter || typeof localVersion === "object" && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
+ return letterForms;
+ }
+ }
+ return null;
+ }
+ }
+ exports2.ParentLetter = ParentLetter;
+ function GrandparentLetter(letter) {
+ if (!isArabic_1.isArabic(letter)) {
+ throw new Error("Not an Arabic letter");
+ }
+ for (let w = 0; w < reference_1.letterList.length; w++) {
+ let letterForms = unicode_arabic_1.default[reference_1.letterList[w]];
+ let versions = Object.keys(letterForms);
+ for (let v = 0; v < versions.length; v++) {
+ let localVersion = letterForms[versions[v]];
+ if (typeof localVersion === "object" && typeof localVersion.indexOf === "undefined") {
+ let embeddedForms = Object.keys(localVersion);
+ for (let ef = 0; ef < embeddedForms.length; ef++) {
+ let form = localVersion[embeddedForms[ef]];
+ if (form === letter || typeof form === "object" && form.indexOf && form.indexOf(letter) > -1) {
+ return letterForms;
+ }
+ }
+ } else if (localVersion === letter || typeof localVersion === "object" && localVersion.indexOf && localVersion.indexOf(letter) > -1) {
+ return letterForms;
+ }
+ }
+ return null;
+ }
+ }
+ exports2.GrandparentLetter = GrandparentLetter;
+ }
+ });
+
+ // node_modules/alif-toolkit/lib/index.js
+ var require_lib = __commonJS({
+ "node_modules/alif-toolkit/lib/index.js"(exports2) {
+ "use strict";
+ Object.defineProperty(exports2, "__esModule", { value: true });
+ var isArabic_1 = require_isArabic();
+ exports2.isArabic = isArabic_1.isArabic;
+ var GlyphSplitter_1 = require_GlyphSplitter();
+ exports2.GlyphSplitter = GlyphSplitter_1.GlyphSplitter;
+ var BaselineSplitter_1 = require_BaselineSplitter();
+ exports2.BaselineSplitter = BaselineSplitter_1.BaselineSplitter;
+ var Normalization_1 = require_Normalization();
+ exports2.Normal = Normalization_1.Normal;
+ var CharShaper_1 = require_CharShaper();
+ exports2.CharShaper = CharShaper_1.CharShaper;
+ var WordShaper_1 = require_WordShaper();
+ exports2.WordShaper = WordShaper_1.WordShaper;
+ var ParentLetter_1 = require_ParentLetter();
+ exports2.ParentLetter = ParentLetter_1.ParentLetter;
+ exports2.GrandparentLetter = ParentLetter_1.GrandparentLetter;
+ }
+ });
+
+ // node_modules/vparse/index.js
+ var require_vparse = __commonJS({
+ "node_modules/vparse/index.js"(exports2, module2) {
+ (function(window2) {
+ "use strict";
+ function parseVersion3(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 compare(v) {
+ if (typeof v === "string") {
+ v = parseVersion3(v);
+ }
+ for (var i2 = 0; i2 < 4; i2++) {
+ if (this.parsed[i2] !== v.parsed[i2]) {
+ return this.parsed[i2] > v.parsed[i2] ? 1 : -1;
+ }
+ }
+ return 0;
+ }
+ if (typeof module2 === "object" && module2 && typeof module2.exports === "object") {
+ module2.exports = parseVersion3;
+ } else {
+ window2.parseVersion = parseVersion3;
+ }
+ })(exports2);
+ }
+ });
+
+ // node_modules/which-polygon/node_modules/quickselect/quickselect.js
+ var require_quickselect = __commonJS({
+ "node_modules/which-polygon/node_modules/quickselect/quickselect.js"(exports2, module2) {
+ (function(global3, factory) {
+ typeof exports2 === "object" && typeof module2 !== "undefined" ? module2.exports = factory() : typeof define === "function" && define.amd ? define(factory) : global3.quickselect = factory();
+ })(exports2, function() {
+ "use strict";
+ function quickselect2(arr, k, left, right, compare) {
+ quickselectStep(arr, k, left || 0, right || arr.length - 1, compare || defaultCompare);
+ }
+ function quickselectStep(arr, k, left, right, compare) {
+ while (right > left) {
+ if (right - left > 600) {
+ var n2 = right - left + 1;
+ var m = k - left + 1;
+ var z = Math.log(n2);
+ var s = 0.5 * Math.exp(2 * z / 3);
+ var sd = 0.5 * Math.sqrt(z * s * (n2 - s) / n2) * (m - n2 / 2 < 0 ? -1 : 1);
+ var newLeft = Math.max(left, Math.floor(k - m * s / n2 + sd));
+ var newRight = Math.min(right, Math.floor(k + (n2 - m) * s / n2 + sd));
+ quickselectStep(arr, k, newLeft, newRight, compare);
+ }
+ var t = arr[k];
+ var i2 = left;
+ var j2 = right;
+ swap2(arr, left, k);
+ if (compare(arr[right], t) > 0)
+ swap2(arr, left, right);
+ while (i2 < j2) {
+ swap2(arr, i2, j2);
+ i2++;
+ j2--;
+ while (compare(arr[i2], t) < 0)
+ i2++;
+ while (compare(arr[j2], t) > 0)
+ j2--;
+ }
+ if (compare(arr[left], t) === 0)
+ swap2(arr, left, j2);
+ else {
+ j2++;
+ swap2(arr, j2, right);
+ }
+ if (j2 <= k)
+ left = j2 + 1;
+ if (k <= j2)
+ right = j2 - 1;
+ }
+ }
+ function swap2(arr, i2, j2) {
+ var tmp = arr[i2];
+ arr[i2] = arr[j2];
+ arr[j2] = tmp;
+ }
+ function defaultCompare(a, b) {
+ return a < b ? -1 : a > b ? 1 : 0;
+ }
+ return quickselect2;
+ });
+ }
+ });
+
+ // node_modules/which-polygon/node_modules/rbush/index.js
+ var require_rbush = __commonJS({
+ "node_modules/which-polygon/node_modules/rbush/index.js"(exports2, module2) {
+ "use strict";
+ module2.exports = rbush;
+ module2.exports.default = rbush;
+ var quickselect2 = require_quickselect();
+ function rbush(maxEntries, format2) {
+ if (!(this instanceof rbush))
+ return new rbush(maxEntries, format2);
+ this._maxEntries = Math.max(4, maxEntries || 9);
+ this._minEntries = Math.max(2, Math.ceil(this._maxEntries * 0.4));
+ if (format2) {
+ this._initFormat(format2);
+ }
+ this.clear();
+ }
+ rbush.prototype = {
+ all: function() {
+ return this._all(this.data, []);
+ },
+ search: function(bbox2) {
+ var node = this.data, result = [], toBBox = this.toBBox;
+ if (!intersects(bbox2, node))
+ return result;
+ var nodesToSearch = [], i2, len, child, childBBox;
+ while (node) {
+ for (i2 = 0, len = node.children.length; i2 < len; i2++) {
+ child = node.children[i2];
+ childBBox = node.leaf ? toBBox(child) : child;
+ if (intersects(bbox2, childBBox)) {
+ if (node.leaf)
+ result.push(child);
+ else if (contains(bbox2, childBBox))
+ this._all(child, result);
+ else
+ nodesToSearch.push(child);
+ }
+ }
+ node = nodesToSearch.pop();
+ }
+ return result;
+ },
+ collides: function(bbox2) {
+ var node = this.data, toBBox = this.toBBox;
+ if (!intersects(bbox2, node))
+ return false;
+ var nodesToSearch = [], i2, len, child, childBBox;
+ while (node) {
+ for (i2 = 0, len = node.children.length; i2 < len; i2++) {
+ child = node.children[i2];
+ childBBox = node.leaf ? toBBox(child) : child;
+ if (intersects(bbox2, childBBox)) {
+ if (node.leaf || contains(bbox2, childBBox))
+ return true;
+ nodesToSearch.push(child);
+ }
+ }
+ node = nodesToSearch.pop();
+ }
+ return false;
+ },
+ load: function(data) {
+ if (!(data && data.length))
+ return this;
+ if (data.length < this._minEntries) {
+ for (var i2 = 0, len = data.length; i2 < len; i2++) {
+ this.insert(data[i2]);
+ }
+ return this;
+ }
+ var node = this._build(data.slice(), 0, data.length - 1, 0);
+ if (!this.data.children.length) {
+ this.data = node;
+ } else if (this.data.height === node.height) {
+ this._splitRoot(this.data, node);
+ } else {
+ if (this.data.height < node.height) {
+ var tmpNode = this.data;
+ this.data = node;
+ node = tmpNode;
+ }
+ this._insert(node, this.data.height - node.height - 1, true);
+ }
+ return this;
+ },
+ insert: function(item) {
+ if (item)
+ this._insert(item, this.data.height - 1);
+ return this;
+ },
+ clear: function() {
+ this.data = createNode([]);
+ return this;
+ },
+ remove: function(item, equalsFn) {
+ if (!item)
+ return this;
+ var node = this.data, bbox2 = this.toBBox(item), path = [], indexes = [], i2, parent, index, goingUp;
+ while (node || path.length) {
+ if (!node) {
+ node = path.pop();
+ parent = path[path.length - 1];
+ i2 = indexes.pop();
+ goingUp = true;
+ }
+ if (node.leaf) {
+ index = findItem(item, node.children, equalsFn);
+ if (index !== -1) {
+ node.children.splice(index, 1);
+ path.push(node);
+ this._condense(path);
+ return this;
+ }
+ }
+ if (!goingUp && !node.leaf && contains(node, bbox2)) {
+ path.push(node);
+ indexes.push(i2);
+ i2 = 0;
+ parent = node;
+ node = node.children[0];
+ } else if (parent) {
+ i2++;
+ node = parent.children[i2];
+ goingUp = false;
+ } else
+ node = null;
+ }
+ return this;
+ },
+ toBBox: function(item) {
+ return item;
+ },
+ compareMinX: compareNodeMinX,
+ compareMinY: compareNodeMinY,
+ toJSON: function() {
+ return this.data;
+ },
+ fromJSON: function(data) {
+ this.data = data;
+ return this;
+ },
+ _all: function(node, result) {
+ var nodesToSearch = [];
+ while (node) {
+ if (node.leaf)
+ result.push.apply(result, node.children);
+ else
+ nodesToSearch.push.apply(nodesToSearch, node.children);
+ node = nodesToSearch.pop();
+ }
+ return result;
+ },
+ _build: function(items, left, right, height) {
+ var N = right - left + 1, M = this._maxEntries, node;
+ if (N <= M) {
+ node = createNode(items.slice(left, right + 1));
+ calcBBox(node, this.toBBox);
+ return node;
+ }
+ if (!height) {
+ height = Math.ceil(Math.log(N) / Math.log(M));
+ M = Math.ceil(N / Math.pow(M, height - 1));
+ }
+ node = createNode([]);
+ node.leaf = false;
+ node.height = height;
+ var N2 = Math.ceil(N / M), N1 = N2 * Math.ceil(Math.sqrt(M)), i2, j2, right2, right3;
+ multiSelect(items, left, right, N1, this.compareMinX);
+ for (i2 = left; i2 <= right; i2 += N1) {
+ right2 = Math.min(i2 + N1 - 1, right);
+ multiSelect(items, i2, right2, N2, this.compareMinY);
+ for (j2 = i2; j2 <= right2; j2 += N2) {
+ right3 = Math.min(j2 + N2 - 1, right2);
+ node.children.push(this._build(items, j2, right3, height - 1));
+ }
+ }
+ calcBBox(node, this.toBBox);
+ return node;
+ },
+ _chooseSubtree: function(bbox2, node, level, path) {
+ var i2, len, child, targetNode, area, enlargement, minArea, minEnlargement;
+ while (true) {
+ path.push(node);
+ if (node.leaf || path.length - 1 === level)
+ break;
+ minArea = minEnlargement = Infinity;
+ for (i2 = 0, len = node.children.length; i2 < len; i2++) {
+ child = node.children[i2];
+ area = bboxArea(child);
+ enlargement = enlargedArea(bbox2, child) - area;
+ if (enlargement < minEnlargement) {
+ minEnlargement = enlargement;
+ minArea = area < minArea ? area : minArea;
+ targetNode = child;
+ } else if (enlargement === minEnlargement) {
+ if (area < minArea) {
+ minArea = area;
+ targetNode = child;
+ }
+ }
+ }
+ node = targetNode || node.children[0];
+ }
+ return node;
+ },
+ _insert: function(item, level, isNode) {
+ var toBBox = this.toBBox, bbox2 = isNode ? item : toBBox(item), insertPath = [];
+ var node = this._chooseSubtree(bbox2, this.data, level, insertPath);
+ node.children.push(item);
+ extend2(node, bbox2);
+ while (level >= 0) {
+ if (insertPath[level].children.length > this._maxEntries) {
+ this._split(insertPath, level);
+ level--;
+ } else
+ break;
+ }
+ this._adjustParentBBoxes(bbox2, insertPath, level);
+ },
+ // split overflowed node into two
+ _split: function(insertPath, level) {
+ var node = insertPath[level], M = node.children.length, m = this._minEntries;
+ this._chooseSplitAxis(node, m, M);
+ var splitIndex = this._chooseSplitIndex(node, m, M);
+ 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);
+ },
+ _splitRoot: function(node, newNode) {
+ this.data = createNode([node, newNode]);
+ this.data.height = node.height + 1;
+ this.data.leaf = false;
+ calcBBox(this.data, this.toBBox);
+ },
+ _chooseSplitIndex: function(node, m, M) {
+ var i2, bbox1, bbox2, overlap, area, minOverlap, minArea, index;
+ minOverlap = minArea = Infinity;
+ for (i2 = m; i2 <= M - m; i2++) {
+ bbox1 = distBBox(node, 0, i2, this.toBBox);
+ bbox2 = distBBox(node, i2, M, this.toBBox);
+ overlap = intersectionArea(bbox1, bbox2);
+ area = bboxArea(bbox1) + bboxArea(bbox2);
+ if (overlap < minOverlap) {
+ minOverlap = overlap;
+ index = i2;
+ minArea = area < minArea ? area : minArea;
+ } else if (overlap === minOverlap) {
+ if (area < minArea) {
+ minArea = area;
+ index = i2;
+ }
+ }
+ }
+ return index;
+ },
+ // sorts node children by the best axis for split
+ _chooseSplitAxis: function(node, m, M) {
+ var compareMinX = node.leaf ? this.compareMinX : compareNodeMinX, compareMinY = node.leaf ? this.compareMinY : compareNodeMinY, xMargin = this._allDistMargin(node, m, M, compareMinX), yMargin = this._allDistMargin(node, m, M, compareMinY);
+ if (xMargin < yMargin)
+ node.children.sort(compareMinX);
+ },
+ // total margin of all possible split distributions where each node is at least m full
+ _allDistMargin: function(node, m, M, compare) {
+ node.children.sort(compare);
+ var toBBox = this.toBBox, leftBBox = distBBox(node, 0, m, toBBox), rightBBox = distBBox(node, M - m, M, toBBox), margin = bboxMargin(leftBBox) + bboxMargin(rightBBox), i2, child;
+ for (i2 = m; i2 < M - m; i2++) {
+ child = node.children[i2];
+ extend2(leftBBox, node.leaf ? toBBox(child) : child);
+ margin += bboxMargin(leftBBox);
+ }
+ for (i2 = M - m - 1; i2 >= m; i2--) {
+ child = node.children[i2];
+ extend2(rightBBox, node.leaf ? toBBox(child) : child);
+ margin += bboxMargin(rightBBox);
+ }
+ return margin;
+ },
+ _adjustParentBBoxes: function(bbox2, path, level) {
+ for (var i2 = level; i2 >= 0; i2--) {
+ extend2(path[i2], bbox2);
+ }
+ },
+ _condense: function(path) {
+ for (var i2 = path.length - 1, siblings; i2 >= 0; i2--) {
+ if (path[i2].children.length === 0) {
+ if (i2 > 0) {
+ siblings = path[i2 - 1].children;
+ siblings.splice(siblings.indexOf(path[i2]), 1);
+ } else
+ this.clear();
+ } else
+ calcBBox(path[i2], this.toBBox);
+ }
+ },
+ _initFormat: function(format2) {
+ var compareArr = ["return a", " - b", ";"];
+ this.compareMinX = new Function("a", "b", compareArr.join(format2[0]));
+ this.compareMinY = new Function("a", "b", compareArr.join(format2[1]));
+ this.toBBox = new Function(
+ "a",
+ "return {minX: a" + format2[0] + ", minY: a" + format2[1] + ", maxX: a" + format2[2] + ", maxY: a" + format2[3] + "};"
+ );
+ }
+ };
+ function findItem(item, items, equalsFn) {
+ if (!equalsFn)
+ return items.indexOf(item);
+ for (var i2 = 0; i2 < items.length; i2++) {
+ if (equalsFn(item, items[i2]))
+ return i2;
+ }
+ return -1;
+ }
+ function calcBBox(node, toBBox) {
+ distBBox(node, 0, node.children.length, toBBox, node);
+ }
+ function distBBox(node, k, p, toBBox, destNode) {
+ if (!destNode)
+ destNode = createNode(null);
+ destNode.minX = Infinity;
+ destNode.minY = Infinity;
+ destNode.maxX = -Infinity;
+ destNode.maxY = -Infinity;
+ for (var i2 = k, child; i2 < p; i2++) {
+ child = node.children[i2];
+ extend2(destNode, node.leaf ? toBBox(child) : child);
+ }
+ return destNode;
+ }
+ function extend2(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 compareNodeMinX(a, b) {
+ return a.minX - b.minX;
+ }
+ function compareNodeMinY(a, b) {
+ return a.minY - b.minY;
+ }
+ function bboxArea(a) {
+ return (a.maxX - a.minX) * (a.maxY - a.minY);
+ }
+ function bboxMargin(a) {
+ return a.maxX - a.minX + (a.maxY - a.minY);
+ }
+ 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 intersectionArea(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);
+ }
+ function contains(a, b) {
+ return a.minX <= b.minX && a.minY <= b.minY && b.maxX <= a.maxX && b.maxY <= a.maxY;
+ }
+ function intersects(a, b) {
+ return b.minX <= a.maxX && b.minY <= a.maxY && b.maxX >= a.minX && b.maxY >= a.minY;
+ }
+ function createNode(children2) {
+ return {
+ children: children2,
+ height: 1,
+ leaf: true,
+ minX: Infinity,
+ minY: Infinity,
+ maxX: -Infinity,
+ maxY: -Infinity
+ };
+ }
+ function multiSelect(arr, left, right, n2, compare) {
+ var stack = [left, right], mid;
+ while (stack.length) {
+ right = stack.pop();
+ left = stack.pop();
+ if (right - left <= n2)
+ continue;
+ mid = left + Math.ceil((right - left) / n2 / 2) * n2;
+ quickselect2(arr, mid, left, right, compare);
+ stack.push(left, mid, mid, right);
+ }
+ }
+ }
+ });
+
+ // node_modules/lineclip/index.js
+ var require_lineclip = __commonJS({
+ "node_modules/lineclip/index.js"(exports2, module2) {
+ "use strict";
+ module2.exports = lineclip2;
+ lineclip2.polyline = lineclip2;
+ lineclip2.polygon = polygonclip2;
+ function lineclip2(points, bbox2, result) {
+ var len = points.length, codeA = bitCode2(points[0], bbox2), part = [], i2, a, b, codeB, lastCode;
+ if (!result)
+ result = [];
+ for (i2 = 1; i2 < len; i2++) {
+ a = points[i2 - 1];
+ b = points[i2];
+ codeB = lastCode = bitCode2(b, bbox2);
+ while (true) {
+ if (!(codeA | codeB)) {
+ part.push(a);
+ if (codeB !== lastCode) {
+ part.push(b);
+ if (i2 < len - 1) {
+ result.push(part);
+ part = [];
+ }
+ } else if (i2 === len - 1) {
+ part.push(b);
+ }
+ break;
+ } else if (codeA & codeB) {
+ break;
+ } else if (codeA) {
+ a = intersect2(a, b, codeA, bbox2);
+ codeA = bitCode2(a, bbox2);
+ } else {
+ b = intersect2(a, b, codeB, bbox2);
+ codeB = bitCode2(b, bbox2);
+ }
+ }
+ codeA = lastCode;
+ }
+ if (part.length)
+ result.push(part);
+ return result;
+ }
+ function polygonclip2(points, bbox2) {
+ var result, edge, prev, prevInside, i2, p, inside;
+ for (edge = 1; edge <= 8; edge *= 2) {
+ result = [];
+ prev = points[points.length - 1];
+ prevInside = !(bitCode2(prev, bbox2) & edge);
+ for (i2 = 0; i2 < points.length; i2++) {
+ p = points[i2];
+ inside = !(bitCode2(p, bbox2) & edge);
+ if (inside !== prevInside)
+ result.push(intersect2(prev, p, edge, bbox2));
+ if (inside)
+ result.push(p);
+ prev = p;
+ prevInside = inside;
+ }
+ points = result;
+ if (!points.length)
+ break;
+ }
+ return result;
+ }
+ function intersect2(a, b, edge, bbox2) {
+ return edge & 8 ? [a[0] + (b[0] - a[0]) * (bbox2[3] - a[1]) / (b[1] - a[1]), bbox2[3]] : (
+ // top
+ edge & 4 ? [a[0] + (b[0] - a[0]) * (bbox2[1] - a[1]) / (b[1] - a[1]), bbox2[1]] : (
+ // bottom
+ edge & 2 ? [bbox2[2], a[1] + (b[1] - a[1]) * (bbox2[2] - a[0]) / (b[0] - a[0])] : (
+ // right
+ edge & 1 ? [bbox2[0], a[1] + (b[1] - a[1]) * (bbox2[0] - a[0]) / (b[0] - a[0])] : (
+ // left
+ null
+ )
+ )
+ )
+ );
+ }
+ function bitCode2(p, bbox2) {
+ var code = 0;
+ if (p[0] < bbox2[0])
+ code |= 1;
+ else if (p[0] > bbox2[2])
+ code |= 2;
+ if (p[1] < bbox2[1])
+ code |= 4;
+ else if (p[1] > bbox2[3])
+ code |= 8;
+ return code;
+ }
+ }
+ });
+
+ // node_modules/which-polygon/index.js
+ var require_which_polygon = __commonJS({
+ "node_modules/which-polygon/index.js"(exports2, module2) {
+ "use strict";
+ var rbush = require_rbush();
+ var lineclip2 = require_lineclip();
+ module2.exports = whichPolygon5;
+ function whichPolygon5(data) {
+ var bboxes = [];
+ for (var i2 = 0; i2 < data.features.length; i2++) {
+ var feature3 = data.features[i2];
+ var coords = feature3.geometry.coordinates;
+ if (feature3.geometry.type === "Polygon") {
+ bboxes.push(treeItem(coords, feature3.properties));
+ } else if (feature3.geometry.type === "MultiPolygon") {
+ for (var j2 = 0; j2 < coords.length; j2++) {
+ bboxes.push(treeItem(coords[j2], feature3.properties));
+ }
+ }
+ }
+ var tree = rbush().load(bboxes);
+ function query(p, multi) {
+ var output = [], result = tree.search({
+ minX: p[0],
+ minY: p[1],
+ maxX: p[0],
+ maxY: p[1]
+ });
+ for (var i3 = 0; i3 < result.length; i3++) {
+ if (insidePolygon(result[i3].coords, p)) {
+ if (multi)
+ output.push(result[i3].props);
+ else
+ return result[i3].props;
+ }
+ }
+ return multi && output.length ? output : null;
+ }
+ query.tree = tree;
+ query.bbox = function queryBBox(bbox2) {
+ var output = [];
+ var result = tree.search({
+ minX: bbox2[0],
+ minY: bbox2[1],
+ maxX: bbox2[2],
+ maxY: bbox2[3]
+ });
+ for (var i3 = 0; i3 < result.length; i3++) {
+ if (polygonIntersectsBBox(result[i3].coords, bbox2)) {
+ output.push(result[i3].props);
+ }
+ }
+ return output;
+ };
+ return query;
+ }
+ function polygonIntersectsBBox(polygon2, bbox2) {
+ var bboxCenter = [
+ (bbox2[0] + bbox2[2]) / 2,
+ (bbox2[1] + bbox2[3]) / 2
+ ];
+ if (insidePolygon(polygon2, bboxCenter))
+ return true;
+ for (var i2 = 0; i2 < polygon2.length; i2++) {
+ if (lineclip2(polygon2[i2], bbox2).length > 0)
+ return true;
+ }
+ return false;
+ }
+ function insidePolygon(rings, p) {
+ var inside = false;
+ for (var i2 = 0, len = rings.length; i2 < len; i2++) {
+ var ring = rings[i2];
+ for (var j2 = 0, len2 = ring.length, k = len2 - 1; j2 < len2; k = j2++) {
+ if (rayIntersect(p, ring[j2], ring[k]))
+ inside = !inside;
+ }
+ }
+ return inside;
+ }
+ 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];
+ }
+ function treeItem(coords, props) {
+ var item = {
+ minX: Infinity,
+ minY: Infinity,
+ maxX: -Infinity,
+ maxY: -Infinity,
+ coords,
+ props
+ };
+ for (var i2 = 0; i2 < coords[0].length; i2++) {
+ var p = coords[0][i2];
+ 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]);
+ }
+ return item;
+ }
+ }
+ });
+
+ // node_modules/wgs84/index.js
+ var require_wgs84 = __commonJS({
+ "node_modules/wgs84/index.js"(exports2, module2) {
+ module2.exports.RADIUS = 6378137;
+ module2.exports.FLATTENING = 1 / 298.257223563;
+ module2.exports.POLAR_RADIUS = 63567523142e-4;
+ }
+ });
+
+ // node_modules/@mapbox/geojson-area/index.js
+ var require_geojson_area = __commonJS({
+ "node_modules/@mapbox/geojson-area/index.js"(exports2, module2) {
+ var wgs84 = require_wgs84();
+ module2.exports.geometry = geometry;
+ module2.exports.ring = ringArea;
+ function geometry(_) {
+ var area = 0, i2;
+ switch (_.type) {
+ case "Polygon":
+ return polygonArea(_.coordinates);
+ case "MultiPolygon":
+ for (i2 = 0; i2 < _.coordinates.length; i2++) {
+ area += polygonArea(_.coordinates[i2]);
+ }
+ return area;
+ case "Point":
+ case "MultiPoint":
+ case "LineString":
+ case "MultiLineString":
+ return 0;
+ case "GeometryCollection":
+ for (i2 = 0; i2 < _.geometries.length; i2++) {
+ area += geometry(_.geometries[i2]);
+ }
+ return area;
+ }
+ }
+ function polygonArea(coords) {
+ var area = 0;
+ if (coords && coords.length > 0) {
+ area += Math.abs(ringArea(coords[0]));
+ for (var i2 = 1; i2 < coords.length; i2++) {
+ area -= Math.abs(ringArea(coords[i2]));
+ }
+ }
+ return area;
+ }
+ function ringArea(coords) {
+ var p1, p2, p3, lowerIndex, middleIndex, upperIndex, i2, area = 0, coordsLength = coords.length;
+ if (coordsLength > 2) {
+ for (i2 = 0; i2 < coordsLength; i2++) {
+ if (i2 === coordsLength - 2) {
+ lowerIndex = coordsLength - 2;
+ middleIndex = coordsLength - 1;
+ upperIndex = 0;
+ } else if (i2 === coordsLength - 1) {
+ lowerIndex = coordsLength - 1;
+ middleIndex = 0;
+ upperIndex = 1;
+ } else {
+ lowerIndex = i2;
+ middleIndex = i2 + 1;
+ upperIndex = i2 + 2;
+ }
+ 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;
+ }
+ return area;
+ }
+ function rad(_) {
+ return _ * Math.PI / 180;
+ }
+ }
+ });
+
+ // node_modules/circle-to-polygon/input-validation/validateCenter.js
+ var require_validateCenter = __commonJS({
+ "node_modules/circle-to-polygon/input-validation/validateCenter.js"(exports2) {
+ exports2.validateCenter = function validateCenter(center) {
+ var validCenterLengths = [2, 3];
+ if (!Array.isArray(center) || !validCenterLengths.includes(center.length)) {
+ throw new Error("ERROR! Center has to be an array of length two or three");
+ }
+ var [lng, lat] = center;
+ if (typeof lng !== "number" || typeof lat !== "number") {
+ throw new Error(
+ `ERROR! Longitude and Latitude has to be numbers but where ${typeof lng} and ${typeof lat}`
+ );
+ }
+ if (lng > 180 || lng < -180) {
+ throw new Error(`ERROR! Longitude has to be between -180 and 180 but was ${lng}`);
+ }
+ if (lat > 90 || lat < -90) {
+ throw new Error(`ERROR! Latitude has to be between -90 and 90 but was ${lat}`);
+ }
+ };
+ }
+ });
+
+ // node_modules/circle-to-polygon/input-validation/validateRadius.js
+ var require_validateRadius = __commonJS({
+ "node_modules/circle-to-polygon/input-validation/validateRadius.js"(exports2) {
+ exports2.validateRadius = function validateRadius(radius) {
+ if (typeof radius !== "number") {
+ throw new Error(`ERROR! Radius has to be a positive number but was: ${typeof radius}`);
+ }
+ if (radius <= 0) {
+ throw new Error(`ERROR! Radius has to be a positive number but was: ${radius}`);
+ }
+ };
+ }
+ });
+
+ // node_modules/circle-to-polygon/input-validation/validateNumberOfEdges.js
+ var require_validateNumberOfEdges = __commonJS({
+ "node_modules/circle-to-polygon/input-validation/validateNumberOfEdges.js"(exports2) {
+ exports2.validateNumberOfEdges = function validateNumberOfEdges(numberOfEdges) {
+ if (typeof numberOfEdges !== "number") {
+ const ARGUMENT_TYPE = Array.isArray(numberOfEdges) ? "array" : typeof numberOfEdges;
+ throw new Error(`ERROR! Number of edges has to be a number but was: ${ARGUMENT_TYPE}`);
+ }
+ if (numberOfEdges < 3) {
+ throw new Error(`ERROR! Number of edges has to be at least 3 but was: ${numberOfEdges}`);
+ }
+ };
+ }
+ });
+
+ // node_modules/circle-to-polygon/input-validation/validateEarthRadius.js
+ var require_validateEarthRadius = __commonJS({
+ "node_modules/circle-to-polygon/input-validation/validateEarthRadius.js"(exports2) {
+ exports2.validateEarthRadius = function validateEarthRadius(earthRadius2) {
+ if (typeof earthRadius2 !== "number") {
+ const ARGUMENT_TYPE = Array.isArray(earthRadius2) ? "array" : typeof earthRadius2;
+ throw new Error(`ERROR! Earth radius has to be a number but was: ${ARGUMENT_TYPE}`);
+ }
+ if (earthRadius2 <= 0) {
+ throw new Error(`ERROR! Earth radius has to be a positive number but was: ${earthRadius2}`);
+ }
+ };
+ }
+ });
+
+ // node_modules/circle-to-polygon/input-validation/validateBearing.js
+ var require_validateBearing = __commonJS({
+ "node_modules/circle-to-polygon/input-validation/validateBearing.js"(exports2) {
+ exports2.validateBearing = function validateBearing(bearing) {
+ if (typeof bearing !== "number") {
+ const ARGUMENT_TYPE = Array.isArray(bearing) ? "array" : typeof bearing;
+ throw new Error(`ERROR! Bearing has to be a number but was: ${ARGUMENT_TYPE}`);
+ }
+ };
+ }
+ });
+
+ // node_modules/circle-to-polygon/input-validation/index.js
+ var require_input_validation = __commonJS({
+ "node_modules/circle-to-polygon/input-validation/index.js"(exports2) {
+ var validateCenter = require_validateCenter().validateCenter;
+ var validateRadius = require_validateRadius().validateRadius;
+ var validateNumberOfEdges = require_validateNumberOfEdges().validateNumberOfEdges;
+ var validateEarthRadius = require_validateEarthRadius().validateEarthRadius;
+ var validateBearing = require_validateBearing().validateBearing;
+ function validateInput({ center, radius, numberOfEdges, earthRadius: earthRadius2, bearing }) {
+ validateCenter(center);
+ validateRadius(radius);
+ validateNumberOfEdges(numberOfEdges);
+ validateEarthRadius(earthRadius2);
+ validateBearing(bearing);
+ }
+ exports2.validateCenter = validateCenter;
+ exports2.validateRadius = validateRadius;
+ exports2.validateNumberOfEdges = validateNumberOfEdges;
+ exports2.validateEarthRadius = validateEarthRadius;
+ exports2.validateBearing = validateBearing;
+ exports2.validateInput = validateInput;
+ }
+ });
+
+ // node_modules/circle-to-polygon/index.js
+ var require_circle_to_polygon = __commonJS({
+ "node_modules/circle-to-polygon/index.js"(exports2, module2) {
+ "use strict";
+ var { validateInput } = require_input_validation();
+ var defaultEarthRadius = 6378137;
+ function toRadians(angleInDegrees) {
+ return angleInDegrees * Math.PI / 180;
+ }
+ function toDegrees(angleInRadians) {
+ return angleInRadians * 180 / Math.PI;
+ }
+ function offset(c1, distance, earthRadius2, bearing) {
+ var lat1 = toRadians(c1[1]);
+ var lon1 = toRadians(c1[0]);
+ var dByR = distance / earthRadius2;
+ 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)];
+ }
+ module2.exports = function circleToPolygon2(center, radius, options2) {
+ var n2 = getNumberOfEdges(options2);
+ var earthRadius2 = getEarthRadius(options2);
+ var bearing = getBearing(options2);
+ var direction = getDirection(options2);
+ validateInput({ center, radius, numberOfEdges: n2, earthRadius: earthRadius2, bearing });
+ var start2 = toRadians(bearing);
+ var coordinates = [];
+ for (var i2 = 0; i2 < n2; ++i2) {
+ coordinates.push(
+ offset(
+ center,
+ radius,
+ earthRadius2,
+ start2 + direction * 2 * Math.PI * -i2 / n2
+ )
+ );
+ }
+ coordinates.push(coordinates[0]);
+ return {
+ type: "Polygon",
+ coordinates: [coordinates]
+ };
+ };
+ function getNumberOfEdges(options2) {
+ if (isUndefinedOrNull(options2)) {
+ return 32;
+ } else if (isObjectNotArray(options2)) {
+ var numberOfEdges = options2.numberOfEdges;
+ return numberOfEdges === void 0 ? 32 : numberOfEdges;
+ }
+ return options2;
+ }
+ function getEarthRadius(options2) {
+ if (isUndefinedOrNull(options2)) {
+ return defaultEarthRadius;
+ } else if (isObjectNotArray(options2)) {
+ var earthRadius2 = options2.earthRadius;
+ return earthRadius2 === void 0 ? defaultEarthRadius : earthRadius2;
+ }
+ return defaultEarthRadius;
+ }
+ function getDirection(options2) {
+ if (isObjectNotArray(options2) && options2.rightHandRule) {
+ return -1;
+ }
+ return 1;
+ }
+ function getBearing(options2) {
+ if (isUndefinedOrNull(options2)) {
+ return 0;
+ } else if (isObjectNotArray(options2)) {
+ var bearing = options2.bearing;
+ return bearing === void 0 ? 0 : bearing;
+ }
+ return 0;
+ }
+ function isObjectNotArray(argument) {
+ return argument !== null && typeof argument === "object" && !Array.isArray(argument);
+ }
+ function isUndefinedOrNull(argument) {
+ return argument === null || argument === void 0;
+ }
+ }
+ });
+
+ // node_modules/polygon-clipping/dist/polygon-clipping.umd.js
+ var require_polygon_clipping_umd = __commonJS({
+ "node_modules/polygon-clipping/dist/polygon-clipping.umd.js"(exports2, module2) {
+ (function(global3, factory) {
+ typeof exports2 === "object" && typeof module2 !== "undefined" ? module2.exports = factory() : typeof define === "function" && define.amd ? define(factory) : (global3 = typeof globalThis !== "undefined" ? globalThis : global3 || self, global3.polygonClipping = factory());
+ })(exports2, function() {
+ "use strict";
+ function _classCallCheck(instance, Constructor) {
+ if (!(instance instanceof Constructor)) {
+ throw new TypeError("Cannot call a class as a function");
+ }
+ }
+ function _defineProperties(target, props) {
+ for (var i2 = 0; i2 < props.length; i2++) {
+ var descriptor = props[i2];
+ descriptor.enumerable = descriptor.enumerable || false;
+ descriptor.configurable = true;
+ if ("value" in descriptor)
+ descriptor.writable = true;
+ Object.defineProperty(target, descriptor.key, descriptor);
+ }
+ }
+ function _createClass(Constructor, protoProps, staticProps) {
+ if (protoProps)
+ _defineProperties(Constructor.prototype, protoProps);
+ if (staticProps)
+ _defineProperties(Constructor, staticProps);
+ return Constructor;
+ }
+ var Node = (
+ /** @class */
+ function() {
+ function Node2(key, data) {
+ this.next = null;
+ this.key = key;
+ this.data = data;
+ this.left = null;
+ this.right = null;
+ }
+ return Node2;
+ }()
+ );
+ function DEFAULT_COMPARE(a, b) {
+ return a > b ? 1 : a < b ? -1 : 0;
+ }
+ function splay(i2, t, comparator) {
+ var N = new Node(null, null);
+ var l = N;
+ var r = N;
+ while (true) {
+ var cmp2 = comparator(i2, t.key);
+ if (cmp2 < 0) {
+ if (t.left === null)
+ break;
+ if (comparator(i2, t.left.key) < 0) {
+ var y = t.left;
+ t.left = y.right;
+ y.right = t;
+ t = y;
+ if (t.left === null)
+ break;
+ }
+ r.left = t;
+ r = t;
+ t = t.left;
+ } else if (cmp2 > 0) {
+ if (t.right === null)
+ break;
+ if (comparator(i2, t.right.key) > 0) {
+ var y = t.right;
+ t.right = y.left;
+ y.left = t;
+ t = y;
+ if (t.right === null)
+ break;
+ }
+ l.right = t;
+ l = t;
+ t = t.right;
+ } else
+ break;
+ }
+ l.right = t.left;
+ r.left = t.right;
+ t.left = N.right;
+ t.right = N.left;
+ return t;
+ }
+ function insert(i2, data, t, comparator) {
+ var node = new Node(i2, data);
+ if (t === null) {
+ node.left = node.right = null;
+ return node;
+ }
+ t = splay(i2, t, comparator);
+ var cmp2 = comparator(i2, t.key);
+ if (cmp2 < 0) {
+ node.left = t.left;
+ node.right = t;
+ t.left = null;
+ } else if (cmp2 >= 0) {
+ node.right = t.right;
+ node.left = t;
+ t.right = null;
+ }
+ return node;
+ }
+ function split(key, v, comparator) {
+ var left = null;
+ var right = null;
+ if (v) {
+ v = splay(key, v, comparator);
+ var cmp2 = comparator(v.key, key);
+ if (cmp2 === 0) {
+ left = v.left;
+ right = v.right;
+ } else if (cmp2 < 0) {
+ right = v.right;
+ v.right = null;
+ left = v;
+ } else {
+ left = v.left;
+ v.left = null;
+ right = v;
+ }
+ }
+ return {
+ left,
+ right
+ };
+ }
+ function merge3(left, right, comparator) {
+ if (right === null)
+ return left;
+ if (left === null)
+ return right;
+ right = splay(left.key, right, comparator);
+ right.left = left;
+ return right;
+ }
+ function printRow(root3, prefix, isTail, out, printNode) {
+ if (root3) {
+ out("" + prefix + (isTail ? "\u2514\u2500\u2500 " : "\u251C\u2500\u2500 ") + printNode(root3) + "\n");
+ var indent2 = prefix + (isTail ? " " : "\u2502 ");
+ if (root3.left)
+ printRow(root3.left, indent2, false, out, printNode);
+ if (root3.right)
+ printRow(root3.right, indent2, true, out, printNode);
+ }
+ }
+ var Tree = (
+ /** @class */
+ function() {
+ function Tree2(comparator) {
+ if (comparator === void 0) {
+ comparator = DEFAULT_COMPARE;
+ }
+ this._root = null;
+ this._size = 0;
+ this._comparator = comparator;
+ }
+ Tree2.prototype.insert = function(key, data) {
+ this._size++;
+ return this._root = insert(key, data, this._root, this._comparator);
+ };
+ Tree2.prototype.add = function(key, data) {
+ var node = new Node(key, data);
+ if (this._root === null) {
+ node.left = node.right = null;
+ this._size++;
+ this._root = node;
+ }
+ var comparator = this._comparator;
+ var t = splay(key, this._root, comparator);
+ var cmp2 = comparator(key, t.key);
+ if (cmp2 === 0)
+ this._root = t;
+ else {
+ if (cmp2 < 0) {
+ node.left = t.left;
+ node.right = t;
+ t.left = null;
+ } else if (cmp2 > 0) {
+ node.right = t.right;
+ node.left = t;
+ t.right = null;
+ }
+ this._size++;
+ this._root = node;
+ }
+ return this._root;
+ };
+ Tree2.prototype.remove = function(key) {
+ this._root = this._remove(key, this._root, this._comparator);
+ };
+ Tree2.prototype._remove = function(i2, t, comparator) {
+ var x;
+ if (t === null)
+ return null;
+ t = splay(i2, t, comparator);
+ var cmp2 = comparator(i2, t.key);
+ if (cmp2 === 0) {
+ if (t.left === null) {
+ x = t.right;
+ } else {
+ x = splay(i2, t.left, comparator);
+ x.right = t.right;
+ }
+ this._size--;
+ return x;
+ }
+ return t;
+ };
+ Tree2.prototype.pop = function() {
+ var node = this._root;
+ if (node) {
+ while (node.left) {
+ node = node.left;
+ }
+ 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;
+ };
+ Tree2.prototype.findStatic = function(key) {
+ var current = this._root;
+ var compare = this._comparator;
+ while (current) {
+ var cmp2 = compare(key, current.key);
+ if (cmp2 === 0)
+ return current;
+ else if (cmp2 < 0)
+ current = current.left;
+ else
+ current = current.right;
+ }
+ return null;
+ };
+ Tree2.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 this._root;
+ };
+ Tree2.prototype.contains = function(key) {
+ var current = this._root;
+ var compare = this._comparator;
+ while (current) {
+ var cmp2 = compare(key, current.key);
+ if (cmp2 === 0)
+ return true;
+ else if (cmp2 < 0)
+ current = current.left;
+ else
+ current = current.right;
+ }
+ return false;
+ };
+ Tree2.prototype.forEach = function(visitor, ctx) {
+ var current = this._root;
+ var Q = [];
+ var done = false;
+ 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;
+ }
+ }
+ return this;
+ };
+ Tree2.prototype.range = function(low, high, fn, ctx) {
+ var Q = [];
+ var compare = this._comparator;
+ var node = this._root;
+ var cmp2;
+ while (Q.length !== 0 || node) {
+ if (node) {
+ Q.push(node);
+ node = node.left;
+ } else {
+ node = Q.pop();
+ cmp2 = compare(node.key, high);
+ if (cmp2 > 0) {
+ break;
+ } else if (compare(node.key, low) >= 0) {
+ if (fn.call(ctx, node))
+ return this;
+ }
+ node = node.right;
+ }
+ }
+ return this;
+ };
+ Tree2.prototype.keys = function() {
+ var keys = [];
+ this.forEach(function(_a) {
+ var key = _a.key;
+ return keys.push(key);
+ });
+ return keys;
+ };
+ Tree2.prototype.values = function() {
+ var values = [];
+ this.forEach(function(_a) {
+ var data = _a.data;
+ return values.push(data);
+ });
+ return values;
+ };
+ Tree2.prototype.min = function() {
+ if (this._root)
+ return this.minNode(this._root).key;
+ return null;
+ };
+ Tree2.prototype.max = function() {
+ if (this._root)
+ return this.maxNode(this._root).key;
+ return null;
+ };
+ Tree2.prototype.minNode = function(t) {
+ if (t === void 0) {
+ t = this._root;
+ }
+ if (t)
+ while (t.left) {
+ t = t.left;
+ }
+ return t;
+ };
+ Tree2.prototype.maxNode = function(t) {
+ if (t === void 0) {
+ t = this._root;
+ }
+ if (t)
+ while (t.right) {
+ t = t.right;
+ }
+ return t;
+ };
+ Tree2.prototype.at = function(index2) {
+ var current = this._root;
+ var done = false;
+ var i2 = 0;
+ var Q = [];
+ while (!done) {
+ if (current) {
+ Q.push(current);
+ current = current.left;
+ } else {
+ if (Q.length > 0) {
+ current = Q.pop();
+ if (i2 === index2)
+ return current;
+ i2++;
+ current = current.right;
+ } else
+ done = true;
+ }
+ }
+ return null;
+ };
+ Tree2.prototype.next = function(d) {
+ var root3 = this._root;
+ var successor = null;
+ if (d.right) {
+ successor = d.right;
+ while (successor.left) {
+ successor = successor.left;
+ }
+ return successor;
+ }
+ var comparator = this._comparator;
+ while (root3) {
+ var cmp2 = comparator(d.key, root3.key);
+ if (cmp2 === 0)
+ break;
+ else if (cmp2 < 0) {
+ successor = root3;
+ root3 = root3.left;
+ } else
+ root3 = root3.right;
+ }
+ return successor;
+ };
+ Tree2.prototype.prev = function(d) {
+ var root3 = this._root;
+ var predecessor = null;
+ if (d.left !== null) {
+ predecessor = d.left;
+ while (predecessor.right) {
+ predecessor = predecessor.right;
+ }
+ return predecessor;
+ }
+ var comparator = this._comparator;
+ while (root3) {
+ var cmp2 = comparator(d.key, root3.key);
+ if (cmp2 === 0)
+ break;
+ else if (cmp2 < 0)
+ root3 = root3.left;
+ else {
+ predecessor = root3;
+ root3 = root3.right;
+ }
+ }
+ return predecessor;
+ };
+ Tree2.prototype.clear = function() {
+ this._root = null;
+ this._size = 0;
+ return this;
+ };
+ Tree2.prototype.toList = function() {
+ return toList(this._root);
+ };
+ Tree2.prototype.load = function(keys, values, presort) {
+ if (values === void 0) {
+ values = [];
+ }
+ if (presort === void 0) {
+ presort = false;
+ }
+ var size = keys.length;
+ var comparator = this._comparator;
+ if (presort)
+ sort(keys, values, 0, size - 1, comparator);
+ if (this._root === null) {
+ this._root = loadRecursive(keys, values, 0, size);
+ this._size = size;
+ } else {
+ var mergedList = mergeLists(this.toList(), createList(keys, values), comparator);
+ size = this._size + size;
+ this._root = sortedListToBST({
+ head: mergedList
+ }, 0, size);
+ }
+ return this;
+ };
+ Tree2.prototype.isEmpty = function() {
+ return this._root === null;
+ };
+ Object.defineProperty(Tree2.prototype, "size", {
+ get: function get4() {
+ return this._size;
+ },
+ enumerable: true,
+ configurable: true
+ });
+ Object.defineProperty(Tree2.prototype, "root", {
+ get: function get4() {
+ return this._root;
+ },
+ enumerable: true,
+ configurable: true
+ });
+ Tree2.prototype.toString = function(printNode) {
+ if (printNode === void 0) {
+ printNode = function printNode2(n2) {
+ return String(n2.key);
+ };
+ }
+ var out = [];
+ printRow(this._root, "", true, function(v) {
+ return out.push(v);
+ }, printNode);
+ return out.join("");
+ };
+ Tree2.prototype.update = function(key, newKey, newData) {
+ var comparator = this._comparator;
+ var _a = split(key, this._root, comparator), left = _a.left, right = _a.right;
+ if (comparator(key, newKey) < 0) {
+ right = insert(newKey, newData, right, comparator);
+ } else {
+ left = insert(newKey, newData, left, comparator);
+ }
+ this._root = merge3(left, right, comparator);
+ };
+ Tree2.prototype.split = function(key) {
+ return split(key, this._root, this._comparator);
+ };
+ return Tree2;
+ }()
+ );
+ function loadRecursive(keys, values, start2, end) {
+ var size = end - start2;
+ if (size > 0) {
+ var middle = start2 + Math.floor(size / 2);
+ var key = keys[middle];
+ var data = values[middle];
+ var node = new Node(key, data);
+ node.left = loadRecursive(keys, values, start2, middle);
+ node.right = loadRecursive(keys, values, middle + 1, end);
+ return node;
+ }
+ return null;
+ }
+ function createList(keys, values) {
+ var head = new Node(null, null);
+ var p = head;
+ for (var i2 = 0; i2 < keys.length; i2++) {
+ p = p.next = new Node(keys[i2], values[i2]);
+ }
+ p.next = null;
+ return head.next;
+ }
+ function toList(root3) {
+ var current = root3;
+ var Q = [];
+ var done = false;
+ var head = new Node(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;
+ }
+ }
+ p.next = null;
+ return head.next;
+ }
+ function sortedListToBST(list, start2, end) {
+ var size = end - start2;
+ if (size > 0) {
+ var middle = start2 + Math.floor(size / 2);
+ var left = sortedListToBST(list, start2, middle);
+ var root3 = list.head;
+ root3.left = left;
+ list.head = list.head.next;
+ root3.right = sortedListToBST(list, middle + 1, end);
+ return root3;
+ }
+ return null;
+ }
+ function mergeLists(l1, l2, compare) {
+ var head = new Node(null, null);
+ 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;
+ }
+ if (p1 !== null) {
+ p.next = p1;
+ } else if (p2 !== null) {
+ p.next = p2;
+ }
+ return head.next;
+ }
+ function sort(keys, values, left, right, compare) {
+ if (left >= right)
+ return;
+ var pivot = keys[left + right >> 1];
+ var i2 = left - 1;
+ var j2 = right + 1;
+ while (true) {
+ do {
+ i2++;
+ } while (compare(keys[i2], pivot) < 0);
+ do {
+ j2--;
+ } while (compare(keys[j2], pivot) > 0);
+ if (i2 >= j2)
+ break;
+ var tmp = keys[i2];
+ keys[i2] = keys[j2];
+ keys[j2] = tmp;
+ tmp = values[i2];
+ values[i2] = values[j2];
+ values[j2] = tmp;
+ }
+ sort(keys, values, left, j2, compare);
+ sort(keys, values, j2 + 1, right, compare);
+ }
+ var isInBbox = function isInBbox2(bbox2, point2) {
+ return bbox2.ll.x <= point2.x && point2.x <= bbox2.ur.x && bbox2.ll.y <= point2.y && point2.y <= bbox2.ur.y;
+ };
+ var getBboxOverlap = function getBboxOverlap2(b1, b2) {
+ 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;
+ 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;
+ 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;
+ return {
+ ll: {
+ x: lowerX,
+ y: lowerY
+ },
+ ur: {
+ x: upperX,
+ y: upperY
+ }
+ };
+ };
+ var epsilon3 = Number.EPSILON;
+ if (epsilon3 === void 0)
+ epsilon3 = Math.pow(2, -52);
+ var EPSILON_SQ = epsilon3 * epsilon3;
+ var cmp = function cmp2(a, b) {
+ if (-epsilon3 < a && a < epsilon3) {
+ if (-epsilon3 < b && b < epsilon3) {
+ return 0;
+ }
+ }
+ var ab = a - b;
+ if (ab * ab < EPSILON_SQ * a * b) {
+ return 0;
+ }
+ return a < b ? -1 : 1;
+ };
+ var PtRounder = /* @__PURE__ */ function() {
+ function PtRounder2() {
+ _classCallCheck(this, PtRounder2);
+ this.reset();
+ }
+ _createClass(PtRounder2, [{
+ 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)
+ };
+ }
+ }]);
+ return PtRounder2;
+ }();
+ var CoordRounder = /* @__PURE__ */ function() {
+ function CoordRounder2() {
+ _classCallCheck(this, CoordRounder2);
+ this.tree = new Tree();
+ this.round(0);
+ }
+ _createClass(CoordRounder2, [{
+ key: "round",
+ value: function round(coord2) {
+ var node = this.tree.add(coord2);
+ var prevNode = this.tree.prev(node);
+ if (prevNode !== null && cmp(node.key, prevNode.key) === 0) {
+ this.tree.remove(coord2);
+ return prevNode.key;
+ }
+ var nextNode = this.tree.next(node);
+ if (nextNode !== null && cmp(node.key, nextNode.key) === 0) {
+ this.tree.remove(coord2);
+ return nextNode.key;
+ }
+ return coord2;
+ }
+ }]);
+ return CoordRounder2;
+ }();
+ var rounder = new PtRounder();
+ var crossProduct = function crossProduct2(a, b) {
+ return a.x * b.y - a.y * b.x;
+ };
+ var dotProduct = function dotProduct2(a, b) {
+ return a.x * b.x + a.y * b.y;
+ };
+ var compareVectorAngles = function compareVectorAngles2(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);
+ };
+ var length = function length2(v) {
+ return Math.sqrt(dotProduct(v, v));
+ };
+ var sineOfAngle = function sineOfAngle2(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);
+ };
+ var cosineOfAngle = function cosineOfAngle2(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);
+ };
+ var horizontalIntersection = function horizontalIntersection2(pt, v, y) {
+ if (v.y === 0)
+ return null;
+ return {
+ x: pt.x + v.x / v.y * (y - pt.y),
+ y
+ };
+ };
+ var verticalIntersection = function verticalIntersection2(pt, v, x) {
+ if (v.x === 0)
+ return null;
+ return {
+ x,
+ y: pt.y + v.y / v.x * (x - pt.x)
+ };
+ };
+ var intersection = function intersection2(pt1, v1, pt2, v2) {
+ 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);
+ 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;
+ var x12 = pt1.x + d2 * v1.x, x2 = pt2.x + d1 * v2.x;
+ var y12 = pt1.y + d2 * v1.y, y2 = pt2.y + d1 * v2.y;
+ var x = (x12 + x2) / 2;
+ var y = (y12 + y2) / 2;
+ return {
+ x,
+ y
+ };
+ };
+ var SweepEvent = /* @__PURE__ */ function() {
+ _createClass(SweepEvent2, null, [{
+ key: "compare",
+ // for ordering sweep events in the sweep event queue
+ value: function compare(a, b) {
+ var ptCmp = SweepEvent2.comparePoints(a.point, b.point);
+ if (ptCmp !== 0)
+ return ptCmp;
+ if (a.point !== b.point)
+ a.link(b);
+ if (a.isLeft !== b.isLeft)
+ return a.isLeft ? 1 : -1;
+ return Segment.compare(a.segment, b.segment);
+ }
+ // for ordering points in sweep line order
+ }, {
+ 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)
+ }]);
+ function SweepEvent2(point2, isLeft) {
+ _classCallCheck(this, SweepEvent2);
+ if (point2.events === void 0)
+ point2.events = [this];
+ else
+ point2.events.push(this);
+ this.point = point2;
+ this.isLeft = isLeft;
+ }
+ _createClass(SweepEvent2, [{
+ key: "link",
+ value: function link2(other) {
+ if (other.point === this.point) {
+ throw new Error("Tried to link already linked events");
+ }
+ var otherEvents = other.point.events;
+ for (var i2 = 0, iMax = otherEvents.length; i2 < iMax; i2++) {
+ var evt = otherEvents[i2];
+ 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. */
+ }, {
+ key: "checkForConsuming",
+ value: function checkForConsuming() {
+ var numEvents = this.point.events.length;
+ for (var i2 = 0; i2 < numEvents; i2++) {
+ var evt1 = this.point.events[i2];
+ if (evt1.segment.consumedBy !== void 0)
+ continue;
+ for (var j2 = i2 + 1; j2 < numEvents; j2++) {
+ var evt2 = this.point.events[j2];
+ if (evt2.consumedBy !== void 0)
+ continue;
+ if (evt1.otherSE.point.events !== evt2.otherSE.point.events)
+ continue;
+ evt1.segment.consume(evt2.segment);
+ }
+ }
+ }
+ }, {
+ key: "getAvailableLinkedEvents",
+ value: function getAvailableLinkedEvents() {
+ var events = [];
+ for (var i2 = 0, iMax = this.point.events.length; i2 < iMax; i2++) {
+ var evt = this.point.events[i2];
+ if (evt !== this && !evt.segment.ringOut && evt.segment.isInResult()) {
+ events.push(evt);
+ }
+ }
+ 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;
+ var cache = /* @__PURE__ */ new Map();
+ var fillCache = function fillCache2(linkedEvent) {
+ var nextEvent = linkedEvent.otherSE;
+ cache.set(linkedEvent, {
+ sine: sineOfAngle(_this.point, baseEvent.point, nextEvent.point),
+ cosine: cosineOfAngle(_this.point, baseEvent.point, nextEvent.point)
+ });
+ };
+ 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;
+ var _cache$get2 = cache.get(b), bsine = _cache$get2.sine, bcosine = _cache$get2.cosine;
+ if (asine >= 0 && bsine >= 0) {
+ if (acosine < bcosine)
+ return 1;
+ if (acosine > bcosine)
+ return -1;
+ return 0;
+ }
+ if (asine < 0 && bsine < 0) {
+ if (acosine < bcosine)
+ return -1;
+ if (acosine > bcosine)
+ return 1;
+ return 0;
+ }
+ if (bsine < asine)
+ return -1;
+ if (bsine > asine)
+ return 1;
+ return 0;
+ };
+ }
+ }]);
+ return SweepEvent2;
+ }();
+ var segmentId = 0;
+ var Segment = /* @__PURE__ */ function() {
+ _createClass(Segment2, null, [{
+ key: "compare",
+ /* 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;
+ 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;
+ if (alx < blx) {
+ if (bly < aly && bly < ary)
+ return 1;
+ if (bly > aly && bly > ary)
+ return -1;
+ var aCmpBLeft = a.comparePoint(b.leftSE.point);
+ if (aCmpBLeft < 0)
+ return 1;
+ if (aCmpBLeft > 0)
+ return -1;
+ var bCmpARight = b.comparePoint(a.rightSE.point);
+ if (bCmpARight !== 0)
+ return bCmpARight;
+ return -1;
+ }
+ if (alx > blx) {
+ if (aly < bly && aly < bry)
+ return -1;
+ if (aly > bly && aly > bry)
+ return 1;
+ var bCmpALeft = b.comparePoint(a.leftSE.point);
+ if (bCmpALeft !== 0)
+ return bCmpALeft;
+ var aCmpBRight = a.comparePoint(b.rightSE.point);
+ if (aCmpBRight < 0)
+ return 1;
+ if (aCmpBRight > 0)
+ return -1;
+ return 1;
+ }
+ if (aly < bly)
+ return -1;
+ if (aly > bly)
+ return 1;
+ if (arx < brx) {
+ var _bCmpARight = b.comparePoint(a.rightSE.point);
+ if (_bCmpARight !== 0)
+ return _bCmpARight;
+ }
+ if (arx > brx) {
+ var _aCmpBRight = a.comparePoint(b.rightSE.point);
+ if (_aCmpBRight < 0)
+ return 1;
+ if (_aCmpBRight > 0)
+ return -1;
+ }
+ if (arx !== brx) {
+ 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;
+ }
+ if (arx > brx)
+ return 1;
+ if (arx < brx)
+ return -1;
+ if (ary < bry)
+ return -1;
+ if (ary > bry)
+ return 1;
+ if (a.id < b.id)
+ return -1;
+ if (a.id > b.id)
+ return 1;
+ return 0;
+ }
+ /* Warning: a reference to ringWindings input will be stored,
+ * and possibly will be later modified */
+ }]);
+ function Segment2(leftSE, rightSE, rings, windings) {
+ _classCallCheck(this, Segment2);
+ 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;
+ }
+ _createClass(Segment2, [{
+ 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 bbox2() {
+ var y12 = this.leftSE.point.y;
+ var y2 = this.rightSE.point.y;
+ return {
+ ll: {
+ x: this.leftSE.point.x,
+ y: y12 < y2 ? y12 : y2
+ },
+ ur: {
+ x: this.rightSE.point.x,
+ y: y12 > y2 ? y12 : y2
+ }
+ };
+ }
+ /* 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
+ };
+ }
+ }, {
+ 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)
+ */
+ }, {
+ key: "comparePoint",
+ value: function comparePoint(point2) {
+ if (this.isAnEndpoint(point2))
+ return 0;
+ var lPt = this.leftSE.point;
+ var rPt = this.rightSE.point;
+ var v = this.vector();
+ if (lPt.x === rPt.x) {
+ if (point2.x === lPt.x)
+ return 0;
+ return point2.x < lPt.x ? 1 : -1;
+ }
+ var yDist = (point2.y - lPt.y) / v.y;
+ var xFromYDist = lPt.x + yDist * v.x;
+ if (point2.x === xFromYDist)
+ return 0;
+ var xDist = (point2.x - lPt.x) / v.x;
+ var yFromXDist = lPt.y + xDist * v.y;
+ if (point2.y === yFromXDist)
+ return 0;
+ return point2.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.
+ */
+ }, {
+ key: "getIntersection",
+ value: function getIntersection(other) {
+ var tBbox = this.bbox();
+ var oBbox = other.bbox();
+ var bboxOverlap = getBboxOverlap(tBbox, oBbox);
+ if (bboxOverlap === null)
+ return null;
+ var tlp = this.leftSE.point;
+ var trp = this.rightSE.point;
+ var olp = other.leftSE.point;
+ var orp = other.rightSE.point;
+ 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;
+ if (touchesThisLSE && touchesOtherLSE) {
+ if (touchesThisRSE && !touchesOtherRSE)
+ return trp;
+ if (!touchesThisRSE && touchesOtherRSE)
+ return orp;
+ return null;
+ }
+ if (touchesThisLSE) {
+ if (touchesOtherRSE) {
+ if (tlp.x === orp.x && tlp.y === orp.y)
+ return null;
+ }
+ return tlp;
+ }
+ if (touchesOtherLSE) {
+ if (touchesThisRSE) {
+ if (trp.x === olp.x && trp.y === olp.y)
+ return null;
+ }
+ return olp;
+ }
+ if (touchesThisRSE && touchesOtherRSE)
+ return null;
+ if (touchesThisRSE)
+ return trp;
+ if (touchesOtherRSE)
+ return orp;
+ var pt = intersection(tlp, this.vector(), olp, other.vector());
+ if (pt === null)
+ return null;
+ if (!isInBbox(bboxOverlap, pt))
+ return null;
+ 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
+ */
+ }, {
+ key: "split",
+ value: function split2(point2) {
+ var newEvents = [];
+ var alreadyLinked = point2.events !== void 0;
+ var newLeftSE = new SweepEvent(point2, true);
+ var newRightSE = new SweepEvent(point2, false);
+ var oldRightSE = this.rightSE;
+ this.replaceRightSE(newRightSE);
+ newEvents.push(newRightSE);
+ newEvents.push(newLeftSE);
+ var newSeg = new Segment2(newLeftSE, oldRightSE, this.rings.slice(), this.windings.slice());
+ if (SweepEvent.comparePoints(newSeg.leftSE.point, newSeg.rightSE.point) > 0) {
+ newSeg.swapEvents();
+ }
+ if (SweepEvent.comparePoints(this.leftSE.point, this.rightSE.point) > 0) {
+ this.swapEvents();
+ }
+ if (alreadyLinked) {
+ newLeftSE.checkForConsuming();
+ newRightSE.checkForConsuming();
+ }
+ 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 i2 = 0, iMax = this.windings.length; i2 < iMax; i2++) {
+ this.windings[i2] *= -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;
+ }
+ while (consumee.consumedBy) {
+ consumee = consumee.consumedBy;
+ }
+ var cmp2 = Segment2.compare(consumer, consumee);
+ if (cmp2 === 0)
+ return;
+ if (cmp2 > 0) {
+ var tmp = consumer;
+ consumer = consumee;
+ consumee = tmp;
+ }
+ if (consumer.prev === consumee) {
+ var _tmp = consumer;
+ consumer = consumee;
+ consumee = _tmp;
+ }
+ for (var i2 = 0, iMax = consumee.rings.length; i2 < iMax; i2++) {
+ var ring = consumee.rings[i2];
+ var winding = consumee.windings[i2];
+ var index2 = consumer.rings.indexOf(ring);
+ if (index2 === -1) {
+ consumer.rings.push(ring);
+ consumer.windings.push(winding);
+ } else
+ consumer.windings[index2] += winding;
+ }
+ consumee.rings = null;
+ consumee.windings = null;
+ consumee.consumedBy = consumer;
+ consumee.leftSE.consumedBy = consumer.leftSE;
+ consumee.rightSE.consumedBy = consumer.rightSE;
+ }
+ /* The first segment previous segment chain that is in the result */
+ }, {
+ key: "prevInResult",
+ value: function prevInResult() {
+ if (this._prevInResult !== void 0)
+ 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 !== void 0)
+ 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 !== void 0)
+ 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;
+ for (var i2 = 0, iMax = this.rings.length; i2 < iMax; i2++) {
+ var ring = this.rings[i2];
+ var winding = this.windings[i2];
+ var index2 = ringsAfter.indexOf(ring);
+ if (index2 === -1) {
+ ringsAfter.push(ring);
+ windingsAfter.push(winding);
+ } else
+ windingsAfter[index2] += winding;
+ }
+ var polysAfter = [];
+ var polysExclude = [];
+ for (var _i = 0, _iMax = ringsAfter.length; _i < _iMax; _i++) {
+ if (windingsAfter[_i] === 0)
+ continue;
+ 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);
+ var _index = polysAfter.indexOf(_ring.poly);
+ if (_index !== -1)
+ polysAfter.splice(_index, 1);
+ }
+ }
+ for (var _i2 = 0, _iMax2 = polysAfter.length; _i2 < _iMax2; _i2++) {
+ var mp = polysAfter[_i2].multiPoly;
+ if (mpsAfter.indexOf(mp) === -1)
+ mpsAfter.push(mp);
+ }
+ return this._afterState;
+ }
+ /* Is this segment part of the final result? */
+ }, {
+ key: "isInResult",
+ value: function isInResult() {
+ if (this.consumedBy)
+ return false;
+ if (this._isInResult !== void 0)
+ return this._isInResult;
+ var mpsBefore = this.beforeState().multiPolys;
+ var mpsAfter = this.afterState().multiPolys;
+ switch (operation.type) {
+ case "union": {
+ var noBefores = mpsBefore.length === 0;
+ var noAfters = mpsAfter.length === 0;
+ this._isInResult = noBefores !== noAfters;
+ break;
+ }
+ case "intersection": {
+ var least;
+ var most;
+ if (mpsBefore.length < mpsAfter.length) {
+ least = mpsBefore.length;
+ most = mpsAfter.length;
+ } else {
+ least = mpsAfter.length;
+ most = mpsBefore.length;
+ }
+ this._isInResult = most === operation.numMultiPolys && least < most;
+ break;
+ }
+ case "xor": {
+ var diff = Math.abs(mpsBefore.length - mpsAfter.length);
+ this._isInResult = diff % 2 === 1;
+ break;
+ }
+ case "difference": {
+ var isJustSubject = function isJustSubject2(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));
+ }
+ return this._isInResult;
+ }
+ }], [{
+ key: "fromRing",
+ value: function fromRing(pt1, pt2, ring) {
+ var leftPt, rightPt, winding;
+ var cmpPts = SweepEvent.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, "]"));
+ var leftSE = new SweepEvent(leftPt, true);
+ var rightSE = new SweepEvent(rightPt, false);
+ return new Segment2(leftSE, rightSE, [ring], [winding]);
+ }
+ }]);
+ return Segment2;
+ }();
+ var RingIn = /* @__PURE__ */ function() {
+ function RingIn2(geomRing, poly, isExterior) {
+ _classCallCheck(this, RingIn2);
+ if (!Array.isArray(geomRing) || geomRing.length === 0) {
+ throw new Error("Input geometry is not a valid Polygon or MultiPolygon");
+ }
+ this.poly = poly;
+ this.isExterior = isExterior;
+ this.segments = [];
+ if (typeof geomRing[0][0] !== "number" || typeof geomRing[0][1] !== "number") {
+ throw new Error("Input geometry is not a valid Polygon or MultiPolygon");
+ }
+ 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 (var i2 = 1, iMax = geomRing.length; i2 < iMax; i2++) {
+ if (typeof geomRing[i2][0] !== "number" || typeof geomRing[i2][1] !== "number") {
+ throw new Error("Input geometry is not a valid Polygon or MultiPolygon");
+ }
+ var point2 = rounder.round(geomRing[i2][0], geomRing[i2][1]);
+ if (point2.x === prevPoint.x && point2.y === prevPoint.y)
+ continue;
+ this.segments.push(Segment.fromRing(prevPoint, point2, this));
+ if (point2.x < this.bbox.ll.x)
+ this.bbox.ll.x = point2.x;
+ if (point2.y < this.bbox.ll.y)
+ this.bbox.ll.y = point2.y;
+ if (point2.x > this.bbox.ur.x)
+ this.bbox.ur.x = point2.x;
+ if (point2.y > this.bbox.ur.y)
+ this.bbox.ur.y = point2.y;
+ prevPoint = point2;
+ }
+ if (firstPoint.x !== prevPoint.x || firstPoint.y !== prevPoint.y) {
+ this.segments.push(Segment.fromRing(prevPoint, firstPoint, this));
+ }
+ }
+ _createClass(RingIn2, [{
+ key: "getSweepEvents",
+ value: function getSweepEvents() {
+ var sweepEvents = [];
+ for (var i2 = 0, iMax = this.segments.length; i2 < iMax; i2++) {
+ var segment = this.segments[i2];
+ sweepEvents.push(segment.leftSE);
+ sweepEvents.push(segment.rightSE);
+ }
+ return sweepEvents;
+ }
+ }]);
+ return RingIn2;
+ }();
+ var PolyIn = /* @__PURE__ */ function() {
+ function PolyIn2(geomPoly, multiPoly) {
+ _classCallCheck(this, PolyIn2);
+ if (!Array.isArray(geomPoly)) {
+ throw new Error("Input geometry is not a valid Polygon or MultiPolygon");
+ }
+ this.exteriorRing = new RingIn(geomPoly[0], this, true);
+ 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
+ }
+ };
+ this.interiorRings = [];
+ for (var i2 = 1, iMax = geomPoly.length; i2 < iMax; i2++) {
+ var ring = new RingIn(geomPoly[i2], 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;
+ }
+ _createClass(PolyIn2, [{
+ key: "getSweepEvents",
+ value: function getSweepEvents() {
+ var sweepEvents = this.exteriorRing.getSweepEvents();
+ for (var i2 = 0, iMax = this.interiorRings.length; i2 < iMax; i2++) {
+ var ringSweepEvents = this.interiorRings[i2].getSweepEvents();
+ for (var j2 = 0, jMax = ringSweepEvents.length; j2 < jMax; j2++) {
+ sweepEvents.push(ringSweepEvents[j2]);
+ }
+ }
+ return sweepEvents;
+ }
+ }]);
+ return PolyIn2;
+ }();
+ var MultiPolyIn = /* @__PURE__ */ function() {
+ function MultiPolyIn2(geom, isSubject) {
+ _classCallCheck(this, MultiPolyIn2);
+ if (!Array.isArray(geom)) {
+ throw new Error("Input geometry is not a valid Polygon or MultiPolygon");
+ }
+ try {
+ if (typeof geom[0][0][0] === "number")
+ geom = [geom];
+ } catch (ex) {
+ }
+ this.polys = [];
+ this.bbox = {
+ ll: {
+ x: Number.POSITIVE_INFINITY,
+ y: Number.POSITIVE_INFINITY
+ },
+ ur: {
+ x: Number.NEGATIVE_INFINITY,
+ y: Number.NEGATIVE_INFINITY
+ }
+ };
+ for (var i2 = 0, iMax = geom.length; i2 < iMax; i2++) {
+ var poly = new PolyIn(geom[i2], 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;
+ }
+ _createClass(MultiPolyIn2, [{
+ key: "getSweepEvents",
+ value: function getSweepEvents() {
+ var sweepEvents = [];
+ for (var i2 = 0, iMax = this.polys.length; i2 < iMax; i2++) {
+ var polySweepEvents = this.polys[i2].getSweepEvents();
+ for (var j2 = 0, jMax = polySweepEvents.length; j2 < jMax; j2++) {
+ sweepEvents.push(polySweepEvents[j2]);
+ }
+ }
+ return sweepEvents;
+ }
+ }]);
+ return MultiPolyIn2;
+ }();
+ var RingOut = /* @__PURE__ */ function() {
+ _createClass(RingOut2, null, [{
+ key: "factory",
+ /* 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 i2 = 0, iMax = allSegments.length; i2 < iMax; i2++) {
+ var segment = allSegments[i2];
+ 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 = [];
+ while (true) {
+ prevEvent = event;
+ event = nextEvent;
+ events.push(event);
+ if (event.point === startingPoint)
+ break;
+ while (true) {
+ var availableLEs = event.getAvailableLinkedEvents();
+ 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, "]."));
+ }
+ if (availableLEs.length === 1) {
+ nextEvent = availableLEs[0].otherSE;
+ break;
+ }
+ var indexLE = null;
+ for (var j2 = 0, jMax = intersectionLEs.length; j2 < jMax; j2++) {
+ if (intersectionLEs[j2].point === event.point) {
+ indexLE = j2;
+ break;
+ }
+ }
+ if (indexLE !== null) {
+ var intersectionLE = intersectionLEs.splice(indexLE)[0];
+ var ringEvents = events.splice(intersectionLE.index);
+ ringEvents.unshift(ringEvents[0].otherSE);
+ ringsOut.push(new RingOut2(ringEvents.reverse()));
+ continue;
+ }
+ intersectionLEs.push({
+ index: events.length,
+ point: event.point
+ });
+ var comparator = event.getLeftmostComparator(prevEvent);
+ nextEvent = availableLEs.sort(comparator)[0].otherSE;
+ break;
+ }
+ }
+ ringsOut.push(new RingOut2(events));
+ }
+ return ringsOut;
+ }
+ }]);
+ function RingOut2(events) {
+ _classCallCheck(this, RingOut2);
+ this.events = events;
+ for (var i2 = 0, iMax = events.length; i2 < iMax; i2++) {
+ events[i2].segment.ringOut = this;
+ }
+ this.poly = null;
+ }
+ _createClass(RingOut2, [{
+ key: "getGeom",
+ value: function getGeom2() {
+ var prevPt = this.events[0].point;
+ var points = [prevPt];
+ for (var i2 = 1, iMax = this.events.length - 1; i2 < iMax; i2++) {
+ var _pt = this.events[i2].point;
+ var _nextPt = this.events[i2 + 1].point;
+ if (compareVectorAngles(_pt, prevPt, _nextPt) === 0)
+ continue;
+ points.push(_pt);
+ prevPt = _pt;
+ }
+ if (points.length === 1)
+ return null;
+ 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 = [];
+ 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 === void 0) {
+ var enclosing = this.enclosingRing();
+ this._isExteriorRing = enclosing ? !enclosing.isExteriorRing() : true;
+ }
+ return this._isExteriorRing;
+ }
+ }, {
+ key: "enclosingRing",
+ value: function enclosingRing() {
+ if (this._enclosingRing === void 0) {
+ this._enclosingRing = this._calcEnclosingRing();
+ }
+ return this._enclosingRing;
+ }
+ /* Returns the ring that encloses this one, if any */
+ }, {
+ key: "_calcEnclosingRing",
+ value: function _calcEnclosingRing() {
+ var leftMostEvt = this.events[0];
+ for (var i2 = 1, iMax = this.events.length; i2 < iMax; i2++) {
+ var evt = this.events[i2];
+ if (SweepEvent.compare(leftMostEvt, evt) > 0)
+ leftMostEvt = evt;
+ }
+ var prevSeg = leftMostEvt.segment.prevInResult();
+ var prevPrevSeg = prevSeg ? prevSeg.prevInResult() : null;
+ while (true) {
+ if (!prevSeg)
+ return null;
+ if (!prevPrevSeg)
+ return prevSeg.ringOut;
+ if (prevPrevSeg.ringOut !== prevSeg.ringOut) {
+ if (prevPrevSeg.ringOut.enclosingRing() !== prevSeg.ringOut) {
+ return prevSeg.ringOut;
+ } else
+ return prevSeg.ringOut.enclosingRing();
+ }
+ prevSeg = prevPrevSeg.prevInResult();
+ prevPrevSeg = prevSeg ? prevSeg.prevInResult() : null;
+ }
+ }
+ }]);
+ return RingOut2;
+ }();
+ var PolyOut = /* @__PURE__ */ function() {
+ function PolyOut2(exteriorRing) {
+ _classCallCheck(this, PolyOut2);
+ this.exteriorRing = exteriorRing;
+ exteriorRing.poly = this;
+ this.interiorRings = [];
+ }
+ _createClass(PolyOut2, [{
+ key: "addInterior",
+ value: function addInterior(ring) {
+ this.interiorRings.push(ring);
+ ring.poly = this;
+ }
+ }, {
+ key: "getGeom",
+ value: function getGeom2() {
+ var geom = [this.exteriorRing.getGeom()];
+ if (geom[0] === null)
+ return null;
+ for (var i2 = 0, iMax = this.interiorRings.length; i2 < iMax; i2++) {
+ var ringGeom = this.interiorRings[i2].getGeom();
+ if (ringGeom === null)
+ continue;
+ geom.push(ringGeom);
+ }
+ return geom;
+ }
+ }]);
+ return PolyOut2;
+ }();
+ var MultiPolyOut = /* @__PURE__ */ function() {
+ function MultiPolyOut2(rings) {
+ _classCallCheck(this, MultiPolyOut2);
+ this.rings = rings;
+ this.polys = this._composePolys(rings);
+ }
+ _createClass(MultiPolyOut2, [{
+ key: "getGeom",
+ value: function getGeom2() {
+ var geom = [];
+ for (var i2 = 0, iMax = this.polys.length; i2 < iMax; i2++) {
+ var polyGeom = this.polys[i2].getGeom();
+ if (polyGeom === null)
+ continue;
+ geom.push(polyGeom);
+ }
+ return geom;
+ }
+ }, {
+ key: "_composePolys",
+ value: function _composePolys(rings) {
+ var polys = [];
+ for (var i2 = 0, iMax = rings.length; i2 < iMax; i2++) {
+ var ring = rings[i2];
+ 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);
+ }
+ }
+ return polys;
+ }
+ }]);
+ return MultiPolyOut2;
+ }();
+ var SweepLine = /* @__PURE__ */ function() {
+ function SweepLine2(queue) {
+ var comparator = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : Segment.compare;
+ _classCallCheck(this, SweepLine2);
+ this.queue = queue;
+ this.tree = new Tree(comparator);
+ this.segments = [];
+ }
+ _createClass(SweepLine2, [{
+ key: "process",
+ value: function process2(event) {
+ var segment = event.segment;
+ var newEvents = [];
+ if (event.consumedBy) {
+ if (event.isLeft)
+ this.queue.remove(event.otherSE);
+ else
+ this.tree.remove(segment);
+ return newEvents;
+ }
+ 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 = void 0;
+ var nextSeg = void 0;
+ while (prevSeg === void 0) {
+ prevNode = this.tree.prev(prevNode);
+ if (prevNode === null)
+ prevSeg = null;
+ else if (prevNode.key.consumedBy === void 0)
+ prevSeg = prevNode.key;
+ }
+ while (nextSeg === void 0) {
+ nextNode = this.tree.next(nextNode);
+ if (nextNode === null)
+ nextSeg = null;
+ else if (nextNode.key.consumedBy === void 0)
+ nextSeg = nextNode.key;
+ }
+ if (event.isLeft) {
+ var prevMySplitter = null;
+ if (prevSeg) {
+ var prevInter = prevSeg.getIntersection(segment);
+ if (prevInter !== null) {
+ if (!segment.isAnEndpoint(prevInter))
+ prevMySplitter = prevInter;
+ if (!prevSeg.isAnEndpoint(prevInter)) {
+ var newEventsFromSplit = this._splitSafely(prevSeg, prevInter);
+ for (var i2 = 0, iMax = newEventsFromSplit.length; i2 < iMax; i2++) {
+ newEvents.push(newEventsFromSplit[i2]);
+ }
+ }
+ }
+ }
+ var nextMySplitter = null;
+ if (nextSeg) {
+ var nextInter = nextSeg.getIntersection(segment);
+ if (nextInter !== null) {
+ if (!segment.isAnEndpoint(nextInter))
+ nextMySplitter = nextInter;
+ if (!nextSeg.isAnEndpoint(nextInter)) {
+ var _newEventsFromSplit = this._splitSafely(nextSeg, nextInter);
+ for (var _i = 0, _iMax = _newEventsFromSplit.length; _i < _iMax; _i++) {
+ newEvents.push(_newEventsFromSplit[_i]);
+ }
+ }
+ }
+ }
+ 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;
+ }
+ this.queue.remove(segment.rightSE);
+ newEvents.push(segment.rightSE);
+ var _newEventsFromSplit2 = segment.split(mySplitter);
+ for (var _i2 = 0, _iMax2 = _newEventsFromSplit2.length; _i2 < _iMax2; _i2++) {
+ newEvents.push(_newEventsFromSplit2[_i2]);
+ }
+ }
+ if (newEvents.length > 0) {
+ this.tree.remove(segment);
+ newEvents.push(event);
+ } else {
+ this.segments.push(segment);
+ segment.prev = prevSeg;
+ }
+ } else {
+ if (prevSeg && nextSeg) {
+ var inter = prevSeg.getIntersection(nextSeg);
+ if (inter !== null) {
+ if (!prevSeg.isAnEndpoint(inter)) {
+ var _newEventsFromSplit3 = this._splitSafely(prevSeg, inter);
+ for (var _i3 = 0, _iMax3 = _newEventsFromSplit3.length; _i3 < _iMax3; _i3++) {
+ newEvents.push(_newEventsFromSplit3[_i3]);
+ }
+ }
+ if (!nextSeg.isAnEndpoint(inter)) {
+ var _newEventsFromSplit4 = this._splitSafely(nextSeg, inter);
+ for (var _i4 = 0, _iMax4 = _newEventsFromSplit4.length; _i4 < _iMax4; _i4++) {
+ newEvents.push(_newEventsFromSplit4[_i4]);
+ }
+ }
+ }
+ }
+ this.tree.remove(segment);
+ }
+ return newEvents;
+ }
+ /* 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) {
+ this.tree.remove(seg);
+ var rightSE = seg.rightSE;
+ this.queue.remove(rightSE);
+ var newEvents = seg.split(pt);
+ newEvents.push(rightSE);
+ if (seg.consumedBy === void 0)
+ this.tree.insert(seg);
+ return newEvents;
+ }
+ }]);
+ return SweepLine2;
+ }();
+ var POLYGON_CLIPPING_MAX_QUEUE_SIZE = typeof process !== "undefined" && process.env.POLYGON_CLIPPING_MAX_QUEUE_SIZE || 1e6;
+ var POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS = typeof process !== "undefined" && process.env.POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS || 1e6;
+ var Operation = /* @__PURE__ */ function() {
+ function Operation2() {
+ _classCallCheck(this, Operation2);
+ }
+ _createClass(Operation2, [{
+ key: "run",
+ value: function run(type2, geom, moreGeoms) {
+ operation.type = type2;
+ rounder.reset();
+ var multipolys = [new MultiPolyIn(geom, true)];
+ for (var i2 = 0, iMax = moreGeoms.length; i2 < iMax; i2++) {
+ multipolys.push(new MultiPolyIn(moreGeoms[i2], false));
+ }
+ operation.numMultiPolys = multipolys.length;
+ if (operation.type === "difference") {
+ var subject = multipolys[0];
+ var _i = 1;
+ while (_i < multipolys.length) {
+ if (getBboxOverlap(multipolys[_i].bbox, subject.bbox) !== null)
+ _i++;
+ else
+ multipolys.splice(_i, 1);
+ }
+ }
+ if (operation.type === "intersection") {
+ for (var _i2 = 0, _iMax = multipolys.length; _i2 < _iMax; _i2++) {
+ var mpA = multipolys[_i2];
+ for (var j2 = _i2 + 1, jMax = multipolys.length; j2 < jMax; j2++) {
+ if (getBboxOverlap(mpA.bbox, multipolys[j2].bbox) === null)
+ return [];
+ }
+ }
+ }
+ var queue = new Tree(SweepEvent.compare);
+ for (var _i3 = 0, _iMax2 = multipolys.length; _i3 < _iMax2; _i3++) {
+ var sweepEvents = multipolys[_i3].getSweepEvents();
+ for (var _j = 0, _jMax = sweepEvents.length; _j < _jMax; _j++) {
+ queue.insert(sweepEvents[_j]);
+ if (queue.size > POLYGON_CLIPPING_MAX_QUEUE_SIZE) {
+ throw new Error("Infinite loop when putting segment endpoints in a priority queue (queue size too big). Please file a bug report.");
+ }
+ }
+ }
+ var sweepLine = new SweepLine(queue);
+ var prevQueueSize = queue.size;
+ var node = queue.pop();
+ while (node) {
+ var evt = node.key;
+ if (queue.size === prevQueueSize) {
+ 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) {
+ throw new Error("Infinite loop when passing sweep line over endpoints (queue size too big). Please file a bug report.");
+ }
+ if (sweepLine.segments.length > POLYGON_CLIPPING_MAX_SWEEPLINE_SEGMENTS) {
+ throw new Error("Infinite loop when passing sweep line over endpoints (too many sweep line segments). Please file a bug report.");
+ }
+ var newEvents = sweepLine.process(evt);
+ for (var _i4 = 0, _iMax3 = newEvents.length; _i4 < _iMax3; _i4++) {
+ var _evt = newEvents[_i4];
+ if (_evt.consumedBy === void 0)
+ queue.insert(_evt);
+ }
+ prevQueueSize = queue.size;
+ node = queue.pop();
+ }
+ rounder.reset();
+ var ringsOut = RingOut.factory(sweepLine.segments);
+ var result = new MultiPolyOut(ringsOut);
+ return result.getGeom();
+ }
+ }]);
+ return Operation2;
+ }();
+ var operation = new Operation();
+ var union = function union2(geom) {
+ for (var _len = arguments.length, moreGeoms = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {
+ moreGeoms[_key - 1] = arguments[_key];
+ }
+ return operation.run("union", geom, moreGeoms);
+ };
+ var intersection$1 = function intersection2(geom) {
+ for (var _len2 = arguments.length, moreGeoms = new Array(_len2 > 1 ? _len2 - 1 : 0), _key2 = 1; _key2 < _len2; _key2++) {
+ moreGeoms[_key2 - 1] = arguments[_key2];
+ }
+ return operation.run("intersection", geom, moreGeoms);
+ };
+ var xor = function xor2(geom) {
+ for (var _len3 = arguments.length, moreGeoms = new Array(_len3 > 1 ? _len3 - 1 : 0), _key3 = 1; _key3 < _len3; _key3++) {
+ moreGeoms[_key3 - 1] = arguments[_key3];
+ }
+ return operation.run("xor", geom, moreGeoms);
+ };
+ var difference = function difference2(subjectGeom) {
+ for (var _len4 = arguments.length, clippingGeoms = new Array(_len4 > 1 ? _len4 - 1 : 0), _key4 = 1; _key4 < _len4; _key4++) {
+ clippingGeoms[_key4 - 1] = arguments[_key4];
+ }
+ return operation.run("difference", subjectGeom, clippingGeoms);
+ };
+ var index = {
+ union,
+ intersection: intersection$1,
+ xor,
+ difference
+ };
+ return index;
+ });
+ }
+ });
+
+ // node_modules/geojson-precision/index.js
+ var require_geojson_precision = __commonJS({
+ "node_modules/geojson-precision/index.js"(exports2, module2) {
+ (function() {
+ function parse(t, coordinatePrecision, extrasPrecision) {
+ function point2(p) {
+ return p.map(function(e, index) {
+ if (index < 2) {
+ return 1 * e.toFixed(coordinatePrecision);
+ } else {
+ return 1 * e.toFixed(extrasPrecision);
+ }
+ });
+ }
+ function multi(l) {
+ return l.map(point2);
+ }
+ function poly(p) {
+ return p.map(multi);
+ }
+ function multiPoly(m) {
+ return m.map(poly);
+ }
+ function geometry(obj) {
+ if (!obj) {
+ return {};
+ }
+ switch (obj.type) {
+ case "Point":
+ obj.coordinates = point2(obj.coordinates);
+ return obj;
+ case "LineString":
+ case "MultiPoint":
+ obj.coordinates = multi(obj.coordinates);
+ return obj;
+ 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 {};
+ }
+ }
+ function feature3(obj) {
+ obj.geometry = geometry(obj.geometry);
+ return obj;
+ }
+ function featureCollection(f2) {
+ f2.features = f2.features.map(feature3);
+ return f2;
+ }
+ function geometryCollection(g) {
+ g.geometries = g.geometries.map(geometry);
+ return g;
+ }
+ if (!t) {
+ return t;
+ }
+ switch (t.type) {
+ case "Feature":
+ return feature3(t);
+ case "GeometryCollection":
+ return geometryCollection(t);
+ case "FeatureCollection":
+ return featureCollection(t);
+ case "Point":
+ case "LineString":
+ case "Polygon":
+ case "MultiPoint":
+ case "MultiPolygon":
+ case "MultiLineString":
+ return geometry(t);
+ default:
+ return t;
+ }
+ }
+ module2.exports = parse;
+ module2.exports.parse = parse;
+ })();
+ }
+ });
+
+ // node_modules/@aitodotai/json-stringify-pretty-compact/index.js
+ var require_json_stringify_pretty_compact = __commonJS({
+ "node_modules/@aitodotai/json-stringify-pretty-compact/index.js"(exports2, module2) {
+ function isObject3(obj) {
+ return typeof obj === "object" && obj !== null;
+ }
+ function forEach(obj, cb) {
+ if (Array.isArray(obj)) {
+ obj.forEach(cb);
+ } else if (isObject3(obj)) {
+ Object.keys(obj).forEach(function(key) {
+ var val = obj[key];
+ cb(val, key);
+ });
+ }
+ }
+ function getTreeDepth(obj) {
+ var depth = 0;
+ if (Array.isArray(obj) || isObject3(obj)) {
+ forEach(obj, function(val) {
+ if (Array.isArray(val) || isObject3(val)) {
+ var tmpDepth = getTreeDepth(val);
+ if (tmpDepth > depth) {
+ depth = tmpDepth;
+ }
+ }
+ });
+ return depth + 1;
+ }
+ return depth;
+ }
+ function stringify3(obj, options2) {
+ options2 = options2 || {};
+ var indent2 = JSON.stringify([1], null, get4(options2, "indent", 2)).slice(2, -3);
+ var addMargin = get4(options2, "margins", false);
+ var addArrayMargin = get4(options2, "arrayMargins", false);
+ var addObjectMargin = get4(options2, "objectMargins", false);
+ var maxLength = indent2 === "" ? Infinity : get4(options2, "maxLength", 80);
+ var maxNesting = get4(options2, "maxNesting", Infinity);
+ return function _stringify(obj2, currentIndent, reserved) {
+ if (obj2 && typeof obj2.toJSON === "function") {
+ obj2 = obj2.toJSON();
+ }
+ var string = JSON.stringify(obj2);
+ if (string === void 0) {
+ return string;
+ }
+ var length = maxLength - currentIndent.length - reserved;
+ var treeDepth = getTreeDepth(obj2);
+ if (treeDepth <= maxNesting && string.length <= length) {
+ var prettified = prettify(string, {
+ addMargin,
+ addArrayMargin,
+ addObjectMargin
+ });
+ if (prettified.length <= length) {
+ return prettified;
+ }
+ }
+ if (isObject3(obj2)) {
+ var nextIndent = currentIndent + indent2;
+ var items = [];
+ var delimiters;
+ var comma = function(array2, index2) {
+ return index2 === array2.length - 1 ? 0 : 1;
+ };
+ if (Array.isArray(obj2)) {
+ for (var index = 0; index < obj2.length; index++) {
+ items.push(
+ _stringify(obj2[index], nextIndent, comma(obj2, index)) || "null"
+ );
+ }
+ delimiters = "[]";
+ } else {
+ Object.keys(obj2).forEach(function(key, index2, array2) {
+ var keyPart = JSON.stringify(key) + ": ";
+ var value = _stringify(
+ obj2[key],
+ nextIndent,
+ keyPart.length + comma(array2, index2)
+ );
+ if (value !== void 0) {
+ items.push(keyPart + value);
+ }
+ });
+ delimiters = "{}";
+ }
+ if (items.length > 0) {
+ return [
+ delimiters[0],
+ indent2 + items.join(",\n" + nextIndent),
+ delimiters[1]
+ ].join("\n" + currentIndent);
+ }
+ }
+ return string;
+ }(obj, "", 0);
+ }
+ var stringOrChar = /("(?:[^\\"]|\\.)*")|[:,\][}{]/g;
+ function prettify(string, options2) {
+ options2 = options2 || {};
+ var tokens = {
+ "{": "{",
+ "}": "}",
+ "[": "[",
+ "]": "]",
+ ",": ", ",
+ ":": ": "
+ };
+ if (options2.addMargin || options2.addObjectMargin) {
+ tokens["{"] = "{ ";
+ tokens["}"] = " }";
+ }
+ if (options2.addMargin || options2.addArrayMargin) {
+ tokens["["] = "[ ";
+ tokens["]"] = " ]";
+ }
+ return string.replace(stringOrChar, function(match, string2) {
+ return string2 ? match : tokens[match];
+ });
+ }
+ function get4(options2, name, defaultValue) {
+ return name in options2 ? options2[name] : defaultValue;
+ }
+ module2.exports = stringify3;
+ }
+ });
+
+ // node_modules/aes-js/index.js
+ var require_aes_js = __commonJS({
+ "node_modules/aes-js/index.js"(exports2, module2) {
+ (function(root3) {
+ "use strict";
+ function checkInt(value) {
+ return parseInt(value) === value;
+ }
+ function checkInts(arrayish) {
+ if (!checkInt(arrayish.length)) {
+ return false;
+ }
+ for (var i2 = 0; i2 < arrayish.length; i2++) {
+ if (!checkInt(arrayish[i2]) || arrayish[i2] < 0 || arrayish[i2] > 255) {
+ return false;
+ }
+ }
+ return true;
+ }
+ function coerceArray(arg, copy2) {
+ if (arg.buffer && arg.name === "Uint8Array") {
+ if (copy2) {
+ if (arg.slice) {
+ arg = arg.slice();
+ } else {
+ arg = Array.prototype.slice.call(arg);
+ }
+ }
+ return arg;
+ }
+ if (Array.isArray(arg)) {
+ if (!checkInts(arg)) {
+ throw new Error("Array contains invalid value: " + arg);
+ }
+ return new Uint8Array(arg);
+ }
+ if (checkInt(arg.length) && checkInts(arg)) {
+ return new Uint8Array(arg);
+ }
+ throw new Error("unsupported array-like object");
+ }
+ function createArray(length) {
+ return new Uint8Array(length);
+ }
+ 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);
+ }
+ }
+ targetArray.set(sourceArray, targetStart);
+ }
+ var convertUtf8 = function() {
+ function toBytes(text2) {
+ var result = [], i2 = 0;
+ text2 = encodeURI(text2);
+ while (i2 < text2.length) {
+ var c = text2.charCodeAt(i2++);
+ if (c === 37) {
+ result.push(parseInt(text2.substr(i2, 2), 16));
+ i2 += 2;
+ } else {
+ result.push(c);
+ }
+ }
+ return coerceArray(result);
+ }
+ function fromBytes(bytes) {
+ var result = [], i2 = 0;
+ while (i2 < bytes.length) {
+ var c = bytes[i2];
+ if (c < 128) {
+ result.push(String.fromCharCode(c));
+ i2++;
+ } else if (c > 191 && c < 224) {
+ result.push(String.fromCharCode((c & 31) << 6 | bytes[i2 + 1] & 63));
+ i2 += 2;
+ } else {
+ result.push(String.fromCharCode((c & 15) << 12 | (bytes[i2 + 1] & 63) << 6 | bytes[i2 + 2] & 63));
+ i2 += 3;
+ }
+ }
+ return result.join("");
+ }
+ return {
+ toBytes,
+ fromBytes
+ };
+ }();
+ var convertHex = function() {
+ function toBytes(text2) {
+ var result = [];
+ for (var i2 = 0; i2 < text2.length; i2 += 2) {
+ result.push(parseInt(text2.substr(i2, 2), 16));
+ }
+ return result;
+ }
+ var Hex = "0123456789abcdef";
+ function fromBytes(bytes) {
+ var result = [];
+ for (var i2 = 0; i2 < bytes.length; i2++) {
+ var v = bytes[i2];
+ result.push(Hex[(v & 240) >> 4] + Hex[v & 15]);
+ }
+ return result.join("");
+ }
+ return {
+ toBytes,
+ fromBytes
+ };
+ }();
+ var numberOfRounds = { 16: 10, 24: 12, 32: 14 };
+ var rcon = [1, 2, 4, 8, 16, 32, 64, 128, 27, 54, 108, 216, 171, 77, 154, 47, 94, 188, 99, 198, 151, 53, 106, 212, 179, 125, 250, 239, 197, 145];
+ var S = [99, 124, 119, 123, 242, 107, 111, 197, 48, 1, 103, 43, 254, 215, 171, 118, 202, 130, 201, 125, 250, 89, 71, 240, 173, 212, 162, 175, 156, 164, 114, 192, 183, 253, 147, 38, 54, 63, 247, 204, 52, 165, 229, 241, 113, 216, 49, 21, 4, 199, 35, 195, 24, 150, 5, 154, 7, 18, 128, 226, 235, 39, 178, 117, 9, 131, 44, 26, 27, 110, 90, 160, 82, 59, 214, 179, 41, 227, 47, 132, 83, 209, 0, 237, 32, 252, 177, 91, 106, 203, 190, 57, 74, 76, 88, 207, 208, 239, 170, 251, 67, 77, 51, 133, 69, 249, 2, 127, 80, 60, 159, 168, 81, 163, 64, 143, 146, 157, 56, 245, 188, 182, 218, 33, 16, 255, 243, 210, 205, 12, 19, 236, 95, 151, 68, 23, 196, 167, 126, 61, 100, 93, 25, 115, 96, 129, 79, 220, 34, 42, 144, 136, 70, 238, 184, 20, 222, 94, 11, 219, 224, 50, 58, 10, 73, 6, 36, 92, 194, 211, 172, 98, 145, 149, 228, 121, 231, 200, 55, 109, 141, 213, 78, 169, 108, 86, 244, 234, 101, 122, 174, 8, 186, 120, 37, 46, 28, 166, 180, 198, 232, 221, 116, 31, 75, 189, 139, 138, 112, 62, 181, 102, 72, 3, 246, 14, 97, 53, 87, 185, 134, 193, 29, 158, 225, 248, 152, 17, 105, 217, 142, 148, 155, 30, 135, 233, 206, 85, 40, 223, 140, 161, 137, 13, 191, 230, 66, 104, 65, 153, 45, 15, 176, 84, 187, 22];
+ var Si = [82, 9, 106, 213, 48, 54, 165, 56, 191, 64, 163, 158, 129, 243, 215, 251, 124, 227, 57, 130, 155, 47, 255, 135, 52, 142, 67, 68, 196, 222, 233, 203, 84, 123, 148, 50, 166, 194, 35, 61, 238, 76, 149, 11, 66, 250, 195, 78, 8, 46, 161, 102, 40, 217, 36, 178, 118, 91, 162, 73, 109, 139, 209, 37, 114, 248, 246, 100, 134, 104, 152, 22, 212, 164, 92, 204, 93, 101, 182, 146, 108, 112, 72, 80, 253, 237, 185, 218, 94, 21, 70, 87, 167, 141, 157, 132, 144, 216, 171, 0, 140, 188, 211, 10, 247, 228, 88, 5, 184, 179, 69, 6, 208, 44, 30, 143, 202, 63, 15, 2, 193, 175, 189, 3, 1, 19, 138, 107, 58, 145, 17, 65, 79, 103, 220, 234, 151, 242, 207, 206, 240, 180, 230, 115, 150, 172, 116, 34, 231, 173, 53, 133, 226, 249, 55, 232, 28, 117, 223, 110, 71, 241, 26, 113, 29, 41, 197, 137, 111, 183, 98, 14, 170, 24, 190, 27, 252, 86, 62, 75, 198, 210, 121, 32, 154, 219, 192, 254, 120, 205, 90, 244, 31, 221, 168, 51, 136, 7, 199, 49, 177, 18, 16, 89, 39, 128, 236, 95, 96, 81, 127, 169, 25, 181, 74, 13, 45, 229, 122, 159, 147, 201, 156, 239, 160, 224, 59, 77, 174, 42, 245, 176, 200, 235, 187, 60, 131, 83, 153, 97, 23, 43, 4, 126, 186, 119, 214, 38, 225, 105, 20, 99, 85, 33, 12, 125];
+ var T1 = [3328402341, 4168907908, 4000806809, 4135287693, 4294111757, 3597364157, 3731845041, 2445657428, 1613770832, 33620227, 3462883241, 1445669757, 3892248089, 3050821474, 1303096294, 3967186586, 2412431941, 528646813, 2311702848, 4202528135, 4026202645, 2992200171, 2387036105, 4226871307, 1101901292, 3017069671, 1604494077, 1169141738, 597466303, 1403299063, 3832705686, 2613100635, 1974974402, 3791519004, 1033081774, 1277568618, 1815492186, 2118074177, 4126668546, 2211236943, 1748251740, 1369810420, 3521504564, 4193382664, 3799085459, 2883115123, 1647391059, 706024767, 134480908, 2512897874, 1176707941, 2646852446, 806885416, 932615841, 168101135, 798661301, 235341577, 605164086, 461406363, 3756188221, 3454790438, 1311188841, 2142417613, 3933566367, 302582043, 495158174, 1479289972, 874125870, 907746093, 3698224818, 3025820398, 1537253627, 2756858614, 1983593293, 3084310113, 2108928974, 1378429307, 3722699582, 1580150641, 327451799, 2790478837, 3117535592, 0, 3253595436, 1075847264, 3825007647, 2041688520, 3059440621, 3563743934, 2378943302, 1740553945, 1916352843, 2487896798, 2555137236, 2958579944, 2244988746, 3151024235, 3320835882, 1336584933, 3992714006, 2252555205, 2588757463, 1714631509, 293963156, 2319795663, 3925473552, 67240454, 4269768577, 2689618160, 2017213508, 631218106, 1269344483, 2723238387, 1571005438, 2151694528, 93294474, 1066570413, 563977660, 1882732616, 4059428100, 1673313503, 2008463041, 2950355573, 1109467491, 537923632, 3858759450, 4260623118, 3218264685, 2177748300, 403442708, 638784309, 3287084079, 3193921505, 899127202, 2286175436, 773265209, 2479146071, 1437050866, 4236148354, 2050833735, 3362022572, 3126681063, 840505643, 3866325909, 3227541664, 427917720, 2655997905, 2749160575, 1143087718, 1412049534, 999329963, 193497219, 2353415882, 3354324521, 1807268051, 672404540, 2816401017, 3160301282, 369822493, 2916866934, 3688947771, 1681011286, 1949973070, 336202270, 2454276571, 201721354, 1210328172, 3093060836, 2680341085, 3184776046, 1135389935, 3294782118, 965841320, 831886756, 3554993207, 4068047243, 3588745010, 2345191491, 1849112409, 3664604599, 26054028, 2983581028, 2622377682, 1235855840, 3630984372, 2891339514, 4092916743, 3488279077, 3395642799, 4101667470, 1202630377, 268961816, 1874508501, 4034427016, 1243948399, 1546530418, 941366308, 1470539505, 1941222599, 2546386513, 3421038627, 2715671932, 3899946140, 1042226977, 2521517021, 1639824860, 227249030, 260737669, 3765465232, 2084453954, 1907733956, 3429263018, 2420656344, 100860677, 4160157185, 470683154, 3261161891, 1781871967, 2924959737, 1773779408, 394692241, 2579611992, 974986535, 664706745, 3655459128, 3958962195, 731420851, 571543859, 3530123707, 2849626480, 126783113, 865375399, 765172662, 1008606754, 361203602, 3387549984, 2278477385, 2857719295, 1344809080, 2782912378, 59542671, 1503764984, 160008576, 437062935, 1707065306, 3622233649, 2218934982, 3496503480, 2185314755, 697932208, 1512910199, 504303377, 2075177163, 2824099068, 1841019862, 739644986];
+ var T2 = [2781242211, 2230877308, 2582542199, 2381740923, 234877682, 3184946027, 2984144751, 1418839493, 1348481072, 50462977, 2848876391, 2102799147, 434634494, 1656084439, 3863849899, 2599188086, 1167051466, 2636087938, 1082771913, 2281340285, 368048890, 3954334041, 3381544775, 201060592, 3963727277, 1739838676, 4250903202, 3930435503, 3206782108, 4149453988, 2531553906, 1536934080, 3262494647, 484572669, 2923271059, 1783375398, 1517041206, 1098792767, 49674231, 1334037708, 1550332980, 4098991525, 886171109, 150598129, 2481090929, 1940642008, 1398944049, 1059722517, 201851908, 1385547719, 1699095331, 1587397571, 674240536, 2704774806, 252314885, 3039795866, 151914247, 908333586, 2602270848, 1038082786, 651029483, 1766729511, 3447698098, 2682942837, 454166793, 2652734339, 1951935532, 775166490, 758520603, 3000790638, 4004797018, 4217086112, 4137964114, 1299594043, 1639438038, 3464344499, 2068982057, 1054729187, 1901997871, 2534638724, 4121318227, 1757008337, 0, 750906861, 1614815264, 535035132, 3363418545, 3988151131, 3201591914, 1183697867, 3647454910, 1265776953, 3734260298, 3566750796, 3903871064, 1250283471, 1807470800, 717615087, 3847203498, 384695291, 3313910595, 3617213773, 1432761139, 2484176261, 3481945413, 283769337, 100925954, 2180939647, 4037038160, 1148730428, 3123027871, 3813386408, 4087501137, 4267549603, 3229630528, 2315620239, 2906624658, 3156319645, 1215313976, 82966005, 3747855548, 3245848246, 1974459098, 1665278241, 807407632, 451280895, 251524083, 1841287890, 1283575245, 337120268, 891687699, 801369324, 3787349855, 2721421207, 3431482436, 959321879, 1469301956, 4065699751, 2197585534, 1199193405, 2898814052, 3887750493, 724703513, 2514908019, 2696962144, 2551808385, 3516813135, 2141445340, 1715741218, 2119445034, 2872807568, 2198571144, 3398190662, 700968686, 3547052216, 1009259540, 2041044702, 3803995742, 487983883, 1991105499, 1004265696, 1449407026, 1316239930, 504629770, 3683797321, 168560134, 1816667172, 3837287516, 1570751170, 1857934291, 4014189740, 2797888098, 2822345105, 2754712981, 936633572, 2347923833, 852879335, 1133234376, 1500395319, 3084545389, 2348912013, 1689376213, 3533459022, 3762923945, 3034082412, 4205598294, 133428468, 634383082, 2949277029, 2398386810, 3913789102, 403703816, 3580869306, 2297460856, 1867130149, 1918643758, 607656988, 4049053350, 3346248884, 1368901318, 600565992, 2090982877, 2632479860, 557719327, 3717614411, 3697393085, 2249034635, 2232388234, 2430627952, 1115438654, 3295786421, 2865522278, 3633334344, 84280067, 33027830, 303828494, 2747425121, 1600795957, 4188952407, 3496589753, 2434238086, 1486471617, 658119965, 3106381470, 953803233, 334231800, 3005978776, 857870609, 3151128937, 1890179545, 2298973838, 2805175444, 3056442267, 574365214, 2450884487, 550103529, 1233637070, 4289353045, 2018519080, 2057691103, 2399374476, 4166623649, 2148108681, 387583245, 3664101311, 836232934, 3330556482, 3100665960, 3280093505, 2955516313, 2002398509, 287182607, 3413881008, 4238890068, 3597515707, 975967766];
+ var T3 = [1671808611, 2089089148, 2006576759, 2072901243, 4061003762, 1807603307, 1873927791, 3310653893, 810573872, 16974337, 1739181671, 729634347, 4263110654, 3613570519, 2883997099, 1989864566, 3393556426, 2191335298, 3376449993, 2106063485, 4195741690, 1508618841, 1204391495, 4027317232, 2917941677, 3563566036, 2734514082, 2951366063, 2629772188, 2767672228, 1922491506, 3227229120, 3082974647, 4246528509, 2477669779, 644500518, 911895606, 1061256767, 4144166391, 3427763148, 878471220, 2784252325, 3845444069, 4043897329, 1905517169, 3631459288, 827548209, 356461077, 67897348, 3344078279, 593839651, 3277757891, 405286936, 2527147926, 84871685, 2595565466, 118033927, 305538066, 2157648768, 3795705826, 3945188843, 661212711, 2999812018, 1973414517, 152769033, 2208177539, 745822252, 439235610, 455947803, 1857215598, 1525593178, 2700827552, 1391895634, 994932283, 3596728278, 3016654259, 695947817, 3812548067, 795958831, 2224493444, 1408607827, 3513301457, 0, 3979133421, 543178784, 4229948412, 2982705585, 1542305371, 1790891114, 3410398667, 3201918910, 961245753, 1256100938, 1289001036, 1491644504, 3477767631, 3496721360, 4012557807, 2867154858, 4212583931, 1137018435, 1305975373, 861234739, 2241073541, 1171229253, 4178635257, 33948674, 2139225727, 1357946960, 1011120188, 2679776671, 2833468328, 1374921297, 2751356323, 1086357568, 2408187279, 2460827538, 2646352285, 944271416, 4110742005, 3168756668, 3066132406, 3665145818, 560153121, 271589392, 4279952895, 4077846003, 3530407890, 3444343245, 202643468, 322250259, 3962553324, 1608629855, 2543990167, 1154254916, 389623319, 3294073796, 2817676711, 2122513534, 1028094525, 1689045092, 1575467613, 422261273, 1939203699, 1621147744, 2174228865, 1339137615, 3699352540, 577127458, 712922154, 2427141008, 2290289544, 1187679302, 3995715566, 3100863416, 339486740, 3732514782, 1591917662, 186455563, 3681988059, 3762019296, 844522546, 978220090, 169743370, 1239126601, 101321734, 611076132, 1558493276, 3260915650, 3547250131, 2901361580, 1655096418, 2443721105, 2510565781, 3828863972, 2039214713, 3878868455, 3359869896, 928607799, 1840765549, 2374762893, 3580146133, 1322425422, 2850048425, 1823791212, 1459268694, 4094161908, 3928346602, 1706019429, 2056189050, 2934523822, 135794696, 3134549946, 2022240376, 628050469, 779246638, 472135708, 2800834470, 3032970164, 3327236038, 3894660072, 3715932637, 1956440180, 522272287, 1272813131, 3185336765, 2340818315, 2323976074, 1888542832, 1044544574, 3049550261, 1722469478, 1222152264, 50660867, 4127324150, 236067854, 1638122081, 895445557, 1475980887, 3117443513, 2257655686, 3243809217, 489110045, 2662934430, 3778599393, 4162055160, 2561878936, 288563729, 1773916777, 3648039385, 2391345038, 2493985684, 2612407707, 505560094, 2274497927, 3911240169, 3460925390, 1442818645, 678973480, 3749357023, 2358182796, 2717407649, 2306869641, 219617805, 3218761151, 3862026214, 1120306242, 1756942440, 1103331905, 2578459033, 762796589, 252780047, 2966125488, 1425844308, 3151392187, 372911126];
+ var T4 = [1667474886, 2088535288, 2004326894, 2071694838, 4075949567, 1802223062, 1869591006, 3318043793, 808472672, 16843522, 1734846926, 724270422, 4278065639, 3621216949, 2880169549, 1987484396, 3402253711, 2189597983, 3385409673, 2105378810, 4210693615, 1499065266, 1195886990, 4042263547, 2913856577, 3570689971, 2728590687, 2947541573, 2627518243, 2762274643, 1920112356, 3233831835, 3082273397, 4261223649, 2475929149, 640051788, 909531756, 1061110142, 4160160501, 3435941763, 875846760, 2779116625, 3857003729, 4059105529, 1903268834, 3638064043, 825316194, 353713962, 67374088, 3351728789, 589522246, 3284360861, 404236336, 2526454071, 84217610, 2593830191, 117901582, 303183396, 2155911963, 3806477791, 3958056653, 656894286, 2998062463, 1970642922, 151591698, 2206440989, 741110872, 437923380, 454765878, 1852748508, 1515908788, 2694904667, 1381168804, 993742198, 3604373943, 3014905469, 690584402, 3823320797, 791638366, 2223281939, 1398011302, 3520161977, 0, 3991743681, 538992704, 4244381667, 2981218425, 1532751286, 1785380564, 3419096717, 3200178535, 960056178, 1246420628, 1280103576, 1482221744, 3486468741, 3503319995, 4025428677, 2863326543, 4227536621, 1128514950, 1296947098, 859002214, 2240123921, 1162203018, 4193849577, 33687044, 2139062782, 1347481760, 1010582648, 2678045221, 2829640523, 1364325282, 2745433693, 1077985408, 2408548869, 2459086143, 2644360225, 943212656, 4126475505, 3166494563, 3065430391, 3671750063, 555836226, 269496352, 4294908645, 4092792573, 3537006015, 3452783745, 202118168, 320025894, 3974901699, 1600119230, 2543297077, 1145359496, 387397934, 3301201811, 2812801621, 2122220284, 1027426170, 1684319432, 1566435258, 421079858, 1936954854, 1616945344, 2172753945, 1330631070, 3705438115, 572679748, 707427924, 2425400123, 2290647819, 1179044492, 4008585671, 3099120491, 336870440, 3739122087, 1583276732, 185277718, 3688593069, 3772791771, 842159716, 976899700, 168435220, 1229577106, 101059084, 606366792, 1549591736, 3267517855, 3553849021, 2897014595, 1650632388, 2442242105, 2509612081, 3840161747, 2038008818, 3890688725, 3368567691, 926374254, 1835907034, 2374863873, 3587531953, 1313788572, 2846482505, 1819063512, 1448540844, 4109633523, 3941213647, 1701162954, 2054852340, 2930698567, 134748176, 3132806511, 2021165296, 623210314, 774795868, 471606328, 2795958615, 3031746419, 3334885783, 3907527627, 3722280097, 1953799400, 522133822, 1263263126, 3183336545, 2341176845, 2324333839, 1886425312, 1044267644, 3048588401, 1718004428, 1212733584, 50529542, 4143317495, 235803164, 1633788866, 892690282, 1465383342, 3115962473, 2256965911, 3250673817, 488449850, 2661202215, 3789633753, 4177007595, 2560144171, 286339874, 1768537042, 3654906025, 2391705863, 2492770099, 2610673197, 505291324, 2273808917, 3924369609, 3469625735, 1431699370, 673740880, 3755965093, 2358021891, 2711746649, 2307489801, 218961690, 3217021541, 3873845719, 1111672452, 1751693520, 1094828930, 2576986153, 757954394, 252645662, 2964376443, 1414855848, 3149649517, 370555436];
+ var T5 = [1374988112, 2118214995, 437757123, 975658646, 1001089995, 530400753, 2902087851, 1273168787, 540080725, 2910219766, 2295101073, 4110568485, 1340463100, 3307916247, 641025152, 3043140495, 3736164937, 632953703, 1172967064, 1576976609, 3274667266, 2169303058, 2370213795, 1809054150, 59727847, 361929877, 3211623147, 2505202138, 3569255213, 1484005843, 1239443753, 2395588676, 1975683434, 4102977912, 2572697195, 666464733, 3202437046, 4035489047, 3374361702, 2110667444, 1675577880, 3843699074, 2538681184, 1649639237, 2976151520, 3144396420, 4269907996, 4178062228, 1883793496, 2403728665, 2497604743, 1383856311, 2876494627, 1917518562, 3810496343, 1716890410, 3001755655, 800440835, 2261089178, 3543599269, 807962610, 599762354, 33778362, 3977675356, 2328828971, 2809771154, 4077384432, 1315562145, 1708848333, 101039829, 3509871135, 3299278474, 875451293, 2733856160, 92987698, 2767645557, 193195065, 1080094634, 1584504582, 3178106961, 1042385657, 2531067453, 3711829422, 1306967366, 2438237621, 1908694277, 67556463, 1615861247, 429456164, 3602770327, 2302690252, 1742315127, 2968011453, 126454664, 3877198648, 2043211483, 2709260871, 2084704233, 4169408201, 0, 159417987, 841739592, 504459436, 1817866830, 4245618683, 260388950, 1034867998, 908933415, 168810852, 1750902305, 2606453969, 607530554, 202008497, 2472011535, 3035535058, 463180190, 2160117071, 1641816226, 1517767529, 470948374, 3801332234, 3231722213, 1008918595, 303765277, 235474187, 4069246893, 766945465, 337553864, 1475418501, 2943682380, 4003061179, 2743034109, 4144047775, 1551037884, 1147550661, 1543208500, 2336434550, 3408119516, 3069049960, 3102011747, 3610369226, 1113818384, 328671808, 2227573024, 2236228733, 3535486456, 2935566865, 3341394285, 496906059, 3702665459, 226906860, 2009195472, 733156972, 2842737049, 294930682, 1206477858, 2835123396, 2700099354, 1451044056, 573804783, 2269728455, 3644379585, 2362090238, 2564033334, 2801107407, 2776292904, 3669462566, 1068351396, 742039012, 1350078989, 1784663195, 1417561698, 4136440770, 2430122216, 775550814, 2193862645, 2673705150, 1775276924, 1876241833, 3475313331, 3366754619, 270040487, 3902563182, 3678124923, 3441850377, 1851332852, 3969562369, 2203032232, 3868552805, 2868897406, 566021896, 4011190502, 3135740889, 1248802510, 3936291284, 699432150, 832877231, 708780849, 3332740144, 899835584, 1951317047, 4236429990, 3767586992, 866637845, 4043610186, 1106041591, 2144161806, 395441711, 1984812685, 1139781709, 3433712980, 3835036895, 2664543715, 1282050075, 3240894392, 1181045119, 2640243204, 25965917, 4203181171, 4211818798, 3009879386, 2463879762, 3910161971, 1842759443, 2597806476, 933301370, 1509430414, 3943906441, 3467192302, 3076639029, 3776767469, 2051518780, 2631065433, 1441952575, 404016761, 1942435775, 1408749034, 1610459739, 3745345300, 2017778566, 3400528769, 3110650942, 941896748, 3265478751, 371049330, 3168937228, 675039627, 4279080257, 967311729, 135050206, 3635733660, 1683407248, 2076935265, 3576870512, 1215061108, 3501741890];
+ var T6 = [1347548327, 1400783205, 3273267108, 2520393566, 3409685355, 4045380933, 2880240216, 2471224067, 1428173050, 4138563181, 2441661558, 636813900, 4233094615, 3620022987, 2149987652, 2411029155, 1239331162, 1730525723, 2554718734, 3781033664, 46346101, 310463728, 2743944855, 3328955385, 3875770207, 2501218972, 3955191162, 3667219033, 768917123, 3545789473, 692707433, 1150208456, 1786102409, 2029293177, 1805211710, 3710368113, 3065962831, 401639597, 1724457132, 3028143674, 409198410, 2196052529, 1620529459, 1164071807, 3769721975, 2226875310, 486441376, 2499348523, 1483753576, 428819965, 2274680428, 3075636216, 598438867, 3799141122, 1474502543, 711349675, 129166120, 53458370, 2592523643, 2782082824, 4063242375, 2988687269, 3120694122, 1559041666, 730517276, 2460449204, 4042459122, 2706270690, 3446004468, 3573941694, 533804130, 2328143614, 2637442643, 2695033685, 839224033, 1973745387, 957055980, 2856345839, 106852767, 1371368976, 4181598602, 1033297158, 2933734917, 1179510461, 3046200461, 91341917, 1862534868, 4284502037, 605657339, 2547432937, 3431546947, 2003294622, 3182487618, 2282195339, 954669403, 3682191598, 1201765386, 3917234703, 3388507166, 0, 2198438022, 1211247597, 2887651696, 1315723890, 4227665663, 1443857720, 507358933, 657861945, 1678381017, 560487590, 3516619604, 975451694, 2970356327, 261314535, 3535072918, 2652609425, 1333838021, 2724322336, 1767536459, 370938394, 182621114, 3854606378, 1128014560, 487725847, 185469197, 2918353863, 3106780840, 3356761769, 2237133081, 1286567175, 3152976349, 4255350624, 2683765030, 3160175349, 3309594171, 878443390, 1988838185, 3704300486, 1756818940, 1673061617, 3403100636, 272786309, 1075025698, 545572369, 2105887268, 4174560061, 296679730, 1841768865, 1260232239, 4091327024, 3960309330, 3497509347, 1814803222, 2578018489, 4195456072, 575138148, 3299409036, 446754879, 3629546796, 4011996048, 3347532110, 3252238545, 4270639778, 915985419, 3483825537, 681933534, 651868046, 2755636671, 3828103837, 223377554, 2607439820, 1649704518, 3270937875, 3901806776, 1580087799, 4118987695, 3198115200, 2087309459, 2842678573, 3016697106, 1003007129, 2802849917, 1860738147, 2077965243, 164439672, 4100872472, 32283319, 2827177882, 1709610350, 2125135846, 136428751, 3874428392, 3652904859, 3460984630, 3572145929, 3593056380, 2939266226, 824852259, 818324884, 3224740454, 930369212, 2801566410, 2967507152, 355706840, 1257309336, 4148292826, 243256656, 790073846, 2373340630, 1296297904, 1422699085, 3756299780, 3818836405, 457992840, 3099667487, 2135319889, 77422314, 1560382517, 1945798516, 788204353, 1521706781, 1385356242, 870912086, 325965383, 2358957921, 2050466060, 2388260884, 2313884476, 4006521127, 901210569, 3990953189, 1014646705, 1503449823, 1062597235, 2031621326, 3212035895, 3931371469, 1533017514, 350174575, 2256028891, 2177544179, 1052338372, 741876788, 1606591296, 1914052035, 213705253, 2334669897, 1107234197, 1899603969, 3725069491, 2631447780, 2422494913, 1635502980, 1893020342, 1950903388, 1120974935];
+ var T7 = [2807058932, 1699970625, 2764249623, 1586903591, 1808481195, 1173430173, 1487645946, 59984867, 4199882800, 1844882806, 1989249228, 1277555970, 3623636965, 3419915562, 1149249077, 2744104290, 1514790577, 459744698, 244860394, 3235995134, 1963115311, 4027744588, 2544078150, 4190530515, 1608975247, 2627016082, 2062270317, 1507497298, 2200818878, 567498868, 1764313568, 3359936201, 2305455554, 2037970062, 1047239e3, 1910319033, 1337376481, 2904027272, 2892417312, 984907214, 1243112415, 830661914, 861968209, 2135253587, 2011214180, 2927934315, 2686254721, 731183368, 1750626376, 4246310725, 1820824798, 4172763771, 3542330227, 48394827, 2404901663, 2871682645, 671593195, 3254988725, 2073724613, 145085239, 2280796200, 2779915199, 1790575107, 2187128086, 472615631, 3029510009, 4075877127, 3802222185, 4107101658, 3201631749, 1646252340, 4270507174, 1402811438, 1436590835, 3778151818, 3950355702, 3963161475, 4020912224, 2667994737, 273792366, 2331590177, 104699613, 95345982, 3175501286, 2377486676, 1560637892, 3564045318, 369057872, 4213447064, 3919042237, 1137477952, 2658625497, 1119727848, 2340947849, 1530455833, 4007360968, 172466556, 266959938, 516552836, 0, 2256734592, 3980931627, 1890328081, 1917742170, 4294704398, 945164165, 3575528878, 958871085, 3647212047, 2787207260, 1423022939, 775562294, 1739656202, 3876557655, 2530391278, 2443058075, 3310321856, 547512796, 1265195639, 437656594, 3121275539, 719700128, 3762502690, 387781147, 218828297, 3350065803, 2830708150, 2848461854, 428169201, 122466165, 3720081049, 1627235199, 648017665, 4122762354, 1002783846, 2117360635, 695634755, 3336358691, 4234721005, 4049844452, 3704280881, 2232435299, 574624663, 287343814, 612205898, 1039717051, 840019705, 2708326185, 793451934, 821288114, 1391201670, 3822090177, 376187827, 3113855344, 1224348052, 1679968233, 2361698556, 1058709744, 752375421, 2431590963, 1321699145, 3519142200, 2734591178, 188127444, 2177869557, 3727205754, 2384911031, 3215212461, 2648976442, 2450346104, 3432737375, 1180849278, 331544205, 3102249176, 4150144569, 2952102595, 2159976285, 2474404304, 766078933, 313773861, 2570832044, 2108100632, 1668212892, 3145456443, 2013908262, 418672217, 3070356634, 2594734927, 1852171925, 3867060991, 3473416636, 3907448597, 2614737639, 919489135, 164948639, 2094410160, 2997825956, 590424639, 2486224549, 1723872674, 3157750862, 3399941250, 3501252752, 3625268135, 2555048196, 3673637356, 1343127501, 4130281361, 3599595085, 2957853679, 1297403050, 81781910, 3051593425, 2283490410, 532201772, 1367295589, 3926170974, 895287692, 1953757831, 1093597963, 492483431, 3528626907, 1446242576, 1192455638, 1636604631, 209336225, 344873464, 1015671571, 669961897, 3375740769, 3857572124, 2973530695, 3747192018, 1933530610, 3464042516, 935293895, 3454686199, 2858115069, 1863638845, 3683022916, 4085369519, 3292445032, 875313188, 1080017571, 3279033885, 621591778, 1233856572, 2504130317, 24197544, 3017672716, 3835484340, 3247465558, 2220981195, 3060847922, 1551124588, 1463996600];
+ var T8 = [4104605777, 1097159550, 396673818, 660510266, 2875968315, 2638606623, 4200115116, 3808662347, 821712160, 1986918061, 3430322568, 38544885, 3856137295, 718002117, 893681702, 1654886325, 2975484382, 3122358053, 3926825029, 4274053469, 796197571, 1290801793, 1184342925, 3556361835, 2405426947, 2459735317, 1836772287, 1381620373, 3196267988, 1948373848, 3764988233, 3385345166, 3263785589, 2390325492, 1480485785, 3111247143, 3780097726, 2293045232, 548169417, 3459953789, 3746175075, 439452389, 1362321559, 1400849762, 1685577905, 1806599355, 2174754046, 137073913, 1214797936, 1174215055, 3731654548, 2079897426, 1943217067, 1258480242, 529487843, 1437280870, 3945269170, 3049390895, 3313212038, 923313619, 679998e3, 3215307299, 57326082, 377642221, 3474729866, 2041877159, 133361907, 1776460110, 3673476453, 96392454, 878845905, 2801699524, 777231668, 4082475170, 2330014213, 4142626212, 2213296395, 1626319424, 1906247262, 1846563261, 562755902, 3708173718, 1040559837, 3871163981, 1418573201, 3294430577, 114585348, 1343618912, 2566595609, 3186202582, 1078185097, 3651041127, 3896688048, 2307622919, 425408743, 3371096953, 2081048481, 1108339068, 2216610296, 0, 2156299017, 736970802, 292596766, 1517440620, 251657213, 2235061775, 2933202493, 758720310, 265905162, 1554391400, 1532285339, 908999204, 174567692, 1474760595, 4002861748, 2610011675, 3234156416, 3693126241, 2001430874, 303699484, 2478443234, 2687165888, 585122620, 454499602, 151849742, 2345119218, 3064510765, 514443284, 4044981591, 1963412655, 2581445614, 2137062819, 19308535, 1928707164, 1715193156, 4219352155, 1126790795, 600235211, 3992742070, 3841024952, 836553431, 1669664834, 2535604243, 3323011204, 1243905413, 3141400786, 4180808110, 698445255, 2653899549, 2989552604, 2253581325, 3252932727, 3004591147, 1891211689, 2487810577, 3915653703, 4237083816, 4030667424, 2100090966, 865136418, 1229899655, 953270745, 3399679628, 3557504664, 4118925222, 2061379749, 3079546586, 2915017791, 983426092, 2022837584, 1607244650, 2118541908, 2366882550, 3635996816, 972512814, 3283088770, 1568718495, 3499326569, 3576539503, 621982671, 2895723464, 410887952, 2623762152, 1002142683, 645401037, 1494807662, 2595684844, 1335535747, 2507040230, 4293295786, 3167684641, 367585007, 3885750714, 1865862730, 2668221674, 2960971305, 2763173681, 1059270954, 2777952454, 2724642869, 1320957812, 2194319100, 2429595872, 2815956275, 77089521, 3973773121, 3444575871, 2448830231, 1305906550, 4021308739, 2857194700, 2516901860, 3518358430, 1787304780, 740276417, 1699839814, 1592394909, 2352307457, 2272556026, 188821243, 1729977011, 3687994002, 274084841, 3594982253, 3613494426, 2701949495, 4162096729, 322734571, 2837966542, 1640576439, 484830689, 1202797690, 3537852828, 4067639125, 349075736, 3342319475, 4157467219, 4255800159, 1030690015, 1155237496, 2951971274, 1757691577, 607398968, 2738905026, 499347990, 3794078908, 1011452712, 227885567, 2818666809, 213114376, 3034881240, 1455525988, 3414450555, 850817237, 1817998408, 3092726480];
+ var U1 = [0, 235474187, 470948374, 303765277, 941896748, 908933415, 607530554, 708780849, 1883793496, 2118214995, 1817866830, 1649639237, 1215061108, 1181045119, 1417561698, 1517767529, 3767586992, 4003061179, 4236429990, 4069246893, 3635733660, 3602770327, 3299278474, 3400528769, 2430122216, 2664543715, 2362090238, 2193862645, 2835123396, 2801107407, 3035535058, 3135740889, 3678124923, 3576870512, 3341394285, 3374361702, 3810496343, 3977675356, 4279080257, 4043610186, 2876494627, 2776292904, 3076639029, 3110650942, 2472011535, 2640243204, 2403728665, 2169303058, 1001089995, 899835584, 666464733, 699432150, 59727847, 226906860, 530400753, 294930682, 1273168787, 1172967064, 1475418501, 1509430414, 1942435775, 2110667444, 1876241833, 1641816226, 2910219766, 2743034109, 2976151520, 3211623147, 2505202138, 2606453969, 2302690252, 2269728455, 3711829422, 3543599269, 3240894392, 3475313331, 3843699074, 3943906441, 4178062228, 4144047775, 1306967366, 1139781709, 1374988112, 1610459739, 1975683434, 2076935265, 1775276924, 1742315127, 1034867998, 866637845, 566021896, 800440835, 92987698, 193195065, 429456164, 395441711, 1984812685, 2017778566, 1784663195, 1683407248, 1315562145, 1080094634, 1383856311, 1551037884, 101039829, 135050206, 437757123, 337553864, 1042385657, 807962610, 573804783, 742039012, 2531067453, 2564033334, 2328828971, 2227573024, 2935566865, 2700099354, 3001755655, 3168937228, 3868552805, 3902563182, 4203181171, 4102977912, 3736164937, 3501741890, 3265478751, 3433712980, 1106041591, 1340463100, 1576976609, 1408749034, 2043211483, 2009195472, 1708848333, 1809054150, 832877231, 1068351396, 766945465, 599762354, 159417987, 126454664, 361929877, 463180190, 2709260871, 2943682380, 3178106961, 3009879386, 2572697195, 2538681184, 2236228733, 2336434550, 3509871135, 3745345300, 3441850377, 3274667266, 3910161971, 3877198648, 4110568485, 4211818798, 2597806476, 2497604743, 2261089178, 2295101073, 2733856160, 2902087851, 3202437046, 2968011453, 3936291284, 3835036895, 4136440770, 4169408201, 3535486456, 3702665459, 3467192302, 3231722213, 2051518780, 1951317047, 1716890410, 1750902305, 1113818384, 1282050075, 1584504582, 1350078989, 168810852, 67556463, 371049330, 404016761, 841739592, 1008918595, 775550814, 540080725, 3969562369, 3801332234, 4035489047, 4269907996, 3569255213, 3669462566, 3366754619, 3332740144, 2631065433, 2463879762, 2160117071, 2395588676, 2767645557, 2868897406, 3102011747, 3069049960, 202008497, 33778362, 270040487, 504459436, 875451293, 975658646, 675039627, 641025152, 2084704233, 1917518562, 1615861247, 1851332852, 1147550661, 1248802510, 1484005843, 1451044056, 933301370, 967311729, 733156972, 632953703, 260388950, 25965917, 328671808, 496906059, 1206477858, 1239443753, 1543208500, 1441952575, 2144161806, 1908694277, 1675577880, 1842759443, 3610369226, 3644379585, 3408119516, 3307916247, 4011190502, 3776767469, 4077384432, 4245618683, 2809771154, 2842737049, 3144396420, 3043140495, 2673705150, 2438237621, 2203032232, 2370213795];
+ var U2 = [0, 185469197, 370938394, 487725847, 741876788, 657861945, 975451694, 824852259, 1483753576, 1400783205, 1315723890, 1164071807, 1950903388, 2135319889, 1649704518, 1767536459, 2967507152, 3152976349, 2801566410, 2918353863, 2631447780, 2547432937, 2328143614, 2177544179, 3901806776, 3818836405, 4270639778, 4118987695, 3299409036, 3483825537, 3535072918, 3652904859, 2077965243, 1893020342, 1841768865, 1724457132, 1474502543, 1559041666, 1107234197, 1257309336, 598438867, 681933534, 901210569, 1052338372, 261314535, 77422314, 428819965, 310463728, 3409685355, 3224740454, 3710368113, 3593056380, 3875770207, 3960309330, 4045380933, 4195456072, 2471224067, 2554718734, 2237133081, 2388260884, 3212035895, 3028143674, 2842678573, 2724322336, 4138563181, 4255350624, 3769721975, 3955191162, 3667219033, 3516619604, 3431546947, 3347532110, 2933734917, 2782082824, 3099667487, 3016697106, 2196052529, 2313884476, 2499348523, 2683765030, 1179510461, 1296297904, 1347548327, 1533017514, 1786102409, 1635502980, 2087309459, 2003294622, 507358933, 355706840, 136428751, 53458370, 839224033, 957055980, 605657339, 790073846, 2373340630, 2256028891, 2607439820, 2422494913, 2706270690, 2856345839, 3075636216, 3160175349, 3573941694, 3725069491, 3273267108, 3356761769, 4181598602, 4063242375, 4011996048, 3828103837, 1033297158, 915985419, 730517276, 545572369, 296679730, 446754879, 129166120, 213705253, 1709610350, 1860738147, 1945798516, 2029293177, 1239331162, 1120974935, 1606591296, 1422699085, 4148292826, 4233094615, 3781033664, 3931371469, 3682191598, 3497509347, 3446004468, 3328955385, 2939266226, 2755636671, 3106780840, 2988687269, 2198438022, 2282195339, 2501218972, 2652609425, 1201765386, 1286567175, 1371368976, 1521706781, 1805211710, 1620529459, 2105887268, 1988838185, 533804130, 350174575, 164439672, 46346101, 870912086, 954669403, 636813900, 788204353, 2358957921, 2274680428, 2592523643, 2441661558, 2695033685, 2880240216, 3065962831, 3182487618, 3572145929, 3756299780, 3270937875, 3388507166, 4174560061, 4091327024, 4006521127, 3854606378, 1014646705, 930369212, 711349675, 560487590, 272786309, 457992840, 106852767, 223377554, 1678381017, 1862534868, 1914052035, 2031621326, 1211247597, 1128014560, 1580087799, 1428173050, 32283319, 182621114, 401639597, 486441376, 768917123, 651868046, 1003007129, 818324884, 1503449823, 1385356242, 1333838021, 1150208456, 1973745387, 2125135846, 1673061617, 1756818940, 2970356327, 3120694122, 2802849917, 2887651696, 2637442643, 2520393566, 2334669897, 2149987652, 3917234703, 3799141122, 4284502037, 4100872472, 3309594171, 3460984630, 3545789473, 3629546796, 2050466060, 1899603969, 1814803222, 1730525723, 1443857720, 1560382517, 1075025698, 1260232239, 575138148, 692707433, 878443390, 1062597235, 243256656, 91341917, 409198410, 325965383, 3403100636, 3252238545, 3704300486, 3620022987, 3874428392, 3990953189, 4042459122, 4227665663, 2460449204, 2578018489, 2226875310, 2411029155, 3198115200, 3046200461, 2827177882, 2743944855];
+ var U3 = [0, 218828297, 437656594, 387781147, 875313188, 958871085, 775562294, 590424639, 1750626376, 1699970625, 1917742170, 2135253587, 1551124588, 1367295589, 1180849278, 1265195639, 3501252752, 3720081049, 3399941250, 3350065803, 3835484340, 3919042237, 4270507174, 4085369519, 3102249176, 3051593425, 2734591178, 2952102595, 2361698556, 2177869557, 2530391278, 2614737639, 3145456443, 3060847922, 2708326185, 2892417312, 2404901663, 2187128086, 2504130317, 2555048196, 3542330227, 3727205754, 3375740769, 3292445032, 3876557655, 3926170974, 4246310725, 4027744588, 1808481195, 1723872674, 1910319033, 2094410160, 1608975247, 1391201670, 1173430173, 1224348052, 59984867, 244860394, 428169201, 344873464, 935293895, 984907214, 766078933, 547512796, 1844882806, 1627235199, 2011214180, 2062270317, 1507497298, 1423022939, 1137477952, 1321699145, 95345982, 145085239, 532201772, 313773861, 830661914, 1015671571, 731183368, 648017665, 3175501286, 2957853679, 2807058932, 2858115069, 2305455554, 2220981195, 2474404304, 2658625497, 3575528878, 3625268135, 3473416636, 3254988725, 3778151818, 3963161475, 4213447064, 4130281361, 3599595085, 3683022916, 3432737375, 3247465558, 3802222185, 4020912224, 4172763771, 4122762354, 3201631749, 3017672716, 2764249623, 2848461854, 2331590177, 2280796200, 2431590963, 2648976442, 104699613, 188127444, 472615631, 287343814, 840019705, 1058709744, 671593195, 621591778, 1852171925, 1668212892, 1953757831, 2037970062, 1514790577, 1463996600, 1080017571, 1297403050, 3673637356, 3623636965, 3235995134, 3454686199, 4007360968, 3822090177, 4107101658, 4190530515, 2997825956, 3215212461, 2830708150, 2779915199, 2256734592, 2340947849, 2627016082, 2443058075, 172466556, 122466165, 273792366, 492483431, 1047239e3, 861968209, 612205898, 695634755, 1646252340, 1863638845, 2013908262, 1963115311, 1446242576, 1530455833, 1277555970, 1093597963, 1636604631, 1820824798, 2073724613, 1989249228, 1436590835, 1487645946, 1337376481, 1119727848, 164948639, 81781910, 331544205, 516552836, 1039717051, 821288114, 669961897, 719700128, 2973530695, 3157750862, 2871682645, 2787207260, 2232435299, 2283490410, 2667994737, 2450346104, 3647212047, 3564045318, 3279033885, 3464042516, 3980931627, 3762502690, 4150144569, 4199882800, 3070356634, 3121275539, 2904027272, 2686254721, 2200818878, 2384911031, 2570832044, 2486224549, 3747192018, 3528626907, 3310321856, 3359936201, 3950355702, 3867060991, 4049844452, 4234721005, 1739656202, 1790575107, 2108100632, 1890328081, 1402811438, 1586903591, 1233856572, 1149249077, 266959938, 48394827, 369057872, 418672217, 1002783846, 919489135, 567498868, 752375421, 209336225, 24197544, 376187827, 459744698, 945164165, 895287692, 574624663, 793451934, 1679968233, 1764313568, 2117360635, 1933530610, 1343127501, 1560637892, 1243112415, 1192455638, 3704280881, 3519142200, 3336358691, 3419915562, 3907448597, 3857572124, 4075877127, 4294704398, 3029510009, 3113855344, 2927934315, 2744104290, 2159976285, 2377486676, 2594734927, 2544078150];
+ var U4 = [0, 151849742, 303699484, 454499602, 607398968, 758720310, 908999204, 1059270954, 1214797936, 1097159550, 1517440620, 1400849762, 1817998408, 1699839814, 2118541908, 2001430874, 2429595872, 2581445614, 2194319100, 2345119218, 3034881240, 3186202582, 2801699524, 2951971274, 3635996816, 3518358430, 3399679628, 3283088770, 4237083816, 4118925222, 4002861748, 3885750714, 1002142683, 850817237, 698445255, 548169417, 529487843, 377642221, 227885567, 77089521, 1943217067, 2061379749, 1640576439, 1757691577, 1474760595, 1592394909, 1174215055, 1290801793, 2875968315, 2724642869, 3111247143, 2960971305, 2405426947, 2253581325, 2638606623, 2487810577, 3808662347, 3926825029, 4044981591, 4162096729, 3342319475, 3459953789, 3576539503, 3693126241, 1986918061, 2137062819, 1685577905, 1836772287, 1381620373, 1532285339, 1078185097, 1229899655, 1040559837, 923313619, 740276417, 621982671, 439452389, 322734571, 137073913, 19308535, 3871163981, 4021308739, 4104605777, 4255800159, 3263785589, 3414450555, 3499326569, 3651041127, 2933202493, 2815956275, 3167684641, 3049390895, 2330014213, 2213296395, 2566595609, 2448830231, 1305906550, 1155237496, 1607244650, 1455525988, 1776460110, 1626319424, 2079897426, 1928707164, 96392454, 213114376, 396673818, 514443284, 562755902, 679998e3, 865136418, 983426092, 3708173718, 3557504664, 3474729866, 3323011204, 4180808110, 4030667424, 3945269170, 3794078908, 2507040230, 2623762152, 2272556026, 2390325492, 2975484382, 3092726480, 2738905026, 2857194700, 3973773121, 3856137295, 4274053469, 4157467219, 3371096953, 3252932727, 3673476453, 3556361835, 2763173681, 2915017791, 3064510765, 3215307299, 2156299017, 2307622919, 2459735317, 2610011675, 2081048481, 1963412655, 1846563261, 1729977011, 1480485785, 1362321559, 1243905413, 1126790795, 878845905, 1030690015, 645401037, 796197571, 274084841, 425408743, 38544885, 188821243, 3613494426, 3731654548, 3313212038, 3430322568, 4082475170, 4200115116, 3780097726, 3896688048, 2668221674, 2516901860, 2366882550, 2216610296, 3141400786, 2989552604, 2837966542, 2687165888, 1202797690, 1320957812, 1437280870, 1554391400, 1669664834, 1787304780, 1906247262, 2022837584, 265905162, 114585348, 499347990, 349075736, 736970802, 585122620, 972512814, 821712160, 2595684844, 2478443234, 2293045232, 2174754046, 3196267988, 3079546586, 2895723464, 2777952454, 3537852828, 3687994002, 3234156416, 3385345166, 4142626212, 4293295786, 3841024952, 3992742070, 174567692, 57326082, 410887952, 292596766, 777231668, 660510266, 1011452712, 893681702, 1108339068, 1258480242, 1343618912, 1494807662, 1715193156, 1865862730, 1948373848, 2100090966, 2701949495, 2818666809, 3004591147, 3122358053, 2235061775, 2352307457, 2535604243, 2653899549, 3915653703, 3764988233, 4219352155, 4067639125, 3444575871, 3294430577, 3746175075, 3594982253, 836553431, 953270745, 600235211, 718002117, 367585007, 484830689, 133361907, 251657213, 2041877159, 1891211689, 1806599355, 1654886325, 1568718495, 1418573201, 1335535747, 1184342925];
+ function convertToInt32(bytes) {
+ var result = [];
+ for (var i2 = 0; i2 < bytes.length; i2 += 4) {
+ result.push(
+ bytes[i2] << 24 | bytes[i2 + 1] << 16 | bytes[i2 + 2] << 8 | bytes[i2 + 3]
+ );
+ }
+ return result;
+ }
+ var AES = function(key) {
+ if (!(this instanceof AES)) {
+ throw Error("AES must be instanitated with `new`");
+ }
+ Object.defineProperty(this, "key", {
+ value: coerceArray(key, true)
+ });
+ this._prepare();
+ };
+ AES.prototype._prepare = function() {
+ var rounds = numberOfRounds[this.key.length];
+ if (rounds == null) {
+ throw new Error("invalid key size (must be 16, 24 or 32 bytes)");
+ }
+ this._Ke = [];
+ this._Kd = [];
+ for (var i2 = 0; i2 <= rounds; i2++) {
+ this._Ke.push([0, 0, 0, 0]);
+ this._Kd.push([0, 0, 0, 0]);
+ }
+ var roundKeyCount = (rounds + 1) * 4;
+ var KC = this.key.length / 4;
+ var tk = convertToInt32(this.key);
+ var index;
+ for (var i2 = 0; i2 < KC; i2++) {
+ index = i2 >> 2;
+ this._Ke[index][i2 % 4] = tk[i2];
+ this._Kd[rounds - index][i2 % 4] = tk[i2];
+ }
+ var rconpointer = 0;
+ var t = KC, tt;
+ while (t < roundKeyCount) {
+ tt = tk[KC - 1];
+ tk[0] ^= S[tt >> 16 & 255] << 24 ^ S[tt >> 8 & 255] << 16 ^ S[tt & 255] << 8 ^ S[tt >> 24 & 255] ^ rcon[rconpointer] << 24;
+ rconpointer += 1;
+ if (KC != 8) {
+ for (var i2 = 1; i2 < KC; i2++) {
+ tk[i2] ^= tk[i2 - 1];
+ }
+ } else {
+ for (var i2 = 1; i2 < KC / 2; i2++) {
+ tk[i2] ^= tk[i2 - 1];
+ }
+ tt = tk[KC / 2 - 1];
+ tk[KC / 2] ^= S[tt & 255] ^ S[tt >> 8 & 255] << 8 ^ S[tt >> 16 & 255] << 16 ^ S[tt >> 24 & 255] << 24;
+ for (var i2 = KC / 2 + 1; i2 < KC; i2++) {
+ tk[i2] ^= tk[i2 - 1];
+ }
+ }
+ var i2 = 0, r, c;
+ while (i2 < KC && t < roundKeyCount) {
+ r = t >> 2;
+ c = t % 4;
+ this._Ke[r][c] = tk[i2];
+ this._Kd[rounds - r][c] = tk[i2++];
+ t++;
+ }
+ }
+ 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 & 255] ^ U2[tt >> 16 & 255] ^ U3[tt >> 8 & 255] ^ U4[tt & 255];
+ }
+ }
+ };
+ 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];
+ var t = convertToInt32(plaintext);
+ for (var i2 = 0; i2 < 4; i2++) {
+ t[i2] ^= this._Ke[0][i2];
+ }
+ for (var r = 1; r < rounds; r++) {
+ for (var i2 = 0; i2 < 4; i2++) {
+ a[i2] = T1[t[i2] >> 24 & 255] ^ T2[t[(i2 + 1) % 4] >> 16 & 255] ^ T3[t[(i2 + 2) % 4] >> 8 & 255] ^ T4[t[(i2 + 3) % 4] & 255] ^ this._Ke[r][i2];
+ }
+ t = a.slice();
+ }
+ var result = createArray(16), tt;
+ for (var i2 = 0; i2 < 4; i2++) {
+ tt = this._Ke[rounds][i2];
+ result[4 * i2] = (S[t[i2] >> 24 & 255] ^ tt >> 24) & 255;
+ result[4 * i2 + 1] = (S[t[(i2 + 1) % 4] >> 16 & 255] ^ tt >> 16) & 255;
+ result[4 * i2 + 2] = (S[t[(i2 + 2) % 4] >> 8 & 255] ^ tt >> 8) & 255;
+ result[4 * i2 + 3] = (S[t[(i2 + 3) % 4] & 255] ^ tt) & 255;
+ }
+ return result;
+ };
+ AES.prototype.decrypt = function(ciphertext) {
+ if (ciphertext.length != 16) {
+ throw new Error("invalid ciphertext size (must be 16 bytes)");
+ }
+ var rounds = this._Kd.length - 1;
+ var a = [0, 0, 0, 0];
+ var t = convertToInt32(ciphertext);
+ for (var i2 = 0; i2 < 4; i2++) {
+ t[i2] ^= this._Kd[0][i2];
+ }
+ for (var r = 1; r < rounds; r++) {
+ for (var i2 = 0; i2 < 4; i2++) {
+ a[i2] = T5[t[i2] >> 24 & 255] ^ T6[t[(i2 + 3) % 4] >> 16 & 255] ^ T7[t[(i2 + 2) % 4] >> 8 & 255] ^ T8[t[(i2 + 1) % 4] & 255] ^ this._Kd[r][i2];
+ }
+ t = a.slice();
+ }
+ var result = createArray(16), tt;
+ for (var i2 = 0; i2 < 4; i2++) {
+ tt = this._Kd[rounds][i2];
+ result[4 * i2] = (Si[t[i2] >> 24 & 255] ^ tt >> 24) & 255;
+ result[4 * i2 + 1] = (Si[t[(i2 + 3) % 4] >> 16 & 255] ^ tt >> 16) & 255;
+ result[4 * i2 + 2] = (Si[t[(i2 + 2) % 4] >> 8 & 255] ^ tt >> 8) & 255;
+ result[4 * i2 + 3] = (Si[t[(i2 + 1) % 4] & 255] ^ tt) & 255;
+ }
+ return result;
+ };
+ var ModeOfOperationECB = function(key) {
+ if (!(this instanceof ModeOfOperationECB)) {
+ throw Error("AES must be instanitated with `new`");
+ }
+ this.description = "Electronic Code Block";
+ this.name = "ecb";
+ this._aes = new AES(key);
+ };
+ ModeOfOperationECB.prototype.encrypt = function(plaintext) {
+ plaintext = coerceArray(plaintext);
+ if (plaintext.length % 16 !== 0) {
+ throw new Error("invalid plaintext size (must be multiple of 16 bytes)");
+ }
+ var ciphertext = createArray(plaintext.length);
+ var block2 = createArray(16);
+ for (var i2 = 0; i2 < plaintext.length; i2 += 16) {
+ copyArray(plaintext, block2, 0, i2, i2 + 16);
+ block2 = this._aes.encrypt(block2);
+ copyArray(block2, ciphertext, i2);
+ }
+ 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)");
+ }
+ var plaintext = createArray(ciphertext.length);
+ var block2 = createArray(16);
+ for (var i2 = 0; i2 < ciphertext.length; i2 += 16) {
+ copyArray(ciphertext, block2, 0, i2, i2 + 16);
+ block2 = this._aes.decrypt(block2);
+ copyArray(block2, plaintext, i2);
+ }
+ return plaintext;
+ };
+ var ModeOfOperationCBC = function(key, iv) {
+ if (!(this instanceof ModeOfOperationCBC)) {
+ throw Error("AES must be instanitated with `new`");
+ }
+ 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);
+ };
+ ModeOfOperationCBC.prototype.encrypt = function(plaintext) {
+ plaintext = coerceArray(plaintext);
+ if (plaintext.length % 16 !== 0) {
+ throw new Error("invalid plaintext size (must be multiple of 16 bytes)");
+ }
+ var ciphertext = createArray(plaintext.length);
+ var block2 = createArray(16);
+ for (var i2 = 0; i2 < plaintext.length; i2 += 16) {
+ copyArray(plaintext, block2, 0, i2, i2 + 16);
+ for (var j2 = 0; j2 < 16; j2++) {
+ block2[j2] ^= this._lastCipherblock[j2];
+ }
+ this._lastCipherblock = this._aes.encrypt(block2);
+ copyArray(this._lastCipherblock, ciphertext, i2);
+ }
+ return ciphertext;
+ };
+ ModeOfOperationCBC.prototype.decrypt = function(ciphertext) {
+ ciphertext = coerceArray(ciphertext);
+ if (ciphertext.length % 16 !== 0) {
+ throw new Error("invalid ciphertext size (must be multiple of 16 bytes)");
+ }
+ var plaintext = createArray(ciphertext.length);
+ var block2 = createArray(16);
+ for (var i2 = 0; i2 < ciphertext.length; i2 += 16) {
+ copyArray(ciphertext, block2, 0, i2, i2 + 16);
+ block2 = this._aes.decrypt(block2);
+ for (var j2 = 0; j2 < 16; j2++) {
+ plaintext[i2 + j2] = block2[j2] ^ this._lastCipherblock[j2];
+ }
+ copyArray(ciphertext, this._lastCipherblock, 0, i2, i2 + 16);
+ }
+ return plaintext;
+ };
+ var ModeOfOperationCFB = function(key, iv, segmentSize) {
+ if (!(this instanceof ModeOfOperationCFB)) {
+ throw Error("AES must be instanitated with `new`");
+ }
+ this.description = "Cipher Feedback";
+ this.name = "cfb";
+ if (!iv) {
+ iv = createArray(16);
+ } else if (iv.length != 16) {
+ throw new Error("invalid initialation vector size (must be 16 size)");
+ }
+ if (!segmentSize) {
+ segmentSize = 1;
+ }
+ 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)");
+ }
+ var encrypted = coerceArray(plaintext, true);
+ var xorSegment;
+ for (var i2 = 0; i2 < encrypted.length; i2 += this.segmentSize) {
+ xorSegment = this._aes.encrypt(this._shiftRegister);
+ for (var j2 = 0; j2 < this.segmentSize; j2++) {
+ encrypted[i2 + j2] ^= xorSegment[j2];
+ }
+ copyArray(this._shiftRegister, this._shiftRegister, 0, this.segmentSize);
+ copyArray(encrypted, this._shiftRegister, 16 - this.segmentSize, i2, i2 + this.segmentSize);
+ }
+ return encrypted;
+ };
+ 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 i2 = 0; i2 < plaintext.length; i2 += this.segmentSize) {
+ xorSegment = this._aes.encrypt(this._shiftRegister);
+ for (var j2 = 0; j2 < this.segmentSize; j2++) {
+ plaintext[i2 + j2] ^= xorSegment[j2];
+ }
+ copyArray(this._shiftRegister, this._shiftRegister, 0, this.segmentSize);
+ copyArray(ciphertext, this._shiftRegister, 16 - this.segmentSize, i2, i2 + this.segmentSize);
+ }
+ return plaintext;
+ };
+ var ModeOfOperationOFB = function(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 i2 = 0; i2 < encrypted.length; i2++) {
+ if (this._lastPrecipherIndex === 16) {
+ this._lastPrecipher = this._aes.encrypt(this._lastPrecipher);
+ this._lastPrecipherIndex = 0;
+ }
+ encrypted[i2] ^= this._lastPrecipher[this._lastPrecipherIndex++];
+ }
+ return encrypted;
+ };
+ ModeOfOperationOFB.prototype.decrypt = ModeOfOperationOFB.prototype.encrypt;
+ var Counter = function(initialValue) {
+ if (!(this instanceof Counter)) {
+ throw Error("Counter must be instanitated with `new`");
+ }
+ if (initialValue !== 0 && !initialValue) {
+ initialValue = 1;
+ }
+ if (typeof initialValue === "number") {
+ this._counter = createArray(16);
+ this.setValue(initialValue);
+ } 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)");
+ }
+ 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 i2 = 15; i2 >= 0; i2--) {
+ if (this._counter[i2] === 255) {
+ this._counter[i2] = 0;
+ } else {
+ this._counter[i2]++;
+ break;
+ }
+ }
+ };
+ var ModeOfOperationCTR = function(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 i2 = 0; i2 < encrypted.length; i2++) {
+ if (this._remainingCounterIndex === 16) {
+ this._remainingCounter = this._aes.encrypt(this._counter._counter);
+ this._remainingCounterIndex = 0;
+ this._counter.increment();
+ }
+ encrypted[i2] ^= this._remainingCounter[this._remainingCounterIndex++];
+ }
+ return encrypted;
+ };
+ ModeOfOperationCTR.prototype.decrypt = ModeOfOperationCTR.prototype.encrypt;
+ function pkcs7pad(data) {
+ data = coerceArray(data, true);
+ var padder = 16 - data.length % 16;
+ var result = createArray(data.length + padder);
+ copyArray(data, result);
+ for (var i2 = data.length; i2 < result.length; i2++) {
+ result[i2] = padder;
+ }
+ 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];
+ if (padder > 16) {
+ throw new Error("PKCS#7 padding byte out of range");
+ }
+ var length = data.length - padder;
+ for (var i2 = 0; i2 < padder; i2++) {
+ if (data[length + i2] !== padder) {
+ throw new Error("PKCS#7 invalid padding byte");
+ }
+ }
+ var result = createArray(length);
+ copyArray(data, result, 0, 0, length);
+ return result;
+ }
+ var aesjs2 = {
+ AES,
+ 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,
+ createArray,
+ copyArray
+ }
+ };
+ if (typeof exports2 !== "undefined") {
+ module2.exports = aesjs2;
+ } else if (typeof define === "function" && define.amd) {
+ define([], function() {
+ return aesjs2;
+ });
+ } else {
+ if (root3.aesjs) {
+ aesjs2._aesjs = root3.aesjs;
+ }
+ root3.aesjs = aesjs2;
+ }
+ })(exports2);
+ }
+ });
+
+ // node_modules/fast-deep-equal/index.js
+ var require_fast_deep_equal = __commonJS({
+ "node_modules/fast-deep-equal/index.js"(exports2, module2) {
+ "use strict";
+ module2.exports = 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, i2, keys;
+ if (Array.isArray(a)) {
+ length = a.length;
+ if (length != b.length)
+ return false;
+ for (i2 = length; i2-- !== 0; )
+ if (!equal(a[i2], b[i2]))
+ return false;
+ return true;
+ }
+ 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;
+ for (i2 = length; i2-- !== 0; )
+ if (!Object.prototype.hasOwnProperty.call(b, keys[i2]))
+ return false;
+ for (i2 = length; i2-- !== 0; ) {
+ var key = keys[i2];
+ if (!equal(a[key], b[key]))
+ return false;
+ }
+ return true;
+ }
+ return a !== a && b !== b;
+ };
+ }
+ });
+
+ // node_modules/lodash/lodash.js
+ var require_lodash = __commonJS({
+ "node_modules/lodash/lodash.js"(exports2, module2) {
+ (function() {
+ var undefined2;
+ var VERSION = "4.17.21";
+ var LARGE_ARRAY_SIZE = 200;
+ var CORE_ERROR_TEXT = "Unsupported core-js use. Try https://npms.io/search?q=ponyfill.", FUNC_ERROR_TEXT3 = "Expected a function", INVALID_TEMPL_VAR_ERROR_TEXT = "Invalid `variable` option passed into `_.template`";
+ var HASH_UNDEFINED = "__lodash_hash_undefined__";
+ var MAX_MEMOIZE_SIZE = 500;
+ var PLACEHOLDER = "__lodash_placeholder__";
+ var CLONE_DEEP_FLAG = 1, CLONE_FLAT_FLAG = 2, CLONE_SYMBOLS_FLAG = 4;
+ var COMPARE_PARTIAL_FLAG = 1, COMPARE_UNORDERED_FLAG = 2;
+ var WRAP_BIND_FLAG = 1, WRAP_BIND_KEY_FLAG = 2, WRAP_CURRY_BOUND_FLAG = 4, WRAP_CURRY_FLAG = 8, WRAP_CURRY_RIGHT_FLAG = 16, WRAP_PARTIAL_FLAG = 32, WRAP_PARTIAL_RIGHT_FLAG = 64, WRAP_ARY_FLAG = 128, WRAP_REARG_FLAG = 256, WRAP_FLIP_FLAG = 512;
+ var DEFAULT_TRUNC_LENGTH = 30, DEFAULT_TRUNC_OMISSION = "...";
+ var HOT_COUNT = 800, HOT_SPAN = 16;
+ var LAZY_FILTER_FLAG = 1, LAZY_MAP_FLAG = 2, LAZY_WHILE_FLAG = 3;
+ var INFINITY2 = 1 / 0, MAX_SAFE_INTEGER = 9007199254740991, MAX_INTEGER = 17976931348623157e292, NAN2 = 0 / 0;
+ var MAX_ARRAY_LENGTH = 4294967295, MAX_ARRAY_INDEX = MAX_ARRAY_LENGTH - 1, HALF_MAX_ARRAY_LENGTH = MAX_ARRAY_LENGTH >>> 1;
+ var wrapFlags = [
+ ["ary", WRAP_ARY_FLAG],
+ ["bind", WRAP_BIND_FLAG],
+ ["bindKey", WRAP_BIND_KEY_FLAG],
+ ["curry", WRAP_CURRY_FLAG],
+ ["curryRight", WRAP_CURRY_RIGHT_FLAG],
+ ["flip", WRAP_FLIP_FLAG],
+ ["partial", WRAP_PARTIAL_FLAG],
+ ["partialRight", WRAP_PARTIAL_RIGHT_FLAG],
+ ["rearg", WRAP_REARG_FLAG]
+ ];
+ var argsTag = "[object Arguments]", arrayTag = "[object Array]", asyncTag = "[object AsyncFunction]", boolTag = "[object Boolean]", dateTag = "[object Date]", domExcTag = "[object DOMException]", errorTag = "[object Error]", funcTag = "[object Function]", genTag = "[object GeneratorFunction]", mapTag = "[object Map]", numberTag = "[object Number]", nullTag2 = "[object Null]", objectTag = "[object Object]", promiseTag = "[object Promise]", proxyTag = "[object Proxy]", regexpTag = "[object RegExp]", setTag = "[object Set]", stringTag = "[object String]", symbolTag2 = "[object Symbol]", undefinedTag2 = "[object Undefined]", weakMapTag = "[object WeakMap]", weakSetTag = "[object WeakSet]";
+ var arrayBufferTag = "[object ArrayBuffer]", dataViewTag = "[object DataView]", float32Tag = "[object Float32Array]", float64Tag = "[object Float64Array]", int8Tag = "[object Int8Array]", int16Tag = "[object Int16Array]", int32Tag = "[object Int32Array]", uint8Tag = "[object Uint8Array]", uint8ClampedTag = "[object Uint8ClampedArray]", uint16Tag = "[object Uint16Array]", uint32Tag = "[object Uint32Array]";
+ var reEmptyStringLeading = /\b__p \+= '';/g, reEmptyStringMiddle = /\b(__p \+=) '' \+/g, reEmptyStringTrailing = /(__e\(.*?\)|\b__t\)) \+\n'';/g;
+ var reEscapedHtml2 = /&(?:amp|lt|gt|quot|#39);/g, reUnescapedHtml2 = /[&<>"']/g, reHasEscapedHtml2 = RegExp(reEscapedHtml2.source), reHasUnescapedHtml2 = RegExp(reUnescapedHtml2.source);
+ var reEscape = /<%-([\s\S]+?)%>/g, reEvaluate = /<%([\s\S]+?)%>/g, reInterpolate = /<%=([\s\S]+?)%>/g;
+ var reIsDeepProp = /\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/, reIsPlainProp = /^\w*$/, rePropName = /[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g;
+ var reRegExpChar = /[\\^$.*+?()[\]{}|]/g, reHasRegExpChar = RegExp(reRegExpChar.source);
+ var reTrimStart2 = /^\s+/;
+ var reWhitespace2 = /\s/;
+ var reWrapComment = /\{(?:\n\/\* \[wrapped with .+\] \*\/)?\n?/, reWrapDetails = /\{\n\/\* \[wrapped with (.+)\] \*/, reSplitDetails = /,? & /;
+ var reAsciiWord = /[^\x00-\x2f\x3a-\x40\x5b-\x60\x7b-\x7f]+/g;
+ var reForbiddenIdentifierChars = /[()=,{}\[\]\/\s]/;
+ var reEscapeChar = /\\(\\)?/g;
+ var reEsTemplate = /\$\{([^\\}]*(?:\\.[^\\}]*)*)\}/g;
+ var reFlags = /\w*$/;
+ var reIsBadHex2 = /^[-+]0x[0-9a-f]+$/i;
+ var reIsBinary2 = /^0b[01]+$/i;
+ var reIsHostCtor = /^\[object .+?Constructor\]$/;
+ var reIsOctal2 = /^0o[0-7]+$/i;
+ var reIsUint = /^(?:0|[1-9]\d*)$/;
+ var reLatin = /[\xc0-\xd6\xd8-\xf6\xf8-\xff\u0100-\u017f]/g;
+ var reNoMatch = /($^)/;
+ var reUnescapedString = /['\n\r\u2028\u2029\\]/g;
+ var rsAstralRange = "\\ud800-\\udfff", rsComboMarksRange = "\\u0300-\\u036f", reComboHalfMarksRange = "\\ufe20-\\ufe2f", rsComboSymbolsRange = "\\u20d0-\\u20ff", rsComboRange = rsComboMarksRange + reComboHalfMarksRange + rsComboSymbolsRange, rsDingbatRange = "\\u2700-\\u27bf", rsLowerRange = "a-z\\xdf-\\xf6\\xf8-\\xff", rsMathOpRange = "\\xac\\xb1\\xd7\\xf7", rsNonCharRange = "\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf", rsPunctuationRange = "\\u2000-\\u206f", rsSpaceRange = " \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000", rsUpperRange = "A-Z\\xc0-\\xd6\\xd8-\\xde", rsVarRange = "\\ufe0e\\ufe0f", rsBreakRange = rsMathOpRange + rsNonCharRange + rsPunctuationRange + rsSpaceRange;
+ var rsApos = "['\u2019]", rsAstral = "[" + rsAstralRange + "]", rsBreak = "[" + rsBreakRange + "]", rsCombo = "[" + rsComboRange + "]", rsDigits = "\\d+", rsDingbat = "[" + rsDingbatRange + "]", rsLower = "[" + rsLowerRange + "]", rsMisc = "[^" + rsAstralRange + rsBreakRange + rsDigits + rsDingbatRange + rsLowerRange + rsUpperRange + "]", rsFitz = "\\ud83c[\\udffb-\\udfff]", rsModifier = "(?:" + rsCombo + "|" + rsFitz + ")", rsNonAstral = "[^" + rsAstralRange + "]", rsRegional = "(?:\\ud83c[\\udde6-\\uddff]){2}", rsSurrPair = "[\\ud800-\\udbff][\\udc00-\\udfff]", rsUpper = "[" + rsUpperRange + "]", rsZWJ = "\\u200d";
+ var rsMiscLower = "(?:" + rsLower + "|" + rsMisc + ")", rsMiscUpper = "(?:" + rsUpper + "|" + rsMisc + ")", rsOptContrLower = "(?:" + rsApos + "(?:d|ll|m|re|s|t|ve))?", rsOptContrUpper = "(?:" + rsApos + "(?:D|LL|M|RE|S|T|VE))?", reOptMod = rsModifier + "?", rsOptVar = "[" + rsVarRange + "]?", rsOptJoin = "(?:" + rsZWJ + "(?:" + [rsNonAstral, rsRegional, rsSurrPair].join("|") + ")" + rsOptVar + reOptMod + ")*", rsOrdLower = "\\d*(?:1st|2nd|3rd|(?![123])\\dth)(?=\\b|[A-Z_])", rsOrdUpper = "\\d*(?:1ST|2ND|3RD|(?![123])\\dTH)(?=\\b|[a-z_])", rsSeq = rsOptVar + reOptMod + rsOptJoin, rsEmoji = "(?:" + [rsDingbat, rsRegional, rsSurrPair].join("|") + ")" + rsSeq, rsSymbol = "(?:" + [rsNonAstral + rsCombo + "?", rsCombo, rsRegional, rsSurrPair, rsAstral].join("|") + ")";
+ var reApos = RegExp(rsApos, "g");
+ var reComboMark = RegExp(rsCombo, "g");
+ var reUnicode = RegExp(rsFitz + "(?=" + rsFitz + ")|" + rsSymbol + rsSeq, "g");
+ var reUnicodeWord = RegExp([
+ rsUpper + "?" + rsLower + "+" + rsOptContrLower + "(?=" + [rsBreak, rsUpper, "$"].join("|") + ")",
+ rsMiscUpper + "+" + rsOptContrUpper + "(?=" + [rsBreak, rsUpper + rsMiscLower, "$"].join("|") + ")",
+ rsUpper + "?" + rsMiscLower + "+" + rsOptContrLower,
+ rsUpper + "+" + rsOptContrUpper,
+ rsOrdUpper,
+ rsOrdLower,
+ rsDigits,
+ rsEmoji
+ ].join("|"), "g");
+ var reHasUnicode = RegExp("[" + rsZWJ + rsAstralRange + rsComboRange + rsVarRange + "]");
+ var reHasUnicodeWord = /[a-z][A-Z]|[A-Z]{2}[a-z]|[0-9][a-zA-Z]|[a-zA-Z][0-9]|[^a-zA-Z0-9 ]/;
+ var contextProps = [
+ "Array",
+ "Buffer",
+ "DataView",
+ "Date",
+ "Error",
+ "Float32Array",
+ "Float64Array",
+ "Function",
+ "Int8Array",
+ "Int16Array",
+ "Int32Array",
+ "Map",
+ "Math",
+ "Object",
+ "Promise",
+ "RegExp",
+ "Set",
+ "String",
+ "Symbol",
+ "TypeError",
+ "Uint8Array",
+ "Uint8ClampedArray",
+ "Uint16Array",
+ "Uint32Array",
+ "WeakMap",
+ "_",
+ "clearTimeout",
+ "isFinite",
+ "parseInt",
+ "setTimeout"
+ ];
+ var templateCounter = -1;
+ var typedArrayTags = {};
+ typedArrayTags[float32Tag] = typedArrayTags[float64Tag] = typedArrayTags[int8Tag] = typedArrayTags[int16Tag] = typedArrayTags[int32Tag] = typedArrayTags[uint8Tag] = typedArrayTags[uint8ClampedTag] = typedArrayTags[uint16Tag] = typedArrayTags[uint32Tag] = true;
+ typedArrayTags[argsTag] = typedArrayTags[arrayTag] = typedArrayTags[arrayBufferTag] = typedArrayTags[boolTag] = typedArrayTags[dataViewTag] = typedArrayTags[dateTag] = typedArrayTags[errorTag] = typedArrayTags[funcTag] = typedArrayTags[mapTag] = typedArrayTags[numberTag] = typedArrayTags[objectTag] = typedArrayTags[regexpTag] = typedArrayTags[setTag] = typedArrayTags[stringTag] = typedArrayTags[weakMapTag] = false;
+ var cloneableTags = {};
+ cloneableTags[argsTag] = cloneableTags[arrayTag] = cloneableTags[arrayBufferTag] = cloneableTags[dataViewTag] = cloneableTags[boolTag] = cloneableTags[dateTag] = cloneableTags[float32Tag] = cloneableTags[float64Tag] = cloneableTags[int8Tag] = cloneableTags[int16Tag] = cloneableTags[int32Tag] = cloneableTags[mapTag] = cloneableTags[numberTag] = cloneableTags[objectTag] = cloneableTags[regexpTag] = cloneableTags[setTag] = cloneableTags[stringTag] = cloneableTags[symbolTag2] = cloneableTags[uint8Tag] = cloneableTags[uint8ClampedTag] = cloneableTags[uint16Tag] = cloneableTags[uint32Tag] = true;
+ cloneableTags[errorTag] = cloneableTags[funcTag] = cloneableTags[weakMapTag] = false;
+ var deburredLetters = {
+ // Latin-1 Supplement block.
+ "\xC0": "A",
+ "\xC1": "A",
+ "\xC2": "A",
+ "\xC3": "A",
+ "\xC4": "A",
+ "\xC5": "A",
+ "\xE0": "a",
+ "\xE1": "a",
+ "\xE2": "a",
+ "\xE3": "a",
+ "\xE4": "a",
+ "\xE5": "a",
+ "\xC7": "C",
+ "\xE7": "c",
+ "\xD0": "D",
+ "\xF0": "d",
+ "\xC8": "E",
+ "\xC9": "E",
+ "\xCA": "E",
+ "\xCB": "E",
+ "\xE8": "e",
+ "\xE9": "e",
+ "\xEA": "e",
+ "\xEB": "e",
+ "\xCC": "I",
+ "\xCD": "I",
+ "\xCE": "I",
+ "\xCF": "I",
+ "\xEC": "i",
+ "\xED": "i",
+ "\xEE": "i",
+ "\xEF": "i",
+ "\xD1": "N",
+ "\xF1": "n",
+ "\xD2": "O",
+ "\xD3": "O",
+ "\xD4": "O",
+ "\xD5": "O",
+ "\xD6": "O",
+ "\xD8": "O",
+ "\xF2": "o",
+ "\xF3": "o",
+ "\xF4": "o",
+ "\xF5": "o",
+ "\xF6": "o",
+ "\xF8": "o",
+ "\xD9": "U",
+ "\xDA": "U",
+ "\xDB": "U",
+ "\xDC": "U",
+ "\xF9": "u",
+ "\xFA": "u",
+ "\xFB": "u",
+ "\xFC": "u",
+ "\xDD": "Y",
+ "\xFD": "y",
+ "\xFF": "y",
+ "\xC6": "Ae",
+ "\xE6": "ae",
+ "\xDE": "Th",
+ "\xFE": "th",
+ "\xDF": "ss",
+ // Latin Extended-A block.
+ "\u0100": "A",
+ "\u0102": "A",
+ "\u0104": "A",
+ "\u0101": "a",
+ "\u0103": "a",
+ "\u0105": "a",
+ "\u0106": "C",
+ "\u0108": "C",
+ "\u010A": "C",
+ "\u010C": "C",
+ "\u0107": "c",
+ "\u0109": "c",
+ "\u010B": "c",
+ "\u010D": "c",
+ "\u010E": "D",
+ "\u0110": "D",
+ "\u010F": "d",
+ "\u0111": "d",
+ "\u0112": "E",
+ "\u0114": "E",
+ "\u0116": "E",
+ "\u0118": "E",
+ "\u011A": "E",
+ "\u0113": "e",
+ "\u0115": "e",
+ "\u0117": "e",
+ "\u0119": "e",
+ "\u011B": "e",
+ "\u011C": "G",
+ "\u011E": "G",
+ "\u0120": "G",
+ "\u0122": "G",
+ "\u011D": "g",
+ "\u011F": "g",
+ "\u0121": "g",
+ "\u0123": "g",
+ "\u0124": "H",
+ "\u0126": "H",
+ "\u0125": "h",
+ "\u0127": "h",
+ "\u0128": "I",
+ "\u012A": "I",
+ "\u012C": "I",
+ "\u012E": "I",
+ "\u0130": "I",
+ "\u0129": "i",
+ "\u012B": "i",
+ "\u012D": "i",
+ "\u012F": "i",
+ "\u0131": "i",
+ "\u0134": "J",
+ "\u0135": "j",
+ "\u0136": "K",
+ "\u0137": "k",
+ "\u0138": "k",
+ "\u0139": "L",
+ "\u013B": "L",
+ "\u013D": "L",
+ "\u013F": "L",
+ "\u0141": "L",
+ "\u013A": "l",
+ "\u013C": "l",
+ "\u013E": "l",
+ "\u0140": "l",
+ "\u0142": "l",
+ "\u0143": "N",
+ "\u0145": "N",
+ "\u0147": "N",
+ "\u014A": "N",
+ "\u0144": "n",
+ "\u0146": "n",
+ "\u0148": "n",
+ "\u014B": "n",
+ "\u014C": "O",
+ "\u014E": "O",
+ "\u0150": "O",
+ "\u014D": "o",
+ "\u014F": "o",
+ "\u0151": "o",
+ "\u0154": "R",
+ "\u0156": "R",
+ "\u0158": "R",
+ "\u0155": "r",
+ "\u0157": "r",
+ "\u0159": "r",
+ "\u015A": "S",
+ "\u015C": "S",
+ "\u015E": "S",
+ "\u0160": "S",
+ "\u015B": "s",
+ "\u015D": "s",
+ "\u015F": "s",
+ "\u0161": "s",
+ "\u0162": "T",
+ "\u0164": "T",
+ "\u0166": "T",
+ "\u0163": "t",
+ "\u0165": "t",
+ "\u0167": "t",
+ "\u0168": "U",
+ "\u016A": "U",
+ "\u016C": "U",
+ "\u016E": "U",
+ "\u0170": "U",
+ "\u0172": "U",
+ "\u0169": "u",
+ "\u016B": "u",
+ "\u016D": "u",
+ "\u016F": "u",
+ "\u0171": "u",
+ "\u0173": "u",
+ "\u0174": "W",
+ "\u0175": "w",
+ "\u0176": "Y",
+ "\u0177": "y",
+ "\u0178": "Y",
+ "\u0179": "Z",
+ "\u017B": "Z",
+ "\u017D": "Z",
+ "\u017A": "z",
+ "\u017C": "z",
+ "\u017E": "z",
+ "\u0132": "IJ",
+ "\u0133": "ij",
+ "\u0152": "Oe",
+ "\u0153": "oe",
+ "\u0149": "'n",
+ "\u017F": "s"
+ };
+ var htmlEscapes2 = {
+ "&": "&",
+ "<": "<",
+ ">": ">",
+ '"': """,
+ "'": "'"
+ };
+ var htmlUnescapes2 = {
+ "&": "&",
+ "<": "<",
+ ">": ">",
+ """: '"',
+ "'": "'"
+ };
+ var stringEscapes = {
+ "\\": "\\",
+ "'": "'",
+ "\n": "n",
+ "\r": "r",
+ "\u2028": "u2028",
+ "\u2029": "u2029"
+ };
+ var freeParseFloat = parseFloat, freeParseInt2 = parseInt;
+ var freeGlobal2 = typeof global == "object" && global && global.Object === Object && global;
+ var freeSelf2 = typeof self == "object" && self && self.Object === Object && self;
+ var root3 = freeGlobal2 || freeSelf2 || Function("return this")();
+ var freeExports = typeof exports2 == "object" && exports2 && !exports2.nodeType && exports2;
+ var freeModule = freeExports && typeof module2 == "object" && module2 && !module2.nodeType && module2;
+ var moduleExports = freeModule && freeModule.exports === freeExports;
+ var freeProcess = moduleExports && freeGlobal2.process;
+ var nodeUtil = function() {
+ try {
+ var types = freeModule && freeModule.require && freeModule.require("util").types;
+ if (types) {
+ return types;
+ }
+ return freeProcess && freeProcess.binding && freeProcess.binding("util");
+ } catch (e) {
+ }
+ }();
+ var nodeIsArrayBuffer = nodeUtil && nodeUtil.isArrayBuffer, nodeIsDate = nodeUtil && nodeUtil.isDate, nodeIsMap = nodeUtil && nodeUtil.isMap, nodeIsRegExp = nodeUtil && nodeUtil.isRegExp, nodeIsSet = nodeUtil && nodeUtil.isSet, nodeIsTypedArray = nodeUtil && nodeUtil.isTypedArray;
+ function apply(func, thisArg, args) {
+ switch (args.length) {
+ case 0:
+ return func.call(thisArg);
+ case 1:
+ return func.call(thisArg, args[0]);
+ case 2:
+ return func.call(thisArg, args[0], args[1]);
+ case 3:
+ return func.call(thisArg, args[0], args[1], args[2]);
+ }
+ return func.apply(thisArg, args);
+ }
+ function arrayAggregator(array2, setter, iteratee, accumulator) {
+ var index = -1, length = array2 == null ? 0 : array2.length;
+ while (++index < length) {
+ var value = array2[index];
+ setter(accumulator, value, iteratee(value), array2);
+ }
+ return accumulator;
+ }
+ function arrayEach(array2, iteratee) {
+ var index = -1, length = array2 == null ? 0 : array2.length;
+ while (++index < length) {
+ if (iteratee(array2[index], index, array2) === false) {
+ break;
+ }
+ }
+ return array2;
+ }
+ function arrayEachRight(array2, iteratee) {
+ var length = array2 == null ? 0 : array2.length;
+ while (length--) {
+ if (iteratee(array2[length], length, array2) === false) {
+ break;
+ }
+ }
+ return array2;
+ }
+ function arrayEvery(array2, predicate) {
+ var index = -1, length = array2 == null ? 0 : array2.length;
+ while (++index < length) {
+ if (!predicate(array2[index], index, array2)) {
+ return false;
+ }
+ }
+ return true;
+ }
+ function arrayFilter(array2, predicate) {
+ var index = -1, length = array2 == null ? 0 : array2.length, resIndex = 0, result = [];
+ while (++index < length) {
+ var value = array2[index];
+ if (predicate(value, index, array2)) {
+ result[resIndex++] = value;
+ }
+ }
+ return result;
+ }
+ function arrayIncludes(array2, value) {
+ var length = array2 == null ? 0 : array2.length;
+ return !!length && baseIndexOf(array2, value, 0) > -1;
+ }
+ function arrayIncludesWith(array2, value, comparator) {
+ var index = -1, length = array2 == null ? 0 : array2.length;
+ while (++index < length) {
+ if (comparator(value, array2[index])) {
+ return true;
+ }
+ }
+ return false;
+ }
+ function arrayMap2(array2, iteratee) {
+ var index = -1, length = array2 == null ? 0 : array2.length, result = Array(length);
+ while (++index < length) {
+ result[index] = iteratee(array2[index], index, array2);
+ }
+ return result;
+ }
+ function arrayPush(array2, values) {
+ var index = -1, length = values.length, offset = array2.length;
+ while (++index < length) {
+ array2[offset + index] = values[index];
+ }
+ return array2;
+ }
+ function arrayReduce(array2, iteratee, accumulator, initAccum) {
+ var index = -1, length = array2 == null ? 0 : array2.length;
+ if (initAccum && length) {
+ accumulator = array2[++index];
+ }
+ while (++index < length) {
+ accumulator = iteratee(accumulator, array2[index], index, array2);
+ }
+ return accumulator;
+ }
+ function arrayReduceRight(array2, iteratee, accumulator, initAccum) {
+ var length = array2 == null ? 0 : array2.length;
+ if (initAccum && length) {
+ accumulator = array2[--length];
+ }
+ while (length--) {
+ accumulator = iteratee(accumulator, array2[length], length, array2);
+ }
+ return accumulator;
+ }
+ function arraySome(array2, predicate) {
+ var index = -1, length = array2 == null ? 0 : array2.length;
+ while (++index < length) {
+ if (predicate(array2[index], index, array2)) {
+ return true;
+ }
+ }
+ return false;
+ }
+ var asciiSize = baseProperty("length");
+ function asciiToArray(string) {
+ return string.split("");
+ }
+ function asciiWords(string) {
+ return string.match(reAsciiWord) || [];
+ }
+ function baseFindKey(collection, predicate, eachFunc) {
+ var result;
+ eachFunc(collection, function(value, key, collection2) {
+ if (predicate(value, key, collection2)) {
+ result = key;
+ return false;
+ }
+ });
+ return result;
+ }
+ function baseFindIndex(array2, predicate, fromIndex, fromRight) {
+ var length = array2.length, index = fromIndex + (fromRight ? 1 : -1);
+ while (fromRight ? index-- : ++index < length) {
+ if (predicate(array2[index], index, array2)) {
+ return index;
+ }
+ }
+ return -1;
+ }
+ function baseIndexOf(array2, value, fromIndex) {
+ return value === value ? strictIndexOf(array2, value, fromIndex) : baseFindIndex(array2, baseIsNaN, fromIndex);
+ }
+ function baseIndexOfWith(array2, value, fromIndex, comparator) {
+ var index = fromIndex - 1, length = array2.length;
+ while (++index < length) {
+ if (comparator(array2[index], value)) {
+ return index;
+ }
+ }
+ return -1;
+ }
+ function baseIsNaN(value) {
+ return value !== value;
+ }
+ function baseMean(array2, iteratee) {
+ var length = array2 == null ? 0 : array2.length;
+ return length ? baseSum(array2, iteratee) / length : NAN2;
+ }
+ function baseProperty(key) {
+ return function(object) {
+ return object == null ? undefined2 : object[key];
+ };
+ }
+ function basePropertyOf2(object) {
+ return function(key) {
+ return object == null ? undefined2 : object[key];
+ };
+ }
+ function baseReduce(collection, iteratee, accumulator, initAccum, eachFunc) {
+ eachFunc(collection, function(value, index, collection2) {
+ accumulator = initAccum ? (initAccum = false, value) : iteratee(accumulator, value, index, collection2);
+ });
+ return accumulator;
+ }
+ function baseSortBy(array2, comparer) {
+ var length = array2.length;
+ array2.sort(comparer);
+ while (length--) {
+ array2[length] = array2[length].value;
+ }
+ return array2;
+ }
+ function baseSum(array2, iteratee) {
+ var result, index = -1, length = array2.length;
+ while (++index < length) {
+ var current = iteratee(array2[index]);
+ if (current !== undefined2) {
+ result = result === undefined2 ? current : result + current;
+ }
+ }
+ return result;
+ }
+ function baseTimes(n2, iteratee) {
+ var index = -1, result = Array(n2);
+ while (++index < n2) {
+ result[index] = iteratee(index);
+ }
+ return result;
+ }
+ function baseToPairs(object, props) {
+ return arrayMap2(props, function(key) {
+ return [key, object[key]];
+ });
+ }
+ function baseTrim2(string) {
+ return string ? string.slice(0, trimmedEndIndex2(string) + 1).replace(reTrimStart2, "") : string;
+ }
+ function baseUnary(func) {
+ return function(value) {
+ return func(value);
+ };
+ }
+ function baseValues(object, props) {
+ return arrayMap2(props, function(key) {
+ return object[key];
+ });
+ }
+ function cacheHas(cache, key) {
+ return cache.has(key);
+ }
+ function charsStartIndex(strSymbols, chrSymbols) {
+ var index = -1, length = strSymbols.length;
+ while (++index < length && baseIndexOf(chrSymbols, strSymbols[index], 0) > -1) {
+ }
+ return index;
+ }
+ function charsEndIndex(strSymbols, chrSymbols) {
+ var index = strSymbols.length;
+ while (index-- && baseIndexOf(chrSymbols, strSymbols[index], 0) > -1) {
+ }
+ return index;
+ }
+ function countHolders(array2, placeholder) {
+ var length = array2.length, result = 0;
+ while (length--) {
+ if (array2[length] === placeholder) {
+ ++result;
+ }
+ }
+ return result;
+ }
+ var deburrLetter = basePropertyOf2(deburredLetters);
+ var escapeHtmlChar2 = basePropertyOf2(htmlEscapes2);
+ function escapeStringChar(chr) {
+ return "\\" + stringEscapes[chr];
+ }
+ function getValue(object, key) {
+ return object == null ? undefined2 : object[key];
+ }
+ function hasUnicode(string) {
+ return reHasUnicode.test(string);
+ }
+ function hasUnicodeWord(string) {
+ return reHasUnicodeWord.test(string);
+ }
+ function iteratorToArray(iterator) {
+ var data, result = [];
+ while (!(data = iterator.next()).done) {
+ result.push(data.value);
+ }
+ return result;
+ }
+ function mapToArray(map2) {
+ var index = -1, result = Array(map2.size);
+ map2.forEach(function(value, key) {
+ result[++index] = [key, value];
+ });
+ return result;
+ }
+ function overArg(func, transform2) {
+ return function(arg) {
+ return func(transform2(arg));
+ };
+ }
+ function replaceHolders(array2, placeholder) {
+ var index = -1, length = array2.length, resIndex = 0, result = [];
+ while (++index < length) {
+ var value = array2[index];
+ if (value === placeholder || value === PLACEHOLDER) {
+ array2[index] = PLACEHOLDER;
+ result[resIndex++] = index;
+ }
+ }
+ return result;
+ }
+ function setToArray(set3) {
+ var index = -1, result = Array(set3.size);
+ set3.forEach(function(value) {
+ result[++index] = value;
+ });
+ return result;
+ }
+ function setToPairs(set3) {
+ var index = -1, result = Array(set3.size);
+ set3.forEach(function(value) {
+ result[++index] = [value, value];
+ });
+ return result;
+ }
+ function strictIndexOf(array2, value, fromIndex) {
+ var index = fromIndex - 1, length = array2.length;
+ while (++index < length) {
+ if (array2[index] === value) {
+ return index;
+ }
+ }
+ return -1;
+ }
+ function strictLastIndexOf(array2, value, fromIndex) {
+ var index = fromIndex + 1;
+ while (index--) {
+ if (array2[index] === value) {
+ return index;
+ }
+ }
+ return index;
+ }
+ function stringSize(string) {
+ return hasUnicode(string) ? unicodeSize(string) : asciiSize(string);
+ }
+ function stringToArray(string) {
+ return hasUnicode(string) ? unicodeToArray(string) : asciiToArray(string);
+ }
+ function trimmedEndIndex2(string) {
+ var index = string.length;
+ while (index-- && reWhitespace2.test(string.charAt(index))) {
+ }
+ return index;
+ }
+ var unescapeHtmlChar2 = basePropertyOf2(htmlUnescapes2);
+ function unicodeSize(string) {
+ var result = reUnicode.lastIndex = 0;
+ while (reUnicode.test(string)) {
+ ++result;
+ }
+ return result;
+ }
+ function unicodeToArray(string) {
+ return string.match(reUnicode) || [];
+ }
+ function unicodeWords(string) {
+ return string.match(reUnicodeWord) || [];
+ }
+ var runInContext = function runInContext2(context) {
+ context = context == null ? root3 : _.defaults(root3.Object(), context, _.pick(root3, contextProps));
+ var Array2 = context.Array, Date2 = context.Date, Error2 = context.Error, Function2 = context.Function, Math2 = context.Math, Object2 = context.Object, RegExp2 = context.RegExp, String2 = context.String, TypeError2 = context.TypeError;
+ var arrayProto = Array2.prototype, funcProto = Function2.prototype, objectProto3 = Object2.prototype;
+ var coreJsData = context["__core-js_shared__"];
+ var funcToString = funcProto.toString;
+ var hasOwnProperty2 = objectProto3.hasOwnProperty;
+ var idCounter = 0;
+ var maskSrcKey = function() {
+ var uid = /[^.]+$/.exec(coreJsData && coreJsData.keys && coreJsData.keys.IE_PROTO || "");
+ return uid ? "Symbol(src)_1." + uid : "";
+ }();
+ var nativeObjectToString3 = objectProto3.toString;
+ var objectCtorString = funcToString.call(Object2);
+ var oldDash = root3._;
+ var reIsNative = RegExp2(
+ "^" + funcToString.call(hasOwnProperty2).replace(reRegExpChar, "\\$&").replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, "$1.*?") + "$"
+ );
+ var Buffer2 = moduleExports ? context.Buffer : undefined2, Symbol3 = context.Symbol, Uint8Array2 = context.Uint8Array, allocUnsafe = Buffer2 ? Buffer2.allocUnsafe : undefined2, getPrototype = overArg(Object2.getPrototypeOf, Object2), objectCreate = Object2.create, propertyIsEnumerable = objectProto3.propertyIsEnumerable, splice = arrayProto.splice, spreadableSymbol = Symbol3 ? Symbol3.isConcatSpreadable : undefined2, symIterator = Symbol3 ? Symbol3.iterator : undefined2, symToStringTag3 = Symbol3 ? Symbol3.toStringTag : undefined2;
+ var defineProperty = function() {
+ try {
+ var func = getNative(Object2, "defineProperty");
+ func({}, "", {});
+ return func;
+ } catch (e) {
+ }
+ }();
+ var ctxClearTimeout = context.clearTimeout !== root3.clearTimeout && context.clearTimeout, ctxNow = Date2 && Date2.now !== root3.Date.now && Date2.now, ctxSetTimeout = context.setTimeout !== root3.setTimeout && context.setTimeout;
+ var nativeCeil = Math2.ceil, nativeFloor = Math2.floor, nativeGetSymbols = Object2.getOwnPropertySymbols, nativeIsBuffer = Buffer2 ? Buffer2.isBuffer : undefined2, nativeIsFinite = context.isFinite, nativeJoin = arrayProto.join, nativeKeys = overArg(Object2.keys, Object2), nativeMax2 = Math2.max, nativeMin2 = Math2.min, nativeNow = Date2.now, nativeParseInt = context.parseInt, nativeRandom = Math2.random, nativeReverse = arrayProto.reverse;
+ var DataView2 = getNative(context, "DataView"), Map2 = getNative(context, "Map"), Promise2 = getNative(context, "Promise"), Set2 = getNative(context, "Set"), WeakMap = getNative(context, "WeakMap"), nativeCreate = getNative(Object2, "create");
+ var metaMap = WeakMap && new WeakMap();
+ var realNames = {};
+ var dataViewCtorString = toSource(DataView2), mapCtorString = toSource(Map2), promiseCtorString = toSource(Promise2), setCtorString = toSource(Set2), weakMapCtorString = toSource(WeakMap);
+ var symbolProto2 = Symbol3 ? Symbol3.prototype : undefined2, symbolValueOf = symbolProto2 ? symbolProto2.valueOf : undefined2, symbolToString2 = symbolProto2 ? symbolProto2.toString : undefined2;
+ function lodash(value) {
+ if (isObjectLike2(value) && !isArray2(value) && !(value instanceof LazyWrapper)) {
+ if (value instanceof LodashWrapper) {
+ return value;
+ }
+ if (hasOwnProperty2.call(value, "__wrapped__")) {
+ return wrapperClone(value);
+ }
+ }
+ return new LodashWrapper(value);
+ }
+ var baseCreate = function() {
+ function object() {
+ }
+ return function(proto) {
+ if (!isObject3(proto)) {
+ return {};
+ }
+ if (objectCreate) {
+ return objectCreate(proto);
+ }
+ object.prototype = proto;
+ var result2 = new object();
+ object.prototype = undefined2;
+ return result2;
+ };
+ }();
+ function baseLodash() {
+ }
+ function LodashWrapper(value, chainAll) {
+ this.__wrapped__ = value;
+ this.__actions__ = [];
+ this.__chain__ = !!chainAll;
+ this.__index__ = 0;
+ this.__values__ = undefined2;
+ }
+ lodash.templateSettings = {
+ /**
+ * Used to detect `data` property values to be HTML-escaped.
+ *
+ * @memberOf _.templateSettings
+ * @type {RegExp}
+ */
+ "escape": reEscape,
+ /**
+ * Used to detect code to be evaluated.
+ *
+ * @memberOf _.templateSettings
+ * @type {RegExp}
+ */
+ "evaluate": reEvaluate,
+ /**
+ * Used to detect `data` property values to inject.
+ *
+ * @memberOf _.templateSettings
+ * @type {RegExp}
+ */
+ "interpolate": reInterpolate,
+ /**
+ * Used to reference the data object in the template text.
+ *
+ * @memberOf _.templateSettings
+ * @type {string}
+ */
+ "variable": "",
+ /**
+ * Used to import variables into the compiled template.
+ *
+ * @memberOf _.templateSettings
+ * @type {Object}
+ */
+ "imports": {
+ /**
+ * A reference to the `lodash` function.
+ *
+ * @memberOf _.templateSettings.imports
+ * @type {Function}
+ */
+ "_": lodash
+ }
+ };
+ lodash.prototype = baseLodash.prototype;
+ lodash.prototype.constructor = lodash;
+ LodashWrapper.prototype = baseCreate(baseLodash.prototype);
+ LodashWrapper.prototype.constructor = LodashWrapper;
+ function LazyWrapper(value) {
+ this.__wrapped__ = value;
+ this.__actions__ = [];
+ this.__dir__ = 1;
+ this.__filtered__ = false;
+ this.__iteratees__ = [];
+ this.__takeCount__ = MAX_ARRAY_LENGTH;
+ this.__views__ = [];
+ }
+ function lazyClone() {
+ var result2 = new LazyWrapper(this.__wrapped__);
+ result2.__actions__ = copyArray(this.__actions__);
+ result2.__dir__ = this.__dir__;
+ result2.__filtered__ = this.__filtered__;
+ result2.__iteratees__ = copyArray(this.__iteratees__);
+ result2.__takeCount__ = this.__takeCount__;
+ result2.__views__ = copyArray(this.__views__);
+ return result2;
+ }
+ function lazyReverse() {
+ if (this.__filtered__) {
+ var result2 = new LazyWrapper(this);
+ result2.__dir__ = -1;
+ result2.__filtered__ = true;
+ } else {
+ result2 = this.clone();
+ result2.__dir__ *= -1;
+ }
+ return result2;
+ }
+ function lazyValue() {
+ var array2 = this.__wrapped__.value(), dir = this.__dir__, isArr = isArray2(array2), isRight = dir < 0, arrLength = isArr ? array2.length : 0, view = getView(0, arrLength, this.__views__), start2 = view.start, end = view.end, length = end - start2, index = isRight ? end : start2 - 1, iteratees = this.__iteratees__, iterLength = iteratees.length, resIndex = 0, takeCount = nativeMin2(length, this.__takeCount__);
+ if (!isArr || !isRight && arrLength == length && takeCount == length) {
+ return baseWrapperValue(array2, this.__actions__);
+ }
+ var result2 = [];
+ outer:
+ while (length-- && resIndex < takeCount) {
+ index += dir;
+ var iterIndex = -1, value = array2[index];
+ while (++iterIndex < iterLength) {
+ var data = iteratees[iterIndex], iteratee2 = data.iteratee, type2 = data.type, computed = iteratee2(value);
+ if (type2 == LAZY_MAP_FLAG) {
+ value = computed;
+ } else if (!computed) {
+ if (type2 == LAZY_FILTER_FLAG) {
+ continue outer;
+ } else {
+ break outer;
+ }
+ }
+ }
+ result2[resIndex++] = value;
+ }
+ return result2;
+ }
+ LazyWrapper.prototype = baseCreate(baseLodash.prototype);
+ LazyWrapper.prototype.constructor = LazyWrapper;
+ function Hash(entries) {
+ var index = -1, length = entries == null ? 0 : entries.length;
+ this.clear();
+ while (++index < length) {
+ var entry = entries[index];
+ this.set(entry[0], entry[1]);
+ }
+ }
+ function hashClear() {
+ this.__data__ = nativeCreate ? nativeCreate(null) : {};
+ this.size = 0;
+ }
+ function hashDelete(key) {
+ var result2 = this.has(key) && delete this.__data__[key];
+ this.size -= result2 ? 1 : 0;
+ return result2;
+ }
+ function hashGet(key) {
+ var data = this.__data__;
+ if (nativeCreate) {
+ var result2 = data[key];
+ return result2 === HASH_UNDEFINED ? undefined2 : result2;
+ }
+ return hasOwnProperty2.call(data, key) ? data[key] : undefined2;
+ }
+ function hashHas(key) {
+ var data = this.__data__;
+ return nativeCreate ? data[key] !== undefined2 : hasOwnProperty2.call(data, key);
+ }
+ function hashSet(key, value) {
+ var data = this.__data__;
+ this.size += this.has(key) ? 0 : 1;
+ data[key] = nativeCreate && value === undefined2 ? HASH_UNDEFINED : value;
+ return this;
+ }
+ Hash.prototype.clear = hashClear;
+ Hash.prototype["delete"] = hashDelete;
+ Hash.prototype.get = hashGet;
+ Hash.prototype.has = hashHas;
+ Hash.prototype.set = hashSet;
+ function ListCache(entries) {
+ var index = -1, length = entries == null ? 0 : entries.length;
+ this.clear();
+ while (++index < length) {
+ var entry = entries[index];
+ this.set(entry[0], entry[1]);
+ }
+ }
+ function listCacheClear() {
+ this.__data__ = [];
+ this.size = 0;
+ }
+ function listCacheDelete(key) {
+ var data = this.__data__, index = assocIndexOf(data, key);
+ if (index < 0) {
+ return false;
+ }
+ var lastIndex = data.length - 1;
+ if (index == lastIndex) {
+ data.pop();
+ } else {
+ splice.call(data, index, 1);
+ }
+ --this.size;
+ return true;
+ }
+ function listCacheGet(key) {
+ var data = this.__data__, index = assocIndexOf(data, key);
+ return index < 0 ? undefined2 : data[index][1];
+ }
+ function listCacheHas(key) {
+ return assocIndexOf(this.__data__, key) > -1;
+ }
+ function listCacheSet(key, value) {
+ var data = this.__data__, index = assocIndexOf(data, key);
+ if (index < 0) {
+ ++this.size;
+ data.push([key, value]);
+ } else {
+ data[index][1] = value;
+ }
+ return this;
+ }
+ ListCache.prototype.clear = listCacheClear;
+ ListCache.prototype["delete"] = listCacheDelete;
+ ListCache.prototype.get = listCacheGet;
+ ListCache.prototype.has = listCacheHas;
+ ListCache.prototype.set = listCacheSet;
+ function MapCache(entries) {
+ var index = -1, length = entries == null ? 0 : entries.length;
+ this.clear();
+ while (++index < length) {
+ var entry = entries[index];
+ this.set(entry[0], entry[1]);
+ }
+ }
+ function mapCacheClear() {
+ this.size = 0;
+ this.__data__ = {
+ "hash": new Hash(),
+ "map": new (Map2 || ListCache)(),
+ "string": new Hash()
+ };
+ }
+ function mapCacheDelete(key) {
+ var result2 = getMapData(this, key)["delete"](key);
+ this.size -= result2 ? 1 : 0;
+ return result2;
+ }
+ function mapCacheGet(key) {
+ return getMapData(this, key).get(key);
+ }
+ function mapCacheHas(key) {
+ return getMapData(this, key).has(key);
+ }
+ function mapCacheSet(key, value) {
+ var data = getMapData(this, key), size2 = data.size;
+ data.set(key, value);
+ this.size += data.size == size2 ? 0 : 1;
+ return this;
+ }
+ MapCache.prototype.clear = mapCacheClear;
+ MapCache.prototype["delete"] = mapCacheDelete;
+ MapCache.prototype.get = mapCacheGet;
+ MapCache.prototype.has = mapCacheHas;
+ MapCache.prototype.set = mapCacheSet;
+ function SetCache(values2) {
+ var index = -1, length = values2 == null ? 0 : values2.length;
+ this.__data__ = new MapCache();
+ while (++index < length) {
+ this.add(values2[index]);
+ }
+ }
+ function setCacheAdd(value) {
+ this.__data__.set(value, HASH_UNDEFINED);
+ return this;
+ }
+ function setCacheHas(value) {
+ return this.__data__.has(value);
+ }
+ SetCache.prototype.add = SetCache.prototype.push = setCacheAdd;
+ SetCache.prototype.has = setCacheHas;
+ function Stack(entries) {
+ var data = this.__data__ = new ListCache(entries);
+ this.size = data.size;
+ }
+ function stackClear() {
+ this.__data__ = new ListCache();
+ this.size = 0;
+ }
+ function stackDelete(key) {
+ var data = this.__data__, result2 = data["delete"](key);
+ this.size = data.size;
+ return result2;
+ }
+ function stackGet(key) {
+ return this.__data__.get(key);
+ }
+ function stackHas(key) {
+ return this.__data__.has(key);
+ }
+ function stackSet(key, value) {
+ var data = this.__data__;
+ if (data instanceof ListCache) {
+ var pairs = data.__data__;
+ if (!Map2 || pairs.length < LARGE_ARRAY_SIZE - 1) {
+ pairs.push([key, value]);
+ this.size = ++data.size;
+ return this;
+ }
+ data = this.__data__ = new MapCache(pairs);
+ }
+ data.set(key, value);
+ this.size = data.size;
+ return this;
+ }
+ Stack.prototype.clear = stackClear;
+ Stack.prototype["delete"] = stackDelete;
+ Stack.prototype.get = stackGet;
+ Stack.prototype.has = stackHas;
+ Stack.prototype.set = stackSet;
+ function arrayLikeKeys(value, inherited) {
+ var isArr = isArray2(value), isArg = !isArr && isArguments(value), isBuff = !isArr && !isArg && isBuffer(value), isType = !isArr && !isArg && !isBuff && isTypedArray(value), skipIndexes = isArr || isArg || isBuff || isType, result2 = skipIndexes ? baseTimes(value.length, String2) : [], length = result2.length;
+ for (var key in value) {
+ if ((inherited || hasOwnProperty2.call(value, key)) && !(skipIndexes && // Safari 9 has enumerable `arguments.length` in strict mode.
+ (key == "length" || // Node.js 0.10 has enumerable non-index properties on buffers.
+ isBuff && (key == "offset" || key == "parent") || // PhantomJS 2 has enumerable non-index properties on typed arrays.
+ isType && (key == "buffer" || key == "byteLength" || key == "byteOffset") || // Skip index properties.
+ isIndex(key, length)))) {
+ result2.push(key);
+ }
+ }
+ return result2;
+ }
+ function arraySample(array2) {
+ var length = array2.length;
+ return length ? array2[baseRandom(0, length - 1)] : undefined2;
+ }
+ function arraySampleSize(array2, n2) {
+ return shuffleSelf(copyArray(array2), baseClamp(n2, 0, array2.length));
+ }
+ function arrayShuffle(array2) {
+ return shuffleSelf(copyArray(array2));
+ }
+ function assignMergeValue(object, key, value) {
+ if (value !== undefined2 && !eq(object[key], value) || value === undefined2 && !(key in object)) {
+ baseAssignValue(object, key, value);
+ }
+ }
+ function assignValue(object, key, value) {
+ var objValue = object[key];
+ if (!(hasOwnProperty2.call(object, key) && eq(objValue, value)) || value === undefined2 && !(key in object)) {
+ baseAssignValue(object, key, value);
+ }
+ }
+ function assocIndexOf(array2, key) {
+ var length = array2.length;
+ while (length--) {
+ if (eq(array2[length][0], key)) {
+ return length;
+ }
+ }
+ return -1;
+ }
+ function baseAggregator(collection, setter, iteratee2, accumulator) {
+ baseEach(collection, function(value, key, collection2) {
+ setter(accumulator, value, iteratee2(value), collection2);
+ });
+ return accumulator;
+ }
+ function baseAssign(object, source) {
+ return object && copyObject(source, keys(source), object);
+ }
+ function baseAssignIn(object, source) {
+ return object && copyObject(source, keysIn(source), object);
+ }
+ function baseAssignValue(object, key, value) {
+ if (key == "__proto__" && defineProperty) {
+ defineProperty(object, key, {
+ "configurable": true,
+ "enumerable": true,
+ "value": value,
+ "writable": true
+ });
+ } else {
+ object[key] = value;
+ }
+ }
+ function baseAt(object, paths) {
+ var index = -1, length = paths.length, result2 = Array2(length), skip = object == null;
+ while (++index < length) {
+ result2[index] = skip ? undefined2 : get4(object, paths[index]);
+ }
+ return result2;
+ }
+ function baseClamp(number3, lower2, upper) {
+ if (number3 === number3) {
+ if (upper !== undefined2) {
+ number3 = number3 <= upper ? number3 : upper;
+ }
+ if (lower2 !== undefined2) {
+ number3 = number3 >= lower2 ? number3 : lower2;
+ }
+ }
+ return number3;
+ }
+ function baseClone(value, bitmask, customizer, key, object, stack) {
+ var result2, isDeep = bitmask & CLONE_DEEP_FLAG, isFlat = bitmask & CLONE_FLAT_FLAG, isFull = bitmask & CLONE_SYMBOLS_FLAG;
+ if (customizer) {
+ result2 = object ? customizer(value, key, object, stack) : customizer(value);
+ }
+ if (result2 !== undefined2) {
+ return result2;
+ }
+ if (!isObject3(value)) {
+ return value;
+ }
+ var isArr = isArray2(value);
+ if (isArr) {
+ result2 = initCloneArray(value);
+ if (!isDeep) {
+ return copyArray(value, result2);
+ }
+ } else {
+ var tag = getTag(value), isFunc = tag == funcTag || tag == genTag;
+ if (isBuffer(value)) {
+ return cloneBuffer(value, isDeep);
+ }
+ if (tag == objectTag || tag == argsTag || isFunc && !object) {
+ result2 = isFlat || isFunc ? {} : initCloneObject(value);
+ if (!isDeep) {
+ return isFlat ? copySymbolsIn(value, baseAssignIn(result2, value)) : copySymbols(value, baseAssign(result2, value));
+ }
+ } else {
+ if (!cloneableTags[tag]) {
+ return object ? value : {};
+ }
+ result2 = initCloneByTag(value, tag, isDeep);
+ }
+ }
+ stack || (stack = new Stack());
+ var stacked = stack.get(value);
+ if (stacked) {
+ return stacked;
+ }
+ stack.set(value, result2);
+ if (isSet(value)) {
+ value.forEach(function(subValue) {
+ result2.add(baseClone(subValue, bitmask, customizer, subValue, value, stack));
+ });
+ } else if (isMap(value)) {
+ value.forEach(function(subValue, key2) {
+ result2.set(key2, baseClone(subValue, bitmask, customizer, key2, value, stack));
+ });
+ }
+ var keysFunc = isFull ? isFlat ? getAllKeysIn : getAllKeys : isFlat ? keysIn : keys;
+ var props = isArr ? undefined2 : keysFunc(value);
+ arrayEach(props || value, function(subValue, key2) {
+ if (props) {
+ key2 = subValue;
+ subValue = value[key2];
+ }
+ assignValue(result2, key2, baseClone(subValue, bitmask, customizer, key2, value, stack));
+ });
+ return result2;
+ }
+ function baseConforms(source) {
+ var props = keys(source);
+ return function(object) {
+ return baseConformsTo(object, source, props);
+ };
+ }
+ function baseConformsTo(object, source, props) {
+ var length = props.length;
+ if (object == null) {
+ return !length;
+ }
+ object = Object2(object);
+ while (length--) {
+ var key = props[length], predicate = source[key], value = object[key];
+ if (value === undefined2 && !(key in object) || !predicate(value)) {
+ return false;
+ }
+ }
+ return true;
+ }
+ function baseDelay(func, wait, args) {
+ if (typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ return setTimeout2(function() {
+ func.apply(undefined2, args);
+ }, wait);
+ }
+ function baseDifference(array2, values2, iteratee2, comparator) {
+ var index = -1, includes2 = arrayIncludes, isCommon = true, length = array2.length, result2 = [], valuesLength = values2.length;
+ if (!length) {
+ return result2;
+ }
+ if (iteratee2) {
+ values2 = arrayMap2(values2, baseUnary(iteratee2));
+ }
+ if (comparator) {
+ includes2 = arrayIncludesWith;
+ isCommon = false;
+ } else if (values2.length >= LARGE_ARRAY_SIZE) {
+ includes2 = cacheHas;
+ isCommon = false;
+ values2 = new SetCache(values2);
+ }
+ outer:
+ while (++index < length) {
+ var value = array2[index], computed = iteratee2 == null ? value : iteratee2(value);
+ value = comparator || value !== 0 ? value : 0;
+ if (isCommon && computed === computed) {
+ var valuesIndex = valuesLength;
+ while (valuesIndex--) {
+ if (values2[valuesIndex] === computed) {
+ continue outer;
+ }
+ }
+ result2.push(value);
+ } else if (!includes2(values2, computed, comparator)) {
+ result2.push(value);
+ }
+ }
+ return result2;
+ }
+ var baseEach = createBaseEach(baseForOwn);
+ var baseEachRight = createBaseEach(baseForOwnRight, true);
+ function baseEvery(collection, predicate) {
+ var result2 = true;
+ baseEach(collection, function(value, index, collection2) {
+ result2 = !!predicate(value, index, collection2);
+ return result2;
+ });
+ return result2;
+ }
+ function baseExtremum(array2, iteratee2, comparator) {
+ var index = -1, length = array2.length;
+ while (++index < length) {
+ var value = array2[index], current = iteratee2(value);
+ if (current != null && (computed === undefined2 ? current === current && !isSymbol2(current) : comparator(current, computed))) {
+ var computed = current, result2 = value;
+ }
+ }
+ return result2;
+ }
+ function baseFill(array2, value, start2, end) {
+ var length = array2.length;
+ start2 = toInteger(start2);
+ if (start2 < 0) {
+ start2 = -start2 > length ? 0 : length + start2;
+ }
+ end = end === undefined2 || end > length ? length : toInteger(end);
+ if (end < 0) {
+ end += length;
+ }
+ end = start2 > end ? 0 : toLength(end);
+ while (start2 < end) {
+ array2[start2++] = value;
+ }
+ return array2;
+ }
+ function baseFilter(collection, predicate) {
+ var result2 = [];
+ baseEach(collection, function(value, index, collection2) {
+ if (predicate(value, index, collection2)) {
+ result2.push(value);
+ }
+ });
+ return result2;
+ }
+ function baseFlatten(array2, depth, predicate, isStrict, result2) {
+ var index = -1, length = array2.length;
+ predicate || (predicate = isFlattenable);
+ result2 || (result2 = []);
+ while (++index < length) {
+ var value = array2[index];
+ if (depth > 0 && predicate(value)) {
+ if (depth > 1) {
+ baseFlatten(value, depth - 1, predicate, isStrict, result2);
+ } else {
+ arrayPush(result2, value);
+ }
+ } else if (!isStrict) {
+ result2[result2.length] = value;
+ }
+ }
+ return result2;
+ }
+ var baseFor = createBaseFor();
+ var baseForRight = createBaseFor(true);
+ function baseForOwn(object, iteratee2) {
+ return object && baseFor(object, iteratee2, keys);
+ }
+ function baseForOwnRight(object, iteratee2) {
+ return object && baseForRight(object, iteratee2, keys);
+ }
+ function baseFunctions(object, props) {
+ return arrayFilter(props, function(key) {
+ return isFunction(object[key]);
+ });
+ }
+ function baseGet(object, path) {
+ path = castPath(path, object);
+ var index = 0, length = path.length;
+ while (object != null && index < length) {
+ object = object[toKey(path[index++])];
+ }
+ return index && index == length ? object : undefined2;
+ }
+ function baseGetAllKeys(object, keysFunc, symbolsFunc) {
+ var result2 = keysFunc(object);
+ return isArray2(object) ? result2 : arrayPush(result2, symbolsFunc(object));
+ }
+ function baseGetTag2(value) {
+ if (value == null) {
+ return value === undefined2 ? undefinedTag2 : nullTag2;
+ }
+ return symToStringTag3 && symToStringTag3 in Object2(value) ? getRawTag2(value) : objectToString2(value);
+ }
+ function baseGt(value, other) {
+ return value > other;
+ }
+ function baseHas(object, key) {
+ return object != null && hasOwnProperty2.call(object, key);
+ }
+ function baseHasIn(object, key) {
+ return object != null && key in Object2(object);
+ }
+ function baseInRange(number3, start2, end) {
+ return number3 >= nativeMin2(start2, end) && number3 < nativeMax2(start2, end);
+ }
+ function baseIntersection(arrays, iteratee2, comparator) {
+ var includes2 = comparator ? arrayIncludesWith : arrayIncludes, length = arrays[0].length, othLength = arrays.length, othIndex = othLength, caches = Array2(othLength), maxLength = Infinity, result2 = [];
+ while (othIndex--) {
+ var array2 = arrays[othIndex];
+ if (othIndex && iteratee2) {
+ array2 = arrayMap2(array2, baseUnary(iteratee2));
+ }
+ maxLength = nativeMin2(array2.length, maxLength);
+ caches[othIndex] = !comparator && (iteratee2 || length >= 120 && array2.length >= 120) ? new SetCache(othIndex && array2) : undefined2;
+ }
+ array2 = arrays[0];
+ var index = -1, seen = caches[0];
+ outer:
+ while (++index < length && result2.length < maxLength) {
+ var value = array2[index], computed = iteratee2 ? iteratee2(value) : value;
+ value = comparator || value !== 0 ? value : 0;
+ if (!(seen ? cacheHas(seen, computed) : includes2(result2, computed, comparator))) {
+ othIndex = othLength;
+ while (--othIndex) {
+ var cache = caches[othIndex];
+ if (!(cache ? cacheHas(cache, computed) : includes2(arrays[othIndex], computed, comparator))) {
+ continue outer;
+ }
+ }
+ if (seen) {
+ seen.push(computed);
+ }
+ result2.push(value);
+ }
+ }
+ return result2;
+ }
+ function baseInverter(object, setter, iteratee2, accumulator) {
+ baseForOwn(object, function(value, key, object2) {
+ setter(accumulator, iteratee2(value), key, object2);
+ });
+ return accumulator;
+ }
+ function baseInvoke(object, path, args) {
+ path = castPath(path, object);
+ object = parent(object, path);
+ var func = object == null ? object : object[toKey(last(path))];
+ return func == null ? undefined2 : apply(func, object, args);
+ }
+ function baseIsArguments(value) {
+ return isObjectLike2(value) && baseGetTag2(value) == argsTag;
+ }
+ function baseIsArrayBuffer(value) {
+ return isObjectLike2(value) && baseGetTag2(value) == arrayBufferTag;
+ }
+ function baseIsDate(value) {
+ return isObjectLike2(value) && baseGetTag2(value) == dateTag;
+ }
+ function baseIsEqual(value, other, bitmask, customizer, stack) {
+ if (value === other) {
+ return true;
+ }
+ if (value == null || other == null || !isObjectLike2(value) && !isObjectLike2(other)) {
+ return value !== value && other !== other;
+ }
+ return baseIsEqualDeep(value, other, bitmask, customizer, baseIsEqual, stack);
+ }
+ function baseIsEqualDeep(object, other, bitmask, customizer, equalFunc, stack) {
+ var objIsArr = isArray2(object), othIsArr = isArray2(other), objTag = objIsArr ? arrayTag : getTag(object), othTag = othIsArr ? arrayTag : getTag(other);
+ objTag = objTag == argsTag ? objectTag : objTag;
+ othTag = othTag == argsTag ? objectTag : othTag;
+ var objIsObj = objTag == objectTag, othIsObj = othTag == objectTag, isSameTag = objTag == othTag;
+ if (isSameTag && isBuffer(object)) {
+ if (!isBuffer(other)) {
+ return false;
+ }
+ objIsArr = true;
+ objIsObj = false;
+ }
+ if (isSameTag && !objIsObj) {
+ stack || (stack = new Stack());
+ return objIsArr || isTypedArray(object) ? equalArrays(object, other, bitmask, customizer, equalFunc, stack) : equalByTag(object, other, objTag, bitmask, customizer, equalFunc, stack);
+ }
+ if (!(bitmask & COMPARE_PARTIAL_FLAG)) {
+ var objIsWrapped = objIsObj && hasOwnProperty2.call(object, "__wrapped__"), othIsWrapped = othIsObj && hasOwnProperty2.call(other, "__wrapped__");
+ if (objIsWrapped || othIsWrapped) {
+ var objUnwrapped = objIsWrapped ? object.value() : object, othUnwrapped = othIsWrapped ? other.value() : other;
+ stack || (stack = new Stack());
+ return equalFunc(objUnwrapped, othUnwrapped, bitmask, customizer, stack);
+ }
+ }
+ if (!isSameTag) {
+ return false;
+ }
+ stack || (stack = new Stack());
+ return equalObjects(object, other, bitmask, customizer, equalFunc, stack);
+ }
+ function baseIsMap(value) {
+ return isObjectLike2(value) && getTag(value) == mapTag;
+ }
+ function baseIsMatch(object, source, matchData, customizer) {
+ var index = matchData.length, length = index, noCustomizer = !customizer;
+ if (object == null) {
+ return !length;
+ }
+ object = Object2(object);
+ while (index--) {
+ var data = matchData[index];
+ if (noCustomizer && data[2] ? data[1] !== object[data[0]] : !(data[0] in object)) {
+ return false;
+ }
+ }
+ while (++index < length) {
+ data = matchData[index];
+ var key = data[0], objValue = object[key], srcValue = data[1];
+ if (noCustomizer && data[2]) {
+ if (objValue === undefined2 && !(key in object)) {
+ return false;
+ }
+ } else {
+ var stack = new Stack();
+ if (customizer) {
+ var result2 = customizer(objValue, srcValue, key, object, source, stack);
+ }
+ if (!(result2 === undefined2 ? baseIsEqual(srcValue, objValue, COMPARE_PARTIAL_FLAG | COMPARE_UNORDERED_FLAG, customizer, stack) : result2)) {
+ return false;
+ }
+ }
+ }
+ return true;
+ }
+ function baseIsNative(value) {
+ if (!isObject3(value) || isMasked(value)) {
+ return false;
+ }
+ var pattern = isFunction(value) ? reIsNative : reIsHostCtor;
+ return pattern.test(toSource(value));
+ }
+ function baseIsRegExp(value) {
+ return isObjectLike2(value) && baseGetTag2(value) == regexpTag;
+ }
+ function baseIsSet(value) {
+ return isObjectLike2(value) && getTag(value) == setTag;
+ }
+ function baseIsTypedArray(value) {
+ return isObjectLike2(value) && isLength(value.length) && !!typedArrayTags[baseGetTag2(value)];
+ }
+ function baseIteratee(value) {
+ if (typeof value == "function") {
+ return value;
+ }
+ if (value == null) {
+ return identity4;
+ }
+ if (typeof value == "object") {
+ return isArray2(value) ? baseMatchesProperty(value[0], value[1]) : baseMatches(value);
+ }
+ return property(value);
+ }
+ function baseKeys(object) {
+ if (!isPrototype(object)) {
+ return nativeKeys(object);
+ }
+ var result2 = [];
+ for (var key in Object2(object)) {
+ if (hasOwnProperty2.call(object, key) && key != "constructor") {
+ result2.push(key);
+ }
+ }
+ return result2;
+ }
+ function baseKeysIn(object) {
+ if (!isObject3(object)) {
+ return nativeKeysIn(object);
+ }
+ var isProto = isPrototype(object), result2 = [];
+ for (var key in object) {
+ if (!(key == "constructor" && (isProto || !hasOwnProperty2.call(object, key)))) {
+ result2.push(key);
+ }
+ }
+ return result2;
+ }
+ function baseLt(value, other) {
+ return value < other;
+ }
+ function baseMap(collection, iteratee2) {
+ var index = -1, result2 = isArrayLike(collection) ? Array2(collection.length) : [];
+ baseEach(collection, function(value, key, collection2) {
+ result2[++index] = iteratee2(value, key, collection2);
+ });
+ return result2;
+ }
+ function baseMatches(source) {
+ var matchData = getMatchData(source);
+ if (matchData.length == 1 && matchData[0][2]) {
+ return matchesStrictComparable(matchData[0][0], matchData[0][1]);
+ }
+ return function(object) {
+ return object === source || baseIsMatch(object, source, matchData);
+ };
+ }
+ function baseMatchesProperty(path, srcValue) {
+ if (isKey(path) && isStrictComparable(srcValue)) {
+ return matchesStrictComparable(toKey(path), srcValue);
+ }
+ return function(object) {
+ var objValue = get4(object, path);
+ return objValue === undefined2 && objValue === srcValue ? hasIn(object, path) : baseIsEqual(srcValue, objValue, COMPARE_PARTIAL_FLAG | COMPARE_UNORDERED_FLAG);
+ };
+ }
+ function baseMerge(object, source, srcIndex, customizer, stack) {
+ if (object === source) {
+ return;
+ }
+ baseFor(source, function(srcValue, key) {
+ stack || (stack = new Stack());
+ if (isObject3(srcValue)) {
+ baseMergeDeep(object, source, key, srcIndex, baseMerge, customizer, stack);
+ } else {
+ var newValue = customizer ? customizer(safeGet(object, key), srcValue, key + "", object, source, stack) : undefined2;
+ if (newValue === undefined2) {
+ newValue = srcValue;
+ }
+ assignMergeValue(object, key, newValue);
+ }
+ }, keysIn);
+ }
+ function baseMergeDeep(object, source, key, srcIndex, mergeFunc, customizer, stack) {
+ var objValue = safeGet(object, key), srcValue = safeGet(source, key), stacked = stack.get(srcValue);
+ if (stacked) {
+ assignMergeValue(object, key, stacked);
+ return;
+ }
+ var newValue = customizer ? customizer(objValue, srcValue, key + "", object, source, stack) : undefined2;
+ var isCommon = newValue === undefined2;
+ if (isCommon) {
+ var isArr = isArray2(srcValue), isBuff = !isArr && isBuffer(srcValue), isTyped = !isArr && !isBuff && isTypedArray(srcValue);
+ newValue = srcValue;
+ if (isArr || isBuff || isTyped) {
+ if (isArray2(objValue)) {
+ newValue = objValue;
+ } else if (isArrayLikeObject(objValue)) {
+ newValue = copyArray(objValue);
+ } else if (isBuff) {
+ isCommon = false;
+ newValue = cloneBuffer(srcValue, true);
+ } else if (isTyped) {
+ isCommon = false;
+ newValue = cloneTypedArray(srcValue, true);
+ } else {
+ newValue = [];
+ }
+ } else if (isPlainObject(srcValue) || isArguments(srcValue)) {
+ newValue = objValue;
+ if (isArguments(objValue)) {
+ newValue = toPlainObject(objValue);
+ } else if (!isObject3(objValue) || isFunction(objValue)) {
+ newValue = initCloneObject(srcValue);
+ }
+ } else {
+ isCommon = false;
+ }
+ }
+ if (isCommon) {
+ stack.set(srcValue, newValue);
+ mergeFunc(newValue, srcValue, srcIndex, customizer, stack);
+ stack["delete"](srcValue);
+ }
+ assignMergeValue(object, key, newValue);
+ }
+ function baseNth(array2, n2) {
+ var length = array2.length;
+ if (!length) {
+ return;
+ }
+ n2 += n2 < 0 ? length : 0;
+ return isIndex(n2, length) ? array2[n2] : undefined2;
+ }
+ function baseOrderBy(collection, iteratees, orders) {
+ if (iteratees.length) {
+ iteratees = arrayMap2(iteratees, function(iteratee2) {
+ if (isArray2(iteratee2)) {
+ return function(value) {
+ return baseGet(value, iteratee2.length === 1 ? iteratee2[0] : iteratee2);
+ };
+ }
+ return iteratee2;
+ });
+ } else {
+ iteratees = [identity4];
+ }
+ var index = -1;
+ iteratees = arrayMap2(iteratees, baseUnary(getIteratee()));
+ var result2 = baseMap(collection, function(value, key, collection2) {
+ var criteria = arrayMap2(iteratees, function(iteratee2) {
+ return iteratee2(value);
+ });
+ return { "criteria": criteria, "index": ++index, "value": value };
+ });
+ return baseSortBy(result2, function(object, other) {
+ return compareMultiple(object, other, orders);
+ });
+ }
+ function basePick(object, paths) {
+ return basePickBy(object, paths, function(value, path) {
+ return hasIn(object, path);
+ });
+ }
+ function basePickBy(object, paths, predicate) {
+ var index = -1, length = paths.length, result2 = {};
+ while (++index < length) {
+ var path = paths[index], value = baseGet(object, path);
+ if (predicate(value, path)) {
+ baseSet(result2, castPath(path, object), value);
+ }
+ }
+ return result2;
+ }
+ function basePropertyDeep(path) {
+ return function(object) {
+ return baseGet(object, path);
+ };
+ }
+ function basePullAll(array2, values2, iteratee2, comparator) {
+ var indexOf2 = comparator ? baseIndexOfWith : baseIndexOf, index = -1, length = values2.length, seen = array2;
+ if (array2 === values2) {
+ values2 = copyArray(values2);
+ }
+ if (iteratee2) {
+ seen = arrayMap2(array2, baseUnary(iteratee2));
+ }
+ while (++index < length) {
+ var fromIndex = 0, value = values2[index], computed = iteratee2 ? iteratee2(value) : value;
+ while ((fromIndex = indexOf2(seen, computed, fromIndex, comparator)) > -1) {
+ if (seen !== array2) {
+ splice.call(seen, fromIndex, 1);
+ }
+ splice.call(array2, fromIndex, 1);
+ }
+ }
+ return array2;
+ }
+ function basePullAt(array2, indexes) {
+ var length = array2 ? indexes.length : 0, lastIndex = length - 1;
+ while (length--) {
+ var index = indexes[length];
+ if (length == lastIndex || index !== previous) {
+ var previous = index;
+ if (isIndex(index)) {
+ splice.call(array2, index, 1);
+ } else {
+ baseUnset(array2, index);
+ }
+ }
+ }
+ return array2;
+ }
+ function baseRandom(lower2, upper) {
+ return lower2 + nativeFloor(nativeRandom() * (upper - lower2 + 1));
+ }
+ function baseRange(start2, end, step, fromRight) {
+ var index = -1, length = nativeMax2(nativeCeil((end - start2) / (step || 1)), 0), result2 = Array2(length);
+ while (length--) {
+ result2[fromRight ? length : ++index] = start2;
+ start2 += step;
+ }
+ return result2;
+ }
+ function baseRepeat(string, n2) {
+ var result2 = "";
+ if (!string || n2 < 1 || n2 > MAX_SAFE_INTEGER) {
+ return result2;
+ }
+ do {
+ if (n2 % 2) {
+ result2 += string;
+ }
+ n2 = nativeFloor(n2 / 2);
+ if (n2) {
+ string += string;
+ }
+ } while (n2);
+ return result2;
+ }
+ function baseRest(func, start2) {
+ return setToString(overRest(func, start2, identity4), func + "");
+ }
+ function baseSample(collection) {
+ return arraySample(values(collection));
+ }
+ function baseSampleSize(collection, n2) {
+ var array2 = values(collection);
+ return shuffleSelf(array2, baseClamp(n2, 0, array2.length));
+ }
+ function baseSet(object, path, value, customizer) {
+ if (!isObject3(object)) {
+ return object;
+ }
+ path = castPath(path, object);
+ var index = -1, length = path.length, lastIndex = length - 1, nested = object;
+ while (nested != null && ++index < length) {
+ var key = toKey(path[index]), newValue = value;
+ if (key === "__proto__" || key === "constructor" || key === "prototype") {
+ return object;
+ }
+ if (index != lastIndex) {
+ var objValue = nested[key];
+ newValue = customizer ? customizer(objValue, key, nested) : undefined2;
+ if (newValue === undefined2) {
+ newValue = isObject3(objValue) ? objValue : isIndex(path[index + 1]) ? [] : {};
+ }
+ }
+ assignValue(nested, key, newValue);
+ nested = nested[key];
+ }
+ return object;
+ }
+ var baseSetData = !metaMap ? identity4 : function(func, data) {
+ metaMap.set(func, data);
+ return func;
+ };
+ var baseSetToString = !defineProperty ? identity4 : function(func, string) {
+ return defineProperty(func, "toString", {
+ "configurable": true,
+ "enumerable": false,
+ "value": constant(string),
+ "writable": true
+ });
+ };
+ function baseShuffle(collection) {
+ return shuffleSelf(values(collection));
+ }
+ function baseSlice(array2, start2, end) {
+ var index = -1, length = array2.length;
+ if (start2 < 0) {
+ start2 = -start2 > length ? 0 : length + start2;
+ }
+ end = end > length ? length : end;
+ if (end < 0) {
+ end += length;
+ }
+ length = start2 > end ? 0 : end - start2 >>> 0;
+ start2 >>>= 0;
+ var result2 = Array2(length);
+ while (++index < length) {
+ result2[index] = array2[index + start2];
+ }
+ return result2;
+ }
+ function baseSome(collection, predicate) {
+ var result2;
+ baseEach(collection, function(value, index, collection2) {
+ result2 = predicate(value, index, collection2);
+ return !result2;
+ });
+ return !!result2;
+ }
+ function baseSortedIndex(array2, value, retHighest) {
+ var low = 0, high = array2 == null ? low : array2.length;
+ if (typeof value == "number" && value === value && high <= HALF_MAX_ARRAY_LENGTH) {
+ while (low < high) {
+ var mid = low + high >>> 1, computed = array2[mid];
+ if (computed !== null && !isSymbol2(computed) && (retHighest ? computed <= value : computed < value)) {
+ low = mid + 1;
+ } else {
+ high = mid;
+ }
+ }
+ return high;
+ }
+ return baseSortedIndexBy(array2, value, identity4, retHighest);
+ }
+ function baseSortedIndexBy(array2, value, iteratee2, retHighest) {
+ var low = 0, high = array2 == null ? 0 : array2.length;
+ if (high === 0) {
+ return 0;
+ }
+ value = iteratee2(value);
+ var valIsNaN = value !== value, valIsNull = value === null, valIsSymbol = isSymbol2(value), valIsUndefined = value === undefined2;
+ while (low < high) {
+ var mid = nativeFloor((low + high) / 2), computed = iteratee2(array2[mid]), othIsDefined = computed !== undefined2, othIsNull = computed === null, othIsReflexive = computed === computed, othIsSymbol = isSymbol2(computed);
+ if (valIsNaN) {
+ var setLow = retHighest || othIsReflexive;
+ } else if (valIsUndefined) {
+ setLow = othIsReflexive && (retHighest || othIsDefined);
+ } else if (valIsNull) {
+ setLow = othIsReflexive && othIsDefined && (retHighest || !othIsNull);
+ } else if (valIsSymbol) {
+ setLow = othIsReflexive && othIsDefined && !othIsNull && (retHighest || !othIsSymbol);
+ } else if (othIsNull || othIsSymbol) {
+ setLow = false;
+ } else {
+ setLow = retHighest ? computed <= value : computed < value;
+ }
+ if (setLow) {
+ low = mid + 1;
+ } else {
+ high = mid;
+ }
+ }
+ return nativeMin2(high, MAX_ARRAY_INDEX);
+ }
+ function baseSortedUniq(array2, iteratee2) {
+ var index = -1, length = array2.length, resIndex = 0, result2 = [];
+ while (++index < length) {
+ var value = array2[index], computed = iteratee2 ? iteratee2(value) : value;
+ if (!index || !eq(computed, seen)) {
+ var seen = computed;
+ result2[resIndex++] = value === 0 ? 0 : value;
+ }
+ }
+ return result2;
+ }
+ function baseToNumber(value) {
+ if (typeof value == "number") {
+ return value;
+ }
+ if (isSymbol2(value)) {
+ return NAN2;
+ }
+ return +value;
+ }
+ function baseToString2(value) {
+ if (typeof value == "string") {
+ return value;
+ }
+ if (isArray2(value)) {
+ return arrayMap2(value, baseToString2) + "";
+ }
+ if (isSymbol2(value)) {
+ return symbolToString2 ? symbolToString2.call(value) : "";
+ }
+ var result2 = value + "";
+ return result2 == "0" && 1 / value == -INFINITY2 ? "-0" : result2;
+ }
+ function baseUniq(array2, iteratee2, comparator) {
+ var index = -1, includes2 = arrayIncludes, length = array2.length, isCommon = true, result2 = [], seen = result2;
+ if (comparator) {
+ isCommon = false;
+ includes2 = arrayIncludesWith;
+ } else if (length >= LARGE_ARRAY_SIZE) {
+ var set4 = iteratee2 ? null : createSet(array2);
+ if (set4) {
+ return setToArray(set4);
+ }
+ isCommon = false;
+ includes2 = cacheHas;
+ seen = new SetCache();
+ } else {
+ seen = iteratee2 ? [] : result2;
+ }
+ outer:
+ while (++index < length) {
+ var value = array2[index], computed = iteratee2 ? iteratee2(value) : value;
+ value = comparator || value !== 0 ? value : 0;
+ if (isCommon && computed === computed) {
+ var seenIndex = seen.length;
+ while (seenIndex--) {
+ if (seen[seenIndex] === computed) {
+ continue outer;
+ }
+ }
+ if (iteratee2) {
+ seen.push(computed);
+ }
+ result2.push(value);
+ } else if (!includes2(seen, computed, comparator)) {
+ if (seen !== result2) {
+ seen.push(computed);
+ }
+ result2.push(value);
+ }
+ }
+ return result2;
+ }
+ function baseUnset(object, path) {
+ path = castPath(path, object);
+ object = parent(object, path);
+ return object == null || delete object[toKey(last(path))];
+ }
+ function baseUpdate(object, path, updater, customizer) {
+ return baseSet(object, path, updater(baseGet(object, path)), customizer);
+ }
+ function baseWhile(array2, predicate, isDrop, fromRight) {
+ var length = array2.length, index = fromRight ? length : -1;
+ while ((fromRight ? index-- : ++index < length) && predicate(array2[index], index, array2)) {
+ }
+ return isDrop ? baseSlice(array2, fromRight ? 0 : index, fromRight ? index + 1 : length) : baseSlice(array2, fromRight ? index + 1 : 0, fromRight ? length : index);
+ }
+ function baseWrapperValue(value, actions) {
+ var result2 = value;
+ if (result2 instanceof LazyWrapper) {
+ result2 = result2.value();
+ }
+ return arrayReduce(actions, function(result3, action) {
+ return action.func.apply(action.thisArg, arrayPush([result3], action.args));
+ }, result2);
+ }
+ function baseXor(arrays, iteratee2, comparator) {
+ var length = arrays.length;
+ if (length < 2) {
+ return length ? baseUniq(arrays[0]) : [];
+ }
+ var index = -1, result2 = Array2(length);
+ while (++index < length) {
+ var array2 = arrays[index], othIndex = -1;
+ while (++othIndex < length) {
+ if (othIndex != index) {
+ result2[index] = baseDifference(result2[index] || array2, arrays[othIndex], iteratee2, comparator);
+ }
+ }
+ }
+ return baseUniq(baseFlatten(result2, 1), iteratee2, comparator);
+ }
+ function baseZipObject(props, values2, assignFunc) {
+ var index = -1, length = props.length, valsLength = values2.length, result2 = {};
+ while (++index < length) {
+ var value = index < valsLength ? values2[index] : undefined2;
+ assignFunc(result2, props[index], value);
+ }
+ return result2;
+ }
+ function castArrayLikeObject(value) {
+ return isArrayLikeObject(value) ? value : [];
+ }
+ function castFunction(value) {
+ return typeof value == "function" ? value : identity4;
+ }
+ function castPath(value, object) {
+ if (isArray2(value)) {
+ return value;
+ }
+ return isKey(value, object) ? [value] : stringToPath(toString2(value));
+ }
+ var castRest = baseRest;
+ function castSlice(array2, start2, end) {
+ var length = array2.length;
+ end = end === undefined2 ? length : end;
+ return !start2 && end >= length ? array2 : baseSlice(array2, start2, end);
+ }
+ var clearTimeout2 = ctxClearTimeout || function(id2) {
+ return root3.clearTimeout(id2);
+ };
+ function cloneBuffer(buffer, isDeep) {
+ if (isDeep) {
+ return buffer.slice();
+ }
+ var length = buffer.length, result2 = allocUnsafe ? allocUnsafe(length) : new buffer.constructor(length);
+ buffer.copy(result2);
+ return result2;
+ }
+ function cloneArrayBuffer(arrayBuffer) {
+ var result2 = new arrayBuffer.constructor(arrayBuffer.byteLength);
+ new Uint8Array2(result2).set(new Uint8Array2(arrayBuffer));
+ return result2;
+ }
+ function cloneDataView(dataView, isDeep) {
+ var buffer = isDeep ? cloneArrayBuffer(dataView.buffer) : dataView.buffer;
+ return new dataView.constructor(buffer, dataView.byteOffset, dataView.byteLength);
+ }
+ function cloneRegExp(regexp) {
+ var result2 = new regexp.constructor(regexp.source, reFlags.exec(regexp));
+ result2.lastIndex = regexp.lastIndex;
+ return result2;
+ }
+ function cloneSymbol(symbol) {
+ return symbolValueOf ? Object2(symbolValueOf.call(symbol)) : {};
+ }
+ function cloneTypedArray(typedArray, isDeep) {
+ var buffer = isDeep ? cloneArrayBuffer(typedArray.buffer) : typedArray.buffer;
+ return new typedArray.constructor(buffer, typedArray.byteOffset, typedArray.length);
+ }
+ function compareAscending(value, other) {
+ if (value !== other) {
+ var valIsDefined = value !== undefined2, valIsNull = value === null, valIsReflexive = value === value, valIsSymbol = isSymbol2(value);
+ var othIsDefined = other !== undefined2, othIsNull = other === null, othIsReflexive = other === other, othIsSymbol = isSymbol2(other);
+ if (!othIsNull && !othIsSymbol && !valIsSymbol && value > other || valIsSymbol && othIsDefined && othIsReflexive && !othIsNull && !othIsSymbol || valIsNull && othIsDefined && othIsReflexive || !valIsDefined && othIsReflexive || !valIsReflexive) {
+ return 1;
+ }
+ if (!valIsNull && !valIsSymbol && !othIsSymbol && value < other || othIsSymbol && valIsDefined && valIsReflexive && !valIsNull && !valIsSymbol || othIsNull && valIsDefined && valIsReflexive || !othIsDefined && valIsReflexive || !othIsReflexive) {
+ return -1;
+ }
+ }
+ return 0;
+ }
+ function compareMultiple(object, other, orders) {
+ var index = -1, objCriteria = object.criteria, othCriteria = other.criteria, length = objCriteria.length, ordersLength = orders.length;
+ while (++index < length) {
+ var result2 = compareAscending(objCriteria[index], othCriteria[index]);
+ if (result2) {
+ if (index >= ordersLength) {
+ return result2;
+ }
+ var order = orders[index];
+ return result2 * (order == "desc" ? -1 : 1);
+ }
+ }
+ return object.index - other.index;
+ }
+ function composeArgs(args, partials, holders, isCurried) {
+ var argsIndex = -1, argsLength = args.length, holdersLength = holders.length, leftIndex = -1, leftLength = partials.length, rangeLength = nativeMax2(argsLength - holdersLength, 0), result2 = Array2(leftLength + rangeLength), isUncurried = !isCurried;
+ while (++leftIndex < leftLength) {
+ result2[leftIndex] = partials[leftIndex];
+ }
+ while (++argsIndex < holdersLength) {
+ if (isUncurried || argsIndex < argsLength) {
+ result2[holders[argsIndex]] = args[argsIndex];
+ }
+ }
+ while (rangeLength--) {
+ result2[leftIndex++] = args[argsIndex++];
+ }
+ return result2;
+ }
+ function composeArgsRight(args, partials, holders, isCurried) {
+ var argsIndex = -1, argsLength = args.length, holdersIndex = -1, holdersLength = holders.length, rightIndex = -1, rightLength = partials.length, rangeLength = nativeMax2(argsLength - holdersLength, 0), result2 = Array2(rangeLength + rightLength), isUncurried = !isCurried;
+ while (++argsIndex < rangeLength) {
+ result2[argsIndex] = args[argsIndex];
+ }
+ var offset = argsIndex;
+ while (++rightIndex < rightLength) {
+ result2[offset + rightIndex] = partials[rightIndex];
+ }
+ while (++holdersIndex < holdersLength) {
+ if (isUncurried || argsIndex < argsLength) {
+ result2[offset + holders[holdersIndex]] = args[argsIndex++];
+ }
+ }
+ return result2;
+ }
+ function copyArray(source, array2) {
+ var index = -1, length = source.length;
+ array2 || (array2 = Array2(length));
+ while (++index < length) {
+ array2[index] = source[index];
+ }
+ return array2;
+ }
+ function copyObject(source, props, object, customizer) {
+ var isNew = !object;
+ object || (object = {});
+ var index = -1, length = props.length;
+ while (++index < length) {
+ var key = props[index];
+ var newValue = customizer ? customizer(object[key], source[key], key, object, source) : undefined2;
+ if (newValue === undefined2) {
+ newValue = source[key];
+ }
+ if (isNew) {
+ baseAssignValue(object, key, newValue);
+ } else {
+ assignValue(object, key, newValue);
+ }
+ }
+ return object;
+ }
+ function copySymbols(source, object) {
+ return copyObject(source, getSymbols(source), object);
+ }
+ function copySymbolsIn(source, object) {
+ return copyObject(source, getSymbolsIn(source), object);
+ }
+ function createAggregator(setter, initializer) {
+ return function(collection, iteratee2) {
+ var func = isArray2(collection) ? arrayAggregator : baseAggregator, accumulator = initializer ? initializer() : {};
+ return func(collection, setter, getIteratee(iteratee2, 2), accumulator);
+ };
+ }
+ function createAssigner(assigner) {
+ return baseRest(function(object, sources) {
+ var index = -1, length = sources.length, customizer = length > 1 ? sources[length - 1] : undefined2, guard = length > 2 ? sources[2] : undefined2;
+ customizer = assigner.length > 3 && typeof customizer == "function" ? (length--, customizer) : undefined2;
+ if (guard && isIterateeCall(sources[0], sources[1], guard)) {
+ customizer = length < 3 ? undefined2 : customizer;
+ length = 1;
+ }
+ object = Object2(object);
+ while (++index < length) {
+ var source = sources[index];
+ if (source) {
+ assigner(object, source, index, customizer);
+ }
+ }
+ return object;
+ });
+ }
+ function createBaseEach(eachFunc, fromRight) {
+ return function(collection, iteratee2) {
+ if (collection == null) {
+ return collection;
+ }
+ if (!isArrayLike(collection)) {
+ return eachFunc(collection, iteratee2);
+ }
+ var length = collection.length, index = fromRight ? length : -1, iterable = Object2(collection);
+ while (fromRight ? index-- : ++index < length) {
+ if (iteratee2(iterable[index], index, iterable) === false) {
+ break;
+ }
+ }
+ return collection;
+ };
+ }
+ function createBaseFor(fromRight) {
+ return function(object, iteratee2, keysFunc) {
+ var index = -1, iterable = Object2(object), props = keysFunc(object), length = props.length;
+ while (length--) {
+ var key = props[fromRight ? length : ++index];
+ if (iteratee2(iterable[key], key, iterable) === false) {
+ break;
+ }
+ }
+ return object;
+ };
+ }
+ function createBind(func, bitmask, thisArg) {
+ var isBind = bitmask & WRAP_BIND_FLAG, Ctor = createCtor(func);
+ function wrapper() {
+ var fn = this && this !== root3 && this instanceof wrapper ? Ctor : func;
+ return fn.apply(isBind ? thisArg : this, arguments);
+ }
+ return wrapper;
+ }
+ function createCaseFirst(methodName) {
+ return function(string) {
+ string = toString2(string);
+ var strSymbols = hasUnicode(string) ? stringToArray(string) : undefined2;
+ var chr = strSymbols ? strSymbols[0] : string.charAt(0);
+ var trailing = strSymbols ? castSlice(strSymbols, 1).join("") : string.slice(1);
+ return chr[methodName]() + trailing;
+ };
+ }
+ function createCompounder(callback) {
+ return function(string) {
+ return arrayReduce(words(deburr(string).replace(reApos, "")), callback, "");
+ };
+ }
+ function createCtor(Ctor) {
+ return function() {
+ var args = arguments;
+ switch (args.length) {
+ case 0:
+ return new Ctor();
+ case 1:
+ return new Ctor(args[0]);
+ case 2:
+ return new Ctor(args[0], args[1]);
+ case 3:
+ return new Ctor(args[0], args[1], args[2]);
+ case 4:
+ return new Ctor(args[0], args[1], args[2], args[3]);
+ case 5:
+ return new Ctor(args[0], args[1], args[2], args[3], args[4]);
+ case 6:
+ return new Ctor(args[0], args[1], args[2], args[3], args[4], args[5]);
+ case 7:
+ return new Ctor(args[0], args[1], args[2], args[3], args[4], args[5], args[6]);
+ }
+ var thisBinding = baseCreate(Ctor.prototype), result2 = Ctor.apply(thisBinding, args);
+ return isObject3(result2) ? result2 : thisBinding;
+ };
+ }
+ function createCurry(func, bitmask, arity) {
+ var Ctor = createCtor(func);
+ function wrapper() {
+ var length = arguments.length, args = Array2(length), index = length, placeholder = getHolder(wrapper);
+ while (index--) {
+ args[index] = arguments[index];
+ }
+ var holders = length < 3 && args[0] !== placeholder && args[length - 1] !== placeholder ? [] : replaceHolders(args, placeholder);
+ length -= holders.length;
+ if (length < arity) {
+ return createRecurry(
+ func,
+ bitmask,
+ createHybrid,
+ wrapper.placeholder,
+ undefined2,
+ args,
+ holders,
+ undefined2,
+ undefined2,
+ arity - length
+ );
+ }
+ var fn = this && this !== root3 && this instanceof wrapper ? Ctor : func;
+ return apply(fn, this, args);
+ }
+ return wrapper;
+ }
+ function createFind(findIndexFunc) {
+ return function(collection, predicate, fromIndex) {
+ var iterable = Object2(collection);
+ if (!isArrayLike(collection)) {
+ var iteratee2 = getIteratee(predicate, 3);
+ collection = keys(collection);
+ predicate = function(key) {
+ return iteratee2(iterable[key], key, iterable);
+ };
+ }
+ var index = findIndexFunc(collection, predicate, fromIndex);
+ return index > -1 ? iterable[iteratee2 ? collection[index] : index] : undefined2;
+ };
+ }
+ function createFlow(fromRight) {
+ return flatRest(function(funcs) {
+ var length = funcs.length, index = length, prereq = LodashWrapper.prototype.thru;
+ if (fromRight) {
+ funcs.reverse();
+ }
+ while (index--) {
+ var func = funcs[index];
+ if (typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ if (prereq && !wrapper && getFuncName(func) == "wrapper") {
+ var wrapper = new LodashWrapper([], true);
+ }
+ }
+ index = wrapper ? index : length;
+ while (++index < length) {
+ func = funcs[index];
+ var funcName = getFuncName(func), data = funcName == "wrapper" ? getData(func) : undefined2;
+ if (data && isLaziable(data[0]) && data[1] == (WRAP_ARY_FLAG | WRAP_CURRY_FLAG | WRAP_PARTIAL_FLAG | WRAP_REARG_FLAG) && !data[4].length && data[9] == 1) {
+ wrapper = wrapper[getFuncName(data[0])].apply(wrapper, data[3]);
+ } else {
+ wrapper = func.length == 1 && isLaziable(func) ? wrapper[funcName]() : wrapper.thru(func);
+ }
+ }
+ return function() {
+ var args = arguments, value = args[0];
+ if (wrapper && args.length == 1 && isArray2(value)) {
+ return wrapper.plant(value).value();
+ }
+ var index2 = 0, result2 = length ? funcs[index2].apply(this, args) : value;
+ while (++index2 < length) {
+ result2 = funcs[index2].call(this, result2);
+ }
+ return result2;
+ };
+ });
+ }
+ function createHybrid(func, bitmask, thisArg, partials, holders, partialsRight, holdersRight, argPos, ary2, arity) {
+ var isAry = bitmask & WRAP_ARY_FLAG, isBind = bitmask & WRAP_BIND_FLAG, isBindKey = bitmask & WRAP_BIND_KEY_FLAG, isCurried = bitmask & (WRAP_CURRY_FLAG | WRAP_CURRY_RIGHT_FLAG), isFlip = bitmask & WRAP_FLIP_FLAG, Ctor = isBindKey ? undefined2 : createCtor(func);
+ function wrapper() {
+ var length = arguments.length, args = Array2(length), index = length;
+ while (index--) {
+ args[index] = arguments[index];
+ }
+ if (isCurried) {
+ var placeholder = getHolder(wrapper), holdersCount = countHolders(args, placeholder);
+ }
+ if (partials) {
+ args = composeArgs(args, partials, holders, isCurried);
+ }
+ if (partialsRight) {
+ args = composeArgsRight(args, partialsRight, holdersRight, isCurried);
+ }
+ length -= holdersCount;
+ if (isCurried && length < arity) {
+ var newHolders = replaceHolders(args, placeholder);
+ return createRecurry(
+ func,
+ bitmask,
+ createHybrid,
+ wrapper.placeholder,
+ thisArg,
+ args,
+ newHolders,
+ argPos,
+ ary2,
+ arity - length
+ );
+ }
+ var thisBinding = isBind ? thisArg : this, fn = isBindKey ? thisBinding[func] : func;
+ length = args.length;
+ if (argPos) {
+ args = reorder(args, argPos);
+ } else if (isFlip && length > 1) {
+ args.reverse();
+ }
+ if (isAry && ary2 < length) {
+ args.length = ary2;
+ }
+ if (this && this !== root3 && this instanceof wrapper) {
+ fn = Ctor || createCtor(fn);
+ }
+ return fn.apply(thisBinding, args);
+ }
+ return wrapper;
+ }
+ function createInverter(setter, toIteratee) {
+ return function(object, iteratee2) {
+ return baseInverter(object, setter, toIteratee(iteratee2), {});
+ };
+ }
+ function createMathOperation(operator, defaultValue) {
+ return function(value, other) {
+ var result2;
+ if (value === undefined2 && other === undefined2) {
+ return defaultValue;
+ }
+ if (value !== undefined2) {
+ result2 = value;
+ }
+ if (other !== undefined2) {
+ if (result2 === undefined2) {
+ return other;
+ }
+ if (typeof value == "string" || typeof other == "string") {
+ value = baseToString2(value);
+ other = baseToString2(other);
+ } else {
+ value = baseToNumber(value);
+ other = baseToNumber(other);
+ }
+ result2 = operator(value, other);
+ }
+ return result2;
+ };
+ }
+ function createOver(arrayFunc) {
+ return flatRest(function(iteratees) {
+ iteratees = arrayMap2(iteratees, baseUnary(getIteratee()));
+ return baseRest(function(args) {
+ var thisArg = this;
+ return arrayFunc(iteratees, function(iteratee2) {
+ return apply(iteratee2, thisArg, args);
+ });
+ });
+ });
+ }
+ function createPadding(length, chars) {
+ chars = chars === undefined2 ? " " : baseToString2(chars);
+ var charsLength = chars.length;
+ if (charsLength < 2) {
+ return charsLength ? baseRepeat(chars, length) : chars;
+ }
+ var result2 = baseRepeat(chars, nativeCeil(length / stringSize(chars)));
+ return hasUnicode(chars) ? castSlice(stringToArray(result2), 0, length).join("") : result2.slice(0, length);
+ }
+ function createPartial(func, bitmask, thisArg, partials) {
+ var isBind = bitmask & WRAP_BIND_FLAG, Ctor = createCtor(func);
+ function wrapper() {
+ var argsIndex = -1, argsLength = arguments.length, leftIndex = -1, leftLength = partials.length, args = Array2(leftLength + argsLength), fn = this && this !== root3 && this instanceof wrapper ? Ctor : func;
+ while (++leftIndex < leftLength) {
+ args[leftIndex] = partials[leftIndex];
+ }
+ while (argsLength--) {
+ args[leftIndex++] = arguments[++argsIndex];
+ }
+ return apply(fn, isBind ? thisArg : this, args);
+ }
+ return wrapper;
+ }
+ function createRange(fromRight) {
+ return function(start2, end, step) {
+ if (step && typeof step != "number" && isIterateeCall(start2, end, step)) {
+ end = step = undefined2;
+ }
+ start2 = toFinite(start2);
+ if (end === undefined2) {
+ end = start2;
+ start2 = 0;
+ } else {
+ end = toFinite(end);
+ }
+ step = step === undefined2 ? start2 < end ? 1 : -1 : toFinite(step);
+ return baseRange(start2, end, step, fromRight);
+ };
+ }
+ function createRelationalOperation(operator) {
+ return function(value, other) {
+ if (!(typeof value == "string" && typeof other == "string")) {
+ value = toNumber2(value);
+ other = toNumber2(other);
+ }
+ return operator(value, other);
+ };
+ }
+ function createRecurry(func, bitmask, wrapFunc, placeholder, thisArg, partials, holders, argPos, ary2, arity) {
+ var isCurry = bitmask & WRAP_CURRY_FLAG, newHolders = isCurry ? holders : undefined2, newHoldersRight = isCurry ? undefined2 : holders, newPartials = isCurry ? partials : undefined2, newPartialsRight = isCurry ? undefined2 : partials;
+ bitmask |= isCurry ? WRAP_PARTIAL_FLAG : WRAP_PARTIAL_RIGHT_FLAG;
+ bitmask &= ~(isCurry ? WRAP_PARTIAL_RIGHT_FLAG : WRAP_PARTIAL_FLAG);
+ if (!(bitmask & WRAP_CURRY_BOUND_FLAG)) {
+ bitmask &= ~(WRAP_BIND_FLAG | WRAP_BIND_KEY_FLAG);
+ }
+ var newData = [
+ func,
+ bitmask,
+ thisArg,
+ newPartials,
+ newHolders,
+ newPartialsRight,
+ newHoldersRight,
+ argPos,
+ ary2,
+ arity
+ ];
+ var result2 = wrapFunc.apply(undefined2, newData);
+ if (isLaziable(func)) {
+ setData(result2, newData);
+ }
+ result2.placeholder = placeholder;
+ return setWrapToString(result2, func, bitmask);
+ }
+ function createRound(methodName) {
+ var func = Math2[methodName];
+ return function(number3, precision2) {
+ number3 = toNumber2(number3);
+ precision2 = precision2 == null ? 0 : nativeMin2(toInteger(precision2), 292);
+ if (precision2 && nativeIsFinite(number3)) {
+ var pair2 = (toString2(number3) + "e").split("e"), value = func(pair2[0] + "e" + (+pair2[1] + precision2));
+ pair2 = (toString2(value) + "e").split("e");
+ return +(pair2[0] + "e" + (+pair2[1] - precision2));
+ }
+ return func(number3);
+ };
+ }
+ var createSet = !(Set2 && 1 / setToArray(new Set2([, -0]))[1] == INFINITY2) ? noop3 : function(values2) {
+ return new Set2(values2);
+ };
+ function createToPairs(keysFunc) {
+ return function(object) {
+ var tag = getTag(object);
+ if (tag == mapTag) {
+ return mapToArray(object);
+ }
+ if (tag == setTag) {
+ return setToPairs(object);
+ }
+ return baseToPairs(object, keysFunc(object));
+ };
+ }
+ function createWrap(func, bitmask, thisArg, partials, holders, argPos, ary2, arity) {
+ var isBindKey = bitmask & WRAP_BIND_KEY_FLAG;
+ if (!isBindKey && typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ var length = partials ? partials.length : 0;
+ if (!length) {
+ bitmask &= ~(WRAP_PARTIAL_FLAG | WRAP_PARTIAL_RIGHT_FLAG);
+ partials = holders = undefined2;
+ }
+ ary2 = ary2 === undefined2 ? ary2 : nativeMax2(toInteger(ary2), 0);
+ arity = arity === undefined2 ? arity : toInteger(arity);
+ length -= holders ? holders.length : 0;
+ if (bitmask & WRAP_PARTIAL_RIGHT_FLAG) {
+ var partialsRight = partials, holdersRight = holders;
+ partials = holders = undefined2;
+ }
+ var data = isBindKey ? undefined2 : getData(func);
+ var newData = [
+ func,
+ bitmask,
+ thisArg,
+ partials,
+ holders,
+ partialsRight,
+ holdersRight,
+ argPos,
+ ary2,
+ arity
+ ];
+ if (data) {
+ mergeData(newData, data);
+ }
+ func = newData[0];
+ bitmask = newData[1];
+ thisArg = newData[2];
+ partials = newData[3];
+ holders = newData[4];
+ arity = newData[9] = newData[9] === undefined2 ? isBindKey ? 0 : func.length : nativeMax2(newData[9] - length, 0);
+ if (!arity && bitmask & (WRAP_CURRY_FLAG | WRAP_CURRY_RIGHT_FLAG)) {
+ bitmask &= ~(WRAP_CURRY_FLAG | WRAP_CURRY_RIGHT_FLAG);
+ }
+ if (!bitmask || bitmask == WRAP_BIND_FLAG) {
+ var result2 = createBind(func, bitmask, thisArg);
+ } else if (bitmask == WRAP_CURRY_FLAG || bitmask == WRAP_CURRY_RIGHT_FLAG) {
+ result2 = createCurry(func, bitmask, arity);
+ } else if ((bitmask == WRAP_PARTIAL_FLAG || bitmask == (WRAP_BIND_FLAG | WRAP_PARTIAL_FLAG)) && !holders.length) {
+ result2 = createPartial(func, bitmask, thisArg, partials);
+ } else {
+ result2 = createHybrid.apply(undefined2, newData);
+ }
+ var setter = data ? baseSetData : setData;
+ return setWrapToString(setter(result2, newData), func, bitmask);
+ }
+ function customDefaultsAssignIn(objValue, srcValue, key, object) {
+ if (objValue === undefined2 || eq(objValue, objectProto3[key]) && !hasOwnProperty2.call(object, key)) {
+ return srcValue;
+ }
+ return objValue;
+ }
+ function customDefaultsMerge(objValue, srcValue, key, object, source, stack) {
+ if (isObject3(objValue) && isObject3(srcValue)) {
+ stack.set(srcValue, objValue);
+ baseMerge(objValue, srcValue, undefined2, customDefaultsMerge, stack);
+ stack["delete"](srcValue);
+ }
+ return objValue;
+ }
+ function customOmitClone(value) {
+ return isPlainObject(value) ? undefined2 : value;
+ }
+ function equalArrays(array2, other, bitmask, customizer, equalFunc, stack) {
+ var isPartial = bitmask & COMPARE_PARTIAL_FLAG, arrLength = array2.length, othLength = other.length;
+ if (arrLength != othLength && !(isPartial && othLength > arrLength)) {
+ return false;
+ }
+ var arrStacked = stack.get(array2);
+ var othStacked = stack.get(other);
+ if (arrStacked && othStacked) {
+ return arrStacked == other && othStacked == array2;
+ }
+ var index = -1, result2 = true, seen = bitmask & COMPARE_UNORDERED_FLAG ? new SetCache() : undefined2;
+ stack.set(array2, other);
+ stack.set(other, array2);
+ while (++index < arrLength) {
+ var arrValue = array2[index], othValue = other[index];
+ if (customizer) {
+ var compared = isPartial ? customizer(othValue, arrValue, index, other, array2, stack) : customizer(arrValue, othValue, index, array2, other, stack);
+ }
+ if (compared !== undefined2) {
+ if (compared) {
+ continue;
+ }
+ result2 = false;
+ break;
+ }
+ if (seen) {
+ if (!arraySome(other, function(othValue2, othIndex) {
+ if (!cacheHas(seen, othIndex) && (arrValue === othValue2 || equalFunc(arrValue, othValue2, bitmask, customizer, stack))) {
+ return seen.push(othIndex);
+ }
+ })) {
+ result2 = false;
+ break;
+ }
+ } else if (!(arrValue === othValue || equalFunc(arrValue, othValue, bitmask, customizer, stack))) {
+ result2 = false;
+ break;
+ }
+ }
+ stack["delete"](array2);
+ stack["delete"](other);
+ return result2;
+ }
+ function equalByTag(object, other, tag, bitmask, customizer, equalFunc, stack) {
+ switch (tag) {
+ case dataViewTag:
+ if (object.byteLength != other.byteLength || object.byteOffset != other.byteOffset) {
+ return false;
+ }
+ object = object.buffer;
+ other = other.buffer;
+ case arrayBufferTag:
+ if (object.byteLength != other.byteLength || !equalFunc(new Uint8Array2(object), new Uint8Array2(other))) {
+ return false;
+ }
+ return true;
+ case boolTag:
+ case dateTag:
+ case numberTag:
+ return eq(+object, +other);
+ case errorTag:
+ return object.name == other.name && object.message == other.message;
+ case regexpTag:
+ case stringTag:
+ return object == other + "";
+ case mapTag:
+ var convert = mapToArray;
+ case setTag:
+ var isPartial = bitmask & COMPARE_PARTIAL_FLAG;
+ convert || (convert = setToArray);
+ if (object.size != other.size && !isPartial) {
+ return false;
+ }
+ var stacked = stack.get(object);
+ if (stacked) {
+ return stacked == other;
+ }
+ bitmask |= COMPARE_UNORDERED_FLAG;
+ stack.set(object, other);
+ var result2 = equalArrays(convert(object), convert(other), bitmask, customizer, equalFunc, stack);
+ stack["delete"](object);
+ return result2;
+ case symbolTag2:
+ if (symbolValueOf) {
+ return symbolValueOf.call(object) == symbolValueOf.call(other);
+ }
+ }
+ return false;
+ }
+ function equalObjects(object, other, bitmask, customizer, equalFunc, stack) {
+ var isPartial = bitmask & COMPARE_PARTIAL_FLAG, objProps = getAllKeys(object), objLength = objProps.length, othProps = getAllKeys(other), othLength = othProps.length;
+ if (objLength != othLength && !isPartial) {
+ return false;
+ }
+ var index = objLength;
+ while (index--) {
+ var key = objProps[index];
+ if (!(isPartial ? key in other : hasOwnProperty2.call(other, key))) {
+ return false;
+ }
+ }
+ var objStacked = stack.get(object);
+ var othStacked = stack.get(other);
+ if (objStacked && othStacked) {
+ return objStacked == other && othStacked == object;
+ }
+ var result2 = true;
+ stack.set(object, other);
+ stack.set(other, object);
+ var skipCtor = isPartial;
+ while (++index < objLength) {
+ key = objProps[index];
+ var objValue = object[key], othValue = other[key];
+ if (customizer) {
+ var compared = isPartial ? customizer(othValue, objValue, key, other, object, stack) : customizer(objValue, othValue, key, object, other, stack);
+ }
+ if (!(compared === undefined2 ? objValue === othValue || equalFunc(objValue, othValue, bitmask, customizer, stack) : compared)) {
+ result2 = false;
+ break;
+ }
+ skipCtor || (skipCtor = key == "constructor");
+ }
+ if (result2 && !skipCtor) {
+ var objCtor = object.constructor, othCtor = other.constructor;
+ if (objCtor != othCtor && ("constructor" in object && "constructor" in other) && !(typeof objCtor == "function" && objCtor instanceof objCtor && typeof othCtor == "function" && othCtor instanceof othCtor)) {
+ result2 = false;
+ }
+ }
+ stack["delete"](object);
+ stack["delete"](other);
+ return result2;
+ }
+ function flatRest(func) {
+ return setToString(overRest(func, undefined2, flatten2), func + "");
+ }
+ function getAllKeys(object) {
+ return baseGetAllKeys(object, keys, getSymbols);
+ }
+ function getAllKeysIn(object) {
+ return baseGetAllKeys(object, keysIn, getSymbolsIn);
+ }
+ var getData = !metaMap ? noop3 : function(func) {
+ return metaMap.get(func);
+ };
+ function getFuncName(func) {
+ var result2 = func.name + "", array2 = realNames[result2], length = hasOwnProperty2.call(realNames, result2) ? array2.length : 0;
+ while (length--) {
+ var data = array2[length], otherFunc = data.func;
+ if (otherFunc == null || otherFunc == func) {
+ return data.name;
+ }
+ }
+ return result2;
+ }
+ function getHolder(func) {
+ var object = hasOwnProperty2.call(lodash, "placeholder") ? lodash : func;
+ return object.placeholder;
+ }
+ function getIteratee() {
+ var result2 = lodash.iteratee || iteratee;
+ result2 = result2 === iteratee ? baseIteratee : result2;
+ return arguments.length ? result2(arguments[0], arguments[1]) : result2;
+ }
+ function getMapData(map3, key) {
+ var data = map3.__data__;
+ return isKeyable(key) ? data[typeof key == "string" ? "string" : "hash"] : data.map;
+ }
+ function getMatchData(object) {
+ var result2 = keys(object), length = result2.length;
+ while (length--) {
+ var key = result2[length], value = object[key];
+ result2[length] = [key, value, isStrictComparable(value)];
+ }
+ return result2;
+ }
+ function getNative(object, key) {
+ var value = getValue(object, key);
+ return baseIsNative(value) ? value : undefined2;
+ }
+ function getRawTag2(value) {
+ var isOwn = hasOwnProperty2.call(value, symToStringTag3), tag = value[symToStringTag3];
+ try {
+ value[symToStringTag3] = undefined2;
+ var unmasked = true;
+ } catch (e) {
+ }
+ var result2 = nativeObjectToString3.call(value);
+ if (unmasked) {
+ if (isOwn) {
+ value[symToStringTag3] = tag;
+ } else {
+ delete value[symToStringTag3];
+ }
+ }
+ return result2;
+ }
+ var getSymbols = !nativeGetSymbols ? stubArray : function(object) {
+ if (object == null) {
+ return [];
+ }
+ object = Object2(object);
+ return arrayFilter(nativeGetSymbols(object), function(symbol) {
+ return propertyIsEnumerable.call(object, symbol);
+ });
+ };
+ var getSymbolsIn = !nativeGetSymbols ? stubArray : function(object) {
+ var result2 = [];
+ while (object) {
+ arrayPush(result2, getSymbols(object));
+ object = getPrototype(object);
+ }
+ return result2;
+ };
+ var getTag = baseGetTag2;
+ if (DataView2 && getTag(new DataView2(new ArrayBuffer(1))) != dataViewTag || Map2 && getTag(new Map2()) != mapTag || Promise2 && getTag(Promise2.resolve()) != promiseTag || Set2 && getTag(new Set2()) != setTag || WeakMap && getTag(new WeakMap()) != weakMapTag) {
+ getTag = function(value) {
+ var result2 = baseGetTag2(value), Ctor = result2 == objectTag ? value.constructor : undefined2, ctorString = Ctor ? toSource(Ctor) : "";
+ if (ctorString) {
+ switch (ctorString) {
+ case dataViewCtorString:
+ return dataViewTag;
+ case mapCtorString:
+ return mapTag;
+ case promiseCtorString:
+ return promiseTag;
+ case setCtorString:
+ return setTag;
+ case weakMapCtorString:
+ return weakMapTag;
+ }
+ }
+ return result2;
+ };
+ }
+ function getView(start2, end, transforms) {
+ var index = -1, length = transforms.length;
+ while (++index < length) {
+ var data = transforms[index], size2 = data.size;
+ switch (data.type) {
+ case "drop":
+ start2 += size2;
+ break;
+ case "dropRight":
+ end -= size2;
+ break;
+ case "take":
+ end = nativeMin2(end, start2 + size2);
+ break;
+ case "takeRight":
+ start2 = nativeMax2(start2, end - size2);
+ break;
+ }
+ }
+ return { "start": start2, "end": end };
+ }
+ function getWrapDetails(source) {
+ var match = source.match(reWrapDetails);
+ return match ? match[1].split(reSplitDetails) : [];
+ }
+ function hasPath(object, path, hasFunc) {
+ path = castPath(path, object);
+ var index = -1, length = path.length, result2 = false;
+ while (++index < length) {
+ var key = toKey(path[index]);
+ if (!(result2 = object != null && hasFunc(object, key))) {
+ break;
+ }
+ object = object[key];
+ }
+ if (result2 || ++index != length) {
+ return result2;
+ }
+ length = object == null ? 0 : object.length;
+ return !!length && isLength(length) && isIndex(key, length) && (isArray2(object) || isArguments(object));
+ }
+ function initCloneArray(array2) {
+ var length = array2.length, result2 = new array2.constructor(length);
+ if (length && typeof array2[0] == "string" && hasOwnProperty2.call(array2, "index")) {
+ result2.index = array2.index;
+ result2.input = array2.input;
+ }
+ return result2;
+ }
+ function initCloneObject(object) {
+ return typeof object.constructor == "function" && !isPrototype(object) ? baseCreate(getPrototype(object)) : {};
+ }
+ function initCloneByTag(object, tag, isDeep) {
+ var Ctor = object.constructor;
+ switch (tag) {
+ case arrayBufferTag:
+ return cloneArrayBuffer(object);
+ case boolTag:
+ case dateTag:
+ return new Ctor(+object);
+ case dataViewTag:
+ return cloneDataView(object, isDeep);
+ case float32Tag:
+ case float64Tag:
+ case int8Tag:
+ case int16Tag:
+ case int32Tag:
+ case uint8Tag:
+ case uint8ClampedTag:
+ case uint16Tag:
+ case uint32Tag:
+ return cloneTypedArray(object, isDeep);
+ case mapTag:
+ return new Ctor();
+ case numberTag:
+ case stringTag:
+ return new Ctor(object);
+ case regexpTag:
+ return cloneRegExp(object);
+ case setTag:
+ return new Ctor();
+ case symbolTag2:
+ return cloneSymbol(object);
+ }
+ }
+ function insertWrapDetails(source, details) {
+ var length = details.length;
+ if (!length) {
+ return source;
+ }
+ var lastIndex = length - 1;
+ details[lastIndex] = (length > 1 ? "& " : "") + details[lastIndex];
+ details = details.join(length > 2 ? ", " : " ");
+ return source.replace(reWrapComment, "{\n/* [wrapped with " + details + "] */\n");
+ }
+ function isFlattenable(value) {
+ return isArray2(value) || isArguments(value) || !!(spreadableSymbol && value && value[spreadableSymbol]);
+ }
+ function isIndex(value, length) {
+ var type2 = typeof value;
+ length = length == null ? MAX_SAFE_INTEGER : length;
+ return !!length && (type2 == "number" || type2 != "symbol" && reIsUint.test(value)) && (value > -1 && value % 1 == 0 && value < length);
+ }
+ function isIterateeCall(value, index, object) {
+ if (!isObject3(object)) {
+ return false;
+ }
+ var type2 = typeof index;
+ if (type2 == "number" ? isArrayLike(object) && isIndex(index, object.length) : type2 == "string" && index in object) {
+ return eq(object[index], value);
+ }
+ return false;
+ }
+ function isKey(value, object) {
+ if (isArray2(value)) {
+ return false;
+ }
+ var type2 = typeof value;
+ if (type2 == "number" || type2 == "symbol" || type2 == "boolean" || value == null || isSymbol2(value)) {
+ return true;
+ }
+ return reIsPlainProp.test(value) || !reIsDeepProp.test(value) || object != null && value in Object2(object);
+ }
+ function isKeyable(value) {
+ var type2 = typeof value;
+ return type2 == "string" || type2 == "number" || type2 == "symbol" || type2 == "boolean" ? value !== "__proto__" : value === null;
+ }
+ function isLaziable(func) {
+ var funcName = getFuncName(func), other = lodash[funcName];
+ if (typeof other != "function" || !(funcName in LazyWrapper.prototype)) {
+ return false;
+ }
+ if (func === other) {
+ return true;
+ }
+ var data = getData(other);
+ return !!data && func === data[0];
+ }
+ function isMasked(func) {
+ return !!maskSrcKey && maskSrcKey in func;
+ }
+ var isMaskable = coreJsData ? isFunction : stubFalse;
+ function isPrototype(value) {
+ var Ctor = value && value.constructor, proto = typeof Ctor == "function" && Ctor.prototype || objectProto3;
+ return value === proto;
+ }
+ function isStrictComparable(value) {
+ return value === value && !isObject3(value);
+ }
+ function matchesStrictComparable(key, srcValue) {
+ return function(object) {
+ if (object == null) {
+ return false;
+ }
+ return object[key] === srcValue && (srcValue !== undefined2 || key in Object2(object));
+ };
+ }
+ function memoizeCapped(func) {
+ var result2 = memoize(func, function(key) {
+ if (cache.size === MAX_MEMOIZE_SIZE) {
+ cache.clear();
+ }
+ return key;
+ });
+ var cache = result2.cache;
+ return result2;
+ }
+ function mergeData(data, source) {
+ var bitmask = data[1], srcBitmask = source[1], newBitmask = bitmask | srcBitmask, isCommon = newBitmask < (WRAP_BIND_FLAG | WRAP_BIND_KEY_FLAG | WRAP_ARY_FLAG);
+ var isCombo = srcBitmask == WRAP_ARY_FLAG && bitmask == WRAP_CURRY_FLAG || srcBitmask == WRAP_ARY_FLAG && bitmask == WRAP_REARG_FLAG && data[7].length <= source[8] || srcBitmask == (WRAP_ARY_FLAG | WRAP_REARG_FLAG) && source[7].length <= source[8] && bitmask == WRAP_CURRY_FLAG;
+ if (!(isCommon || isCombo)) {
+ return data;
+ }
+ if (srcBitmask & WRAP_BIND_FLAG) {
+ data[2] = source[2];
+ newBitmask |= bitmask & WRAP_BIND_FLAG ? 0 : WRAP_CURRY_BOUND_FLAG;
+ }
+ var value = source[3];
+ if (value) {
+ var partials = data[3];
+ data[3] = partials ? composeArgs(partials, value, source[4]) : value;
+ data[4] = partials ? replaceHolders(data[3], PLACEHOLDER) : source[4];
+ }
+ value = source[5];
+ if (value) {
+ partials = data[5];
+ data[5] = partials ? composeArgsRight(partials, value, source[6]) : value;
+ data[6] = partials ? replaceHolders(data[5], PLACEHOLDER) : source[6];
+ }
+ value = source[7];
+ if (value) {
+ data[7] = value;
+ }
+ if (srcBitmask & WRAP_ARY_FLAG) {
+ data[8] = data[8] == null ? source[8] : nativeMin2(data[8], source[8]);
+ }
+ if (data[9] == null) {
+ data[9] = source[9];
+ }
+ data[0] = source[0];
+ data[1] = newBitmask;
+ return data;
+ }
+ function nativeKeysIn(object) {
+ var result2 = [];
+ if (object != null) {
+ for (var key in Object2(object)) {
+ result2.push(key);
+ }
+ }
+ return result2;
+ }
+ function objectToString2(value) {
+ return nativeObjectToString3.call(value);
+ }
+ function overRest(func, start2, transform3) {
+ start2 = nativeMax2(start2 === undefined2 ? func.length - 1 : start2, 0);
+ return function() {
+ var args = arguments, index = -1, length = nativeMax2(args.length - start2, 0), array2 = Array2(length);
+ while (++index < length) {
+ array2[index] = args[start2 + index];
+ }
+ index = -1;
+ var otherArgs = Array2(start2 + 1);
+ while (++index < start2) {
+ otherArgs[index] = args[index];
+ }
+ otherArgs[start2] = transform3(array2);
+ return apply(func, this, otherArgs);
+ };
+ }
+ function parent(object, path) {
+ return path.length < 2 ? object : baseGet(object, baseSlice(path, 0, -1));
+ }
+ function reorder(array2, indexes) {
+ var arrLength = array2.length, length = nativeMin2(indexes.length, arrLength), oldArray = copyArray(array2);
+ while (length--) {
+ var index = indexes[length];
+ array2[length] = isIndex(index, arrLength) ? oldArray[index] : undefined2;
+ }
+ return array2;
+ }
+ function safeGet(object, key) {
+ if (key === "constructor" && typeof object[key] === "function") {
+ return;
+ }
+ if (key == "__proto__") {
+ return;
+ }
+ return object[key];
+ }
+ var setData = shortOut(baseSetData);
+ var setTimeout2 = ctxSetTimeout || function(func, wait) {
+ return root3.setTimeout(func, wait);
+ };
+ var setToString = shortOut(baseSetToString);
+ function setWrapToString(wrapper, reference, bitmask) {
+ var source = reference + "";
+ return setToString(wrapper, insertWrapDetails(source, updateWrapDetails(getWrapDetails(source), bitmask)));
+ }
+ function shortOut(func) {
+ var count = 0, lastCalled = 0;
+ return function() {
+ var stamp = nativeNow(), remaining = HOT_SPAN - (stamp - lastCalled);
+ lastCalled = stamp;
+ if (remaining > 0) {
+ if (++count >= HOT_COUNT) {
+ return arguments[0];
+ }
+ } else {
+ count = 0;
+ }
+ return func.apply(undefined2, arguments);
+ };
+ }
+ function shuffleSelf(array2, size2) {
+ var index = -1, length = array2.length, lastIndex = length - 1;
+ size2 = size2 === undefined2 ? length : size2;
+ while (++index < size2) {
+ var rand = baseRandom(index, lastIndex), value = array2[rand];
+ array2[rand] = array2[index];
+ array2[index] = value;
+ }
+ array2.length = size2;
+ return array2;
+ }
+ var stringToPath = memoizeCapped(function(string) {
+ var result2 = [];
+ if (string.charCodeAt(0) === 46) {
+ result2.push("");
+ }
+ string.replace(rePropName, function(match, number3, quote2, subString) {
+ result2.push(quote2 ? subString.replace(reEscapeChar, "$1") : number3 || match);
+ });
+ return result2;
+ });
+ function toKey(value) {
+ if (typeof value == "string" || isSymbol2(value)) {
+ return value;
+ }
+ var result2 = value + "";
+ return result2 == "0" && 1 / value == -INFINITY2 ? "-0" : result2;
+ }
+ function toSource(func) {
+ if (func != null) {
+ try {
+ return funcToString.call(func);
+ } catch (e) {
+ }
+ try {
+ return func + "";
+ } catch (e) {
+ }
+ }
+ return "";
+ }
+ function updateWrapDetails(details, bitmask) {
+ arrayEach(wrapFlags, function(pair2) {
+ var value = "_." + pair2[0];
+ if (bitmask & pair2[1] && !arrayIncludes(details, value)) {
+ details.push(value);
+ }
+ });
+ return details.sort();
+ }
+ function wrapperClone(wrapper) {
+ if (wrapper instanceof LazyWrapper) {
+ return wrapper.clone();
+ }
+ var result2 = new LodashWrapper(wrapper.__wrapped__, wrapper.__chain__);
+ result2.__actions__ = copyArray(wrapper.__actions__);
+ result2.__index__ = wrapper.__index__;
+ result2.__values__ = wrapper.__values__;
+ return result2;
+ }
+ function chunk(array2, size2, guard) {
+ if (guard ? isIterateeCall(array2, size2, guard) : size2 === undefined2) {
+ size2 = 1;
+ } else {
+ size2 = nativeMax2(toInteger(size2), 0);
+ }
+ var length = array2 == null ? 0 : array2.length;
+ if (!length || size2 < 1) {
+ return [];
+ }
+ var index = 0, resIndex = 0, result2 = Array2(nativeCeil(length / size2));
+ while (index < length) {
+ result2[resIndex++] = baseSlice(array2, index, index += size2);
+ }
+ return result2;
+ }
+ function compact(array2) {
+ var index = -1, length = array2 == null ? 0 : array2.length, resIndex = 0, result2 = [];
+ while (++index < length) {
+ var value = array2[index];
+ if (value) {
+ result2[resIndex++] = value;
+ }
+ }
+ return result2;
+ }
+ function concat() {
+ var length = arguments.length;
+ if (!length) {
+ return [];
+ }
+ var args = Array2(length - 1), array2 = arguments[0], index = length;
+ while (index--) {
+ args[index - 1] = arguments[index];
+ }
+ return arrayPush(isArray2(array2) ? copyArray(array2) : [array2], baseFlatten(args, 1));
+ }
+ var difference = baseRest(function(array2, values2) {
+ return isArrayLikeObject(array2) ? baseDifference(array2, baseFlatten(values2, 1, isArrayLikeObject, true)) : [];
+ });
+ var differenceBy = baseRest(function(array2, values2) {
+ var iteratee2 = last(values2);
+ if (isArrayLikeObject(iteratee2)) {
+ iteratee2 = undefined2;
+ }
+ return isArrayLikeObject(array2) ? baseDifference(array2, baseFlatten(values2, 1, isArrayLikeObject, true), getIteratee(iteratee2, 2)) : [];
+ });
+ var differenceWith = baseRest(function(array2, values2) {
+ var comparator = last(values2);
+ if (isArrayLikeObject(comparator)) {
+ comparator = undefined2;
+ }
+ return isArrayLikeObject(array2) ? baseDifference(array2, baseFlatten(values2, 1, isArrayLikeObject, true), undefined2, comparator) : [];
+ });
+ function drop(array2, n2, guard) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return [];
+ }
+ n2 = guard || n2 === undefined2 ? 1 : toInteger(n2);
+ return baseSlice(array2, n2 < 0 ? 0 : n2, length);
+ }
+ function dropRight(array2, n2, guard) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return [];
+ }
+ n2 = guard || n2 === undefined2 ? 1 : toInteger(n2);
+ n2 = length - n2;
+ return baseSlice(array2, 0, n2 < 0 ? 0 : n2);
+ }
+ function dropRightWhile(array2, predicate) {
+ return array2 && array2.length ? baseWhile(array2, getIteratee(predicate, 3), true, true) : [];
+ }
+ function dropWhile(array2, predicate) {
+ return array2 && array2.length ? baseWhile(array2, getIteratee(predicate, 3), true) : [];
+ }
+ function fill(array2, value, start2, end) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return [];
+ }
+ if (start2 && typeof start2 != "number" && isIterateeCall(array2, value, start2)) {
+ start2 = 0;
+ end = length;
+ }
+ return baseFill(array2, value, start2, end);
+ }
+ function findIndex(array2, predicate, fromIndex) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return -1;
+ }
+ var index = fromIndex == null ? 0 : toInteger(fromIndex);
+ if (index < 0) {
+ index = nativeMax2(length + index, 0);
+ }
+ return baseFindIndex(array2, getIteratee(predicate, 3), index);
+ }
+ function findLastIndex(array2, predicate, fromIndex) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return -1;
+ }
+ var index = length - 1;
+ if (fromIndex !== undefined2) {
+ index = toInteger(fromIndex);
+ index = fromIndex < 0 ? nativeMax2(length + index, 0) : nativeMin2(index, length - 1);
+ }
+ return baseFindIndex(array2, getIteratee(predicate, 3), index, true);
+ }
+ function flatten2(array2) {
+ var length = array2 == null ? 0 : array2.length;
+ return length ? baseFlatten(array2, 1) : [];
+ }
+ function flattenDeep(array2) {
+ var length = array2 == null ? 0 : array2.length;
+ return length ? baseFlatten(array2, INFINITY2) : [];
+ }
+ function flattenDepth(array2, depth) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return [];
+ }
+ depth = depth === undefined2 ? 1 : toInteger(depth);
+ return baseFlatten(array2, depth);
+ }
+ function fromPairs(pairs) {
+ var index = -1, length = pairs == null ? 0 : pairs.length, result2 = {};
+ while (++index < length) {
+ var pair2 = pairs[index];
+ result2[pair2[0]] = pair2[1];
+ }
+ return result2;
+ }
+ function head(array2) {
+ return array2 && array2.length ? array2[0] : undefined2;
+ }
+ function indexOf(array2, value, fromIndex) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return -1;
+ }
+ var index = fromIndex == null ? 0 : toInteger(fromIndex);
+ if (index < 0) {
+ index = nativeMax2(length + index, 0);
+ }
+ return baseIndexOf(array2, value, index);
+ }
+ function initial(array2) {
+ var length = array2 == null ? 0 : array2.length;
+ return length ? baseSlice(array2, 0, -1) : [];
+ }
+ var intersection = baseRest(function(arrays) {
+ var mapped = arrayMap2(arrays, castArrayLikeObject);
+ return mapped.length && mapped[0] === arrays[0] ? baseIntersection(mapped) : [];
+ });
+ var intersectionBy = baseRest(function(arrays) {
+ var iteratee2 = last(arrays), mapped = arrayMap2(arrays, castArrayLikeObject);
+ if (iteratee2 === last(mapped)) {
+ iteratee2 = undefined2;
+ } else {
+ mapped.pop();
+ }
+ return mapped.length && mapped[0] === arrays[0] ? baseIntersection(mapped, getIteratee(iteratee2, 2)) : [];
+ });
+ var intersectionWith = baseRest(function(arrays) {
+ var comparator = last(arrays), mapped = arrayMap2(arrays, castArrayLikeObject);
+ comparator = typeof comparator == "function" ? comparator : undefined2;
+ if (comparator) {
+ mapped.pop();
+ }
+ return mapped.length && mapped[0] === arrays[0] ? baseIntersection(mapped, undefined2, comparator) : [];
+ });
+ function join(array2, separator) {
+ return array2 == null ? "" : nativeJoin.call(array2, separator);
+ }
+ function last(array2) {
+ var length = array2 == null ? 0 : array2.length;
+ return length ? array2[length - 1] : undefined2;
+ }
+ function lastIndexOf(array2, value, fromIndex) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return -1;
+ }
+ var index = length;
+ if (fromIndex !== undefined2) {
+ index = toInteger(fromIndex);
+ index = index < 0 ? nativeMax2(length + index, 0) : nativeMin2(index, length - 1);
+ }
+ return value === value ? strictLastIndexOf(array2, value, index) : baseFindIndex(array2, baseIsNaN, index, true);
+ }
+ function nth(array2, n2) {
+ return array2 && array2.length ? baseNth(array2, toInteger(n2)) : undefined2;
+ }
+ var pull = baseRest(pullAll);
+ function pullAll(array2, values2) {
+ return array2 && array2.length && values2 && values2.length ? basePullAll(array2, values2) : array2;
+ }
+ function pullAllBy(array2, values2, iteratee2) {
+ return array2 && array2.length && values2 && values2.length ? basePullAll(array2, values2, getIteratee(iteratee2, 2)) : array2;
+ }
+ function pullAllWith(array2, values2, comparator) {
+ return array2 && array2.length && values2 && values2.length ? basePullAll(array2, values2, undefined2, comparator) : array2;
+ }
+ var pullAt = flatRest(function(array2, indexes) {
+ var length = array2 == null ? 0 : array2.length, result2 = baseAt(array2, indexes);
+ basePullAt(array2, arrayMap2(indexes, function(index) {
+ return isIndex(index, length) ? +index : index;
+ }).sort(compareAscending));
+ return result2;
+ });
+ function remove2(array2, predicate) {
+ var result2 = [];
+ if (!(array2 && array2.length)) {
+ return result2;
+ }
+ var index = -1, indexes = [], length = array2.length;
+ predicate = getIteratee(predicate, 3);
+ while (++index < length) {
+ var value = array2[index];
+ if (predicate(value, index, array2)) {
+ result2.push(value);
+ indexes.push(index);
+ }
+ }
+ basePullAt(array2, indexes);
+ return result2;
+ }
+ function reverse(array2) {
+ return array2 == null ? array2 : nativeReverse.call(array2);
+ }
+ function slice(array2, start2, end) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return [];
+ }
+ if (end && typeof end != "number" && isIterateeCall(array2, start2, end)) {
+ start2 = 0;
+ end = length;
+ } else {
+ start2 = start2 == null ? 0 : toInteger(start2);
+ end = end === undefined2 ? length : toInteger(end);
+ }
+ return baseSlice(array2, start2, end);
+ }
+ function sortedIndex(array2, value) {
+ return baseSortedIndex(array2, value);
+ }
+ function sortedIndexBy(array2, value, iteratee2) {
+ return baseSortedIndexBy(array2, value, getIteratee(iteratee2, 2));
+ }
+ function sortedIndexOf(array2, value) {
+ var length = array2 == null ? 0 : array2.length;
+ if (length) {
+ var index = baseSortedIndex(array2, value);
+ if (index < length && eq(array2[index], value)) {
+ return index;
+ }
+ }
+ return -1;
+ }
+ function sortedLastIndex(array2, value) {
+ return baseSortedIndex(array2, value, true);
+ }
+ function sortedLastIndexBy(array2, value, iteratee2) {
+ return baseSortedIndexBy(array2, value, getIteratee(iteratee2, 2), true);
+ }
+ function sortedLastIndexOf(array2, value) {
+ var length = array2 == null ? 0 : array2.length;
+ if (length) {
+ var index = baseSortedIndex(array2, value, true) - 1;
+ if (eq(array2[index], value)) {
+ return index;
+ }
+ }
+ return -1;
+ }
+ function sortedUniq(array2) {
+ return array2 && array2.length ? baseSortedUniq(array2) : [];
+ }
+ function sortedUniqBy(array2, iteratee2) {
+ return array2 && array2.length ? baseSortedUniq(array2, getIteratee(iteratee2, 2)) : [];
+ }
+ function tail(array2) {
+ var length = array2 == null ? 0 : array2.length;
+ return length ? baseSlice(array2, 1, length) : [];
+ }
+ function take(array2, n2, guard) {
+ if (!(array2 && array2.length)) {
+ return [];
+ }
+ n2 = guard || n2 === undefined2 ? 1 : toInteger(n2);
+ return baseSlice(array2, 0, n2 < 0 ? 0 : n2);
+ }
+ function takeRight(array2, n2, guard) {
+ var length = array2 == null ? 0 : array2.length;
+ if (!length) {
+ return [];
+ }
+ n2 = guard || n2 === undefined2 ? 1 : toInteger(n2);
+ n2 = length - n2;
+ return baseSlice(array2, n2 < 0 ? 0 : n2, length);
+ }
+ function takeRightWhile(array2, predicate) {
+ return array2 && array2.length ? baseWhile(array2, getIteratee(predicate, 3), false, true) : [];
+ }
+ function takeWhile(array2, predicate) {
+ return array2 && array2.length ? baseWhile(array2, getIteratee(predicate, 3)) : [];
+ }
+ var union = baseRest(function(arrays) {
+ return baseUniq(baseFlatten(arrays, 1, isArrayLikeObject, true));
+ });
+ var unionBy = baseRest(function(arrays) {
+ var iteratee2 = last(arrays);
+ if (isArrayLikeObject(iteratee2)) {
+ iteratee2 = undefined2;
+ }
+ return baseUniq(baseFlatten(arrays, 1, isArrayLikeObject, true), getIteratee(iteratee2, 2));
+ });
+ var unionWith = baseRest(function(arrays) {
+ var comparator = last(arrays);
+ comparator = typeof comparator == "function" ? comparator : undefined2;
+ return baseUniq(baseFlatten(arrays, 1, isArrayLikeObject, true), undefined2, comparator);
+ });
+ function uniq(array2) {
+ return array2 && array2.length ? baseUniq(array2) : [];
+ }
+ function uniqBy(array2, iteratee2) {
+ return array2 && array2.length ? baseUniq(array2, getIteratee(iteratee2, 2)) : [];
+ }
+ function uniqWith(array2, comparator) {
+ comparator = typeof comparator == "function" ? comparator : undefined2;
+ return array2 && array2.length ? baseUniq(array2, undefined2, comparator) : [];
+ }
+ function unzip(array2) {
+ if (!(array2 && array2.length)) {
+ return [];
+ }
+ var length = 0;
+ array2 = arrayFilter(array2, function(group) {
+ if (isArrayLikeObject(group)) {
+ length = nativeMax2(group.length, length);
+ return true;
+ }
+ });
+ return baseTimes(length, function(index) {
+ return arrayMap2(array2, baseProperty(index));
+ });
+ }
+ function unzipWith(array2, iteratee2) {
+ if (!(array2 && array2.length)) {
+ return [];
+ }
+ var result2 = unzip(array2);
+ if (iteratee2 == null) {
+ return result2;
+ }
+ return arrayMap2(result2, function(group) {
+ return apply(iteratee2, undefined2, group);
+ });
+ }
+ var without = baseRest(function(array2, values2) {
+ return isArrayLikeObject(array2) ? baseDifference(array2, values2) : [];
+ });
+ var xor = baseRest(function(arrays) {
+ return baseXor(arrayFilter(arrays, isArrayLikeObject));
+ });
+ var xorBy = baseRest(function(arrays) {
+ var iteratee2 = last(arrays);
+ if (isArrayLikeObject(iteratee2)) {
+ iteratee2 = undefined2;
+ }
+ return baseXor(arrayFilter(arrays, isArrayLikeObject), getIteratee(iteratee2, 2));
+ });
+ var xorWith = baseRest(function(arrays) {
+ var comparator = last(arrays);
+ comparator = typeof comparator == "function" ? comparator : undefined2;
+ return baseXor(arrayFilter(arrays, isArrayLikeObject), undefined2, comparator);
+ });
+ var zip = baseRest(unzip);
+ function zipObject(props, values2) {
+ return baseZipObject(props || [], values2 || [], assignValue);
+ }
+ function zipObjectDeep(props, values2) {
+ return baseZipObject(props || [], values2 || [], baseSet);
+ }
+ var zipWith = baseRest(function(arrays) {
+ var length = arrays.length, iteratee2 = length > 1 ? arrays[length - 1] : undefined2;
+ iteratee2 = typeof iteratee2 == "function" ? (arrays.pop(), iteratee2) : undefined2;
+ return unzipWith(arrays, iteratee2);
+ });
+ function chain(value) {
+ var result2 = lodash(value);
+ result2.__chain__ = true;
+ return result2;
+ }
+ function tap(value, interceptor) {
+ interceptor(value);
+ return value;
+ }
+ function thru(value, interceptor) {
+ return interceptor(value);
+ }
+ var wrapperAt = flatRest(function(paths) {
+ var length = paths.length, start2 = length ? paths[0] : 0, value = this.__wrapped__, interceptor = function(object) {
+ return baseAt(object, paths);
+ };
+ if (length > 1 || this.__actions__.length || !(value instanceof LazyWrapper) || !isIndex(start2)) {
+ return this.thru(interceptor);
+ }
+ value = value.slice(start2, +start2 + (length ? 1 : 0));
+ value.__actions__.push({
+ "func": thru,
+ "args": [interceptor],
+ "thisArg": undefined2
+ });
+ return new LodashWrapper(value, this.__chain__).thru(function(array2) {
+ if (length && !array2.length) {
+ array2.push(undefined2);
+ }
+ return array2;
+ });
+ });
+ function wrapperChain() {
+ return chain(this);
+ }
+ function wrapperCommit() {
+ return new LodashWrapper(this.value(), this.__chain__);
+ }
+ function wrapperNext() {
+ if (this.__values__ === undefined2) {
+ this.__values__ = toArray(this.value());
+ }
+ var done = this.__index__ >= this.__values__.length, value = done ? undefined2 : this.__values__[this.__index__++];
+ return { "done": done, "value": value };
+ }
+ function wrapperToIterator() {
+ return this;
+ }
+ function wrapperPlant(value) {
+ var result2, parent2 = this;
+ while (parent2 instanceof baseLodash) {
+ var clone2 = wrapperClone(parent2);
+ clone2.__index__ = 0;
+ clone2.__values__ = undefined2;
+ if (result2) {
+ previous.__wrapped__ = clone2;
+ } else {
+ result2 = clone2;
+ }
+ var previous = clone2;
+ parent2 = parent2.__wrapped__;
+ }
+ previous.__wrapped__ = value;
+ return result2;
+ }
+ function wrapperReverse() {
+ var value = this.__wrapped__;
+ if (value instanceof LazyWrapper) {
+ var wrapped = value;
+ if (this.__actions__.length) {
+ wrapped = new LazyWrapper(this);
+ }
+ wrapped = wrapped.reverse();
+ wrapped.__actions__.push({
+ "func": thru,
+ "args": [reverse],
+ "thisArg": undefined2
+ });
+ return new LodashWrapper(wrapped, this.__chain__);
+ }
+ return this.thru(reverse);
+ }
+ function wrapperValue() {
+ return baseWrapperValue(this.__wrapped__, this.__actions__);
+ }
+ var countBy = createAggregator(function(result2, value, key) {
+ if (hasOwnProperty2.call(result2, key)) {
+ ++result2[key];
+ } else {
+ baseAssignValue(result2, key, 1);
+ }
+ });
+ function every(collection, predicate, guard) {
+ var func = isArray2(collection) ? arrayEvery : baseEvery;
+ if (guard && isIterateeCall(collection, predicate, guard)) {
+ predicate = undefined2;
+ }
+ return func(collection, getIteratee(predicate, 3));
+ }
+ function filter2(collection, predicate) {
+ var func = isArray2(collection) ? arrayFilter : baseFilter;
+ return func(collection, getIteratee(predicate, 3));
+ }
+ var find2 = createFind(findIndex);
+ var findLast = createFind(findLastIndex);
+ function flatMap(collection, iteratee2) {
+ return baseFlatten(map2(collection, iteratee2), 1);
+ }
+ function flatMapDeep(collection, iteratee2) {
+ return baseFlatten(map2(collection, iteratee2), INFINITY2);
+ }
+ function flatMapDepth(collection, iteratee2, depth) {
+ depth = depth === undefined2 ? 1 : toInteger(depth);
+ return baseFlatten(map2(collection, iteratee2), depth);
+ }
+ function forEach(collection, iteratee2) {
+ var func = isArray2(collection) ? arrayEach : baseEach;
+ return func(collection, getIteratee(iteratee2, 3));
+ }
+ function forEachRight(collection, iteratee2) {
+ var func = isArray2(collection) ? arrayEachRight : baseEachRight;
+ return func(collection, getIteratee(iteratee2, 3));
+ }
+ var groupBy = createAggregator(function(result2, value, key) {
+ if (hasOwnProperty2.call(result2, key)) {
+ result2[key].push(value);
+ } else {
+ baseAssignValue(result2, key, [value]);
+ }
+ });
+ function includes(collection, value, fromIndex, guard) {
+ collection = isArrayLike(collection) ? collection : values(collection);
+ fromIndex = fromIndex && !guard ? toInteger(fromIndex) : 0;
+ var length = collection.length;
+ if (fromIndex < 0) {
+ fromIndex = nativeMax2(length + fromIndex, 0);
+ }
+ return isString(collection) ? fromIndex <= length && collection.indexOf(value, fromIndex) > -1 : !!length && baseIndexOf(collection, value, fromIndex) > -1;
+ }
+ var invokeMap = baseRest(function(collection, path, args) {
+ var index = -1, isFunc = typeof path == "function", result2 = isArrayLike(collection) ? Array2(collection.length) : [];
+ baseEach(collection, function(value) {
+ result2[++index] = isFunc ? apply(path, value, args) : baseInvoke(value, path, args);
+ });
+ return result2;
+ });
+ var keyBy = createAggregator(function(result2, value, key) {
+ baseAssignValue(result2, key, value);
+ });
+ function map2(collection, iteratee2) {
+ var func = isArray2(collection) ? arrayMap2 : baseMap;
+ return func(collection, getIteratee(iteratee2, 3));
+ }
+ function orderBy(collection, iteratees, orders, guard) {
+ if (collection == null) {
+ return [];
+ }
+ if (!isArray2(iteratees)) {
+ iteratees = iteratees == null ? [] : [iteratees];
+ }
+ orders = guard ? undefined2 : orders;
+ if (!isArray2(orders)) {
+ orders = orders == null ? [] : [orders];
+ }
+ return baseOrderBy(collection, iteratees, orders);
+ }
+ var partition = createAggregator(function(result2, value, key) {
+ result2[key ? 0 : 1].push(value);
+ }, function() {
+ return [[], []];
+ });
+ function reduce(collection, iteratee2, accumulator) {
+ var func = isArray2(collection) ? arrayReduce : baseReduce, initAccum = arguments.length < 3;
+ return func(collection, getIteratee(iteratee2, 4), accumulator, initAccum, baseEach);
+ }
+ function reduceRight(collection, iteratee2, accumulator) {
+ var func = isArray2(collection) ? arrayReduceRight : baseReduce, initAccum = arguments.length < 3;
+ return func(collection, getIteratee(iteratee2, 4), accumulator, initAccum, baseEachRight);
+ }
+ function reject(collection, predicate) {
+ var func = isArray2(collection) ? arrayFilter : baseFilter;
+ return func(collection, negate(getIteratee(predicate, 3)));
+ }
+ function sample(collection) {
+ var func = isArray2(collection) ? arraySample : baseSample;
+ return func(collection);
+ }
+ function sampleSize(collection, n2, guard) {
+ if (guard ? isIterateeCall(collection, n2, guard) : n2 === undefined2) {
+ n2 = 1;
+ } else {
+ n2 = toInteger(n2);
+ }
+ var func = isArray2(collection) ? arraySampleSize : baseSampleSize;
+ return func(collection, n2);
+ }
+ function shuffle(collection) {
+ var func = isArray2(collection) ? arrayShuffle : baseShuffle;
+ return func(collection);
+ }
+ function size(collection) {
+ if (collection == null) {
+ return 0;
+ }
+ if (isArrayLike(collection)) {
+ return isString(collection) ? stringSize(collection) : collection.length;
+ }
+ var tag = getTag(collection);
+ if (tag == mapTag || tag == setTag) {
+ return collection.size;
+ }
+ return baseKeys(collection).length;
+ }
+ function some(collection, predicate, guard) {
+ var func = isArray2(collection) ? arraySome : baseSome;
+ if (guard && isIterateeCall(collection, predicate, guard)) {
+ predicate = undefined2;
+ }
+ return func(collection, getIteratee(predicate, 3));
+ }
+ var sortBy = baseRest(function(collection, iteratees) {
+ if (collection == null) {
+ return [];
+ }
+ var length = iteratees.length;
+ if (length > 1 && isIterateeCall(collection, iteratees[0], iteratees[1])) {
+ iteratees = [];
+ } else if (length > 2 && isIterateeCall(iteratees[0], iteratees[1], iteratees[2])) {
+ iteratees = [iteratees[0]];
+ }
+ return baseOrderBy(collection, baseFlatten(iteratees, 1), []);
+ });
+ var now3 = ctxNow || function() {
+ return root3.Date.now();
+ };
+ function after(n2, func) {
+ if (typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ n2 = toInteger(n2);
+ return function() {
+ if (--n2 < 1) {
+ return func.apply(this, arguments);
+ }
+ };
+ }
+ function ary(func, n2, guard) {
+ n2 = guard ? undefined2 : n2;
+ n2 = func && n2 == null ? func.length : n2;
+ return createWrap(func, WRAP_ARY_FLAG, undefined2, undefined2, undefined2, undefined2, n2);
+ }
+ function before(n2, func) {
+ var result2;
+ if (typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ n2 = toInteger(n2);
+ return function() {
+ if (--n2 > 0) {
+ result2 = func.apply(this, arguments);
+ }
+ if (n2 <= 1) {
+ func = undefined2;
+ }
+ return result2;
+ };
+ }
+ var bind = baseRest(function(func, thisArg, partials) {
+ var bitmask = WRAP_BIND_FLAG;
+ if (partials.length) {
+ var holders = replaceHolders(partials, getHolder(bind));
+ bitmask |= WRAP_PARTIAL_FLAG;
+ }
+ return createWrap(func, bitmask, thisArg, partials, holders);
+ });
+ var bindKey2 = baseRest(function(object, key, partials) {
+ var bitmask = WRAP_BIND_FLAG | WRAP_BIND_KEY_FLAG;
+ if (partials.length) {
+ var holders = replaceHolders(partials, getHolder(bindKey2));
+ bitmask |= WRAP_PARTIAL_FLAG;
+ }
+ return createWrap(key, bitmask, object, partials, holders);
+ });
+ function curry(func, arity, guard) {
+ arity = guard ? undefined2 : arity;
+ var result2 = createWrap(func, WRAP_CURRY_FLAG, undefined2, undefined2, undefined2, undefined2, undefined2, arity);
+ result2.placeholder = curry.placeholder;
+ return result2;
+ }
+ function curryRight(func, arity, guard) {
+ arity = guard ? undefined2 : arity;
+ var result2 = createWrap(func, WRAP_CURRY_RIGHT_FLAG, undefined2, undefined2, undefined2, undefined2, undefined2, arity);
+ result2.placeholder = curryRight.placeholder;
+ return result2;
+ }
+ function debounce2(func, wait, options2) {
+ var lastArgs, lastThis, maxWait, result2, timerId, lastCallTime, lastInvokeTime = 0, leading = false, maxing = false, trailing = true;
+ if (typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ wait = toNumber2(wait) || 0;
+ if (isObject3(options2)) {
+ leading = !!options2.leading;
+ maxing = "maxWait" in options2;
+ maxWait = maxing ? nativeMax2(toNumber2(options2.maxWait) || 0, wait) : maxWait;
+ trailing = "trailing" in options2 ? !!options2.trailing : trailing;
+ }
+ function invokeFunc(time) {
+ var args = lastArgs, thisArg = lastThis;
+ lastArgs = lastThis = undefined2;
+ lastInvokeTime = time;
+ result2 = func.apply(thisArg, args);
+ return result2;
+ }
+ function leadingEdge(time) {
+ lastInvokeTime = time;
+ timerId = setTimeout2(timerExpired, wait);
+ return leading ? invokeFunc(time) : result2;
+ }
+ function remainingWait(time) {
+ var timeSinceLastCall = time - lastCallTime, timeSinceLastInvoke = time - lastInvokeTime, timeWaiting = wait - timeSinceLastCall;
+ return maxing ? nativeMin2(timeWaiting, maxWait - timeSinceLastInvoke) : timeWaiting;
+ }
+ function shouldInvoke(time) {
+ var timeSinceLastCall = time - lastCallTime, timeSinceLastInvoke = time - lastInvokeTime;
+ return lastCallTime === undefined2 || timeSinceLastCall >= wait || timeSinceLastCall < 0 || maxing && timeSinceLastInvoke >= maxWait;
+ }
+ function timerExpired() {
+ var time = now3();
+ if (shouldInvoke(time)) {
+ return trailingEdge(time);
+ }
+ timerId = setTimeout2(timerExpired, remainingWait(time));
+ }
+ function trailingEdge(time) {
+ timerId = undefined2;
+ if (trailing && lastArgs) {
+ return invokeFunc(time);
+ }
+ lastArgs = lastThis = undefined2;
+ return result2;
+ }
+ function cancel() {
+ if (timerId !== undefined2) {
+ clearTimeout2(timerId);
+ }
+ lastInvokeTime = 0;
+ lastArgs = lastCallTime = lastThis = timerId = undefined2;
+ }
+ function flush() {
+ return timerId === undefined2 ? result2 : trailingEdge(now3());
+ }
+ function debounced() {
+ var time = now3(), isInvoking = shouldInvoke(time);
+ lastArgs = arguments;
+ lastThis = this;
+ lastCallTime = time;
+ if (isInvoking) {
+ if (timerId === undefined2) {
+ return leadingEdge(lastCallTime);
+ }
+ if (maxing) {
+ clearTimeout2(timerId);
+ timerId = setTimeout2(timerExpired, wait);
+ return invokeFunc(lastCallTime);
+ }
+ }
+ if (timerId === undefined2) {
+ timerId = setTimeout2(timerExpired, wait);
+ }
+ return result2;
+ }
+ debounced.cancel = cancel;
+ debounced.flush = flush;
+ return debounced;
+ }
+ var defer = baseRest(function(func, args) {
+ return baseDelay(func, 1, args);
+ });
+ var delay = baseRest(function(func, wait, args) {
+ return baseDelay(func, toNumber2(wait) || 0, args);
+ });
+ function flip(func) {
+ return createWrap(func, WRAP_FLIP_FLAG);
+ }
+ function memoize(func, resolver) {
+ if (typeof func != "function" || resolver != null && typeof resolver != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ var memoized = function() {
+ var args = arguments, key = resolver ? resolver.apply(this, args) : args[0], cache = memoized.cache;
+ if (cache.has(key)) {
+ return cache.get(key);
+ }
+ var result2 = func.apply(this, args);
+ memoized.cache = cache.set(key, result2) || cache;
+ return result2;
+ };
+ memoized.cache = new (memoize.Cache || MapCache)();
+ return memoized;
+ }
+ memoize.Cache = MapCache;
+ function negate(predicate) {
+ if (typeof predicate != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ return function() {
+ var args = arguments;
+ switch (args.length) {
+ case 0:
+ return !predicate.call(this);
+ case 1:
+ return !predicate.call(this, args[0]);
+ case 2:
+ return !predicate.call(this, args[0], args[1]);
+ case 3:
+ return !predicate.call(this, args[0], args[1], args[2]);
+ }
+ return !predicate.apply(this, args);
+ };
+ }
+ function once(func) {
+ return before(2, func);
+ }
+ var overArgs = castRest(function(func, transforms) {
+ transforms = transforms.length == 1 && isArray2(transforms[0]) ? arrayMap2(transforms[0], baseUnary(getIteratee())) : arrayMap2(baseFlatten(transforms, 1), baseUnary(getIteratee()));
+ var funcsLength = transforms.length;
+ return baseRest(function(args) {
+ var index = -1, length = nativeMin2(args.length, funcsLength);
+ while (++index < length) {
+ args[index] = transforms[index].call(this, args[index]);
+ }
+ return apply(func, this, args);
+ });
+ });
+ var partial = baseRest(function(func, partials) {
+ var holders = replaceHolders(partials, getHolder(partial));
+ return createWrap(func, WRAP_PARTIAL_FLAG, undefined2, partials, holders);
+ });
+ var partialRight = baseRest(function(func, partials) {
+ var holders = replaceHolders(partials, getHolder(partialRight));
+ return createWrap(func, WRAP_PARTIAL_RIGHT_FLAG, undefined2, partials, holders);
+ });
+ var rearg = flatRest(function(func, indexes) {
+ return createWrap(func, WRAP_REARG_FLAG, undefined2, undefined2, undefined2, indexes);
+ });
+ function rest(func, start2) {
+ if (typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ start2 = start2 === undefined2 ? start2 : toInteger(start2);
+ return baseRest(func, start2);
+ }
+ function spread(func, start2) {
+ if (typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ start2 = start2 == null ? 0 : nativeMax2(toInteger(start2), 0);
+ return baseRest(function(args) {
+ var array2 = args[start2], otherArgs = castSlice(args, 0, start2);
+ if (array2) {
+ arrayPush(otherArgs, array2);
+ }
+ return apply(func, this, otherArgs);
+ });
+ }
+ function throttle2(func, wait, options2) {
+ var leading = true, trailing = true;
+ if (typeof func != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ if (isObject3(options2)) {
+ leading = "leading" in options2 ? !!options2.leading : leading;
+ trailing = "trailing" in options2 ? !!options2.trailing : trailing;
+ }
+ return debounce2(func, wait, {
+ "leading": leading,
+ "maxWait": wait,
+ "trailing": trailing
+ });
+ }
+ function unary(func) {
+ return ary(func, 1);
+ }
+ function wrap2(value, wrapper) {
+ return partial(castFunction(wrapper), value);
+ }
+ function castArray() {
+ if (!arguments.length) {
+ return [];
+ }
+ var value = arguments[0];
+ return isArray2(value) ? value : [value];
+ }
+ function clone(value) {
+ return baseClone(value, CLONE_SYMBOLS_FLAG);
+ }
+ function cloneWith(value, customizer) {
+ customizer = typeof customizer == "function" ? customizer : undefined2;
+ return baseClone(value, CLONE_SYMBOLS_FLAG, customizer);
+ }
+ function cloneDeep(value) {
+ return baseClone(value, CLONE_DEEP_FLAG | CLONE_SYMBOLS_FLAG);
+ }
+ function cloneDeepWith(value, customizer) {
+ customizer = typeof customizer == "function" ? customizer : undefined2;
+ return baseClone(value, CLONE_DEEP_FLAG | CLONE_SYMBOLS_FLAG, customizer);
+ }
+ function conformsTo(object, source) {
+ return source == null || baseConformsTo(object, source, keys(source));
+ }
+ function eq(value, other) {
+ return value === other || value !== value && other !== other;
+ }
+ var gt = createRelationalOperation(baseGt);
+ var gte = createRelationalOperation(function(value, other) {
+ return value >= other;
+ });
+ var isArguments = baseIsArguments(function() {
+ return arguments;
+ }()) ? baseIsArguments : function(value) {
+ return isObjectLike2(value) && hasOwnProperty2.call(value, "callee") && !propertyIsEnumerable.call(value, "callee");
+ };
+ var isArray2 = Array2.isArray;
+ var isArrayBuffer = nodeIsArrayBuffer ? baseUnary(nodeIsArrayBuffer) : baseIsArrayBuffer;
+ function isArrayLike(value) {
+ return value != null && isLength(value.length) && !isFunction(value);
+ }
+ function isArrayLikeObject(value) {
+ return isObjectLike2(value) && isArrayLike(value);
+ }
+ function isBoolean(value) {
+ return value === true || value === false || isObjectLike2(value) && baseGetTag2(value) == boolTag;
+ }
+ var isBuffer = nativeIsBuffer || stubFalse;
+ var isDate = nodeIsDate ? baseUnary(nodeIsDate) : baseIsDate;
+ function isElement2(value) {
+ return isObjectLike2(value) && value.nodeType === 1 && !isPlainObject(value);
+ }
+ function isEmpty(value) {
+ if (value == null) {
+ return true;
+ }
+ if (isArrayLike(value) && (isArray2(value) || typeof value == "string" || typeof value.splice == "function" || isBuffer(value) || isTypedArray(value) || isArguments(value))) {
+ return !value.length;
+ }
+ var tag = getTag(value);
+ if (tag == mapTag || tag == setTag) {
+ return !value.size;
+ }
+ if (isPrototype(value)) {
+ return !baseKeys(value).length;
+ }
+ for (var key in value) {
+ if (hasOwnProperty2.call(value, key)) {
+ return false;
+ }
+ }
+ return true;
+ }
+ function isEqual(value, other) {
+ return baseIsEqual(value, other);
+ }
+ function isEqualWith(value, other, customizer) {
+ customizer = typeof customizer == "function" ? customizer : undefined2;
+ var result2 = customizer ? customizer(value, other) : undefined2;
+ return result2 === undefined2 ? baseIsEqual(value, other, undefined2, customizer) : !!result2;
+ }
+ function isError(value) {
+ if (!isObjectLike2(value)) {
+ return false;
+ }
+ var tag = baseGetTag2(value);
+ return tag == errorTag || tag == domExcTag || typeof value.message == "string" && typeof value.name == "string" && !isPlainObject(value);
+ }
+ function isFinite2(value) {
+ return typeof value == "number" && nativeIsFinite(value);
+ }
+ function isFunction(value) {
+ if (!isObject3(value)) {
+ return false;
+ }
+ var tag = baseGetTag2(value);
+ return tag == funcTag || tag == genTag || tag == asyncTag || tag == proxyTag;
+ }
+ function isInteger(value) {
+ return typeof value == "number" && value == toInteger(value);
+ }
+ function isLength(value) {
+ return typeof value == "number" && value > -1 && value % 1 == 0 && value <= MAX_SAFE_INTEGER;
+ }
+ function isObject3(value) {
+ var type2 = typeof value;
+ return value != null && (type2 == "object" || type2 == "function");
+ }
+ function isObjectLike2(value) {
+ return value != null && typeof value == "object";
+ }
+ var isMap = nodeIsMap ? baseUnary(nodeIsMap) : baseIsMap;
+ function isMatch(object, source) {
+ return object === source || baseIsMatch(object, source, getMatchData(source));
+ }
+ function isMatchWith(object, source, customizer) {
+ customizer = typeof customizer == "function" ? customizer : undefined2;
+ return baseIsMatch(object, source, getMatchData(source), customizer);
+ }
+ function isNaN2(value) {
+ return isNumber2(value) && value != +value;
+ }
+ function isNative(value) {
+ if (isMaskable(value)) {
+ throw new Error2(CORE_ERROR_TEXT);
+ }
+ return baseIsNative(value);
+ }
+ function isNull(value) {
+ return value === null;
+ }
+ function isNil(value) {
+ return value == null;
+ }
+ function isNumber2(value) {
+ return typeof value == "number" || isObjectLike2(value) && baseGetTag2(value) == numberTag;
+ }
+ function isPlainObject(value) {
+ if (!isObjectLike2(value) || baseGetTag2(value) != objectTag) {
+ return false;
+ }
+ var proto = getPrototype(value);
+ if (proto === null) {
+ return true;
+ }
+ var Ctor = hasOwnProperty2.call(proto, "constructor") && proto.constructor;
+ return typeof Ctor == "function" && Ctor instanceof Ctor && funcToString.call(Ctor) == objectCtorString;
+ }
+ var isRegExp = nodeIsRegExp ? baseUnary(nodeIsRegExp) : baseIsRegExp;
+ function isSafeInteger(value) {
+ return isInteger(value) && value >= -MAX_SAFE_INTEGER && value <= MAX_SAFE_INTEGER;
+ }
+ var isSet = nodeIsSet ? baseUnary(nodeIsSet) : baseIsSet;
+ function isString(value) {
+ return typeof value == "string" || !isArray2(value) && isObjectLike2(value) && baseGetTag2(value) == stringTag;
+ }
+ function isSymbol2(value) {
+ return typeof value == "symbol" || isObjectLike2(value) && baseGetTag2(value) == symbolTag2;
+ }
+ var isTypedArray = nodeIsTypedArray ? baseUnary(nodeIsTypedArray) : baseIsTypedArray;
+ function isUndefined(value) {
+ return value === undefined2;
+ }
+ function isWeakMap(value) {
+ return isObjectLike2(value) && getTag(value) == weakMapTag;
+ }
+ function isWeakSet(value) {
+ return isObjectLike2(value) && baseGetTag2(value) == weakSetTag;
+ }
+ var lt = createRelationalOperation(baseLt);
+ var lte = createRelationalOperation(function(value, other) {
+ return value <= other;
+ });
+ function toArray(value) {
+ if (!value) {
+ return [];
+ }
+ if (isArrayLike(value)) {
+ return isString(value) ? stringToArray(value) : copyArray(value);
+ }
+ if (symIterator && value[symIterator]) {
+ return iteratorToArray(value[symIterator]());
+ }
+ var tag = getTag(value), func = tag == mapTag ? mapToArray : tag == setTag ? setToArray : values;
+ return func(value);
+ }
+ function toFinite(value) {
+ if (!value) {
+ return value === 0 ? value : 0;
+ }
+ value = toNumber2(value);
+ if (value === INFINITY2 || value === -INFINITY2) {
+ var sign2 = value < 0 ? -1 : 1;
+ return sign2 * MAX_INTEGER;
+ }
+ return value === value ? value : 0;
+ }
+ function toInteger(value) {
+ var result2 = toFinite(value), remainder = result2 % 1;
+ return result2 === result2 ? remainder ? result2 - remainder : result2 : 0;
+ }
+ function toLength(value) {
+ return value ? baseClamp(toInteger(value), 0, MAX_ARRAY_LENGTH) : 0;
+ }
+ function toNumber2(value) {
+ if (typeof value == "number") {
+ return value;
+ }
+ if (isSymbol2(value)) {
+ return NAN2;
+ }
+ if (isObject3(value)) {
+ var other = typeof value.valueOf == "function" ? value.valueOf() : value;
+ value = isObject3(other) ? other + "" : other;
+ }
+ if (typeof value != "string") {
+ return value === 0 ? value : +value;
+ }
+ value = baseTrim2(value);
+ var isBinary = reIsBinary2.test(value);
+ return isBinary || reIsOctal2.test(value) ? freeParseInt2(value.slice(2), isBinary ? 2 : 8) : reIsBadHex2.test(value) ? NAN2 : +value;
+ }
+ function toPlainObject(value) {
+ return copyObject(value, keysIn(value));
+ }
+ function toSafeInteger(value) {
+ return value ? baseClamp(toInteger(value), -MAX_SAFE_INTEGER, MAX_SAFE_INTEGER) : value === 0 ? value : 0;
+ }
+ function toString2(value) {
+ return value == null ? "" : baseToString2(value);
+ }
+ var assign = createAssigner(function(object, source) {
+ if (isPrototype(source) || isArrayLike(source)) {
+ copyObject(source, keys(source), object);
+ return;
+ }
+ for (var key in source) {
+ if (hasOwnProperty2.call(source, key)) {
+ assignValue(object, key, source[key]);
+ }
+ }
+ });
+ var assignIn = createAssigner(function(object, source) {
+ copyObject(source, keysIn(source), object);
+ });
+ var assignInWith = createAssigner(function(object, source, srcIndex, customizer) {
+ copyObject(source, keysIn(source), object, customizer);
+ });
+ var assignWith = createAssigner(function(object, source, srcIndex, customizer) {
+ copyObject(source, keys(source), object, customizer);
+ });
+ var at = flatRest(baseAt);
+ function create2(prototype, properties) {
+ var result2 = baseCreate(prototype);
+ return properties == null ? result2 : baseAssign(result2, properties);
+ }
+ var defaults2 = baseRest(function(object, sources) {
+ object = Object2(object);
+ var index = -1;
+ var length = sources.length;
+ var guard = length > 2 ? sources[2] : undefined2;
+ if (guard && isIterateeCall(sources[0], sources[1], guard)) {
+ length = 1;
+ }
+ while (++index < length) {
+ var source = sources[index];
+ var props = keysIn(source);
+ var propsIndex = -1;
+ var propsLength = props.length;
+ while (++propsIndex < propsLength) {
+ var key = props[propsIndex];
+ var value = object[key];
+ if (value === undefined2 || eq(value, objectProto3[key]) && !hasOwnProperty2.call(object, key)) {
+ object[key] = source[key];
+ }
+ }
+ }
+ return object;
+ });
+ var defaultsDeep = baseRest(function(args) {
+ args.push(undefined2, customDefaultsMerge);
+ return apply(mergeWith, undefined2, args);
+ });
+ function findKey(object, predicate) {
+ return baseFindKey(object, getIteratee(predicate, 3), baseForOwn);
+ }
+ function findLastKey(object, predicate) {
+ return baseFindKey(object, getIteratee(predicate, 3), baseForOwnRight);
+ }
+ function forIn(object, iteratee2) {
+ return object == null ? object : baseFor(object, getIteratee(iteratee2, 3), keysIn);
+ }
+ function forInRight(object, iteratee2) {
+ return object == null ? object : baseForRight(object, getIteratee(iteratee2, 3), keysIn);
+ }
+ function forOwn(object, iteratee2) {
+ return object && baseForOwn(object, getIteratee(iteratee2, 3));
+ }
+ function forOwnRight(object, iteratee2) {
+ return object && baseForOwnRight(object, getIteratee(iteratee2, 3));
+ }
+ function functions(object) {
+ return object == null ? [] : baseFunctions(object, keys(object));
+ }
+ function functionsIn(object) {
+ return object == null ? [] : baseFunctions(object, keysIn(object));
+ }
+ function get4(object, path, defaultValue) {
+ var result2 = object == null ? undefined2 : baseGet(object, path);
+ return result2 === undefined2 ? defaultValue : result2;
+ }
+ function has(object, path) {
+ return object != null && hasPath(object, path, baseHas);
+ }
+ function hasIn(object, path) {
+ return object != null && hasPath(object, path, baseHasIn);
+ }
+ var invert = createInverter(function(result2, value, key) {
+ if (value != null && typeof value.toString != "function") {
+ value = nativeObjectToString3.call(value);
+ }
+ result2[value] = key;
+ }, constant(identity4));
+ var invertBy = createInverter(function(result2, value, key) {
+ if (value != null && typeof value.toString != "function") {
+ value = nativeObjectToString3.call(value);
+ }
+ if (hasOwnProperty2.call(result2, value)) {
+ result2[value].push(key);
+ } else {
+ result2[value] = [key];
+ }
+ }, getIteratee);
+ var invoke = baseRest(baseInvoke);
+ function keys(object) {
+ return isArrayLike(object) ? arrayLikeKeys(object) : baseKeys(object);
+ }
+ function keysIn(object) {
+ return isArrayLike(object) ? arrayLikeKeys(object, true) : baseKeysIn(object);
+ }
+ function mapKeys(object, iteratee2) {
+ var result2 = {};
+ iteratee2 = getIteratee(iteratee2, 3);
+ baseForOwn(object, function(value, key, object2) {
+ baseAssignValue(result2, iteratee2(value, key, object2), value);
+ });
+ return result2;
+ }
+ function mapValues(object, iteratee2) {
+ var result2 = {};
+ iteratee2 = getIteratee(iteratee2, 3);
+ baseForOwn(object, function(value, key, object2) {
+ baseAssignValue(result2, key, iteratee2(value, key, object2));
+ });
+ return result2;
+ }
+ var merge3 = createAssigner(function(object, source, srcIndex) {
+ baseMerge(object, source, srcIndex);
+ });
+ var mergeWith = createAssigner(function(object, source, srcIndex, customizer) {
+ baseMerge(object, source, srcIndex, customizer);
+ });
+ var omit = flatRest(function(object, paths) {
+ var result2 = {};
+ if (object == null) {
+ return result2;
+ }
+ var isDeep = false;
+ paths = arrayMap2(paths, function(path) {
+ path = castPath(path, object);
+ isDeep || (isDeep = path.length > 1);
+ return path;
+ });
+ copyObject(object, getAllKeysIn(object), result2);
+ if (isDeep) {
+ result2 = baseClone(result2, CLONE_DEEP_FLAG | CLONE_FLAT_FLAG | CLONE_SYMBOLS_FLAG, customOmitClone);
+ }
+ var length = paths.length;
+ while (length--) {
+ baseUnset(result2, paths[length]);
+ }
+ return result2;
+ });
+ function omitBy(object, predicate) {
+ return pickBy(object, negate(getIteratee(predicate)));
+ }
+ var pick = flatRest(function(object, paths) {
+ return object == null ? {} : basePick(object, paths);
+ });
+ function pickBy(object, predicate) {
+ if (object == null) {
+ return {};
+ }
+ var props = arrayMap2(getAllKeysIn(object), function(prop) {
+ return [prop];
+ });
+ predicate = getIteratee(predicate);
+ return basePickBy(object, props, function(value, path) {
+ return predicate(value, path[0]);
+ });
+ }
+ function result(object, path, defaultValue) {
+ path = castPath(path, object);
+ var index = -1, length = path.length;
+ if (!length) {
+ length = 1;
+ object = undefined2;
+ }
+ while (++index < length) {
+ var value = object == null ? undefined2 : object[toKey(path[index])];
+ if (value === undefined2) {
+ index = length;
+ value = defaultValue;
+ }
+ object = isFunction(value) ? value.call(object) : value;
+ }
+ return object;
+ }
+ function set3(object, path, value) {
+ return object == null ? object : baseSet(object, path, value);
+ }
+ function setWith(object, path, value, customizer) {
+ customizer = typeof customizer == "function" ? customizer : undefined2;
+ return object == null ? object : baseSet(object, path, value, customizer);
+ }
+ var toPairs = createToPairs(keys);
+ var toPairsIn = createToPairs(keysIn);
+ function transform2(object, iteratee2, accumulator) {
+ var isArr = isArray2(object), isArrLike = isArr || isBuffer(object) || isTypedArray(object);
+ iteratee2 = getIteratee(iteratee2, 4);
+ if (accumulator == null) {
+ var Ctor = object && object.constructor;
+ if (isArrLike) {
+ accumulator = isArr ? new Ctor() : [];
+ } else if (isObject3(object)) {
+ accumulator = isFunction(Ctor) ? baseCreate(getPrototype(object)) : {};
+ } else {
+ accumulator = {};
+ }
+ }
+ (isArrLike ? arrayEach : baseForOwn)(object, function(value, index, object2) {
+ return iteratee2(accumulator, value, index, object2);
+ });
+ return accumulator;
+ }
+ function unset(object, path) {
+ return object == null ? true : baseUnset(object, path);
+ }
+ function update(object, path, updater) {
+ return object == null ? object : baseUpdate(object, path, castFunction(updater));
+ }
+ function updateWith(object, path, updater, customizer) {
+ customizer = typeof customizer == "function" ? customizer : undefined2;
+ return object == null ? object : baseUpdate(object, path, castFunction(updater), customizer);
+ }
+ function values(object) {
+ return object == null ? [] : baseValues(object, keys(object));
+ }
+ function valuesIn(object) {
+ return object == null ? [] : baseValues(object, keysIn(object));
+ }
+ function clamp3(number3, lower2, upper) {
+ if (upper === undefined2) {
+ upper = lower2;
+ lower2 = undefined2;
+ }
+ if (upper !== undefined2) {
+ upper = toNumber2(upper);
+ upper = upper === upper ? upper : 0;
+ }
+ if (lower2 !== undefined2) {
+ lower2 = toNumber2(lower2);
+ lower2 = lower2 === lower2 ? lower2 : 0;
+ }
+ return baseClamp(toNumber2(number3), lower2, upper);
+ }
+ function inRange(number3, start2, end) {
+ start2 = toFinite(start2);
+ if (end === undefined2) {
+ end = start2;
+ start2 = 0;
+ } else {
+ end = toFinite(end);
+ }
+ number3 = toNumber2(number3);
+ return baseInRange(number3, start2, end);
+ }
+ function random(lower2, upper, floating) {
+ if (floating && typeof floating != "boolean" && isIterateeCall(lower2, upper, floating)) {
+ upper = floating = undefined2;
+ }
+ if (floating === undefined2) {
+ if (typeof upper == "boolean") {
+ floating = upper;
+ upper = undefined2;
+ } else if (typeof lower2 == "boolean") {
+ floating = lower2;
+ lower2 = undefined2;
+ }
+ }
+ if (lower2 === undefined2 && upper === undefined2) {
+ lower2 = 0;
+ upper = 1;
+ } else {
+ lower2 = toFinite(lower2);
+ if (upper === undefined2) {
+ upper = lower2;
+ lower2 = 0;
+ } else {
+ upper = toFinite(upper);
+ }
+ }
+ if (lower2 > upper) {
+ var temp = lower2;
+ lower2 = upper;
+ upper = temp;
+ }
+ if (floating || lower2 % 1 || upper % 1) {
+ var rand = nativeRandom();
+ return nativeMin2(lower2 + rand * (upper - lower2 + freeParseFloat("1e-" + ((rand + "").length - 1))), upper);
+ }
+ return baseRandom(lower2, upper);
+ }
+ var camelCase = createCompounder(function(result2, word, index) {
+ word = word.toLowerCase();
+ return result2 + (index ? capitalize(word) : word);
+ });
+ function capitalize(string) {
+ return upperFirst(toString2(string).toLowerCase());
+ }
+ function deburr(string) {
+ string = toString2(string);
+ return string && string.replace(reLatin, deburrLetter).replace(reComboMark, "");
+ }
+ function endsWith(string, target, position) {
+ string = toString2(string);
+ target = baseToString2(target);
+ var length = string.length;
+ position = position === undefined2 ? length : baseClamp(toInteger(position), 0, length);
+ var end = position;
+ position -= target.length;
+ return position >= 0 && string.slice(position, end) == target;
+ }
+ function escape6(string) {
+ string = toString2(string);
+ return string && reHasUnescapedHtml2.test(string) ? string.replace(reUnescapedHtml2, escapeHtmlChar2) : string;
+ }
+ function escapeRegExp(string) {
+ string = toString2(string);
+ return string && reHasRegExpChar.test(string) ? string.replace(reRegExpChar, "\\$&") : string;
+ }
+ var kebabCase = createCompounder(function(result2, word, index) {
+ return result2 + (index ? "-" : "") + word.toLowerCase();
+ });
+ var lowerCase = createCompounder(function(result2, word, index) {
+ return result2 + (index ? " " : "") + word.toLowerCase();
+ });
+ var lowerFirst = createCaseFirst("toLowerCase");
+ function pad2(string, length, chars) {
+ string = toString2(string);
+ length = toInteger(length);
+ var strLength = length ? stringSize(string) : 0;
+ if (!length || strLength >= length) {
+ return string;
+ }
+ var mid = (length - strLength) / 2;
+ return createPadding(nativeFloor(mid), chars) + string + createPadding(nativeCeil(mid), chars);
+ }
+ function padEnd(string, length, chars) {
+ string = toString2(string);
+ length = toInteger(length);
+ var strLength = length ? stringSize(string) : 0;
+ return length && strLength < length ? string + createPadding(length - strLength, chars) : string;
+ }
+ function padStart(string, length, chars) {
+ string = toString2(string);
+ length = toInteger(length);
+ var strLength = length ? stringSize(string) : 0;
+ return length && strLength < length ? createPadding(length - strLength, chars) + string : string;
+ }
+ function parseInt2(string, radix, guard) {
+ if (guard || radix == null) {
+ radix = 0;
+ } else if (radix) {
+ radix = +radix;
+ }
+ return nativeParseInt(toString2(string).replace(reTrimStart2, ""), radix || 0);
+ }
+ function repeat(string, n2, guard) {
+ if (guard ? isIterateeCall(string, n2, guard) : n2 === undefined2) {
+ n2 = 1;
+ } else {
+ n2 = toInteger(n2);
+ }
+ return baseRepeat(toString2(string), n2);
+ }
+ function replace() {
+ var args = arguments, string = toString2(args[0]);
+ return args.length < 3 ? string : string.replace(args[1], args[2]);
+ }
+ var snakeCase = createCompounder(function(result2, word, index) {
+ return result2 + (index ? "_" : "") + word.toLowerCase();
+ });
+ function split(string, separator, limit) {
+ if (limit && typeof limit != "number" && isIterateeCall(string, separator, limit)) {
+ separator = limit = undefined2;
+ }
+ limit = limit === undefined2 ? MAX_ARRAY_LENGTH : limit >>> 0;
+ if (!limit) {
+ return [];
+ }
+ string = toString2(string);
+ if (string && (typeof separator == "string" || separator != null && !isRegExp(separator))) {
+ separator = baseToString2(separator);
+ if (!separator && hasUnicode(string)) {
+ return castSlice(stringToArray(string), 0, limit);
+ }
+ }
+ return string.split(separator, limit);
+ }
+ var startCase = createCompounder(function(result2, word, index) {
+ return result2 + (index ? " " : "") + upperFirst(word);
+ });
+ function startsWith(string, target, position) {
+ string = toString2(string);
+ position = position == null ? 0 : baseClamp(toInteger(position), 0, string.length);
+ target = baseToString2(target);
+ return string.slice(position, position + target.length) == target;
+ }
+ function template(string, options2, guard) {
+ var settings = lodash.templateSettings;
+ if (guard && isIterateeCall(string, options2, guard)) {
+ options2 = undefined2;
+ }
+ string = toString2(string);
+ options2 = assignInWith({}, options2, settings, customDefaultsAssignIn);
+ var imports = assignInWith({}, options2.imports, settings.imports, customDefaultsAssignIn), importsKeys = keys(imports), importsValues = baseValues(imports, importsKeys);
+ var isEscaping, isEvaluating, index = 0, interpolate = options2.interpolate || reNoMatch, source = "__p += '";
+ var reDelimiters = RegExp2(
+ (options2.escape || reNoMatch).source + "|" + interpolate.source + "|" + (interpolate === reInterpolate ? reEsTemplate : reNoMatch).source + "|" + (options2.evaluate || reNoMatch).source + "|$",
+ "g"
+ );
+ var sourceURL = "//# sourceURL=" + (hasOwnProperty2.call(options2, "sourceURL") ? (options2.sourceURL + "").replace(/\s/g, " ") : "lodash.templateSources[" + ++templateCounter + "]") + "\n";
+ string.replace(reDelimiters, function(match, escapeValue, interpolateValue, esTemplateValue, evaluateValue, offset) {
+ interpolateValue || (interpolateValue = esTemplateValue);
+ source += string.slice(index, offset).replace(reUnescapedString, escapeStringChar);
+ if (escapeValue) {
+ isEscaping = true;
+ source += "' +\n__e(" + escapeValue + ") +\n'";
+ }
+ if (evaluateValue) {
+ isEvaluating = true;
+ source += "';\n" + evaluateValue + ";\n__p += '";
+ }
+ if (interpolateValue) {
+ source += "' +\n((__t = (" + interpolateValue + ")) == null ? '' : __t) +\n'";
+ }
+ index = offset + match.length;
+ return match;
+ });
+ source += "';\n";
+ var variable = hasOwnProperty2.call(options2, "variable") && options2.variable;
+ if (!variable) {
+ source = "with (obj) {\n" + source + "\n}\n";
+ } else if (reForbiddenIdentifierChars.test(variable)) {
+ throw new Error2(INVALID_TEMPL_VAR_ERROR_TEXT);
+ }
+ source = (isEvaluating ? source.replace(reEmptyStringLeading, "") : source).replace(reEmptyStringMiddle, "$1").replace(reEmptyStringTrailing, "$1;");
+ source = "function(" + (variable || "obj") + ") {\n" + (variable ? "" : "obj || (obj = {});\n") + "var __t, __p = ''" + (isEscaping ? ", __e = _.escape" : "") + (isEvaluating ? ", __j = Array.prototype.join;\nfunction print() { __p += __j.call(arguments, '') }\n" : ";\n") + source + "return __p\n}";
+ var result2 = attempt(function() {
+ return Function2(importsKeys, sourceURL + "return " + source).apply(undefined2, importsValues);
+ });
+ result2.source = source;
+ if (isError(result2)) {
+ throw result2;
+ }
+ return result2;
+ }
+ function toLower(value) {
+ return toString2(value).toLowerCase();
+ }
+ function toUpper(value) {
+ return toString2(value).toUpperCase();
+ }
+ function trim(string, chars, guard) {
+ string = toString2(string);
+ if (string && (guard || chars === undefined2)) {
+ return baseTrim2(string);
+ }
+ if (!string || !(chars = baseToString2(chars))) {
+ return string;
+ }
+ var strSymbols = stringToArray(string), chrSymbols = stringToArray(chars), start2 = charsStartIndex(strSymbols, chrSymbols), end = charsEndIndex(strSymbols, chrSymbols) + 1;
+ return castSlice(strSymbols, start2, end).join("");
+ }
+ function trimEnd(string, chars, guard) {
+ string = toString2(string);
+ if (string && (guard || chars === undefined2)) {
+ return string.slice(0, trimmedEndIndex2(string) + 1);
+ }
+ if (!string || !(chars = baseToString2(chars))) {
+ return string;
+ }
+ var strSymbols = stringToArray(string), end = charsEndIndex(strSymbols, stringToArray(chars)) + 1;
+ return castSlice(strSymbols, 0, end).join("");
+ }
+ function trimStart(string, chars, guard) {
+ string = toString2(string);
+ if (string && (guard || chars === undefined2)) {
+ return string.replace(reTrimStart2, "");
+ }
+ if (!string || !(chars = baseToString2(chars))) {
+ return string;
+ }
+ var strSymbols = stringToArray(string), start2 = charsStartIndex(strSymbols, stringToArray(chars));
+ return castSlice(strSymbols, start2).join("");
+ }
+ function truncate(string, options2) {
+ var length = DEFAULT_TRUNC_LENGTH, omission = DEFAULT_TRUNC_OMISSION;
+ if (isObject3(options2)) {
+ var separator = "separator" in options2 ? options2.separator : separator;
+ length = "length" in options2 ? toInteger(options2.length) : length;
+ omission = "omission" in options2 ? baseToString2(options2.omission) : omission;
+ }
+ string = toString2(string);
+ var strLength = string.length;
+ if (hasUnicode(string)) {
+ var strSymbols = stringToArray(string);
+ strLength = strSymbols.length;
+ }
+ if (length >= strLength) {
+ return string;
+ }
+ var end = length - stringSize(omission);
+ if (end < 1) {
+ return omission;
+ }
+ var result2 = strSymbols ? castSlice(strSymbols, 0, end).join("") : string.slice(0, end);
+ if (separator === undefined2) {
+ return result2 + omission;
+ }
+ if (strSymbols) {
+ end += result2.length - end;
+ }
+ if (isRegExp(separator)) {
+ if (string.slice(end).search(separator)) {
+ var match, substring = result2;
+ if (!separator.global) {
+ separator = RegExp2(separator.source, toString2(reFlags.exec(separator)) + "g");
+ }
+ separator.lastIndex = 0;
+ while (match = separator.exec(substring)) {
+ var newEnd = match.index;
+ }
+ result2 = result2.slice(0, newEnd === undefined2 ? end : newEnd);
+ }
+ } else if (string.indexOf(baseToString2(separator), end) != end) {
+ var index = result2.lastIndexOf(separator);
+ if (index > -1) {
+ result2 = result2.slice(0, index);
+ }
+ }
+ return result2 + omission;
+ }
+ function unescape4(string) {
+ string = toString2(string);
+ return string && reHasEscapedHtml2.test(string) ? string.replace(reEscapedHtml2, unescapeHtmlChar2) : string;
+ }
+ var upperCase = createCompounder(function(result2, word, index) {
+ return result2 + (index ? " " : "") + word.toUpperCase();
+ });
+ var upperFirst = createCaseFirst("toUpperCase");
+ function words(string, pattern, guard) {
+ string = toString2(string);
+ pattern = guard ? undefined2 : pattern;
+ if (pattern === undefined2) {
+ return hasUnicodeWord(string) ? unicodeWords(string) : asciiWords(string);
+ }
+ return string.match(pattern) || [];
+ }
+ var attempt = baseRest(function(func, args) {
+ try {
+ return apply(func, undefined2, args);
+ } catch (e) {
+ return isError(e) ? e : new Error2(e);
+ }
+ });
+ var bindAll = flatRest(function(object, methodNames) {
+ arrayEach(methodNames, function(key) {
+ key = toKey(key);
+ baseAssignValue(object, key, bind(object[key], object));
+ });
+ return object;
+ });
+ function cond(pairs) {
+ var length = pairs == null ? 0 : pairs.length, toIteratee = getIteratee();
+ pairs = !length ? [] : arrayMap2(pairs, function(pair2) {
+ if (typeof pair2[1] != "function") {
+ throw new TypeError2(FUNC_ERROR_TEXT3);
+ }
+ return [toIteratee(pair2[0]), pair2[1]];
+ });
+ return baseRest(function(args) {
+ var index = -1;
+ while (++index < length) {
+ var pair2 = pairs[index];
+ if (apply(pair2[0], this, args)) {
+ return apply(pair2[1], this, args);
+ }
+ }
+ });
+ }
+ function conforms(source) {
+ return baseConforms(baseClone(source, CLONE_DEEP_FLAG));
+ }
+ function constant(value) {
+ return function() {
+ return value;
+ };
+ }
+ function defaultTo(value, defaultValue) {
+ return value == null || value !== value ? defaultValue : value;
+ }
+ var flow = createFlow();
+ var flowRight = createFlow(true);
+ function identity4(value) {
+ return value;
+ }
+ function iteratee(func) {
+ return baseIteratee(typeof func == "function" ? func : baseClone(func, CLONE_DEEP_FLAG));
+ }
+ function matches(source) {
+ return baseMatches(baseClone(source, CLONE_DEEP_FLAG));
+ }
+ function matchesProperty(path, srcValue) {
+ return baseMatchesProperty(path, baseClone(srcValue, CLONE_DEEP_FLAG));
+ }
+ var method = baseRest(function(path, args) {
+ return function(object) {
+ return baseInvoke(object, path, args);
+ };
+ });
+ var methodOf = baseRest(function(object, args) {
+ return function(path) {
+ return baseInvoke(object, path, args);
+ };
+ });
+ function mixin(object, source, options2) {
+ var props = keys(source), methodNames = baseFunctions(source, props);
+ if (options2 == null && !(isObject3(source) && (methodNames.length || !props.length))) {
+ options2 = source;
+ source = object;
+ object = this;
+ methodNames = baseFunctions(source, keys(source));
+ }
+ var chain2 = !(isObject3(options2) && "chain" in options2) || !!options2.chain, isFunc = isFunction(object);
+ arrayEach(methodNames, function(methodName) {
+ var func = source[methodName];
+ object[methodName] = func;
+ if (isFunc) {
+ object.prototype[methodName] = function() {
+ var chainAll = this.__chain__;
+ if (chain2 || chainAll) {
+ var result2 = object(this.__wrapped__), actions = result2.__actions__ = copyArray(this.__actions__);
+ actions.push({ "func": func, "args": arguments, "thisArg": object });
+ result2.__chain__ = chainAll;
+ return result2;
+ }
+ return func.apply(object, arrayPush([this.value()], arguments));
+ };
+ }
+ });
+ return object;
+ }
+ function noConflict() {
+ if (root3._ === this) {
+ root3._ = oldDash;
+ }
+ return this;
+ }
+ function noop3() {
+ }
+ function nthArg(n2) {
+ n2 = toInteger(n2);
+ return baseRest(function(args) {
+ return baseNth(args, n2);
+ });
+ }
+ var over = createOver(arrayMap2);
+ var overEvery = createOver(arrayEvery);
+ var overSome = createOver(arraySome);
+ function property(path) {
+ return isKey(path) ? baseProperty(toKey(path)) : basePropertyDeep(path);
+ }
+ function propertyOf(object) {
+ return function(path) {
+ return object == null ? undefined2 : baseGet(object, path);
+ };
+ }
+ var range3 = createRange();
+ var rangeRight = createRange(true);
+ function stubArray() {
+ return [];
+ }
+ function stubFalse() {
+ return false;
+ }
+ function stubObject() {
+ return {};
+ }
+ function stubString() {
+ return "";
+ }
+ function stubTrue() {
+ return true;
+ }
+ function times(n2, iteratee2) {
+ n2 = toInteger(n2);
+ if (n2 < 1 || n2 > MAX_SAFE_INTEGER) {
+ return [];
+ }
+ var index = MAX_ARRAY_LENGTH, length = nativeMin2(n2, MAX_ARRAY_LENGTH);
+ iteratee2 = getIteratee(iteratee2);
+ n2 -= MAX_ARRAY_LENGTH;
+ var result2 = baseTimes(length, iteratee2);
+ while (++index < n2) {
+ iteratee2(index);
+ }
+ return result2;
+ }
+ function toPath(value) {
+ if (isArray2(value)) {
+ return arrayMap2(value, toKey);
+ }
+ return isSymbol2(value) ? [value] : copyArray(stringToPath(toString2(value)));
+ }
+ function uniqueId(prefix) {
+ var id2 = ++idCounter;
+ return toString2(prefix) + id2;
+ }
+ var add = createMathOperation(function(augend, addend) {
+ return augend + addend;
+ }, 0);
+ var ceil = createRound("ceil");
+ var divide = createMathOperation(function(dividend, divisor) {
+ return dividend / divisor;
+ }, 1);
+ var floor = createRound("floor");
+ function max3(array2) {
+ return array2 && array2.length ? baseExtremum(array2, identity4, baseGt) : undefined2;
+ }
+ function maxBy(array2, iteratee2) {
+ return array2 && array2.length ? baseExtremum(array2, getIteratee(iteratee2, 2), baseGt) : undefined2;
+ }
+ function mean(array2) {
+ return baseMean(array2, identity4);
+ }
+ function meanBy(array2, iteratee2) {
+ return baseMean(array2, getIteratee(iteratee2, 2));
+ }
+ function min3(array2) {
+ return array2 && array2.length ? baseExtremum(array2, identity4, baseLt) : undefined2;
+ }
+ function minBy(array2, iteratee2) {
+ return array2 && array2.length ? baseExtremum(array2, getIteratee(iteratee2, 2), baseLt) : undefined2;
+ }
+ var multiply = createMathOperation(function(multiplier, multiplicand) {
+ return multiplier * multiplicand;
+ }, 1);
+ var round = createRound("round");
+ var subtract = createMathOperation(function(minuend, subtrahend) {
+ return minuend - subtrahend;
+ }, 0);
+ function sum(array2) {
+ return array2 && array2.length ? baseSum(array2, identity4) : 0;
+ }
+ function sumBy(array2, iteratee2) {
+ return array2 && array2.length ? baseSum(array2, getIteratee(iteratee2, 2)) : 0;
+ }
+ lodash.after = after;
+ lodash.ary = ary;
+ lodash.assign = assign;
+ lodash.assignIn = assignIn;
+ lodash.assignInWith = assignInWith;
+ lodash.assignWith = assignWith;
+ lodash.at = at;
+ lodash.before = before;
+ lodash.bind = bind;
+ lodash.bindAll = bindAll;
+ lodash.bindKey = bindKey2;
+ lodash.castArray = castArray;
+ lodash.chain = chain;
+ lodash.chunk = chunk;
+ lodash.compact = compact;
+ lodash.concat = concat;
+ lodash.cond = cond;
+ lodash.conforms = conforms;
+ lodash.constant = constant;
+ lodash.countBy = countBy;
+ lodash.create = create2;
+ lodash.curry = curry;
+ lodash.curryRight = curryRight;
+ lodash.debounce = debounce2;
+ lodash.defaults = defaults2;
+ lodash.defaultsDeep = defaultsDeep;
+ lodash.defer = defer;
+ lodash.delay = delay;
+ lodash.difference = difference;
+ lodash.differenceBy = differenceBy;
+ lodash.differenceWith = differenceWith;
+ lodash.drop = drop;
+ lodash.dropRight = dropRight;
+ lodash.dropRightWhile = dropRightWhile;
+ lodash.dropWhile = dropWhile;
+ lodash.fill = fill;
+ lodash.filter = filter2;
+ lodash.flatMap = flatMap;
+ lodash.flatMapDeep = flatMapDeep;
+ lodash.flatMapDepth = flatMapDepth;
+ lodash.flatten = flatten2;
+ lodash.flattenDeep = flattenDeep;
+ lodash.flattenDepth = flattenDepth;
+ lodash.flip = flip;
+ lodash.flow = flow;
+ lodash.flowRight = flowRight;
+ lodash.fromPairs = fromPairs;
+ lodash.functions = functions;
+ lodash.functionsIn = functionsIn;
+ lodash.groupBy = groupBy;
+ lodash.initial = initial;
+ lodash.intersection = intersection;
+ lodash.intersectionBy = intersectionBy;
+ lodash.intersectionWith = intersectionWith;
+ lodash.invert = invert;
+ lodash.invertBy = invertBy;
+ lodash.invokeMap = invokeMap;
+ lodash.iteratee = iteratee;
+ lodash.keyBy = keyBy;
+ lodash.keys = keys;
+ lodash.keysIn = keysIn;
+ lodash.map = map2;
+ lodash.mapKeys = mapKeys;
+ lodash.mapValues = mapValues;
+ lodash.matches = matches;
+ lodash.matchesProperty = matchesProperty;
+ lodash.memoize = memoize;
+ lodash.merge = merge3;
+ lodash.mergeWith = mergeWith;
+ lodash.method = method;
+ lodash.methodOf = methodOf;
+ lodash.mixin = mixin;
+ lodash.negate = negate;
+ lodash.nthArg = nthArg;
+ lodash.omit = omit;
+ lodash.omitBy = omitBy;
+ lodash.once = once;
+ lodash.orderBy = orderBy;
+ lodash.over = over;
+ lodash.overArgs = overArgs;
+ lodash.overEvery = overEvery;
+ lodash.overSome = overSome;
+ lodash.partial = partial;
+ lodash.partialRight = partialRight;
+ lodash.partition = partition;
+ lodash.pick = pick;
+ lodash.pickBy = pickBy;
+ lodash.property = property;
+ lodash.propertyOf = propertyOf;
+ lodash.pull = pull;
+ lodash.pullAll = pullAll;
+ lodash.pullAllBy = pullAllBy;
+ lodash.pullAllWith = pullAllWith;
+ lodash.pullAt = pullAt;
+ lodash.range = range3;
+ lodash.rangeRight = rangeRight;
+ lodash.rearg = rearg;
+ lodash.reject = reject;
+ lodash.remove = remove2;
+ lodash.rest = rest;
+ lodash.reverse = reverse;
+ lodash.sampleSize = sampleSize;
+ lodash.set = set3;
+ lodash.setWith = setWith;
+ lodash.shuffle = shuffle;
+ lodash.slice = slice;
+ lodash.sortBy = sortBy;
+ lodash.sortedUniq = sortedUniq;
+ lodash.sortedUniqBy = sortedUniqBy;
+ lodash.split = split;
+ lodash.spread = spread;
+ lodash.tail = tail;
+ lodash.take = take;
+ lodash.takeRight = takeRight;
+ lodash.takeRightWhile = takeRightWhile;
+ lodash.takeWhile = takeWhile;
+ lodash.tap = tap;
+ lodash.throttle = throttle2;
+ lodash.thru = thru;
+ lodash.toArray = toArray;
+ lodash.toPairs = toPairs;
+ lodash.toPairsIn = toPairsIn;
+ lodash.toPath = toPath;
+ lodash.toPlainObject = toPlainObject;
+ lodash.transform = transform2;
+ lodash.unary = unary;
+ lodash.union = union;
+ lodash.unionBy = unionBy;
+ lodash.unionWith = unionWith;
+ lodash.uniq = uniq;
+ lodash.uniqBy = uniqBy;
+ lodash.uniqWith = uniqWith;
+ lodash.unset = unset;
+ lodash.unzip = unzip;
+ lodash.unzipWith = unzipWith;
+ lodash.update = update;
+ lodash.updateWith = updateWith;
+ lodash.values = values;
+ lodash.valuesIn = valuesIn;
+ lodash.without = without;
+ lodash.words = words;
+ lodash.wrap = wrap2;
+ lodash.xor = xor;
+ lodash.xorBy = xorBy;
+ lodash.xorWith = xorWith;
+ lodash.zip = zip;
+ lodash.zipObject = zipObject;
+ lodash.zipObjectDeep = zipObjectDeep;
+ lodash.zipWith = zipWith;
+ lodash.entries = toPairs;
+ lodash.entriesIn = toPairsIn;
+ lodash.extend = assignIn;
+ lodash.extendWith = assignInWith;
+ mixin(lodash, lodash);
+ lodash.add = add;
+ lodash.attempt = attempt;
+ lodash.camelCase = camelCase;
+ lodash.capitalize = capitalize;
+ lodash.ceil = ceil;
+ lodash.clamp = clamp3;
+ lodash.clone = clone;
+ lodash.cloneDeep = cloneDeep;
+ lodash.cloneDeepWith = cloneDeepWith;
+ lodash.cloneWith = cloneWith;
+ lodash.conformsTo = conformsTo;
+ lodash.deburr = deburr;
+ lodash.defaultTo = defaultTo;
+ lodash.divide = divide;
+ lodash.endsWith = endsWith;
+ lodash.eq = eq;
+ lodash.escape = escape6;
+ lodash.escapeRegExp = escapeRegExp;
+ lodash.every = every;
+ lodash.find = find2;
+ lodash.findIndex = findIndex;
+ lodash.findKey = findKey;
+ lodash.findLast = findLast;
+ lodash.findLastIndex = findLastIndex;
+ lodash.findLastKey = findLastKey;
+ lodash.floor = floor;
+ lodash.forEach = forEach;
+ lodash.forEachRight = forEachRight;
+ lodash.forIn = forIn;
+ lodash.forInRight = forInRight;
+ lodash.forOwn = forOwn;
+ lodash.forOwnRight = forOwnRight;
+ lodash.get = get4;
+ lodash.gt = gt;
+ lodash.gte = gte;
+ lodash.has = has;
+ lodash.hasIn = hasIn;
+ lodash.head = head;
+ lodash.identity = identity4;
+ lodash.includes = includes;
+ lodash.indexOf = indexOf;
+ lodash.inRange = inRange;
+ lodash.invoke = invoke;
+ lodash.isArguments = isArguments;
+ lodash.isArray = isArray2;
+ lodash.isArrayBuffer = isArrayBuffer;
+ lodash.isArrayLike = isArrayLike;
+ lodash.isArrayLikeObject = isArrayLikeObject;
+ lodash.isBoolean = isBoolean;
+ lodash.isBuffer = isBuffer;
+ lodash.isDate = isDate;
+ lodash.isElement = isElement2;
+ lodash.isEmpty = isEmpty;
+ lodash.isEqual = isEqual;
+ lodash.isEqualWith = isEqualWith;
+ lodash.isError = isError;
+ lodash.isFinite = isFinite2;
+ lodash.isFunction = isFunction;
+ lodash.isInteger = isInteger;
+ lodash.isLength = isLength;
+ lodash.isMap = isMap;
+ lodash.isMatch = isMatch;
+ lodash.isMatchWith = isMatchWith;
+ lodash.isNaN = isNaN2;
+ lodash.isNative = isNative;
+ lodash.isNil = isNil;
+ lodash.isNull = isNull;
+ lodash.isNumber = isNumber2;
+ lodash.isObject = isObject3;
+ lodash.isObjectLike = isObjectLike2;
+ lodash.isPlainObject = isPlainObject;
+ lodash.isRegExp = isRegExp;
+ lodash.isSafeInteger = isSafeInteger;
+ lodash.isSet = isSet;
+ lodash.isString = isString;
+ lodash.isSymbol = isSymbol2;
+ lodash.isTypedArray = isTypedArray;
+ lodash.isUndefined = isUndefined;
+ lodash.isWeakMap = isWeakMap;
+ lodash.isWeakSet = isWeakSet;
+ lodash.join = join;
+ lodash.kebabCase = kebabCase;
+ lodash.last = last;
+ lodash.lastIndexOf = lastIndexOf;
+ lodash.lowerCase = lowerCase;
+ lodash.lowerFirst = lowerFirst;
+ lodash.lt = lt;
+ lodash.lte = lte;
+ lodash.max = max3;
+ lodash.maxBy = maxBy;
+ lodash.mean = mean;
+ lodash.meanBy = meanBy;
+ lodash.min = min3;
+ lodash.minBy = minBy;
+ lodash.stubArray = stubArray;
+ lodash.stubFalse = stubFalse;
+ lodash.stubObject = stubObject;
+ lodash.stubString = stubString;
+ lodash.stubTrue = stubTrue;
+ lodash.multiply = multiply;
+ lodash.nth = nth;
+ lodash.noConflict = noConflict;
+ lodash.noop = noop3;
+ lodash.now = now3;
+ lodash.pad = pad2;
+ lodash.padEnd = padEnd;
+ lodash.padStart = padStart;
+ lodash.parseInt = parseInt2;
+ lodash.random = random;
+ lodash.reduce = reduce;
+ lodash.reduceRight = reduceRight;
+ lodash.repeat = repeat;
+ lodash.replace = replace;
+ lodash.result = result;
+ lodash.round = round;
+ lodash.runInContext = runInContext2;
+ lodash.sample = sample;
+ lodash.size = size;
+ lodash.snakeCase = snakeCase;
+ lodash.some = some;
+ lodash.sortedIndex = sortedIndex;
+ lodash.sortedIndexBy = sortedIndexBy;
+ lodash.sortedIndexOf = sortedIndexOf;
+ lodash.sortedLastIndex = sortedLastIndex;
+ lodash.sortedLastIndexBy = sortedLastIndexBy;
+ lodash.sortedLastIndexOf = sortedLastIndexOf;
+ lodash.startCase = startCase;
+ lodash.startsWith = startsWith;
+ lodash.subtract = subtract;
+ lodash.sum = sum;
+ lodash.sumBy = sumBy;
+ lodash.template = template;
+ lodash.times = times;
+ lodash.toFinite = toFinite;
+ lodash.toInteger = toInteger;
+ lodash.toLength = toLength;
+ lodash.toLower = toLower;
+ lodash.toNumber = toNumber2;
+ lodash.toSafeInteger = toSafeInteger;
+ lodash.toString = toString2;
+ lodash.toUpper = toUpper;
+ lodash.trim = trim;
+ lodash.trimEnd = trimEnd;
+ lodash.trimStart = trimStart;
+ lodash.truncate = truncate;
+ lodash.unescape = unescape4;
+ lodash.uniqueId = uniqueId;
+ lodash.upperCase = upperCase;
+ lodash.upperFirst = upperFirst;
+ lodash.each = forEach;
+ lodash.eachRight = forEachRight;
+ lodash.first = head;
+ mixin(lodash, function() {
+ var source = {};
+ baseForOwn(lodash, function(func, methodName) {
+ if (!hasOwnProperty2.call(lodash.prototype, methodName)) {
+ source[methodName] = func;
+ }
+ });
+ return source;
+ }(), { "chain": false });
+ lodash.VERSION = VERSION;
+ arrayEach(["bind", "bindKey", "curry", "curryRight", "partial", "partialRight"], function(methodName) {
+ lodash[methodName].placeholder = lodash;
+ });
+ arrayEach(["drop", "take"], function(methodName, index) {
+ LazyWrapper.prototype[methodName] = function(n2) {
+ n2 = n2 === undefined2 ? 1 : nativeMax2(toInteger(n2), 0);
+ var result2 = this.__filtered__ && !index ? new LazyWrapper(this) : this.clone();
+ if (result2.__filtered__) {
+ result2.__takeCount__ = nativeMin2(n2, result2.__takeCount__);
+ } else {
+ result2.__views__.push({
+ "size": nativeMin2(n2, MAX_ARRAY_LENGTH),
+ "type": methodName + (result2.__dir__ < 0 ? "Right" : "")
+ });
+ }
+ return result2;
+ };
+ LazyWrapper.prototype[methodName + "Right"] = function(n2) {
+ return this.reverse()[methodName](n2).reverse();
+ };
+ });
+ arrayEach(["filter", "map", "takeWhile"], function(methodName, index) {
+ var type2 = index + 1, isFilter = type2 == LAZY_FILTER_FLAG || type2 == LAZY_WHILE_FLAG;
+ LazyWrapper.prototype[methodName] = function(iteratee2) {
+ var result2 = this.clone();
+ result2.__iteratees__.push({
+ "iteratee": getIteratee(iteratee2, 3),
+ "type": type2
+ });
+ result2.__filtered__ = result2.__filtered__ || isFilter;
+ return result2;
+ };
+ });
+ arrayEach(["head", "last"], function(methodName, index) {
+ var takeName = "take" + (index ? "Right" : "");
+ LazyWrapper.prototype[methodName] = function() {
+ return this[takeName](1).value()[0];
+ };
+ });
+ arrayEach(["initial", "tail"], function(methodName, index) {
+ var dropName = "drop" + (index ? "" : "Right");
+ LazyWrapper.prototype[methodName] = function() {
+ return this.__filtered__ ? new LazyWrapper(this) : this[dropName](1);
+ };
+ });
+ LazyWrapper.prototype.compact = function() {
+ return this.filter(identity4);
+ };
+ LazyWrapper.prototype.find = function(predicate) {
+ return this.filter(predicate).head();
+ };
+ LazyWrapper.prototype.findLast = function(predicate) {
+ return this.reverse().find(predicate);
+ };
+ LazyWrapper.prototype.invokeMap = baseRest(function(path, args) {
+ if (typeof path == "function") {
+ return new LazyWrapper(this);
+ }
+ return this.map(function(value) {
+ return baseInvoke(value, path, args);
+ });
+ });
+ LazyWrapper.prototype.reject = function(predicate) {
+ return this.filter(negate(getIteratee(predicate)));
+ };
+ LazyWrapper.prototype.slice = function(start2, end) {
+ start2 = toInteger(start2);
+ var result2 = this;
+ if (result2.__filtered__ && (start2 > 0 || end < 0)) {
+ return new LazyWrapper(result2);
+ }
+ if (start2 < 0) {
+ result2 = result2.takeRight(-start2);
+ } else if (start2) {
+ result2 = result2.drop(start2);
+ }
+ if (end !== undefined2) {
+ end = toInteger(end);
+ result2 = end < 0 ? result2.dropRight(-end) : result2.take(end - start2);
+ }
+ return result2;
+ };
+ LazyWrapper.prototype.takeRightWhile = function(predicate) {
+ return this.reverse().takeWhile(predicate).reverse();
+ };
+ LazyWrapper.prototype.toArray = function() {
+ return this.take(MAX_ARRAY_LENGTH);
+ };
+ baseForOwn(LazyWrapper.prototype, function(func, methodName) {
+ var checkIteratee = /^(?:filter|find|map|reject)|While$/.test(methodName), isTaker = /^(?:head|last)$/.test(methodName), lodashFunc = lodash[isTaker ? "take" + (methodName == "last" ? "Right" : "") : methodName], retUnwrapped = isTaker || /^find/.test(methodName);
+ if (!lodashFunc) {
+ return;
+ }
+ lodash.prototype[methodName] = function() {
+ var value = this.__wrapped__, args = isTaker ? [1] : arguments, isLazy = value instanceof LazyWrapper, iteratee2 = args[0], useLazy = isLazy || isArray2(value);
+ var interceptor = function(value2) {
+ var result3 = lodashFunc.apply(lodash, arrayPush([value2], args));
+ return isTaker && chainAll ? result3[0] : result3;
+ };
+ if (useLazy && checkIteratee && typeof iteratee2 == "function" && iteratee2.length != 1) {
+ isLazy = useLazy = false;
+ }
+ var chainAll = this.__chain__, isHybrid = !!this.__actions__.length, isUnwrapped = retUnwrapped && !chainAll, onlyLazy = isLazy && !isHybrid;
+ if (!retUnwrapped && useLazy) {
+ value = onlyLazy ? value : new LazyWrapper(this);
+ var result2 = func.apply(value, args);
+ result2.__actions__.push({ "func": thru, "args": [interceptor], "thisArg": undefined2 });
+ return new LodashWrapper(result2, chainAll);
+ }
+ if (isUnwrapped && onlyLazy) {
+ return func.apply(this, args);
+ }
+ result2 = this.thru(interceptor);
+ return isUnwrapped ? isTaker ? result2.value()[0] : result2.value() : result2;
+ };
+ });
+ arrayEach(["pop", "push", "shift", "sort", "splice", "unshift"], function(methodName) {
+ var func = arrayProto[methodName], chainName = /^(?:push|sort|unshift)$/.test(methodName) ? "tap" : "thru", retUnwrapped = /^(?:pop|shift)$/.test(methodName);
+ lodash.prototype[methodName] = function() {
+ var args = arguments;
+ if (retUnwrapped && !this.__chain__) {
+ var value = this.value();
+ return func.apply(isArray2(value) ? value : [], args);
+ }
+ return this[chainName](function(value2) {
+ return func.apply(isArray2(value2) ? value2 : [], args);
+ });
+ };
+ });
+ baseForOwn(LazyWrapper.prototype, function(func, methodName) {
+ var lodashFunc = lodash[methodName];
+ if (lodashFunc) {
+ var key = lodashFunc.name + "";
+ if (!hasOwnProperty2.call(realNames, key)) {
+ realNames[key] = [];
+ }
+ realNames[key].push({ "name": methodName, "func": lodashFunc });
+ }
+ });
+ realNames[createHybrid(undefined2, WRAP_BIND_KEY_FLAG).name] = [{
+ "name": "wrapper",
+ "func": undefined2
+ }];
+ LazyWrapper.prototype.clone = lazyClone;
+ LazyWrapper.prototype.reverse = lazyReverse;
+ LazyWrapper.prototype.value = lazyValue;
+ lodash.prototype.at = wrapperAt;
+ lodash.prototype.chain = wrapperChain;
+ lodash.prototype.commit = wrapperCommit;
+ lodash.prototype.next = wrapperNext;
+ lodash.prototype.plant = wrapperPlant;
+ lodash.prototype.reverse = wrapperReverse;
+ lodash.prototype.toJSON = lodash.prototype.valueOf = lodash.prototype.value = wrapperValue;
+ lodash.prototype.first = lodash.prototype.head;
+ if (symIterator) {
+ lodash.prototype[symIterator] = wrapperToIterator;
+ }
+ return lodash;
+ };
+ var _ = runInContext();
+ if (typeof define == "function" && typeof define.amd == "object" && define.amd) {
+ root3._ = _;
+ define(function() {
+ return _;
+ });
+ } else if (freeModule) {
+ (freeModule.exports = _)._ = _;
+ freeExports._ = _;
+ } else {
+ root3._ = _;
+ }
+ }).call(exports2);
+ }
+ });
+
+ // node_modules/rbush/rbush.min.js
+ var require_rbush_min = __commonJS({
+ "node_modules/rbush/rbush.min.js"(exports2, module2) {
+ !function(t, i2) {
+ "object" == typeof exports2 && "undefined" != typeof module2 ? module2.exports = i2() : "function" == typeof define && define.amd ? define(i2) : (t = t || self).RBush = i2();
+ }(exports2, function() {
+ "use strict";
+ function t(t2, r2, e3, a2, h2) {
+ !function t3(n3, r3, e4, a3, h3) {
+ for (; a3 > e4; ) {
+ if (a3 - e4 > 600) {
+ var o2 = a3 - e4 + 1, s2 = r3 - e4 + 1, l2 = Math.log(o2), f3 = 0.5 * Math.exp(2 * l2 / 3), u2 = 0.5 * Math.sqrt(l2 * f3 * (o2 - f3) / o2) * (s2 - o2 / 2 < 0 ? -1 : 1), m2 = Math.max(e4, Math.floor(r3 - s2 * f3 / o2 + u2)), c2 = Math.min(a3, Math.floor(r3 + (o2 - s2) * f3 / o2 + u2));
+ t3(n3, r3, m2, c2, h3);
+ }
+ var p2 = n3[r3], d2 = e4, x = a3;
+ for (i2(n3, e4, r3), h3(n3[a3], p2) > 0 && i2(n3, e4, a3); d2 < x; ) {
+ for (i2(n3, d2, x), d2++, x--; h3(n3[d2], p2) < 0; )
+ d2++;
+ for (; h3(n3[x], p2) > 0; )
+ x--;
+ }
+ 0 === h3(n3[e4], p2) ? i2(n3, e4, x) : i2(n3, ++x, a3), x <= r3 && (e4 = x + 1), r3 <= x && (a3 = x - 1);
+ }
+ }(t2, r2, e3 || 0, a2 || t2.length - 1, h2 || n2);
+ }
+ function i2(t2, i3, n3) {
+ var r2 = t2[i3];
+ t2[i3] = t2[n3], t2[n3] = r2;
+ }
+ function n2(t2, i3) {
+ return t2 < i3 ? -1 : t2 > i3 ? 1 : 0;
+ }
+ var r = function(t2) {
+ void 0 === t2 && (t2 = 9), this._maxEntries = Math.max(4, t2), this._minEntries = Math.max(2, Math.ceil(0.4 * this._maxEntries)), this.clear();
+ };
+ function e(t2, i3, n3) {
+ if (!n3)
+ return i3.indexOf(t2);
+ for (var r2 = 0; r2 < i3.length; r2++)
+ if (n3(t2, i3[r2]))
+ return r2;
+ return -1;
+ }
+ function a(t2, i3) {
+ h(t2, 0, t2.children.length, i3, t2);
+ }
+ function h(t2, i3, n3, r2, e3) {
+ e3 || (e3 = p(null)), e3.minX = 1 / 0, e3.minY = 1 / 0, e3.maxX = -1 / 0, e3.maxY = -1 / 0;
+ for (var a2 = i3; a2 < n3; a2++) {
+ var h2 = t2.children[a2];
+ o(e3, t2.leaf ? r2(h2) : h2);
+ }
+ return e3;
+ }
+ function o(t2, i3) {
+ return t2.minX = Math.min(t2.minX, i3.minX), t2.minY = Math.min(t2.minY, i3.minY), t2.maxX = Math.max(t2.maxX, i3.maxX), t2.maxY = Math.max(t2.maxY, i3.maxY), t2;
+ }
+ function s(t2, i3) {
+ return t2.minX - i3.minX;
+ }
+ function l(t2, i3) {
+ return t2.minY - i3.minY;
+ }
+ function f2(t2) {
+ return (t2.maxX - t2.minX) * (t2.maxY - t2.minY);
+ }
+ function u(t2) {
+ return t2.maxX - t2.minX + (t2.maxY - t2.minY);
+ }
+ function m(t2, i3) {
+ return t2.minX <= i3.minX && t2.minY <= i3.minY && i3.maxX <= t2.maxX && i3.maxY <= t2.maxY;
+ }
+ function c(t2, i3) {
+ return i3.minX <= t2.maxX && i3.minY <= t2.maxY && i3.maxX >= t2.minX && i3.maxY >= t2.minY;
+ }
+ function p(t2) {
+ return { children: t2, height: 1, leaf: true, minX: 1 / 0, minY: 1 / 0, maxX: -1 / 0, maxY: -1 / 0 };
+ }
+ function d(i3, n3, r2, e3, a2) {
+ for (var h2 = [n3, r2]; h2.length; )
+ if (!((r2 = h2.pop()) - (n3 = h2.pop()) <= e3)) {
+ var o2 = n3 + Math.ceil((r2 - n3) / e3 / 2) * e3;
+ t(i3, o2, n3, r2, a2), h2.push(n3, o2, o2, r2);
+ }
+ }
+ return r.prototype.all = function() {
+ return this._all(this.data, []);
+ }, r.prototype.search = function(t2) {
+ var i3 = this.data, n3 = [];
+ if (!c(t2, i3))
+ return n3;
+ for (var r2 = this.toBBox, e3 = []; i3; ) {
+ for (var a2 = 0; a2 < i3.children.length; a2++) {
+ var h2 = i3.children[a2], o2 = i3.leaf ? r2(h2) : h2;
+ c(t2, o2) && (i3.leaf ? n3.push(h2) : m(t2, o2) ? this._all(h2, n3) : e3.push(h2));
+ }
+ i3 = e3.pop();
+ }
+ return n3;
+ }, r.prototype.collides = function(t2) {
+ var i3 = this.data;
+ if (!c(t2, i3))
+ return false;
+ for (var n3 = []; i3; ) {
+ for (var r2 = 0; r2 < i3.children.length; r2++) {
+ var e3 = i3.children[r2], a2 = i3.leaf ? this.toBBox(e3) : e3;
+ if (c(t2, a2)) {
+ if (i3.leaf || m(t2, a2))
+ return true;
+ n3.push(e3);
+ }
+ }
+ i3 = n3.pop();
+ }
+ return false;
+ }, r.prototype.load = function(t2) {
+ if (!t2 || !t2.length)
+ return this;
+ if (t2.length < this._minEntries) {
+ for (var i3 = 0; i3 < t2.length; i3++)
+ this.insert(t2[i3]);
+ return this;
+ }
+ var n3 = this._build(t2.slice(), 0, t2.length - 1, 0);
+ if (this.data.children.length)
+ if (this.data.height === n3.height)
+ this._splitRoot(this.data, n3);
+ else {
+ if (this.data.height < n3.height) {
+ var r2 = this.data;
+ this.data = n3, n3 = r2;
+ }
+ this._insert(n3, this.data.height - n3.height - 1, true);
+ }
+ else
+ this.data = n3;
+ return this;
+ }, r.prototype.insert = function(t2) {
+ return t2 && this._insert(t2, this.data.height - 1), this;
+ }, r.prototype.clear = function() {
+ return this.data = p([]), this;
+ }, r.prototype.remove = function(t2, i3) {
+ if (!t2)
+ return this;
+ for (var n3, r2, a2, h2 = this.data, o2 = this.toBBox(t2), s2 = [], l2 = []; h2 || s2.length; ) {
+ if (h2 || (h2 = s2.pop(), r2 = s2[s2.length - 1], n3 = l2.pop(), a2 = true), h2.leaf) {
+ var f3 = e(t2, h2.children, i3);
+ if (-1 !== f3)
+ return h2.children.splice(f3, 1), s2.push(h2), this._condense(s2), this;
+ }
+ a2 || h2.leaf || !m(h2, o2) ? r2 ? (n3++, h2 = r2.children[n3], a2 = false) : h2 = null : (s2.push(h2), l2.push(n3), n3 = 0, r2 = h2, h2 = h2.children[0]);
+ }
+ return this;
+ }, r.prototype.toBBox = function(t2) {
+ return t2;
+ }, r.prototype.compareMinX = function(t2, i3) {
+ return t2.minX - i3.minX;
+ }, r.prototype.compareMinY = function(t2, i3) {
+ return t2.minY - i3.minY;
+ }, r.prototype.toJSON = function() {
+ return this.data;
+ }, r.prototype.fromJSON = function(t2) {
+ return this.data = t2, this;
+ }, r.prototype._all = function(t2, i3) {
+ for (var n3 = []; t2; )
+ t2.leaf ? i3.push.apply(i3, t2.children) : n3.push.apply(n3, t2.children), t2 = n3.pop();
+ return i3;
+ }, r.prototype._build = function(t2, i3, n3, r2) {
+ var e3, h2 = n3 - i3 + 1, o2 = this._maxEntries;
+ if (h2 <= o2)
+ return a(e3 = p(t2.slice(i3, n3 + 1)), this.toBBox), e3;
+ r2 || (r2 = Math.ceil(Math.log(h2) / Math.log(o2)), o2 = Math.ceil(h2 / Math.pow(o2, r2 - 1))), (e3 = p([])).leaf = false, e3.height = r2;
+ var s2 = Math.ceil(h2 / o2), l2 = s2 * Math.ceil(Math.sqrt(o2));
+ d(t2, i3, n3, l2, this.compareMinX);
+ for (var f3 = i3; f3 <= n3; f3 += l2) {
+ var u2 = Math.min(f3 + l2 - 1, n3);
+ d(t2, f3, u2, s2, this.compareMinY);
+ for (var m2 = f3; m2 <= u2; m2 += s2) {
+ var c2 = Math.min(m2 + s2 - 1, u2);
+ e3.children.push(this._build(t2, m2, c2, r2 - 1));
+ }
+ }
+ return a(e3, this.toBBox), e3;
+ }, r.prototype._chooseSubtree = function(t2, i3, n3, r2) {
+ for (; r2.push(i3), !i3.leaf && r2.length - 1 !== n3; ) {
+ for (var e3 = 1 / 0, a2 = 1 / 0, h2 = void 0, o2 = 0; o2 < i3.children.length; o2++) {
+ var s2 = i3.children[o2], l2 = f2(s2), u2 = (m2 = t2, c2 = s2, (Math.max(c2.maxX, m2.maxX) - Math.min(c2.minX, m2.minX)) * (Math.max(c2.maxY, m2.maxY) - Math.min(c2.minY, m2.minY)) - l2);
+ u2 < a2 ? (a2 = u2, e3 = l2 < e3 ? l2 : e3, h2 = s2) : u2 === a2 && l2 < e3 && (e3 = l2, h2 = s2);
+ }
+ i3 = h2 || i3.children[0];
+ }
+ var m2, c2;
+ return i3;
+ }, r.prototype._insert = function(t2, i3, n3) {
+ var r2 = n3 ? t2 : this.toBBox(t2), e3 = [], a2 = this._chooseSubtree(r2, this.data, i3, e3);
+ for (a2.children.push(t2), o(a2, r2); i3 >= 0 && e3[i3].children.length > this._maxEntries; )
+ this._split(e3, i3), i3--;
+ this._adjustParentBBoxes(r2, e3, i3);
+ }, r.prototype._split = function(t2, i3) {
+ var n3 = t2[i3], r2 = n3.children.length, e3 = this._minEntries;
+ this._chooseSplitAxis(n3, e3, r2);
+ var h2 = this._chooseSplitIndex(n3, e3, r2), o2 = p(n3.children.splice(h2, n3.children.length - h2));
+ o2.height = n3.height, o2.leaf = n3.leaf, a(n3, this.toBBox), a(o2, this.toBBox), i3 ? t2[i3 - 1].children.push(o2) : this._splitRoot(n3, o2);
+ }, r.prototype._splitRoot = function(t2, i3) {
+ this.data = p([t2, i3]), this.data.height = t2.height + 1, this.data.leaf = false, a(this.data, this.toBBox);
+ }, r.prototype._chooseSplitIndex = function(t2, i3, n3) {
+ for (var r2, e3, a2, o2, s2, l2, u2, m2 = 1 / 0, c2 = 1 / 0, p2 = i3; p2 <= n3 - i3; p2++) {
+ var d2 = h(t2, 0, p2, this.toBBox), x = h(t2, p2, n3, this.toBBox), v = (e3 = d2, a2 = x, o2 = void 0, s2 = void 0, l2 = void 0, u2 = void 0, o2 = Math.max(e3.minX, a2.minX), s2 = Math.max(e3.minY, a2.minY), l2 = Math.min(e3.maxX, a2.maxX), u2 = Math.min(e3.maxY, a2.maxY), Math.max(0, l2 - o2) * Math.max(0, u2 - s2)), M = f2(d2) + f2(x);
+ v < m2 ? (m2 = v, r2 = p2, c2 = M < c2 ? M : c2) : v === m2 && M < c2 && (c2 = M, r2 = p2);
+ }
+ return r2 || n3 - i3;
+ }, r.prototype._chooseSplitAxis = function(t2, i3, n3) {
+ var r2 = t2.leaf ? this.compareMinX : s, e3 = t2.leaf ? this.compareMinY : l;
+ this._allDistMargin(t2, i3, n3, r2) < this._allDistMargin(t2, i3, n3, e3) && t2.children.sort(r2);
+ }, r.prototype._allDistMargin = function(t2, i3, n3, r2) {
+ t2.children.sort(r2);
+ for (var e3 = this.toBBox, a2 = h(t2, 0, i3, e3), s2 = h(t2, n3 - i3, n3, e3), l2 = u(a2) + u(s2), f3 = i3; f3 < n3 - i3; f3++) {
+ var m2 = t2.children[f3];
+ o(a2, t2.leaf ? e3(m2) : m2), l2 += u(a2);
+ }
+ for (var c2 = n3 - i3 - 1; c2 >= i3; c2--) {
+ var p2 = t2.children[c2];
+ o(s2, t2.leaf ? e3(p2) : p2), l2 += u(s2);
+ }
+ return l2;
+ }, r.prototype._adjustParentBBoxes = function(t2, i3, n3) {
+ for (var r2 = n3; r2 >= 0; r2--)
+ o(i3[r2], t2);
+ }, r.prototype._condense = function(t2) {
+ for (var i3 = t2.length - 1, n3 = void 0; i3 >= 0; i3--)
+ 0 === t2[i3].children.length ? i3 > 0 ? (n3 = t2[i3 - 1].children).splice(n3.indexOf(t2[i3]), 1) : this.clear() : a(t2[i3], this.toBBox);
+ }, r;
+ });
+ }
+ });
+
+ // node_modules/ieee754/index.js
+ var require_ieee754 = __commonJS({
+ "node_modules/ieee754/index.js"(exports2) {
+ exports2.read = function(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 i2 = isLE ? nBytes - 1 : 0;
+ var d = isLE ? -1 : 1;
+ var s = buffer[offset + i2];
+ i2 += d;
+ e = s & (1 << -nBits) - 1;
+ s >>= -nBits;
+ nBits += eLen;
+ for (; nBits > 0; e = e * 256 + buffer[offset + i2], i2 += d, nBits -= 8) {
+ }
+ m = e & (1 << -nBits) - 1;
+ e >>= -nBits;
+ nBits += mLen;
+ for (; nBits > 0; m = m * 256 + buffer[offset + i2], i2 += 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;
+ }
+ return (s ? -1 : 1) * m * Math.pow(2, e - mLen);
+ };
+ exports2.write = function(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 i2 = 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);
+ if (value * (c = Math.pow(2, -e)) < 1) {
+ e--;
+ c *= 2;
+ }
+ if (e + eBias >= 1) {
+ value += rt / c;
+ } else {
+ value += rt * Math.pow(2, 1 - eBias);
+ }
+ if (value * c >= 2) {
+ e++;
+ c /= 2;
+ }
+ 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 + i2] = m & 255, i2 += d, m /= 256, mLen -= 8) {
+ }
+ e = e << mLen | m;
+ eLen += mLen;
+ for (; eLen > 0; buffer[offset + i2] = e & 255, i2 += d, e /= 256, eLen -= 8) {
+ }
+ buffer[offset + i2 - d] |= s * 128;
+ };
+ }
+ });
+
+ // node_modules/pbf/index.js
+ var require_pbf = __commonJS({
+ "node_modules/pbf/index.js"(exports2, module2) {
+ "use strict";
+ module2.exports = Pbf;
+ var ieee754 = require_ieee754();
+ 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;
+ }
+ Pbf.Varint = 0;
+ Pbf.Fixed64 = 1;
+ Pbf.Bytes = 2;
+ Pbf.Fixed32 = 5;
+ var SHIFT_LEFT_32 = (1 << 16) * (1 << 16);
+ var SHIFT_RIGHT_32 = 1 / SHIFT_LEFT_32;
+ var TEXT_DECODER_MIN_LENGTH = 12;
+ var utf8TextDecoder = typeof TextDecoder === "undefined" ? null : new TextDecoder("utf8");
+ Pbf.prototype = {
+ destroy: function() {
+ this.buf = null;
+ },
+ // === READING =================================================================
+ readFields: function(readField, result, end) {
+ end = end || this.length;
+ while (this.pos < end) {
+ var val = this.readVarint(), tag = val >> 3, startPos = this.pos;
+ this.type = val & 7;
+ readField(tag, result, this);
+ if (this.pos === startPos)
+ this.skip(val);
+ }
+ return result;
+ },
+ readMessage: function(readField, result) {
+ return this.readFields(readField, result, this.readVarint() + this.pos);
+ },
+ readFixed32: function() {
+ var val = readUInt32(this.buf, this.pos);
+ this.pos += 4;
+ return val;
+ },
+ readSFixed32: function() {
+ 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() {
+ var val = readUInt32(this.buf, this.pos) + readUInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;
+ this.pos += 8;
+ return val;
+ },
+ readSFixed64: function() {
+ var val = readUInt32(this.buf, this.pos) + readInt32(this.buf, this.pos + 4) * SHIFT_LEFT_32;
+ this.pos += 8;
+ return val;
+ },
+ readFloat: function() {
+ var val = ieee754.read(this.buf, this.pos, true, 23, 4);
+ this.pos += 4;
+ return val;
+ },
+ readDouble: function() {
+ var val = ieee754.read(this.buf, this.pos, true, 52, 8);
+ this.pos += 8;
+ return val;
+ },
+ readVarint: function(isSigned) {
+ var buf = this.buf, val, b;
+ b = buf[this.pos++];
+ val = b & 127;
+ if (b < 128)
+ return val;
+ b = buf[this.pos++];
+ val |= (b & 127) << 7;
+ if (b < 128)
+ return val;
+ b = buf[this.pos++];
+ val |= (b & 127) << 14;
+ if (b < 128)
+ return val;
+ b = buf[this.pos++];
+ val |= (b & 127) << 21;
+ if (b < 128)
+ return val;
+ b = buf[this.pos];
+ val |= (b & 15) << 28;
+ return readVarintRemainder(val, isSigned, this);
+ },
+ readVarint64: function() {
+ return this.readVarint(true);
+ },
+ readSVarint: function() {
+ var num = this.readVarint();
+ return num % 2 === 1 ? (num + 1) / -2 : num / 2;
+ },
+ readBoolean: function() {
+ return Boolean(this.readVarint());
+ },
+ readString: function() {
+ var end = this.readVarint() + this.pos;
+ var pos = this.pos;
+ this.pos = end;
+ if (end - pos >= TEXT_DECODER_MIN_LENGTH && utf8TextDecoder) {
+ return readUtf8TextDecoder(this.buf, pos, end);
+ }
+ return readUtf8(this.buf, pos, end);
+ },
+ readBytes: function() {
+ 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(arr, isSigned) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readVarint(isSigned));
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readVarint(isSigned));
+ return arr;
+ },
+ readPackedSVarint: function(arr) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readSVarint());
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readSVarint());
+ return arr;
+ },
+ readPackedBoolean: function(arr) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readBoolean());
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readBoolean());
+ return arr;
+ },
+ readPackedFloat: function(arr) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readFloat());
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readFloat());
+ return arr;
+ },
+ readPackedDouble: function(arr) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readDouble());
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readDouble());
+ return arr;
+ },
+ readPackedFixed32: function(arr) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readFixed32());
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readFixed32());
+ return arr;
+ },
+ readPackedSFixed32: function(arr) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readSFixed32());
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readSFixed32());
+ return arr;
+ },
+ readPackedFixed64: function(arr) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readFixed64());
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readFixed64());
+ return arr;
+ },
+ readPackedSFixed64: function(arr) {
+ if (this.type !== Pbf.Bytes)
+ return arr.push(this.readSFixed64());
+ var end = readPackedEnd(this);
+ arr = arr || [];
+ while (this.pos < end)
+ arr.push(this.readSFixed64());
+ return arr;
+ },
+ skip: function(val) {
+ var type2 = val & 7;
+ if (type2 === Pbf.Varint)
+ while (this.buf[this.pos++] > 127) {
+ }
+ else if (type2 === Pbf.Bytes)
+ this.pos = this.readVarint() + this.pos;
+ else if (type2 === Pbf.Fixed32)
+ this.pos += 4;
+ else if (type2 === Pbf.Fixed64)
+ this.pos += 8;
+ else
+ throw new Error("Unimplemented type: " + type2);
+ },
+ // === WRITING =================================================================
+ writeTag: function(tag, type2) {
+ this.writeVarint(tag << 3 | type2);
+ },
+ realloc: function(min3) {
+ var length = this.length || 16;
+ while (length < this.pos + min3)
+ length *= 2;
+ if (length !== this.length) {
+ var buf = new Uint8Array(length);
+ buf.set(this.buf);
+ this.buf = buf;
+ this.length = length;
+ }
+ },
+ finish: function() {
+ this.length = this.pos;
+ this.pos = 0;
+ return this.buf.subarray(0, this.length);
+ },
+ writeFixed32: function(val) {
+ this.realloc(4);
+ writeInt32(this.buf, val, this.pos);
+ this.pos += 4;
+ },
+ writeSFixed32: function(val) {
+ this.realloc(4);
+ writeInt32(this.buf, val, this.pos);
+ this.pos += 4;
+ },
+ writeFixed64: function(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(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(val) {
+ val = +val || 0;
+ if (val > 268435455 || val < 0) {
+ writeBigVarint(val, this);
+ return;
+ }
+ this.realloc(4);
+ this.buf[this.pos++] = val & 127 | (val > 127 ? 128 : 0);
+ if (val <= 127)
+ return;
+ this.buf[this.pos++] = (val >>>= 7) & 127 | (val > 127 ? 128 : 0);
+ if (val <= 127)
+ return;
+ this.buf[this.pos++] = (val >>>= 7) & 127 | (val > 127 ? 128 : 0);
+ if (val <= 127)
+ return;
+ this.buf[this.pos++] = val >>> 7 & 127;
+ },
+ writeSVarint: function(val) {
+ this.writeVarint(val < 0 ? -val * 2 - 1 : val * 2);
+ },
+ writeBoolean: function(val) {
+ this.writeVarint(Boolean(val));
+ },
+ writeString: function(str2) {
+ str2 = String(str2);
+ this.realloc(str2.length * 4);
+ this.pos++;
+ var startPos = this.pos;
+ this.pos = writeUtf8(this.buf, str2, this.pos);
+ var len = this.pos - startPos;
+ if (len >= 128)
+ makeRoomForExtraLength(startPos, len, this);
+ this.pos = startPos - 1;
+ this.writeVarint(len);
+ this.pos += len;
+ },
+ writeFloat: function(val) {
+ this.realloc(4);
+ ieee754.write(this.buf, val, this.pos, true, 23, 4);
+ this.pos += 4;
+ },
+ writeDouble: function(val) {
+ this.realloc(8);
+ ieee754.write(this.buf, val, this.pos, true, 52, 8);
+ this.pos += 8;
+ },
+ writeBytes: function(buffer) {
+ var len = buffer.length;
+ this.writeVarint(len);
+ this.realloc(len);
+ for (var i2 = 0; i2 < len; i2++)
+ this.buf[this.pos++] = buffer[i2];
+ },
+ writeRawMessage: function(fn, obj) {
+ this.pos++;
+ var startPos = this.pos;
+ fn(obj, this);
+ var len = this.pos - startPos;
+ if (len >= 128)
+ makeRoomForExtraLength(startPos, len, this);
+ this.pos = startPos - 1;
+ this.writeVarint(len);
+ this.pos += len;
+ },
+ writeMessage: function(tag, fn, obj) {
+ this.writeTag(tag, Pbf.Bytes);
+ this.writeRawMessage(fn, obj);
+ },
+ writePackedVarint: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedVarint, arr);
+ },
+ writePackedSVarint: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedSVarint, arr);
+ },
+ writePackedBoolean: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedBoolean, arr);
+ },
+ writePackedFloat: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedFloat, arr);
+ },
+ writePackedDouble: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedDouble, arr);
+ },
+ writePackedFixed32: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedFixed32, arr);
+ },
+ writePackedSFixed32: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedSFixed32, arr);
+ },
+ writePackedFixed64: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedFixed64, arr);
+ },
+ writePackedSFixed64: function(tag, arr) {
+ if (arr.length)
+ this.writeMessage(tag, writePackedSFixed64, arr);
+ },
+ writeBytesField: function(tag, buffer) {
+ this.writeTag(tag, Pbf.Bytes);
+ this.writeBytes(buffer);
+ },
+ writeFixed32Field: function(tag, val) {
+ this.writeTag(tag, Pbf.Fixed32);
+ this.writeFixed32(val);
+ },
+ writeSFixed32Field: function(tag, val) {
+ this.writeTag(tag, Pbf.Fixed32);
+ this.writeSFixed32(val);
+ },
+ writeFixed64Field: function(tag, val) {
+ this.writeTag(tag, Pbf.Fixed64);
+ this.writeFixed64(val);
+ },
+ writeSFixed64Field: function(tag, val) {
+ this.writeTag(tag, Pbf.Fixed64);
+ this.writeSFixed64(val);
+ },
+ writeVarintField: function(tag, val) {
+ this.writeTag(tag, Pbf.Varint);
+ this.writeVarint(val);
+ },
+ writeSVarintField: function(tag, val) {
+ this.writeTag(tag, Pbf.Varint);
+ this.writeSVarint(val);
+ },
+ writeStringField: function(tag, str2) {
+ this.writeTag(tag, Pbf.Bytes);
+ this.writeString(str2);
+ },
+ writeFloatField: function(tag, val) {
+ this.writeTag(tag, Pbf.Fixed32);
+ this.writeFloat(val);
+ },
+ writeDoubleField: function(tag, val) {
+ this.writeTag(tag, Pbf.Fixed64);
+ this.writeDouble(val);
+ },
+ writeBooleanField: function(tag, val) {
+ this.writeVarintField(tag, Boolean(val));
+ }
+ };
+ function readVarintRemainder(l, s, p) {
+ var buf = p.buf, h, b;
+ b = buf[p.pos++];
+ h = (b & 112) >> 4;
+ if (b < 128)
+ return toNum(l, h, s);
+ b = buf[p.pos++];
+ h |= (b & 127) << 3;
+ if (b < 128)
+ return toNum(l, h, s);
+ b = buf[p.pos++];
+ h |= (b & 127) << 10;
+ if (b < 128)
+ return toNum(l, h, s);
+ b = buf[p.pos++];
+ h |= (b & 127) << 17;
+ if (b < 128)
+ return toNum(l, h, s);
+ b = buf[p.pos++];
+ h |= (b & 127) << 24;
+ if (b < 128)
+ return toNum(l, h, s);
+ b = buf[p.pos++];
+ h |= (b & 1) << 31;
+ if (b < 128)
+ return toNum(l, h, s);
+ throw new Error("Expected varint not more than 10 bytes");
+ }
+ function readPackedEnd(pbf) {
+ return pbf.type === Pbf.Bytes ? pbf.readVarint() + pbf.pos : pbf.pos + 1;
+ }
+ function toNum(low, high, isSigned) {
+ if (isSigned) {
+ return high * 4294967296 + (low >>> 0);
+ }
+ return (high >>> 0) * 4294967296 + (low >>> 0);
+ }
+ function writeBigVarint(val, pbf) {
+ var low, high;
+ if (val >= 0) {
+ low = val % 4294967296 | 0;
+ high = val / 4294967296 | 0;
+ } else {
+ low = ~(-val % 4294967296);
+ high = ~(-val / 4294967296);
+ if (low ^ 4294967295) {
+ low = low + 1 | 0;
+ } else {
+ low = 0;
+ high = high + 1 | 0;
+ }
+ }
+ if (val >= 18446744073709552e3 || val < -18446744073709552e3) {
+ throw new Error("Given varint doesn't fit into 10 bytes");
+ }
+ pbf.realloc(10);
+ writeBigVarintLow(low, high, pbf);
+ writeBigVarintHigh(high, pbf);
+ }
+ function writeBigVarintLow(low, high, pbf) {
+ pbf.buf[pbf.pos++] = low & 127 | 128;
+ low >>>= 7;
+ pbf.buf[pbf.pos++] = low & 127 | 128;
+ low >>>= 7;
+ pbf.buf[pbf.pos++] = low & 127 | 128;
+ low >>>= 7;
+ pbf.buf[pbf.pos++] = low & 127 | 128;
+ low >>>= 7;
+ pbf.buf[pbf.pos] = low & 127;
+ }
+ function writeBigVarintHigh(high, pbf) {
+ var lsb = (high & 7) << 4;
+ pbf.buf[pbf.pos++] |= lsb | ((high >>>= 3) ? 128 : 0);
+ if (!high)
+ return;
+ pbf.buf[pbf.pos++] = high & 127 | ((high >>>= 7) ? 128 : 0);
+ if (!high)
+ return;
+ pbf.buf[pbf.pos++] = high & 127 | ((high >>>= 7) ? 128 : 0);
+ if (!high)
+ return;
+ pbf.buf[pbf.pos++] = high & 127 | ((high >>>= 7) ? 128 : 0);
+ if (!high)
+ return;
+ pbf.buf[pbf.pos++] = high & 127 | ((high >>>= 7) ? 128 : 0);
+ if (!high)
+ return;
+ pbf.buf[pbf.pos++] = high & 127;
+ }
+ function makeRoomForExtraLength(startPos, len, pbf) {
+ var extraLen = len <= 16383 ? 1 : len <= 2097151 ? 2 : len <= 268435455 ? 3 : Math.floor(Math.log(len) / (Math.LN2 * 7));
+ pbf.realloc(extraLen);
+ for (var i2 = pbf.pos - 1; i2 >= startPos; i2--)
+ pbf.buf[i2 + extraLen] = pbf.buf[i2];
+ }
+ function writePackedVarint(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeVarint(arr[i2]);
+ }
+ function writePackedSVarint(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeSVarint(arr[i2]);
+ }
+ function writePackedFloat(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeFloat(arr[i2]);
+ }
+ function writePackedDouble(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeDouble(arr[i2]);
+ }
+ function writePackedBoolean(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeBoolean(arr[i2]);
+ }
+ function writePackedFixed32(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeFixed32(arr[i2]);
+ }
+ function writePackedSFixed32(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeSFixed32(arr[i2]);
+ }
+ function writePackedFixed64(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeFixed64(arr[i2]);
+ }
+ function writePackedSFixed64(arr, pbf) {
+ for (var i2 = 0; i2 < arr.length; i2++)
+ pbf.writeSFixed64(arr[i2]);
+ }
+ function readUInt32(buf, pos) {
+ return (buf[pos] | buf[pos + 1] << 8 | buf[pos + 2] << 16) + buf[pos + 3] * 16777216;
+ }
+ function writeInt32(buf, val, pos) {
+ buf[pos] = val;
+ buf[pos + 1] = val >>> 8;
+ buf[pos + 2] = val >>> 16;
+ buf[pos + 3] = val >>> 24;
+ }
+ function readInt32(buf, pos) {
+ return (buf[pos] | buf[pos + 1] << 8 | buf[pos + 2] << 16) + (buf[pos + 3] << 24);
+ }
+ function readUtf8(buf, pos, end) {
+ var str2 = "";
+ var i2 = pos;
+ while (i2 < end) {
+ var b0 = buf[i2];
+ var c = null;
+ var bytesPerSequence = b0 > 239 ? 4 : b0 > 223 ? 3 : b0 > 191 ? 2 : 1;
+ if (i2 + bytesPerSequence > end)
+ break;
+ var b1, b2, b3;
+ if (bytesPerSequence === 1) {
+ if (b0 < 128) {
+ c = b0;
+ }
+ } else if (bytesPerSequence === 2) {
+ b1 = buf[i2 + 1];
+ if ((b1 & 192) === 128) {
+ c = (b0 & 31) << 6 | b1 & 63;
+ if (c <= 127) {
+ c = null;
+ }
+ }
+ } else if (bytesPerSequence === 3) {
+ b1 = buf[i2 + 1];
+ b2 = buf[i2 + 2];
+ if ((b1 & 192) === 128 && (b2 & 192) === 128) {
+ c = (b0 & 15) << 12 | (b1 & 63) << 6 | b2 & 63;
+ if (c <= 2047 || c >= 55296 && c <= 57343) {
+ c = null;
+ }
+ }
+ } else if (bytesPerSequence === 4) {
+ b1 = buf[i2 + 1];
+ b2 = buf[i2 + 2];
+ b3 = buf[i2 + 3];
+ if ((b1 & 192) === 128 && (b2 & 192) === 128 && (b3 & 192) === 128) {
+ c = (b0 & 15) << 18 | (b1 & 63) << 12 | (b2 & 63) << 6 | b3 & 63;
+ if (c <= 65535 || c >= 1114112) {
+ c = null;
+ }
+ }
+ }
+ if (c === null) {
+ c = 65533;
+ bytesPerSequence = 1;
+ } else if (c > 65535) {
+ c -= 65536;
+ str2 += String.fromCharCode(c >>> 10 & 1023 | 55296);
+ c = 56320 | c & 1023;
+ }
+ str2 += String.fromCharCode(c);
+ i2 += bytesPerSequence;
+ }
+ return str2;
+ }
+ function readUtf8TextDecoder(buf, pos, end) {
+ return utf8TextDecoder.decode(buf.subarray(pos, end));
+ }
+ function writeUtf8(buf, str2, pos) {
+ for (var i2 = 0, c, lead; i2 < str2.length; i2++) {
+ c = str2.charCodeAt(i2);
+ if (c > 55295 && c < 57344) {
+ if (lead) {
+ if (c < 56320) {
+ buf[pos++] = 239;
+ buf[pos++] = 191;
+ buf[pos++] = 189;
+ lead = c;
+ continue;
+ } else {
+ c = lead - 55296 << 10 | c - 56320 | 65536;
+ lead = null;
+ }
+ } else {
+ if (c > 56319 || i2 + 1 === str2.length) {
+ buf[pos++] = 239;
+ buf[pos++] = 191;
+ buf[pos++] = 189;
+ } else {
+ lead = c;
+ }
+ continue;
+ }
+ } else if (lead) {
+ buf[pos++] = 239;
+ buf[pos++] = 191;
+ buf[pos++] = 189;
+ lead = null;
+ }
+ if (c < 128) {
+ buf[pos++] = c;
+ } else {
+ if (c < 2048) {
+ buf[pos++] = c >> 6 | 192;
+ } else {
+ if (c < 65536) {
+ buf[pos++] = c >> 12 | 224;
+ } else {
+ buf[pos++] = c >> 18 | 240;
+ buf[pos++] = c >> 12 & 63 | 128;
+ }
+ buf[pos++] = c >> 6 & 63 | 128;
+ }
+ buf[pos++] = c & 63 | 128;
+ }
+ }
+ return pos;
+ }
+ }
+ });
+
+ // node_modules/@mapbox/point-geometry/index.js
+ var require_point_geometry = __commonJS({
+ "node_modules/@mapbox/point-geometry/index.js"(exports2, module2) {
+ "use strict";
+ module2.exports = Point;
+ function Point(x, y) {
+ this.x = x;
+ this.y = y;
+ }
+ Point.prototype = {
+ /**
+ * Clone this point, returning a new point that can be modified
+ * without affecting the old one.
+ * @return {Point} the clone
+ */
+ clone: function() {
+ return new Point(this.x, this.y);
+ },
+ /**
+ * 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(p) {
+ return this.clone()._add(p);
+ },
+ /**
+ * 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(p) {
+ return this.clone()._sub(p);
+ },
+ /**
+ * 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(p) {
+ return this.clone()._multByPoint(p);
+ },
+ /**
+ * 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(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(k) {
+ return this.clone()._mult(k);
+ },
+ /**
+ * Divide this point's x & y coordinates by a factor,
+ * yielding a new point.
+ * @param {Point} k factor
+ * @return {Point} output point
+ */
+ div: function(k) {
+ return this.clone()._div(k);
+ },
+ /**
+ * 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(a) {
+ return this.clone()._rotate(a);
+ },
+ /**
+ * 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(a, p) {
+ return this.clone()._rotateAround(a, p);
+ },
+ /**
+ * Multiply this point by a 4x1 transformation matrix
+ * @param {Array<Number>} m transformation matrix
+ * @return {Point} output point
+ */
+ matMult: function(m) {
+ return this.clone()._matMult(m);
+ },
+ /**
+ * 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() {
+ return this.clone()._unit();
+ },
+ /**
+ * 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() {
+ return this.clone()._perp();
+ },
+ /**
+ * Return a version of this point with the x & y coordinates
+ * rounded to integers.
+ * @return {Point} rounded point
+ */
+ round: function() {
+ return this.clone()._round();
+ },
+ /**
+ * 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() {
+ return Math.sqrt(this.x * this.x + this.y * this.y);
+ },
+ /**
+ * 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(other) {
+ return this.x === other.x && this.y === other.y;
+ },
+ /**
+ * Calculate the distance from this point to another point
+ * @param {Point} p the other point
+ * @return {Number} distance
+ */
+ dist: function(p) {
+ return Math.sqrt(this.distSqr(p));
+ },
+ /**
+ * 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(p) {
+ var dx = p.x - this.x, dy = p.y - this.y;
+ return dx * dx + dy * dy;
+ },
+ /**
+ * Get the angle from the 0, 0 coordinate to this point, in radians
+ * coordinates.
+ * @return {Number} angle
+ */
+ angle: function() {
+ return Math.atan2(this.y, this.x);
+ },
+ /**
+ * Get the angle from this point to another point, in radians
+ * @param {Point} b the other point
+ * @return {Number} angle
+ */
+ angleTo: function(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(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(x, y) {
+ return Math.atan2(
+ this.x * y - this.y * x,
+ this.x * x + this.y * y
+ );
+ },
+ _matMult: function(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(p) {
+ this.x += p.x;
+ this.y += p.y;
+ return this;
+ },
+ _sub: function(p) {
+ this.x -= p.x;
+ this.y -= p.y;
+ return this;
+ },
+ _mult: function(k) {
+ this.x *= k;
+ this.y *= k;
+ return this;
+ },
+ _div: function(k) {
+ this.x /= k;
+ this.y /= k;
+ return this;
+ },
+ _multByPoint: function(p) {
+ this.x *= p.x;
+ this.y *= p.y;
+ return this;
+ },
+ _divByPoint: function(p) {
+ this.x /= p.x;
+ this.y /= p.y;
+ return this;
+ },
+ _unit: function() {
+ this._div(this.mag());
+ return this;
+ },
+ _perp: function() {
+ var y = this.y;
+ this.y = this.x;
+ this.x = -y;
+ return this;
+ },
+ _rotate: function(angle2) {
+ var cos2 = Math.cos(angle2), sin2 = Math.sin(angle2), x = cos2 * this.x - sin2 * this.y, y = sin2 * this.x + cos2 * this.y;
+ this.x = x;
+ this.y = y;
+ return this;
+ },
+ _rotateAround: function(angle2, p) {
+ var cos2 = Math.cos(angle2), sin2 = Math.sin(angle2), x = p.x + cos2 * (this.x - p.x) - sin2 * (this.y - p.y), y = p.y + sin2 * (this.x - p.x) + cos2 * (this.y - p.y);
+ this.x = x;
+ this.y = y;
+ return this;
+ },
+ _round: function() {
+ this.x = Math.round(this.x);
+ this.y = Math.round(this.y);
+ return this;
+ }
+ };
+ Point.convert = function(a) {
+ if (a instanceof Point) {
+ return a;
+ }
+ if (Array.isArray(a)) {
+ return new Point(a[0], a[1]);
+ }
+ return a;
+ };
+ }
+ });
+
+ // node_modules/@mapbox/vector-tile/lib/vectortilefeature.js
+ var require_vectortilefeature = __commonJS({
+ "node_modules/@mapbox/vector-tile/lib/vectortilefeature.js"(exports2, module2) {
+ "use strict";
+ var Point = require_point_geometry();
+ module2.exports = VectorTileFeature;
+ function VectorTileFeature(pbf, end, extent, keys, values) {
+ this.properties = {};
+ this.extent = extent;
+ this.type = 0;
+ this._pbf = pbf;
+ this._geometry = -1;
+ this._keys = keys;
+ this._values = values;
+ pbf.readFields(readFeature, this, end);
+ }
+ function readFeature(tag, feature3, pbf) {
+ if (tag == 1)
+ feature3.id = pbf.readVarint();
+ else if (tag == 2)
+ readTag(pbf, feature3);
+ else if (tag == 3)
+ feature3.type = pbf.readVarint();
+ else if (tag == 4)
+ feature3._geometry = pbf.pos;
+ }
+ function readTag(pbf, feature3) {
+ var end = pbf.readVarint() + pbf.pos;
+ while (pbf.pos < end) {
+ var key = feature3._keys[pbf.readVarint()], value = feature3._values[pbf.readVarint()];
+ feature3.properties[key] = value;
+ }
+ }
+ VectorTileFeature.types = ["Unknown", "Point", "LineString", "Polygon"];
+ 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 & 7;
+ length = cmdLen >> 3;
+ }
+ length--;
+ if (cmd === 1 || cmd === 2) {
+ x += pbf.readSVarint();
+ y += pbf.readSVarint();
+ if (cmd === 1) {
+ if (line)
+ lines.push(line);
+ line = [];
+ }
+ line.push(new Point(x, y));
+ } else if (cmd === 7) {
+ if (line) {
+ line.push(line[0].clone());
+ }
+ } else {
+ throw new Error("unknown command " + cmd);
+ }
+ }
+ if (line)
+ lines.push(line);
+ return lines;
+ };
+ 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, x12 = Infinity, x2 = -Infinity, y12 = Infinity, y2 = -Infinity;
+ while (pbf.pos < end) {
+ if (length <= 0) {
+ var cmdLen = pbf.readVarint();
+ cmd = cmdLen & 7;
+ length = cmdLen >> 3;
+ }
+ length--;
+ if (cmd === 1 || cmd === 2) {
+ x += pbf.readSVarint();
+ y += pbf.readSVarint();
+ if (x < x12)
+ x12 = x;
+ if (x > x2)
+ x2 = x;
+ if (y < y12)
+ y12 = y;
+ if (y > y2)
+ y2 = y;
+ } else if (cmd !== 7) {
+ throw new Error("unknown command " + cmd);
+ }
+ }
+ return [x12, y12, x2, y2];
+ };
+ VectorTileFeature.prototype.toGeoJSON = function(x, y, z) {
+ var size = this.extent * Math.pow(2, z), x05 = this.extent * x, y05 = this.extent * y, coords = this.loadGeometry(), type2 = VectorTileFeature.types[this.type], i2, j2;
+ function project(line) {
+ for (var j3 = 0; j3 < line.length; j3++) {
+ var p = line[j3], y2 = 180 - (p.y + y05) * 360 / size;
+ line[j3] = [
+ (p.x + x05) * 360 / size - 180,
+ 360 / Math.PI * Math.atan(Math.exp(y2 * Math.PI / 180)) - 90
+ ];
+ }
+ }
+ switch (this.type) {
+ case 1:
+ var points = [];
+ for (i2 = 0; i2 < coords.length; i2++) {
+ points[i2] = coords[i2][0];
+ }
+ coords = points;
+ project(coords);
+ break;
+ case 2:
+ for (i2 = 0; i2 < coords.length; i2++) {
+ project(coords[i2]);
+ }
+ break;
+ case 3:
+ coords = classifyRings(coords);
+ for (i2 = 0; i2 < coords.length; i2++) {
+ for (j2 = 0; j2 < coords[i2].length; j2++) {
+ project(coords[i2][j2]);
+ }
+ }
+ break;
+ }
+ if (coords.length === 1) {
+ coords = coords[0];
+ } else {
+ type2 = "Multi" + type2;
+ }
+ var result = {
+ type: "Feature",
+ geometry: {
+ type: type2,
+ coordinates: coords
+ },
+ properties: this.properties
+ };
+ if ("id" in this) {
+ result.id = this.id;
+ }
+ return result;
+ };
+ function classifyRings(rings) {
+ var len = rings.length;
+ if (len <= 1)
+ return [rings];
+ var polygons = [], polygon2, ccw;
+ for (var i2 = 0; i2 < len; i2++) {
+ var area = signedArea(rings[i2]);
+ if (area === 0)
+ continue;
+ if (ccw === void 0)
+ ccw = area < 0;
+ if (ccw === area < 0) {
+ if (polygon2)
+ polygons.push(polygon2);
+ polygon2 = [rings[i2]];
+ } else {
+ polygon2.push(rings[i2]);
+ }
+ }
+ if (polygon2)
+ polygons.push(polygon2);
+ return polygons;
+ }
+ function signedArea(ring) {
+ var sum = 0;
+ for (var i2 = 0, len = ring.length, j2 = len - 1, p1, p2; i2 < len; j2 = i2++) {
+ p1 = ring[i2];
+ p2 = ring[j2];
+ sum += (p2.x - p1.x) * (p1.y + p2.y);
+ }
+ return sum;
+ }
+ }
+ });
+
+ // node_modules/@mapbox/vector-tile/lib/vectortilelayer.js
+ var require_vectortilelayer = __commonJS({
+ "node_modules/@mapbox/vector-tile/lib/vectortilelayer.js"(exports2, module2) {
+ "use strict";
+ var VectorTileFeature = require_vectortilefeature();
+ module2.exports = VectorTileLayer;
+ function VectorTileLayer(pbf, end) {
+ this.version = 1;
+ this.name = null;
+ this.extent = 4096;
+ this.length = 0;
+ this._pbf = pbf;
+ this._keys = [];
+ this._values = [];
+ this._features = [];
+ pbf.readFields(readLayer, this, end);
+ this.length = this._features.length;
+ }
+ 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 readValueMessage(pbf) {
+ var value = null, end = pbf.readVarint() + pbf.pos;
+ 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;
+ }
+ return value;
+ }
+ VectorTileLayer.prototype.feature = function(i2) {
+ if (i2 < 0 || i2 >= this._features.length)
+ throw new Error("feature index out of bounds");
+ this._pbf.pos = this._features[i2];
+ var end = this._pbf.readVarint() + this._pbf.pos;
+ return new VectorTileFeature(this._pbf, end, this.extent, this._keys, this._values);
+ };
+ }
+ });
+
+ // node_modules/@mapbox/vector-tile/lib/vectortile.js
+ var require_vectortile = __commonJS({
+ "node_modules/@mapbox/vector-tile/lib/vectortile.js"(exports2, module2) {
+ "use strict";
+ var VectorTileLayer = require_vectortilelayer();
+ module2.exports = VectorTile2;
+ function VectorTile2(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;
+ }
+ }
+ }
+ });
+
+ // node_modules/@mapbox/vector-tile/index.js
+ var require_vector_tile = __commonJS({
+ "node_modules/@mapbox/vector-tile/index.js"(exports2, module2) {
+ module2.exports.VectorTile = require_vectortile();
+ module2.exports.VectorTileFeature = require_vectortilefeature();
+ module2.exports.VectorTileLayer = require_vectortilelayer();
+ }
+ });
+
+ // node_modules/fast-json-stable-stringify/index.js
+ var require_fast_json_stable_stringify = __commonJS({
+ "node_modules/fast-json-stable-stringify/index.js"(exports2, module2) {
+ "use strict";
+ module2.exports = function(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(f2) {
+ return function(node) {
+ return function(a, b) {
+ var aobj = { key: a, value: node[a] };
+ var bobj = { key: b, value: node[b] };
+ return f2(aobj, bobj);
+ };
+ };
+ }(opts.cmp);
+ var seen = [];
+ return function stringify3(node) {
+ if (node && node.toJSON && typeof node.toJSON === "function") {
+ node = node.toJSON();
+ }
+ if (node === void 0)
+ return;
+ if (typeof node == "number")
+ return isFinite(node) ? "" + node : "null";
+ if (typeof node !== "object")
+ return JSON.stringify(node);
+ var i2, out;
+ if (Array.isArray(node)) {
+ out = "[";
+ for (i2 = 0; i2 < node.length; i2++) {
+ if (i2)
+ out += ",";
+ out += stringify3(node[i2]) || "null";
+ }
+ return out + "]";
+ }
+ if (node === null)
+ return "null";
+ if (seen.indexOf(node) !== -1) {
+ if (cycles)
+ return JSON.stringify("__cycle__");
+ throw new TypeError("Converting circular structure to JSON");
+ }
+ var seenIndex = seen.push(node) - 1;
+ var keys = Object.keys(node).sort(cmp && cmp(node));
+ out = "";
+ for (i2 = 0; i2 < keys.length; i2++) {
+ var key = keys[i2];
+ var value = stringify3(node[key]);
+ if (!value)
+ continue;
+ if (out)
+ out += ",";
+ out += JSON.stringify(key) + ":" + value;
+ }
+ seen.splice(seenIndex, 1);
+ return "{" + out + "}";
+ }(data);
+ };
+ }
+ });
+
+ // node_modules/@mapbox/sexagesimal/index.js
+ var require_sexagesimal = __commonJS({
+ "node_modules/@mapbox/sexagesimal/index.js"(exports2, module2) {
+ module2.exports = element;
+ module2.exports.pair = pair2;
+ module2.exports.format = format2;
+ module2.exports.formatPair = formatPair;
+ module2.exports.coordToDMS = coordToDMS;
+ function element(input, dims) {
+ var result = search(input, dims);
+ return result === null ? null : result.val;
+ }
+ function formatPair(input) {
+ return format2(input.lat, "lat") + " " + format2(input.lon, "lon");
+ }
+ function format2(input, dim) {
+ var dms = coordToDMS(input, dim);
+ return dms.whole + "\xB0 " + (dms.minutes ? dms.minutes + "' " : "") + (dms.seconds ? dms.seconds + '" ' : "") + dms.dir;
+ }
+ function coordToDMS(input, dim) {
+ var dirs = { lat: ["N", "S"], lon: ["E", "W"] }[dim] || "";
+ var dir = dirs[input >= 0 ? 0 : 1];
+ var abs3 = Math.abs(input);
+ var whole = Math.floor(abs3);
+ var fraction = abs3 - whole;
+ var fractionMinutes = fraction * 60;
+ var minutes = Math.floor(fractionMinutes);
+ var seconds = Math.floor((fractionMinutes - minutes) * 60);
+ return {
+ whole,
+ minutes,
+ seconds,
+ dir
+ };
+ }
+ 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;
+ var matched = m[0];
+ var dim;
+ if (m[1] && m[5]) {
+ dim = m[1];
+ matched = matched.slice(0, -1);
+ } else {
+ dim = m[1] || m[5];
+ }
+ if (dim && dims.indexOf(dim) === -1)
+ return null;
+ var deg = m[2] ? parseFloat(m[2]) : 0;
+ var min3 = m[3] ? parseFloat(m[3]) / 60 : 0;
+ var sec = m[4] ? parseFloat(m[4]) / 3600 : 0;
+ var sign2 = deg < 0 ? -1 : 1;
+ if (dim === "S" || dim === "W")
+ sign2 *= -1;
+ return {
+ val: (Math.abs(deg) + min3 + sec) * sign2,
+ dim,
+ matched,
+ remain: input.slice(matched.length)
+ };
+ }
+ function pair2(input, dims) {
+ input = input.trim();
+ var one2 = search(input, dims);
+ if (!one2)
+ return null;
+ input = one2.remain.trim();
+ var two = search(input, dims);
+ if (!two || two.remain)
+ return null;
+ if (one2.dim) {
+ return swapdim(one2.val, two.val, one2.dim);
+ } else {
+ return [one2.val, two.val];
+ }
+ }
+ function swapdim(a, b, dim) {
+ if (dim === "N" || dim === "S")
+ return [a, b];
+ if (dim === "W" || dim === "E")
+ return [b, a];
+ }
+ }
+ });
+
+ // node_modules/store/src/util.js
+ var require_util = __commonJS({
+ "node_modules/store/src/util.js"(exports2, module2) {
+ var assign = make_assign();
+ var create2 = make_create();
+ var trim = make_trim();
+ var Global = typeof window !== "undefined" ? window : global;
+ module2.exports = {
+ assign,
+ create: create2,
+ trim,
+ bind,
+ slice,
+ each,
+ map: map2,
+ pluck,
+ isList,
+ isFunction,
+ isObject: isObject3,
+ Global
+ };
+ function make_assign() {
+ if (Object.assign) {
+ return Object.assign;
+ } else {
+ return function shimAssign(obj, props1, props2, etc) {
+ for (var i2 = 1; i2 < arguments.length; i2++) {
+ each(Object(arguments[i2]), function(val, key) {
+ obj[key] = val;
+ });
+ }
+ return obj;
+ };
+ }
+ }
+ function make_create() {
+ if (Object.create) {
+ return function create3(obj, assignProps1, assignProps2, etc) {
+ var assignArgsList = slice(arguments, 1);
+ return assign.apply(this, [Object.create(obj)].concat(assignArgsList));
+ };
+ } else {
+ let F2 = function() {
+ };
+ var F = F2;
+ return function create3(obj, assignProps1, assignProps2, etc) {
+ var assignArgsList = slice(arguments, 1);
+ F2.prototype = obj;
+ return assign.apply(this, [new F2()].concat(assignArgsList));
+ };
+ }
+ }
+ function make_trim() {
+ if (String.prototype.trim) {
+ return function trim2(str2) {
+ return String.prototype.trim.call(str2);
+ };
+ } else {
+ return function trim2(str2) {
+ return str2.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, "");
+ };
+ }
+ }
+ function bind(obj, fn) {
+ return function() {
+ return fn.apply(obj, Array.prototype.slice.call(arguments, 0));
+ };
+ }
+ function slice(arr, index) {
+ return Array.prototype.slice.call(arr, index || 0);
+ }
+ function each(obj, fn) {
+ pluck(obj, function(val, key) {
+ fn(val, key);
+ return false;
+ });
+ }
+ function map2(obj, fn) {
+ var res = isList(obj) ? [] : {};
+ pluck(obj, function(v, k) {
+ res[k] = fn(v, k);
+ return false;
+ });
+ return res;
+ }
+ function pluck(obj, fn) {
+ if (isList(obj)) {
+ for (var i2 = 0; i2 < obj.length; i2++) {
+ if (fn(obj[i2], i2)) {
+ return obj[i2];
+ }
+ }
+ } else {
+ for (var key in obj) {
+ if (obj.hasOwnProperty(key)) {
+ if (fn(obj[key], key)) {
+ return obj[key];
+ }
+ }
+ }
+ }
+ }
+ function isList(val) {
+ return val != null && typeof val != "function" && typeof val.length == "number";
+ }
+ function isFunction(val) {
+ return val && {}.toString.call(val) === "[object Function]";
+ }
+ function isObject3(val) {
+ return val && {}.toString.call(val) === "[object Object]";
+ }
+ }
+ });
+
+ // node_modules/store/src/store-engine.js
+ var require_store_engine = __commonJS({
+ "node_modules/store/src/store-engine.js"(exports2, module2) {
+ var util = require_util();
+ var slice = util.slice;
+ var pluck = util.pluck;
+ var each = util.each;
+ var bind = util.bind;
+ var create2 = util.create;
+ var isList = util.isList;
+ var isFunction = util.isFunction;
+ var isObject3 = util.isObject;
+ module2.exports = {
+ 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(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(key, value) {
+ if (value === void 0) {
+ return this.remove(key);
+ }
+ this.storage.write(this._namespacePrefix + key, this._serialize(value));
+ return value;
+ },
+ // remove deletes the key and value stored at the given key.
+ remove: function(key) {
+ this.storage.remove(this._namespacePrefix + key);
+ },
+ // each will call the given callback once for each key-value pair
+ // in this store.
+ each: function(callback) {
+ var self2 = this;
+ this.storage.each(function(val, namespacedKey) {
+ callback.call(self2, self2._deserialize(val), (namespacedKey || "").replace(self2._namespaceRegexp, ""));
+ });
+ },
+ // clearAll will remove all the stored key-value pairs in this store.
+ clearAll: function() {
+ this.storage.clearAll();
+ },
+ // additional functionality that can't live in plugins
+ // ---------------------------------------------------
+ // hasNamespace returns true if this store instance has the given namespace.
+ hasNamespace: function(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() {
+ return createStore.apply(this, arguments);
+ },
+ addPlugin: function(plugin) {
+ this._addPlugin(plugin);
+ },
+ namespace: function(namespace) {
+ return createStore(this.storage, this.plugins, namespace);
+ }
+ };
+ function _warn() {
+ var _console = typeof console == "undefined" ? null : console;
+ if (!_console) {
+ return;
+ }
+ var fn = _console.warn ? _console.warn : _console.log;
+ fn.apply(_console, arguments);
+ }
+ function createStore(storages, plugins, namespace) {
+ if (!namespace) {
+ namespace = "";
+ }
+ if (storages && !isList(storages)) {
+ storages = [storages];
+ }
+ if (plugins && !isList(plugins)) {
+ plugins = [plugins];
+ }
+ var namespacePrefix = namespace ? "__storejs_" + namespace + "_" : "";
+ var namespaceRegexp = namespace ? new RegExp("^" + namespacePrefix) : null;
+ var legalNamespaces = /^[a-zA-Z0-9_\-]*$/;
+ 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(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(pluginFnProp, propName) {
+ var oldFn = this[propName];
+ this[propName] = function pluginFn() {
+ var args = slice(arguments, 0);
+ var self2 = this;
+ function super_fn() {
+ if (!oldFn) {
+ return;
+ }
+ each(arguments, function(arg, i2) {
+ args[i2] = arg;
+ });
+ return oldFn.apply(self2, args);
+ }
+ var newFnArgs = [super_fn].concat(args);
+ return pluginFnProp.apply(self2, newFnArgs);
+ };
+ },
+ _serialize: function(obj) {
+ return JSON.stringify(obj);
+ },
+ _deserialize: function(strVal, defaultVal) {
+ if (!strVal) {
+ return defaultVal;
+ }
+ var val = "";
+ try {
+ val = JSON.parse(strVal);
+ } catch (e) {
+ val = strVal;
+ }
+ return val !== void 0 ? val : defaultVal;
+ },
+ _addStorage: function(storage) {
+ if (this.enabled) {
+ return;
+ }
+ if (this._testStorage(storage)) {
+ this.storage = storage;
+ this.enabled = true;
+ }
+ },
+ _addPlugin: function(plugin) {
+ var self2 = this;
+ if (isList(plugin)) {
+ each(plugin, function(plugin2) {
+ self2._addPlugin(plugin2);
+ });
+ return;
+ }
+ var seenPlugin = pluck(this.plugins, function(seenPlugin2) {
+ return plugin === seenPlugin2;
+ });
+ if (seenPlugin) {
+ return;
+ }
+ this.plugins.push(plugin);
+ if (!isFunction(plugin)) {
+ throw new Error("Plugins must be function values that return objects");
+ }
+ var pluginProperties = plugin.call(this);
+ if (!isObject3(pluginProperties)) {
+ throw new Error("Plugins must return an object of function properties");
+ }
+ each(pluginProperties, function(pluginFnProp, propName) {
+ if (!isFunction(pluginFnProp)) {
+ throw new Error("Bad plugin property: " + propName + " from plugin " + plugin.name + ". Plugins should only return functions.");
+ }
+ self2._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(storage) {
+ _warn("store.addStorage(storage) is deprecated. Use createStore([storages])");
+ this._addStorage(storage);
+ }
+ };
+ var store2 = create2(_privateStoreProps, storeAPI, {
+ plugins: []
+ });
+ store2.raw = {};
+ each(store2, function(prop, propName) {
+ if (isFunction(prop)) {
+ store2.raw[propName] = bind(store2, prop);
+ }
+ });
+ each(storages, function(storage) {
+ store2._addStorage(storage);
+ });
+ each(plugins, function(plugin) {
+ store2._addPlugin(plugin);
+ });
+ return store2;
+ }
+ }
+ });
+
+ // node_modules/store/storages/localStorage.js
+ var require_localStorage = __commonJS({
+ "node_modules/store/storages/localStorage.js"(exports2, module2) {
+ var util = require_util();
+ var Global = util.Global;
+ module2.exports = {
+ name: "localStorage",
+ read,
+ write,
+ each,
+ remove: remove2,
+ clearAll
+ };
+ function localStorage2() {
+ return Global.localStorage;
+ }
+ function read(key) {
+ return localStorage2().getItem(key);
+ }
+ function write(key, data) {
+ return localStorage2().setItem(key, data);
+ }
+ function each(fn) {
+ for (var i2 = localStorage2().length - 1; i2 >= 0; i2--) {
+ var key = localStorage2().key(i2);
+ fn(read(key), key);
+ }
+ }
+ function remove2(key) {
+ return localStorage2().removeItem(key);
+ }
+ function clearAll() {
+ return localStorage2().clear();
+ }
+ }
+ });
+
+ // node_modules/store/storages/oldFF-globalStorage.js
+ var require_oldFF_globalStorage = __commonJS({
+ "node_modules/store/storages/oldFF-globalStorage.js"(exports2, module2) {
+ var util = require_util();
+ var Global = util.Global;
+ module2.exports = {
+ name: "oldFF-globalStorage",
+ read,
+ write,
+ each,
+ remove: remove2,
+ clearAll
+ };
+ var globalStorage = Global.globalStorage;
+ function read(key) {
+ return globalStorage[key];
+ }
+ function write(key, data) {
+ globalStorage[key] = data;
+ }
+ function each(fn) {
+ for (var i2 = globalStorage.length - 1; i2 >= 0; i2--) {
+ var key = globalStorage.key(i2);
+ fn(globalStorage[key], key);
+ }
+ }
+ function remove2(key) {
+ return globalStorage.removeItem(key);
+ }
+ function clearAll() {
+ each(function(key, _) {
+ delete globalStorage[key];
+ });
+ }
+ }
+ });
+
+ // node_modules/store/storages/oldIE-userDataStorage.js
+ var require_oldIE_userDataStorage = __commonJS({
+ "node_modules/store/storages/oldIE-userDataStorage.js"(exports2, module2) {
+ var util = require_util();
+ var Global = util.Global;
+ module2.exports = {
+ name: "oldIE-userDataStorage",
+ write,
+ read,
+ each,
+ remove: remove2,
+ clearAll
+ };
+ var storageName = "storejs";
+ var doc = Global.document;
+ var _withStorageEl = _makeIEStorageElFunction();
+ var disable = (Global.navigator ? Global.navigator.userAgent : "").match(/ (MSIE 8|MSIE 9|MSIE 10)\./);
+ function write(unfixedKey, data) {
+ if (disable) {
+ return;
+ }
+ var fixedKey = fixKey(unfixedKey);
+ _withStorageEl(function(storageEl) {
+ storageEl.setAttribute(fixedKey, data);
+ storageEl.save(storageName);
+ });
+ }
+ function read(unfixedKey) {
+ if (disable) {
+ return;
+ }
+ var fixedKey = fixKey(unfixedKey);
+ var res = null;
+ _withStorageEl(function(storageEl) {
+ res = storageEl.getAttribute(fixedKey);
+ });
+ return res;
+ }
+ function each(callback) {
+ _withStorageEl(function(storageEl) {
+ var attributes = storageEl.XMLDocument.documentElement.attributes;
+ for (var i2 = attributes.length - 1; i2 >= 0; i2--) {
+ var attr = attributes[i2];
+ callback(storageEl.getAttribute(attr.name), attr.name);
+ }
+ });
+ }
+ function remove2(unfixedKey) {
+ var fixedKey = fixKey(unfixedKey);
+ _withStorageEl(function(storageEl) {
+ storageEl.removeAttribute(fixedKey);
+ storageEl.save(storageName);
+ });
+ }
+ function clearAll() {
+ _withStorageEl(function(storageEl) {
+ var attributes = storageEl.XMLDocument.documentElement.attributes;
+ storageEl.load(storageName);
+ for (var i2 = attributes.length - 1; i2 >= 0; i2--) {
+ storageEl.removeAttribute(attributes[i2].name);
+ }
+ storageEl.save(storageName);
+ });
+ }
+ var forbiddenCharsRegex = new RegExp("[!\"#$%&'()*+,/\\\\:;<=>?@[\\]^`{|}~]", "g");
+ function fixKey(key) {
+ return key.replace(/^\d/, "___$&").replace(forbiddenCharsRegex, "___");
+ }
+ function _makeIEStorageElFunction() {
+ if (!doc || !doc.documentElement || !doc.documentElement.addBehavior) {
+ return null;
+ }
+ var scriptTag = "script", storageOwner, storageContainer, storageEl;
+ try {
+ 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) {
+ storageEl = doc.createElement("div");
+ storageOwner = doc.body;
+ }
+ return function(storeFunction) {
+ var args = [].slice.call(arguments, 0);
+ args.unshift(storageEl);
+ storageOwner.appendChild(storageEl);
+ storageEl.addBehavior("#default#userData");
+ storageEl.load(storageName);
+ storeFunction.apply(this, args);
+ storageOwner.removeChild(storageEl);
+ return;
+ };
+ }
+ }
+ });
+
+ // node_modules/store/storages/cookieStorage.js
+ var require_cookieStorage = __commonJS({
+ "node_modules/store/storages/cookieStorage.js"(exports2, module2) {
+ var util = require_util();
+ var Global = util.Global;
+ var trim = util.trim;
+ module2.exports = {
+ name: "cookieStorage",
+ read,
+ write,
+ each,
+ remove: remove2,
+ clearAll
+ };
+ var doc = Global.document;
+ function read(key) {
+ if (!key || !_has(key)) {
+ return null;
+ }
+ var regexpStr = "(?:^|.*;\\s*)" + escape(key).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=\\s*((?:[^;](?!;))*[^;]?).*";
+ return unescape(doc.cookie.replace(new RegExp(regexpStr), "$1"));
+ }
+ function each(callback) {
+ var cookies = doc.cookie.split(/; ?/g);
+ for (var i2 = cookies.length - 1; i2 >= 0; i2--) {
+ if (!trim(cookies[i2])) {
+ continue;
+ }
+ var kvp = cookies[i2].split("=");
+ var key = unescape(kvp[0]);
+ var val = unescape(kvp[1]);
+ callback(val, key);
+ }
+ }
+ function write(key, data) {
+ if (!key) {
+ return;
+ }
+ doc.cookie = escape(key) + "=" + escape(data) + "; expires=Tue, 19 Jan 2038 03:14:07 GMT; path=/";
+ }
+ function remove2(key) {
+ if (!key || !_has(key)) {
+ return;
+ }
+ doc.cookie = escape(key) + "=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/";
+ }
+ function clearAll() {
+ each(function(_, key) {
+ remove2(key);
+ });
+ }
+ function _has(key) {
+ return new RegExp("(?:^|;\\s*)" + escape(key).replace(/[\-\.\+\*]/g, "\\$&") + "\\s*\\=").test(doc.cookie);
+ }
+ }
+ });
+
+ // node_modules/store/storages/sessionStorage.js
+ var require_sessionStorage = __commonJS({
+ "node_modules/store/storages/sessionStorage.js"(exports2, module2) {
+ var util = require_util();
+ var Global = util.Global;
+ module2.exports = {
+ name: "sessionStorage",
+ read,
+ write,
+ each,
+ remove: remove2,
+ clearAll
+ };
+ function sessionStorage() {
+ return Global.sessionStorage;
+ }
+ function read(key) {
+ return sessionStorage().getItem(key);
+ }
+ function write(key, data) {
+ return sessionStorage().setItem(key, data);
+ }
+ function each(fn) {
+ for (var i2 = sessionStorage().length - 1; i2 >= 0; i2--) {
+ var key = sessionStorage().key(i2);
+ fn(read(key), key);
+ }
+ }
+ function remove2(key) {
+ return sessionStorage().removeItem(key);
+ }
+ function clearAll() {
+ return sessionStorage().clear();
+ }
+ }
+ });
+
+ // node_modules/store/storages/memoryStorage.js
+ var require_memoryStorage = __commonJS({
+ "node_modules/store/storages/memoryStorage.js"(exports2, module2) {
+ module2.exports = {
+ name: "memoryStorage",
+ read,
+ write,
+ each,
+ remove: remove2,
+ clearAll
+ };
+ var memoryStorage = {};
+ function read(key) {
+ return memoryStorage[key];
+ }
+ function write(key, data) {
+ memoryStorage[key] = data;
+ }
+ function each(callback) {
+ for (var key in memoryStorage) {
+ if (memoryStorage.hasOwnProperty(key)) {
+ callback(memoryStorage[key], key);
+ }
+ }
+ }
+ function remove2(key) {
+ delete memoryStorage[key];
+ }
+ function clearAll(key) {
+ memoryStorage = {};
+ }
+ }
+ });
+
+ // node_modules/store/storages/all.js
+ var require_all = __commonJS({
+ "node_modules/store/storages/all.js"(exports2, module2) {
+ module2.exports = [
+ // Listed in order of usage preference
+ require_localStorage(),
+ require_oldFF_globalStorage(),
+ require_oldIE_userDataStorage(),
+ require_cookieStorage(),
+ require_sessionStorage(),
+ require_memoryStorage()
+ ];
+ }
+ });
+
+ // node_modules/store/plugins/lib/json2.js
+ var require_json2 = __commonJS({
+ "node_modules/store/plugins/lib/json2.js"(exports, module) {
+ if (typeof JSON !== "object") {
+ JSON = {};
+ }
+ (function() {
+ "use strict";
+ 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;
+ function f(n2) {
+ return n2 < 10 ? "0" + n2 : n2;
+ }
+ function this_value() {
+ return this.valueOf();
+ }
+ 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;
+ };
+ Boolean.prototype.toJSON = this_value;
+ Number.prototype.toJSON = this_value;
+ String.prototype.toJSON = this_value;
+ }
+ var gap;
+ var indent;
+ var meta;
+ var rep;
+ function quote(string) {
+ 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 + '"';
+ }
+ function str(key, holder) {
+ var i2;
+ var k;
+ var v;
+ var length;
+ var mind = gap;
+ var partial;
+ var value = holder[key];
+ if (value && typeof value === "object" && typeof value.toJSON === "function") {
+ value = value.toJSON(key);
+ }
+ if (typeof rep === "function") {
+ value = rep.call(holder, key, value);
+ }
+ switch (typeof value) {
+ case "string":
+ return quote(value);
+ case "number":
+ return isFinite(value) ? String(value) : "null";
+ case "boolean":
+ case "null":
+ return String(value);
+ case "object":
+ if (!value) {
+ return "null";
+ }
+ gap += indent;
+ partial = [];
+ if (Object.prototype.toString.apply(value) === "[object Array]") {
+ length = value.length;
+ for (i2 = 0; i2 < length; i2 += 1) {
+ partial[i2] = str(i2, value) || "null";
+ }
+ v = partial.length === 0 ? "[]" : gap ? "[\n" + gap + partial.join(",\n" + gap) + "\n" + mind + "]" : "[" + partial.join(",") + "]";
+ gap = mind;
+ return v;
+ }
+ if (rep && typeof rep === "object") {
+ length = rep.length;
+ for (i2 = 0; i2 < length; i2 += 1) {
+ if (typeof rep[i2] === "string") {
+ k = rep[i2];
+ v = str(k, value);
+ if (v) {
+ partial.push(quote(k) + (gap ? ": " : ":") + v);
+ }
+ }
+ }
+ } else {
+ for (k in value) {
+ if (Object.prototype.hasOwnProperty.call(value, k)) {
+ v = str(k, value);
+ if (v) {
+ partial.push(quote(k) + (gap ? ": " : ":") + v);
+ }
+ }
+ }
+ }
+ v = partial.length === 0 ? "{}" : gap ? "{\n" + gap + partial.join(",\n" + gap) + "\n" + mind + "}" : "{" + partial.join(",") + "}";
+ gap = mind;
+ return v;
+ }
+ }
+ if (typeof JSON.stringify !== "function") {
+ meta = {
+ // table of character substitutions
+ "\b": "\\b",
+ " ": "\\t",
+ "\n": "\\n",
+ "\f": "\\f",
+ "\r": "\\r",
+ '"': '\\"',
+ "\\": "\\\\"
+ };
+ JSON.stringify = function(value, replacer, space) {
+ var i2;
+ gap = "";
+ indent = "";
+ if (typeof space === "number") {
+ for (i2 = 0; i2 < space; i2 += 1) {
+ indent += " ";
+ }
+ } else if (typeof space === "string") {
+ indent = space;
+ }
+ rep = replacer;
+ if (replacer && typeof replacer !== "function" && (typeof replacer !== "object" || typeof replacer.length !== "number")) {
+ throw new Error("JSON.stringify");
+ }
+ return str("", { "": value });
+ };
+ }
+ if (typeof JSON.parse !== "function") {
+ JSON.parse = function(text, reviver) {
+ var j;
+ function walk(holder, key) {
+ var k;
+ var v;
+ var value = holder[key];
+ if (value && typeof value === "object") {
+ for (k in value) {
+ if (Object.prototype.hasOwnProperty.call(value, k)) {
+ v = walk(value, k);
+ if (v !== void 0) {
+ value[k] = v;
+ } else {
+ delete value[k];
+ }
+ }
+ }
+ }
+ return reviver.call(holder, key, value);
+ }
+ text = String(text);
+ rx_dangerous.lastIndex = 0;
+ if (rx_dangerous.test(text)) {
+ text = text.replace(rx_dangerous, function(a) {
+ return "\\u" + ("0000" + a.charCodeAt(0).toString(16)).slice(-4);
+ });
+ }
+ if (rx_one.test(
+ text.replace(rx_two, "@").replace(rx_three, "]").replace(rx_four, "")
+ )) {
+ j = eval("(" + text + ")");
+ return typeof reviver === "function" ? walk({ "": j }, "") : j;
+ }
+ throw new SyntaxError("JSON.parse");
+ };
+ }
+ })();
+ }
+ });
+
+ // node_modules/store/plugins/json2.js
+ var require_json22 = __commonJS({
+ "node_modules/store/plugins/json2.js"(exports2, module2) {
+ module2.exports = json2Plugin;
+ function json2Plugin() {
+ require_json2();
+ return {};
+ }
+ }
+ });
+
+ // node_modules/store/dist/store.legacy.js
+ var require_store_legacy = __commonJS({
+ "node_modules/store/dist/store.legacy.js"(exports2, module2) {
+ var engine = require_store_engine();
+ var storages = require_all();
+ var plugins = [require_json22()];
+ module2.exports = engine.createStore(storages, plugins);
+ }
+ });
+
+ // node_modules/whatwg-fetch/fetch.js
+ var global2 = typeof globalThis !== "undefined" && globalThis || typeof self !== "undefined" && self || typeof global2 !== "undefined" && global2;
+ var support = {
+ searchParams: "URLSearchParams" in global2,
+ iterable: "Symbol" in global2 && "iterator" in Symbol,
+ blob: "FileReader" in global2 && "Blob" in global2 && function() {
+ try {
+ new Blob();
+ return true;
+ } catch (e) {
+ return false;
+ }
+ }(),
+ formData: "FormData" in global2,
+ arrayBuffer: "ArrayBuffer" in global2
+ };
+ function isDataView(obj) {
+ return obj && DataView.prototype.isPrototypeOf(obj);
+ }
+ if (support.arrayBuffer) {
+ viewClasses = [
+ "[object Int8Array]",
+ "[object Uint8Array]",
+ "[object Uint8ClampedArray]",
+ "[object Int16Array]",
+ "[object Uint16Array]",
+ "[object Int32Array]",
+ "[object Uint32Array]",
+ "[object Float32Array]",
+ "[object Float64Array]"
+ ];
+ isArrayBufferView = ArrayBuffer.isView || function(obj) {
+ return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1;
+ };
+ }
+ var viewClasses;
+ var isArrayBufferView;
+ function normalizeName(name) {
+ if (typeof name !== "string") {
+ name = String(name);
+ }
+ if (/[^a-z0-9\-#$%&'*+.^_`|~!]/i.test(name) || name === "") {
+ throw new TypeError('Invalid character in header field name: "' + name + '"');
+ }
+ return name.toLowerCase();
+ }
+ function normalizeValue(value) {
+ if (typeof value !== "string") {
+ value = String(value);
+ }
+ return value;
+ }
+ function iteratorFor(items) {
+ var iterator = {
+ next: function() {
+ var value = items.shift();
+ return { done: value === void 0, value };
+ }
+ };
+ if (support.iterable) {
+ iterator[Symbol.iterator] = function() {
+ return iterator;
+ };
+ }
+ return iterator;
+ }
+ function Headers(headers) {
+ this.map = {};
+ if (headers instanceof Headers) {
+ headers.forEach(function(value, name) {
+ this.append(name, value);
+ }, this);
+ } else if (Array.isArray(headers)) {
+ headers.forEach(function(header) {
+ this.append(header[0], header[1]);
+ }, this);
+ } else if (headers) {
+ Object.getOwnPropertyNames(headers).forEach(function(name) {
+ this.append(name, headers[name]);
+ }, this);
+ }
+ }
+ 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.prototype["delete"] = function(name) {
+ delete this.map[normalizeName(name)];
+ };
+ Headers.prototype.get = function(name) {
+ name = normalizeName(name);
+ return this.has(name) ? this.map[name] : null;
+ };
+ Headers.prototype.has = function(name) {
+ return this.map.hasOwnProperty(normalizeName(name));
+ };
+ Headers.prototype.set = function(name, value) {
+ this.map[normalizeName(name)] = normalizeValue(value);
+ };
+ 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.prototype.keys = function() {
+ var items = [];
+ this.forEach(function(value, name) {
+ items.push(name);
+ });
+ return iteratorFor(items);
+ };
+ Headers.prototype.values = function() {
+ var items = [];
+ this.forEach(function(value) {
+ items.push(value);
+ });
+ return iteratorFor(items);
+ };
+ Headers.prototype.entries = function() {
+ var items = [];
+ this.forEach(function(value, name) {
+ items.push([name, value]);
+ });
+ return iteratorFor(items);
+ };
+ if (support.iterable) {
+ Headers.prototype[Symbol.iterator] = Headers.prototype.entries;
+ }
+ function consumed(body) {
+ if (body.bodyUsed) {
+ return Promise.reject(new TypeError("Already read"));
+ }
+ body.bodyUsed = true;
+ }
+ function fileReaderReady(reader) {
+ return new Promise(function(resolve, reject) {
+ reader.onload = function() {
+ resolve(reader.result);
+ };
+ reader.onerror = function() {
+ reject(reader.error);
+ };
+ });
+ }
+ function readBlobAsArrayBuffer(blob) {
+ var reader = new FileReader();
+ var promise = fileReaderReady(reader);
+ reader.readAsArrayBuffer(blob);
+ return promise;
+ }
+ function readBlobAsText(blob) {
+ var reader = new FileReader();
+ var promise = fileReaderReady(reader);
+ reader.readAsText(blob);
+ return promise;
+ }
+ function readArrayBufferAsText(buf) {
+ var view = new Uint8Array(buf);
+ var chars = new Array(view.length);
+ for (var i2 = 0; i2 < view.length; i2++) {
+ chars[i2] = String.fromCharCode(view[i2]);
+ }
+ return chars.join("");
+ }
+ function bufferClone(buf) {
+ if (buf.slice) {
+ return buf.slice(0);
+ } else {
+ var view = new Uint8Array(buf.byteLength);
+ view.set(new Uint8Array(buf));
+ return view.buffer;
+ }
+ }
+ function Body() {
+ this.bodyUsed = false;
+ this._initBody = function(body) {
+ this.bodyUsed = this.bodyUsed;
+ this._bodyInit = body;
+ if (!body) {
+ this._bodyText = "";
+ } else if (typeof body === "string") {
+ this._bodyText = body;
+ } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {
+ this._bodyBlob = body;
+ } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {
+ this._bodyFormData = body;
+ } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
+ this._bodyText = body.toString();
+ } else if (support.arrayBuffer && support.blob && isDataView(body)) {
+ this._bodyArrayBuffer = bufferClone(body.buffer);
+ this._bodyInit = new Blob([this._bodyArrayBuffer]);
+ } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {
+ this._bodyArrayBuffer = bufferClone(body);
+ } else {
+ this._bodyText = body = Object.prototype.toString.call(body);
+ }
+ if (!this.headers.get("content-type")) {
+ if (typeof body === "string") {
+ this.headers.set("content-type", "text/plain;charset=UTF-8");
+ } else if (this._bodyBlob && this._bodyBlob.type) {
+ this.headers.set("content-type", this._bodyBlob.type);
+ } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
+ this.headers.set("content-type", "application/x-www-form-urlencoded;charset=UTF-8");
+ }
+ }
+ };
+ if (support.blob) {
+ this.blob = function() {
+ var rejected = consumed(this);
+ if (rejected) {
+ return rejected;
+ }
+ if (this._bodyBlob) {
+ return Promise.resolve(this._bodyBlob);
+ } else if (this._bodyArrayBuffer) {
+ return Promise.resolve(new Blob([this._bodyArrayBuffer]));
+ } else if (this._bodyFormData) {
+ throw new Error("could not read FormData body as blob");
+ } else {
+ return Promise.resolve(new Blob([this._bodyText]));
+ }
+ };
+ this.arrayBuffer = function() {
+ if (this._bodyArrayBuffer) {
+ var isConsumed = consumed(this);
+ if (isConsumed) {
+ return isConsumed;
+ }
+ if (ArrayBuffer.isView(this._bodyArrayBuffer)) {
+ return Promise.resolve(
+ this._bodyArrayBuffer.buffer.slice(
+ this._bodyArrayBuffer.byteOffset,
+ this._bodyArrayBuffer.byteOffset + this._bodyArrayBuffer.byteLength
+ )
+ );
+ } else {
+ return Promise.resolve(this._bodyArrayBuffer);
+ }
+ } else {
+ return this.blob().then(readBlobAsArrayBuffer);
+ }
+ };
+ }
+ this.text = function() {
+ var rejected = consumed(this);
+ if (rejected) {
+ return rejected;
+ }
+ if (this._bodyBlob) {
+ return readBlobAsText(this._bodyBlob);
+ } else if (this._bodyArrayBuffer) {
+ return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer));
+ } else if (this._bodyFormData) {
+ throw new Error("could not read FormData body as text");
+ } else {
+ return Promise.resolve(this._bodyText);
+ }
+ };
+ if (support.formData) {
+ this.formData = function() {
+ return this.text().then(decode);
+ };
+ }
+ this.json = function() {
+ return this.text().then(JSON.parse);
+ };
+ return this;
+ }
+ var methods = ["DELETE", "GET", "HEAD", "OPTIONS", "POST", "PUT"];
+ function normalizeMethod(method) {
+ var upcased = method.toUpperCase();
+ return methods.indexOf(upcased) > -1 ? upcased : method;
+ }
+ function Request(input, options2) {
+ if (!(this instanceof Request)) {
+ throw new TypeError('Please use the "new" operator, this DOM object constructor cannot be called as a function.');
+ }
+ options2 = options2 || {};
+ var body = options2.body;
+ if (input instanceof Request) {
+ if (input.bodyUsed) {
+ throw new TypeError("Already read");
+ }
+ this.url = input.url;
+ this.credentials = input.credentials;
+ if (!options2.headers) {
+ this.headers = new Headers(input.headers);
+ }
+ this.method = input.method;
+ this.mode = input.mode;
+ this.signal = input.signal;
+ if (!body && input._bodyInit != null) {
+ body = input._bodyInit;
+ input.bodyUsed = true;
+ }
+ } else {
+ this.url = String(input);
+ }
+ this.credentials = options2.credentials || this.credentials || "same-origin";
+ if (options2.headers || !this.headers) {
+ this.headers = new Headers(options2.headers);
+ }
+ this.method = normalizeMethod(options2.method || this.method || "GET");
+ this.mode = options2.mode || this.mode || null;
+ this.signal = options2.signal || this.signal;
+ this.referrer = null;
+ if ((this.method === "GET" || this.method === "HEAD") && body) {
+ throw new TypeError("Body not allowed for GET or HEAD requests");
+ }
+ this._initBody(body);
+ if (this.method === "GET" || this.method === "HEAD") {
+ if (options2.cache === "no-store" || options2.cache === "no-cache") {
+ var reParamSearch = /([?&])_=[^&]*/;
+ if (reParamSearch.test(this.url)) {
+ this.url = this.url.replace(reParamSearch, "$1_=" + new Date().getTime());
+ } else {
+ var reQueryString = /\?/;
+ this.url += (reQueryString.test(this.url) ? "&" : "?") + "_=" + new Date().getTime();
+ }
+ }
+ }
+ }
+ Request.prototype.clone = function() {
+ return new Request(this, { body: this._bodyInit });
+ };
+ function decode(body) {
+ var form = new FormData();
+ body.trim().split("&").forEach(function(bytes) {
+ if (bytes) {
+ var split = bytes.split("=");
+ var name = split.shift().replace(/\+/g, " ");
+ var value = split.join("=").replace(/\+/g, " ");
+ form.append(decodeURIComponent(name), decodeURIComponent(value));
+ }
+ });
+ return form;
+ }
+ function parseHeaders(rawHeaders) {
+ var headers = new Headers();
+ var preProcessedHeaders = rawHeaders.replace(/\r?\n[\t ]+/g, " ");
+ preProcessedHeaders.split("\r").map(function(header) {
+ return header.indexOf("\n") === 0 ? header.substr(1, header.length) : header;
+ }).forEach(function(line) {
+ var parts = line.split(":");
+ var key = parts.shift().trim();
+ if (key) {
+ var value = parts.join(":").trim();
+ headers.append(key, value);
+ }
+ });
+ return headers;
+ }
+ Body.call(Request.prototype);
+ function Response(bodyInit, options2) {
+ if (!(this instanceof Response)) {
+ throw new TypeError('Please use the "new" operator, this DOM object constructor cannot be called as a function.');
+ }
+ if (!options2) {
+ options2 = {};
+ }
+ this.type = "default";
+ this.status = options2.status === void 0 ? 200 : options2.status;
+ this.ok = this.status >= 200 && this.status < 300;
+ this.statusText = options2.statusText === void 0 ? "" : "" + options2.statusText;
+ this.headers = new Headers(options2.headers);
+ this.url = options2.url || "";
+ this._initBody(bodyInit);
+ }
+ Body.call(Response.prototype);
+ Response.prototype.clone = function() {
+ return new Response(this._bodyInit, {
+ status: this.status,
+ statusText: this.statusText,
+ headers: new Headers(this.headers),
+ url: this.url
+ });
+ };
+ Response.error = function() {
+ var response = new Response(null, { status: 0, statusText: "" });
+ response.type = "error";
+ return response;
+ };
+ var redirectStatuses = [301, 302, 303, 307, 308];
+ Response.redirect = function(url, status) {
+ if (redirectStatuses.indexOf(status) === -1) {
+ throw new RangeError("Invalid status code");
+ }
+ return new Response(null, { status, headers: { location: url } });
+ };
+ var DOMException2 = global2.DOMException;
+ try {
+ new DOMException2();
+ } catch (err) {
+ DOMException2 = function(message, name) {
+ this.message = message;
+ this.name = name;
+ var error = Error(message);
+ this.stack = error.stack;
+ };
+ DOMException2.prototype = Object.create(Error.prototype);
+ DOMException2.prototype.constructor = DOMException2;
+ }
+ function fetch2(input, init2) {
+ return new Promise(function(resolve, reject) {
+ var request3 = new Request(input, init2);
+ if (request3.signal && request3.signal.aborted) {
+ return reject(new DOMException2("Aborted", "AbortError"));
+ }
+ var xhr = new XMLHttpRequest();
+ function abortXhr() {
+ xhr.abort();
+ }
+ xhr.onload = function() {
+ var options2 = {
+ status: xhr.status,
+ statusText: xhr.statusText,
+ headers: parseHeaders(xhr.getAllResponseHeaders() || "")
+ };
+ options2.url = "responseURL" in xhr ? xhr.responseURL : options2.headers.get("X-Request-URL");
+ var body = "response" in xhr ? xhr.response : xhr.responseText;
+ setTimeout(function() {
+ resolve(new Response(body, options2));
+ }, 0);
+ };
+ xhr.onerror = function() {
+ setTimeout(function() {
+ reject(new TypeError("Network request failed"));
+ }, 0);
+ };
+ xhr.ontimeout = function() {
+ setTimeout(function() {
+ reject(new TypeError("Network request failed"));
+ }, 0);
+ };
+ xhr.onabort = function() {
+ setTimeout(function() {
+ reject(new DOMException2("Aborted", "AbortError"));
+ }, 0);
+ };
+ function fixUrl(url) {
+ try {
+ return url === "" && global2.location.href ? global2.location.href : url;
+ } catch (e) {
+ return url;
+ }
+ }
+ xhr.open(request3.method, fixUrl(request3.url), true);
+ if (request3.credentials === "include") {
+ xhr.withCredentials = true;
+ } else if (request3.credentials === "omit") {
+ xhr.withCredentials = false;
+ }
+ if ("responseType" in xhr) {
+ if (support.blob) {
+ xhr.responseType = "blob";
+ } else if (support.arrayBuffer && request3.headers.get("Content-Type") && request3.headers.get("Content-Type").indexOf("application/octet-stream") !== -1) {
+ xhr.responseType = "arraybuffer";
+ }
+ }
+ if (init2 && typeof init2.headers === "object" && !(init2.headers instanceof Headers)) {
+ Object.getOwnPropertyNames(init2.headers).forEach(function(name) {
+ xhr.setRequestHeader(name, normalizeValue(init2.headers[name]));
+ });
+ } else {
+ request3.headers.forEach(function(value, name) {
+ xhr.setRequestHeader(name, value);
+ });
+ }
+ if (request3.signal) {
+ request3.signal.addEventListener("abort", abortXhr);
+ xhr.onreadystatechange = function() {
+ if (xhr.readyState === 4) {
+ request3.signal.removeEventListener("abort", abortXhr);
+ }
+ };
+ }
+ xhr.send(typeof request3._bodyInit === "undefined" ? null : request3._bodyInit);
+ });
+ }
+ fetch2.polyfill = true;
+ if (!global2.fetch) {
+ global2.fetch = fetch2;
+ global2.Headers = Headers;
+ global2.Request = Request;
+ global2.Response = Response;
+ }
+
+ // node_modules/abortcontroller-polyfill/dist/polyfill-patch-fetch.js
+ (function(factory) {
+ typeof define === "function" && define.amd ? define(factory) : factory();
+ })(function() {
+ "use strict";
+ function _classCallCheck(instance, Constructor) {
+ if (!(instance instanceof Constructor)) {
+ throw new TypeError("Cannot call a class as a function");
+ }
+ }
+ function _defineProperties(target, props) {
+ for (var i2 = 0; i2 < props.length; i2++) {
+ var descriptor = props[i2];
+ descriptor.enumerable = descriptor.enumerable || false;
+ descriptor.configurable = true;
+ if ("value" in descriptor)
+ descriptor.writable = true;
+ Object.defineProperty(target, descriptor.key, descriptor);
+ }
+ }
+ function _createClass(Constructor, protoProps, staticProps) {
+ if (protoProps)
+ _defineProperties(Constructor.prototype, protoProps);
+ if (staticProps)
+ _defineProperties(Constructor, staticProps);
+ return Constructor;
+ }
+ function _inherits(subClass, superClass) {
+ if (typeof superClass !== "function" && superClass !== null) {
+ throw new TypeError("Super expression must either be null or a function");
+ }
+ subClass.prototype = Object.create(superClass && superClass.prototype, {
+ constructor: {
+ value: subClass,
+ writable: true,
+ configurable: true
+ }
+ });
+ if (superClass)
+ _setPrototypeOf(subClass, superClass);
+ }
+ function _getPrototypeOf(o) {
+ _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf2(o2) {
+ return o2.__proto__ || Object.getPrototypeOf(o2);
+ };
+ return _getPrototypeOf(o);
+ }
+ function _setPrototypeOf(o, p) {
+ _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf2(o2, p2) {
+ o2.__proto__ = p2;
+ return o2;
+ };
+ return _setPrototypeOf(o, p);
+ }
+ function _isNativeReflectConstruct() {
+ if (typeof Reflect === "undefined" || !Reflect.construct)
+ return false;
+ if (Reflect.construct.sham)
+ return false;
+ if (typeof Proxy === "function")
+ return true;
+ try {
+ Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function() {
+ }));
+ return true;
+ } catch (e) {
+ return false;
+ }
+ }
+ function _assertThisInitialized(self2) {
+ if (self2 === void 0) {
+ throw new ReferenceError("this hasn't been initialised - super() hasn't been called");
+ }
+ return self2;
+ }
+ function _possibleConstructorReturn(self2, call) {
+ if (call && (typeof call === "object" || typeof call === "function")) {
+ return call;
+ }
+ return _assertThisInitialized(self2);
+ }
+ function _createSuper(Derived) {
+ var hasNativeReflectConstruct = _isNativeReflectConstruct();
+ return function _createSuperInternal() {
+ var Super = _getPrototypeOf(Derived), result;
+ if (hasNativeReflectConstruct) {
+ var NewTarget = _getPrototypeOf(this).constructor;
+ result = Reflect.construct(Super, arguments, NewTarget);
+ } else {
+ result = Super.apply(this, arguments);
+ }
+ return _possibleConstructorReturn(this, result);
+ };
+ }
+ function _superPropBase(object, property) {
+ while (!Object.prototype.hasOwnProperty.call(object, property)) {
+ object = _getPrototypeOf(object);
+ if (object === null)
+ break;
+ }
+ return object;
+ }
+ function _get(target, property, receiver) {
+ if (typeof Reflect !== "undefined" && Reflect.get) {
+ _get = Reflect.get;
+ } else {
+ _get = function _get2(target2, property2, receiver2) {
+ var base = _superPropBase(target2, property2);
+ if (!base)
+ return;
+ var desc = Object.getOwnPropertyDescriptor(base, property2);
+ if (desc.get) {
+ return desc.get.call(receiver2);
+ }
+ return desc.value;
+ };
+ }
+ return _get(target, property, receiver || target);
+ }
+ var Emitter = /* @__PURE__ */ function() {
+ function Emitter2() {
+ _classCallCheck(this, Emitter2);
+ Object.defineProperty(this, "listeners", {
+ value: {},
+ writable: true,
+ configurable: true
+ });
+ }
+ _createClass(Emitter2, [{
+ key: "addEventListener",
+ value: function addEventListener(type2, callback, options2) {
+ if (!(type2 in this.listeners)) {
+ this.listeners[type2] = [];
+ }
+ this.listeners[type2].push({
+ callback,
+ options: options2
+ });
+ }
+ }, {
+ key: "removeEventListener",
+ value: function removeEventListener(type2, callback) {
+ if (!(type2 in this.listeners)) {
+ return;
+ }
+ var stack = this.listeners[type2];
+ for (var i2 = 0, l = stack.length; i2 < l; i2++) {
+ if (stack[i2].callback === callback) {
+ stack.splice(i2, 1);
+ return;
+ }
+ }
+ }
+ }, {
+ key: "dispatchEvent",
+ value: function dispatchEvent2(event) {
+ if (!(event.type in this.listeners)) {
+ return;
+ }
+ var stack = this.listeners[event.type];
+ var stackToCall = stack.slice();
+ for (var i2 = 0, l = stackToCall.length; i2 < l; i2++) {
+ var listener = stackToCall[i2];
+ try {
+ listener.callback.call(this, event);
+ } catch (e) {
+ Promise.resolve().then(function() {
+ throw e;
+ });
+ }
+ if (listener.options && listener.options.once) {
+ this.removeEventListener(event.type, listener.callback);
+ }
+ }
+ return !event.defaultPrevented;
+ }
+ }]);
+ return Emitter2;
+ }();
+ var AbortSignal = /* @__PURE__ */ function(_Emitter) {
+ _inherits(AbortSignal2, _Emitter);
+ var _super = _createSuper(AbortSignal2);
+ function AbortSignal2() {
+ var _this;
+ _classCallCheck(this, AbortSignal2);
+ _this = _super.call(this);
+ if (!_this.listeners) {
+ Emitter.call(_assertThisInitialized(_this));
+ }
+ Object.defineProperty(_assertThisInitialized(_this), "aborted", {
+ value: false,
+ writable: true,
+ configurable: true
+ });
+ Object.defineProperty(_assertThisInitialized(_this), "onabort", {
+ value: null,
+ writable: true,
+ configurable: true
+ });
+ return _this;
+ }
+ _createClass(AbortSignal2, [{
+ key: "toString",
+ value: function toString2() {
+ return "[object AbortSignal]";
+ }
+ }, {
+ key: "dispatchEvent",
+ value: function dispatchEvent2(event) {
+ if (event.type === "abort") {
+ this.aborted = true;
+ if (typeof this.onabort === "function") {
+ this.onabort.call(this, event);
+ }
+ }
+ _get(_getPrototypeOf(AbortSignal2.prototype), "dispatchEvent", this).call(this, event);
+ }
+ }]);
+ return AbortSignal2;
+ }(Emitter);
+ var AbortController2 = /* @__PURE__ */ function() {
+ function AbortController3() {
+ _classCallCheck(this, AbortController3);
+ Object.defineProperty(this, "signal", {
+ value: new AbortSignal(),
+ writable: true,
+ configurable: true
+ });
+ }
+ _createClass(AbortController3, [{
+ key: "abort",
+ value: function abort() {
+ var event;
+ try {
+ event = new Event("abort");
+ } catch (e) {
+ if (typeof document !== "undefined") {
+ if (!document.createEvent) {
+ event = document.createEventObject();
+ event.type = "abort";
+ } else {
+ event = document.createEvent("Event");
+ event.initEvent("abort", false, false);
+ }
+ } else {
+ event = {
+ type: "abort",
+ bubbles: false,
+ cancelable: false
+ };
+ }
+ }
+ this.signal.dispatchEvent(event);
+ }
+ }, {
+ key: "toString",
+ value: function toString2() {
+ return "[object AbortController]";
+ }
+ }]);
+ return AbortController3;
+ }();
+ if (typeof Symbol !== "undefined" && Symbol.toStringTag) {
+ AbortController2.prototype[Symbol.toStringTag] = "AbortController";
+ AbortSignal.prototype[Symbol.toStringTag] = "AbortSignal";
+ }
+ function polyfillNeeded(self2) {
+ if (self2.__FORCE_INSTALL_ABORTCONTROLLER_POLYFILL) {
+ console.log("__FORCE_INSTALL_ABORTCONTROLLER_POLYFILL=true is set, will force install polyfill");
+ return true;
+ }
+ return typeof self2.Request === "function" && !self2.Request.prototype.hasOwnProperty("signal") || !self2.AbortController;
+ }
+ function abortableFetchDecorator(patchTargets) {
+ if ("function" === typeof patchTargets) {
+ patchTargets = {
+ fetch: patchTargets
+ };
+ }
+ var _patchTargets = patchTargets, fetch3 = _patchTargets.fetch, _patchTargets$Request = _patchTargets.Request, NativeRequest = _patchTargets$Request === void 0 ? fetch3.Request : _patchTargets$Request, NativeAbortController = _patchTargets.AbortController, _patchTargets$__FORCE = _patchTargets.__FORCE_INSTALL_ABORTCONTROLLER_POLYFILL, __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL = _patchTargets$__FORCE === void 0 ? false : _patchTargets$__FORCE;
+ if (!polyfillNeeded({
+ fetch: fetch3,
+ Request: NativeRequest,
+ AbortController: NativeAbortController,
+ __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL
+ })) {
+ return {
+ fetch: fetch3,
+ Request: Request2
+ };
+ }
+ var Request2 = NativeRequest;
+ if (Request2 && !Request2.prototype.hasOwnProperty("signal") || __FORCE_INSTALL_ABORTCONTROLLER_POLYFILL) {
+ Request2 = function Request3(input, init2) {
+ var signal;
+ if (init2 && init2.signal) {
+ signal = init2.signal;
+ delete init2.signal;
+ }
+ var request3 = new NativeRequest(input, init2);
+ if (signal) {
+ Object.defineProperty(request3, "signal", {
+ writable: false,
+ enumerable: false,
+ configurable: true,
+ value: signal
+ });
+ }
+ return request3;
+ };
+ Request2.prototype = NativeRequest.prototype;
+ }
+ var realFetch = fetch3;
+ var abortableFetch = function abortableFetch2(input, init2) {
+ var signal = Request2 && Request2.prototype.isPrototypeOf(input) ? input.signal : init2 ? init2.signal : void 0;
+ if (signal) {
+ var abortError;
+ try {
+ abortError = new DOMException("Aborted", "AbortError");
+ } catch (err) {
+ abortError = new Error("Aborted");
+ abortError.name = "AbortError";
+ }
+ if (signal.aborted) {
+ return Promise.reject(abortError);
+ }
+ var cancellation = new Promise(function(_, reject) {
+ signal.addEventListener("abort", function() {
+ return reject(abortError);
+ }, {
+ once: true
+ });
+ });
+ if (init2 && init2.signal) {
+ delete init2.signal;
+ }
+ return Promise.race([cancellation, realFetch(input, init2)]);
+ }
+ return realFetch(input, init2);
+ };
+ return {
+ fetch: abortableFetch,
+ Request: Request2
+ };
+ }
+ (function(self2) {
+ if (!polyfillNeeded(self2)) {
+ return;
+ }
+ if (!self2.fetch) {
+ console.warn("fetch() is not available, cannot install abortcontroller-polyfill");
+ return;
+ }
+ var _abortableFetch = abortableFetchDecorator(self2), fetch3 = _abortableFetch.fetch, Request2 = _abortableFetch.Request;
+ self2.fetch = fetch3;
+ self2.Request = Request2;
+ Object.defineProperty(self2, "AbortController", {
+ writable: true,
+ enumerable: false,
+ configurable: true,
+ value: AbortController2
+ });
+ Object.defineProperty(self2, "AbortSignal", {
+ writable: true,
+ enumerable: false,
+ configurable: true,
+ value: AbortSignal
+ });
+ })(typeof self !== "undefined" ? self : global);
+ });
+
+ // modules/index.js
+ var modules_exports = {};
+ __export(modules_exports, {
+ LocationManager: () => LocationManager,
+ QAItem: () => QAItem,
+ actionAddEntity: () => actionAddEntity,
+ actionAddMember: () => actionAddMember,
+ actionAddMidpoint: () => actionAddMidpoint,
+ actionAddVertex: () => actionAddVertex,
+ actionChangeMember: () => actionChangeMember,
+ actionChangePreset: () => actionChangePreset,
+ actionChangeTags: () => actionChangeTags,
+ actionCircularize: () => actionCircularize,
+ actionConnect: () => actionConnect,
+ actionCopyEntities: () => actionCopyEntities,
+ actionDeleteMember: () => actionDeleteMember,
+ actionDeleteMultiple: () => actionDeleteMultiple,
+ actionDeleteNode: () => actionDeleteNode,
+ actionDeleteRelation: () => actionDeleteRelation,
+ actionDeleteWay: () => actionDeleteWay,
+ actionDiscardTags: () => actionDiscardTags,
+ actionDisconnect: () => actionDisconnect,
+ actionExtract: () => actionExtract,
+ actionJoin: () => actionJoin,
+ actionMerge: () => actionMerge,
+ actionMergeNodes: () => actionMergeNodes,
+ actionMergePolygon: () => actionMergePolygon,
+ actionMergeRemoteChanges: () => actionMergeRemoteChanges,
+ actionMove: () => actionMove,
+ actionMoveMember: () => actionMoveMember,
+ actionMoveNode: () => actionMoveNode,
+ actionNoop: () => actionNoop,
+ actionOrthogonalize: () => actionOrthogonalize,
+ actionReflect: () => actionReflect,
+ actionRestrictTurn: () => actionRestrictTurn,
+ actionReverse: () => actionReverse,
+ actionRevert: () => actionRevert,
+ actionRotate: () => actionRotate,
+ actionScale: () => actionScale,
+ actionSplit: () => actionSplit,
+ actionStraightenNodes: () => actionStraightenNodes,
+ actionStraightenWay: () => actionStraightenWay,
+ actionUnrestrictTurn: () => actionUnrestrictTurn,
+ actionUpgradeTags: () => actionUpgradeTags,
+ behaviorAddWay: () => behaviorAddWay,
+ behaviorBreathe: () => behaviorBreathe,
+ behaviorDrag: () => behaviorDrag,
+ behaviorDraw: () => behaviorDraw,
+ behaviorDrawWay: () => behaviorDrawWay,
+ behaviorEdit: () => behaviorEdit,
+ behaviorHash: () => behaviorHash,
+ behaviorHover: () => behaviorHover,
+ behaviorLasso: () => behaviorLasso,
+ behaviorOperation: () => behaviorOperation,
+ behaviorPaste: () => behaviorPaste,
+ behaviorSelect: () => behaviorSelect,
+ coreContext: () => coreContext,
+ coreDifference: () => coreDifference,
+ coreFileFetcher: () => coreFileFetcher,
+ coreGraph: () => coreGraph,
+ coreHistory: () => coreHistory,
+ coreLocalizer: () => coreLocalizer,
+ coreTree: () => coreTree,
+ coreUploader: () => coreUploader,
+ coreValidator: () => coreValidator,
+ d3: () => d3,
+ debug: () => debug,
+ fileFetcher: () => _mainFileFetcher,
+ geoAngle: () => geoAngle,
+ geoChooseEdge: () => geoChooseEdge,
+ geoEdgeEqual: () => geoEdgeEqual,
+ geoExtent: () => geoExtent,
+ geoGetSmallestSurroundingRectangle: () => geoGetSmallestSurroundingRectangle,
+ geoHasLineIntersections: () => geoHasLineIntersections,
+ geoHasSelfIntersections: () => geoHasSelfIntersections,
+ geoLatToMeters: () => geoLatToMeters,
+ geoLineIntersection: () => geoLineIntersection,
+ geoLonToMeters: () => geoLonToMeters,
+ geoMetersToLat: () => geoMetersToLat,
+ geoMetersToLon: () => geoMetersToLon,
+ geoMetersToOffset: () => geoMetersToOffset,
+ geoOffsetToMeters: () => geoOffsetToMeters,
+ geoOrthoCalcScore: () => geoOrthoCalcScore,
+ geoOrthoCanOrthogonalize: () => geoOrthoCanOrthogonalize,
+ geoOrthoMaxOffsetAngle: () => geoOrthoMaxOffsetAngle,
+ geoOrthoNormalizedDotProduct: () => geoOrthoNormalizedDotProduct,
+ geoPathHasIntersections: () => geoPathHasIntersections,
+ geoPathIntersections: () => geoPathIntersections,
+ geoPathLength: () => geoPathLength,
+ geoPointInPolygon: () => geoPointInPolygon,
+ geoPolygonContainsPolygon: () => geoPolygonContainsPolygon,
+ geoPolygonIntersectsPolygon: () => geoPolygonIntersectsPolygon,
+ geoRawMercator: () => geoRawMercator,
+ geoRotate: () => geoRotate,
+ geoScaleToZoom: () => geoScaleToZoom,
+ geoSphericalClosestNode: () => geoSphericalClosestNode,
+ geoSphericalDistance: () => geoSphericalDistance,
+ geoVecAdd: () => geoVecAdd,
+ geoVecAngle: () => geoVecAngle,
+ geoVecCross: () => geoVecCross,
+ geoVecDot: () => geoVecDot,
+ geoVecEqual: () => geoVecEqual,
+ geoVecFloor: () => geoVecFloor,
+ geoVecInterp: () => geoVecInterp,
+ geoVecLength: () => geoVecLength,
+ geoVecLengthSquare: () => geoVecLengthSquare,
+ geoVecNormalize: () => geoVecNormalize,
+ geoVecNormalizedDot: () => geoVecNormalizedDot,
+ geoVecProject: () => geoVecProject,
+ geoVecScale: () => geoVecScale,
+ geoVecSubtract: () => geoVecSubtract,
+ geoViewportEdge: () => geoViewportEdge,
+ geoZoomToScale: () => geoZoomToScale,
+ localizer: () => _mainLocalizer,
+ locationManager: () => _sharedLocationManager,
+ modeAddArea: () => modeAddArea,
+ modeAddLine: () => modeAddLine,
+ modeAddNote: () => modeAddNote,
+ modeAddPoint: () => modeAddPoint,
+ modeBrowse: () => modeBrowse,
+ modeDragNode: () => modeDragNode,
+ modeDragNote: () => modeDragNote,
+ modeDrawArea: () => modeDrawArea,
+ modeDrawLine: () => modeDrawLine,
+ modeMove: () => modeMove,
+ modeRotate: () => modeRotate,
+ modeSave: () => modeSave,
+ modeSelect: () => modeSelect,
+ modeSelectData: () => modeSelectData,
+ modeSelectError: () => modeSelectError,
+ modeSelectNote: () => modeSelectNote,
+ operationCircularize: () => operationCircularize,
+ operationContinue: () => operationContinue,
+ operationCopy: () => operationCopy,
+ operationDelete: () => operationDelete,
+ operationDisconnect: () => operationDisconnect,
+ operationDowngrade: () => operationDowngrade,
+ operationExtract: () => operationExtract,
+ operationMerge: () => operationMerge,
+ operationMove: () => operationMove,
+ operationOrthogonalize: () => operationOrthogonalize,
+ operationPaste: () => operationPaste,
+ operationReflectLong: () => operationReflectLong,
+ operationReflectShort: () => operationReflectShort,
+ operationReverse: () => operationReverse,
+ operationRotate: () => operationRotate,
+ operationSplit: () => operationSplit,
+ operationStraighten: () => operationStraighten,
+ osmAreaKeys: () => osmAreaKeys,
+ osmChangeset: () => osmChangeset,
+ osmEntity: () => osmEntity,
+ osmFlowingWaterwayTagValues: () => osmFlowingWaterwayTagValues,
+ osmInferRestriction: () => osmInferRestriction,
+ osmIntersection: () => osmIntersection,
+ osmIsInterestingTag: () => osmIsInterestingTag,
+ osmIsOldMultipolygonOuterMember: () => osmIsOldMultipolygonOuterMember,
+ osmJoinWays: () => osmJoinWays,
+ osmLanes: () => osmLanes,
+ osmLifecyclePrefixes: () => osmLifecyclePrefixes,
+ osmNode: () => osmNode,
+ osmNodeGeometriesForTags: () => osmNodeGeometriesForTags,
+ osmNote: () => osmNote,
+ osmOldMultipolygonOuterMember: () => osmOldMultipolygonOuterMember,
+ osmOldMultipolygonOuterMemberOfRelation: () => osmOldMultipolygonOuterMemberOfRelation,
+ osmOneWayTags: () => osmOneWayTags,
+ osmPavedTags: () => osmPavedTags,
+ osmPointTags: () => osmPointTags,
+ osmRailwayTrackTagValues: () => osmRailwayTrackTagValues,
+ osmRelation: () => osmRelation,
+ osmRemoveLifecyclePrefix: () => osmRemoveLifecyclePrefix,
+ osmRoutableHighwayTagValues: () => osmRoutableHighwayTagValues,
+ osmSetAreaKeys: () => osmSetAreaKeys,
+ osmSetPointTags: () => osmSetPointTags,
+ osmSetVertexTags: () => osmSetVertexTags,
+ osmTagSuggestingArea: () => osmTagSuggestingArea,
+ osmTurn: () => osmTurn,
+ osmVertexTags: () => osmVertexTags,
+ osmWay: () => osmWay,
+ prefs: () => corePreferences,
+ presetCategory: () => presetCategory,
+ presetCollection: () => presetCollection,
+ presetField: () => presetField,
+ presetIndex: () => presetIndex,
+ presetManager: () => _mainPresetIndex,
+ presetPreset: () => presetPreset,
+ rendererBackground: () => rendererBackground,
+ rendererBackgroundSource: () => rendererBackgroundSource,
+ rendererFeatures: () => rendererFeatures,
+ rendererMap: () => rendererMap,
+ rendererPhotos: () => rendererPhotos,
+ rendererTileLayer: () => rendererTileLayer,
+ serviceImproveOSM: () => improveOSM_default,
+ serviceKartaview: () => kartaview_default,
+ serviceKeepRight: () => keepRight_default,
+ serviceMapRules: () => maprules_default,
+ serviceMapillary: () => mapillary_default,
+ serviceNominatim: () => nominatim_default,
+ serviceNsi: () => nsi_default,
+ serviceOsm: () => osm_default,
+ serviceOsmWikibase: () => osm_wikibase_default,
+ serviceOsmose: () => osmose_default,
+ serviceStreetside: () => streetside_default,
+ serviceTaginfo: () => taginfo_default,
+ serviceVectorTile: () => vector_tile_default,
+ serviceWikidata: () => wikidata_default,
+ serviceWikipedia: () => wikipedia_default,
+ services: () => services,
+ svgAreas: () => svgAreas,
+ svgData: () => svgData,
+ svgDebug: () => svgDebug,
+ svgDefs: () => svgDefs,
+ svgGeolocate: () => svgGeolocate,
+ svgIcon: () => svgIcon,
+ svgKartaviewImages: () => svgKartaviewImages,
+ svgKeepRight: () => svgKeepRight,
+ svgLabels: () => svgLabels,
+ svgLayers: () => svgLayers,
+ svgLines: () => svgLines,
+ svgMapillaryImages: () => svgMapillaryImages,
+ svgMapillarySigns: () => svgMapillarySigns,
+ svgMarkerSegments: () => svgMarkerSegments,
+ svgMidpoints: () => svgMidpoints,
+ svgNotes: () => svgNotes,
+ svgOsm: () => svgOsm,
+ svgPassiveVertex: () => svgPassiveVertex,
+ svgPath: () => svgPath,
+ svgPointTransform: () => svgPointTransform,
+ svgPoints: () => svgPoints,
+ svgRelationMemberTags: () => svgRelationMemberTags,
+ svgSegmentWay: () => svgSegmentWay,
+ svgStreetside: () => svgStreetside,
+ svgTagClasses: () => svgTagClasses,
+ svgTagPattern: () => svgTagPattern,
+ svgTouch: () => svgTouch,
+ svgTurns: () => svgTurns,
+ svgVertices: () => svgVertices,
+ t: () => _t,
+ uiAccount: () => uiAccount,
+ uiAttribution: () => uiAttribution,
+ uiChangesetEditor: () => uiChangesetEditor,
+ uiCmd: () => uiCmd,
+ uiCombobox: () => uiCombobox,
+ uiCommit: () => uiCommit,
+ uiCommitWarnings: () => uiCommitWarnings,
+ uiConfirm: () => uiConfirm,
+ uiConflicts: () => uiConflicts,
+ uiContributors: () => uiContributors,
+ uiCurtain: () => uiCurtain,
+ uiDataEditor: () => uiDataEditor,
+ uiDataHeader: () => uiDataHeader,
+ uiDisclosure: () => uiDisclosure,
+ uiEditMenu: () => uiEditMenu,
+ uiEntityEditor: () => uiEntityEditor,
+ uiFeatureInfo: () => uiFeatureInfo,
+ uiFeatureList: () => uiFeatureList,
+ uiField: () => uiField,
+ uiFieldAccess: () => uiFieldAccess,
+ uiFieldAddress: () => uiFieldAddress,
+ uiFieldCheck: () => uiFieldCheck,
+ uiFieldColour: () => uiFieldText,
+ uiFieldCombo: () => uiFieldCombo,
+ uiFieldDefaultCheck: () => uiFieldCheck,
+ uiFieldDirectionalCombo: () => uiFieldDirectionalCombo,
+ uiFieldEmail: () => uiFieldText,
+ uiFieldHelp: () => uiFieldHelp,
+ uiFieldIdentifier: () => uiFieldText,
+ uiFieldLanes: () => uiFieldLanes,
+ uiFieldLocalized: () => uiFieldLocalized,
+ uiFieldManyCombo: () => uiFieldCombo,
+ uiFieldMultiCombo: () => uiFieldCombo,
+ uiFieldNetworkCombo: () => uiFieldCombo,
+ uiFieldNumber: () => uiFieldText,
+ uiFieldOnewayCheck: () => uiFieldCheck,
+ uiFieldRadio: () => uiFieldRadio,
+ uiFieldRestrictions: () => uiFieldRestrictions,
+ uiFieldRoadheight: () => uiFieldRoadheight,
+ uiFieldRoadspeed: () => uiFieldRoadspeed,
+ uiFieldSemiCombo: () => uiFieldCombo,
+ uiFieldStructureRadio: () => uiFieldRadio,
+ uiFieldTel: () => uiFieldText,
+ uiFieldText: () => uiFieldText,
+ uiFieldTextarea: () => uiFieldTextarea,
+ uiFieldTypeCombo: () => uiFieldCombo,
+ uiFieldUrl: () => uiFieldText,
+ uiFieldWikidata: () => uiFieldWikidata,
+ uiFieldWikipedia: () => uiFieldWikipedia,
+ uiFields: () => uiFields,
+ uiFlash: () => uiFlash,
+ uiFormFields: () => uiFormFields,
+ uiFullScreen: () => uiFullScreen,
+ uiGeolocate: () => uiGeolocate,
+ uiImproveOsmComments: () => uiImproveOsmComments,
+ uiImproveOsmDetails: () => uiImproveOsmDetails,
+ uiImproveOsmEditor: () => uiImproveOsmEditor,
+ uiImproveOsmHeader: () => uiImproveOsmHeader,
+ uiInfo: () => uiInfo,
+ uiInfoPanels: () => uiInfoPanels,
+ uiInit: () => uiInit,
+ uiInspector: () => uiInspector,
+ uiIntro: () => uiIntro,
+ uiIssuesInfo: () => uiIssuesInfo,
+ uiKeepRightDetails: () => uiKeepRightDetails,
+ uiKeepRightEditor: () => uiKeepRightEditor,
+ uiKeepRightHeader: () => uiKeepRightHeader,
+ uiLasso: () => uiLasso,
+ uiLengthIndicator: () => uiLengthIndicator,
+ uiLoading: () => uiLoading,
+ uiMapInMap: () => uiMapInMap,
+ uiModal: () => uiModal,
+ uiNoteComments: () => uiNoteComments,
+ uiNoteEditor: () => uiNoteEditor,
+ uiNoteHeader: () => uiNoteHeader,
+ uiNoteReport: () => uiNoteReport,
+ uiNotice: () => uiNotice,
+ uiPaneBackground: () => uiPaneBackground,
+ uiPaneHelp: () => uiPaneHelp,
+ uiPaneIssues: () => uiPaneIssues,
+ uiPaneMapData: () => uiPaneMapData,
+ uiPanePreferences: () => uiPanePreferences,
+ uiPanelBackground: () => uiPanelBackground,
+ uiPanelHistory: () => uiPanelHistory,
+ uiPanelLocation: () => uiPanelLocation,
+ uiPanelMeasurement: () => uiPanelMeasurement,
+ uiPopover: () => uiPopover,
+ uiPresetIcon: () => uiPresetIcon,
+ uiPresetList: () => uiPresetList,
+ uiRestore: () => uiRestore,
+ uiScale: () => uiScale,
+ uiSectionBackgroundDisplayOptions: () => uiSectionBackgroundDisplayOptions,
+ uiSectionBackgroundList: () => uiSectionBackgroundList,
+ uiSectionBackgroundOffset: () => uiSectionBackgroundOffset,
+ uiSectionChanges: () => uiSectionChanges,
+ uiSectionDataLayers: () => uiSectionDataLayers,
+ uiSectionEntityIssues: () => uiSectionEntityIssues,
+ uiSectionFeatureType: () => uiSectionFeatureType,
+ uiSectionMapFeatures: () => uiSectionMapFeatures,
+ uiSectionMapStyleOptions: () => uiSectionMapStyleOptions,
+ uiSectionOverlayList: () => uiSectionOverlayList,
+ uiSectionPhotoOverlays: () => uiSectionPhotoOverlays,
+ uiSectionPresetFields: () => uiSectionPresetFields,
+ uiSectionPrivacy: () => uiSectionPrivacy,
+ uiSectionRawMemberEditor: () => uiSectionRawMemberEditor,
+ uiSectionRawMembershipEditor: () => uiSectionRawMembershipEditor,
+ uiSectionRawTagEditor: () => uiSectionRawTagEditor,
+ uiSectionSelectionList: () => uiSectionSelectionList,
+ uiSectionValidationIssues: () => uiSectionValidationIssues,
+ uiSectionValidationOptions: () => uiSectionValidationOptions,
+ uiSectionValidationRules: () => uiSectionValidationRules,
+ uiSectionValidationStatus: () => uiSectionValidationStatus,
+ uiSettingsCustomBackground: () => uiSettingsCustomBackground,
+ uiSettingsCustomData: () => uiSettingsCustomData,
+ uiSidebar: () => uiSidebar,
+ uiSourceSwitch: () => uiSourceSwitch,
+ uiSpinner: () => uiSpinner,
+ uiSplash: () => uiSplash,
+ uiStatus: () => uiStatus,
+ uiSuccess: () => uiSuccess,
+ uiTagReference: () => uiTagReference,
+ uiToggle: () => uiToggle,
+ uiTooltip: () => uiTooltip,
+ uiVersion: () => uiVersion,
+ uiViewOnKeepRight: () => uiViewOnKeepRight,
+ uiViewOnOSM: () => uiViewOnOSM,
+ uiZoom: () => uiZoom,
+ utilAesDecrypt: () => utilAesDecrypt,
+ utilAesEncrypt: () => utilAesEncrypt,
+ utilArrayChunk: () => utilArrayChunk,
+ utilArrayDifference: () => utilArrayDifference,
+ utilArrayFlatten: () => utilArrayFlatten,
+ utilArrayGroupBy: () => utilArrayGroupBy,
+ utilArrayIdentical: () => utilArrayIdentical,
+ utilArrayIntersection: () => utilArrayIntersection,
+ utilArrayUnion: () => utilArrayUnion,
+ utilArrayUniq: () => utilArrayUniq,
+ utilArrayUniqBy: () => utilArrayUniqBy,
+ utilAsyncMap: () => utilAsyncMap,
+ utilCleanOsmString: () => utilCleanOsmString,
+ utilCleanTags: () => utilCleanTags,
+ utilCombinedTags: () => utilCombinedTags,
+ utilCompareIDs: () => utilCompareIDs,
+ utilDeepMemberSelector: () => utilDeepMemberSelector,
+ utilDetect: () => utilDetect,
+ utilDisplayLabel: () => utilDisplayLabel,
+ utilDisplayName: () => utilDisplayName,
+ utilDisplayNameForPath: () => utilDisplayNameForPath,
+ utilDisplayType: () => utilDisplayType,
+ utilEditDistance: () => utilEditDistance,
+ utilEntityAndDeepMemberIDs: () => utilEntityAndDeepMemberIDs,
+ utilEntityOrDeepMemberSelector: () => utilEntityOrDeepMemberSelector,
+ utilEntityOrMemberSelector: () => utilEntityOrMemberSelector,
+ utilEntityRoot: () => utilEntityRoot,
+ utilEntitySelector: () => utilEntitySelector,
+ utilFastMouse: () => utilFastMouse,
+ utilFunctor: () => utilFunctor,
+ utilGetAllNodes: () => utilGetAllNodes,
+ utilGetSetValue: () => utilGetSetValue,
+ utilHashcode: () => utilHashcode,
+ utilHighlightEntities: () => utilHighlightEntities,
+ utilKeybinding: () => utilKeybinding,
+ utilNoAuto: () => utilNoAuto,
+ utilObjectOmit: () => utilObjectOmit,
+ utilOldestID: () => utilOldestID,
+ utilPrefixCSSProperty: () => utilPrefixCSSProperty,
+ utilPrefixDOMProperty: () => utilPrefixDOMProperty,
+ utilQsString: () => utilQsString,
+ utilRebind: () => utilRebind,
+ utilSafeClassName: () => utilSafeClassName,
+ utilSessionMutex: () => utilSessionMutex,
+ utilSetTransform: () => utilSetTransform,
+ utilStringQs: () => utilStringQs,
+ utilTagDiff: () => utilTagDiff,
+ utilTagText: () => utilTagText,
+ utilTiler: () => utilTiler,
+ utilTotalExtent: () => utilTotalExtent,
+ utilTriggerEvent: () => utilTriggerEvent,
+ utilUnicodeCharsCount: () => utilUnicodeCharsCount,
+ utilUnicodeCharsTruncated: () => utilUnicodeCharsTruncated,
+ utilUniqueDomId: () => utilUniqueDomId,
+ utilWrap: () => utilWrap,
+ 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
+ });
+
+ // modules/actions/add_entity.js
+ function actionAddEntity(way) {
+ return function(graph) {
+ return graph.replace(way);
+ };
+ }
+
+ // modules/actions/reverse.js
+ function actionReverse(entityID, options2) {
+ var ignoreKey = /^.*(_|:)?(description|name|note|website|ref|source|comment|watch|attribution)(_|:)?/;
+ var numeric = /^([+\-]?)(?=[\d.])/;
+ var directionKey = /direction$/;
+ var turn_lanes = /^turn:lanes:?/;
+ var keyReplacements = [
+ [/:right$/, ":left"],
+ [/:left$/, ":right"],
+ [/:forward$/, ":backward"],
+ [/:backward$/, ":forward"],
+ [/:right:/, ":left:"],
+ [/:left:/, ":right:"],
+ [/:forward:/, ":backward:"],
+ [/:backward:/, ":forward:"]
+ ];
+ var valueReplacements = {
+ left: "right",
+ right: "left",
+ up: "down",
+ down: "up",
+ forward: "backward",
+ backward: "forward",
+ forwards: "backward",
+ backwards: "forward"
+ };
+ var roleReplacements = {
+ forward: "backward",
+ backward: "forward",
+ forwards: "backward",
+ backwards: "forward"
+ };
+ var onewayReplacements = {
+ yes: "-1",
+ "1": "-1",
+ "-1": "yes"
+ };
+ var compassReplacements = {
+ N: "S",
+ NNE: "SSW",
+ NE: "SW",
+ ENE: "WSW",
+ E: "W",
+ ESE: "WNW",
+ SE: "NW",
+ SSE: "NNW",
+ S: "N",
+ SSW: "NNE",
+ SW: "NE",
+ WSW: "ENE",
+ W: "E",
+ WNW: "ESE",
+ NW: "SE",
+ NNW: "SSE"
+ };
+ function reverseKey(key) {
+ for (var i2 = 0; i2 < keyReplacements.length; ++i2) {
+ var replacement = keyReplacements[i2];
+ if (replacement[0].test(key)) {
+ return key.replace(replacement[0], replacement[1]);
+ }
+ }
+ return key;
+ }
+ function reverseValue(key, value, includeAbsolute) {
+ if (ignoreKey.test(key))
+ return value;
+ if (turn_lanes.test(key)) {
+ return value;
+ } else if (key === "incline" && numeric.test(value)) {
+ return value.replace(numeric, function(_, sign2) {
+ return sign2 === "-" ? "" : "-";
+ });
+ } else if (options2 && options2.reverseOneway && key === "oneway") {
+ return onewayReplacements[value] || value;
+ } else if (includeAbsolute && directionKey.test(key)) {
+ return value.split(";").map((value2) => {
+ if (compassReplacements[value2])
+ return compassReplacements[value2];
+ var degrees3 = Number(value2);
+ if (isFinite(degrees3)) {
+ if (degrees3 < 180) {
+ degrees3 += 180;
+ } else {
+ degrees3 -= 180;
+ }
+ return degrees3.toString();
+ } else {
+ return valueReplacements[value2] || value2;
+ }
+ }).join(";");
+ }
+ return valueReplacements[value] || value;
+ }
+ function reverseNodeTags(graph, nodeIDs) {
+ for (var i2 = 0; i2 < nodeIDs.length; i2++) {
+ var node = graph.hasEntity(nodeIDs[i2]);
+ if (!node || !Object.keys(node.tags).length)
+ continue;
+ var tags = {};
+ for (var key in node.tags) {
+ tags[reverseKey(key)] = reverseValue(key, node.tags[key], node.id === entityID);
+ }
+ graph = graph.replace(node.update({ tags }));
+ }
+ return graph;
+ }
+ function reverseWay(graph, way) {
+ var nodes = way.nodes.slice().reverse();
+ var tags = {};
+ var role;
+ for (var key in way.tags) {
+ tags[reverseKey(key)] = reverseValue(key, way.tags[key]);
+ }
+ graph.parentRelations(way).forEach(function(relation) {
+ relation.members.forEach(function(member, index) {
+ if (member.id === way.id && (role = roleReplacements[member.role])) {
+ relation = relation.updateMember({ role }, index);
+ graph = graph.replace(relation);
+ }
+ });
+ });
+ return reverseNodeTags(graph, nodes).replace(way.update({ nodes, tags }));
+ }
+ var action = function(graph) {
+ var entity = graph.entity(entityID);
+ if (entity.type === "way") {
+ return reverseWay(graph, entity);
+ }
+ return reverseNodeTags(graph, [entityID]);
+ };
+ action.disabled = function(graph) {
+ var entity = graph.hasEntity(entityID);
+ if (!entity || entity.type === "way")
+ return false;
+ for (var key in entity.tags) {
+ var value = entity.tags[key];
+ if (reverseKey(key) !== key || reverseValue(key, value, true) !== value) {
+ return false;
+ }
+ }
+ return "nondirectional_node";
+ };
+ action.entityID = function() {
+ return entityID;
+ };
+ return action;
+ }
+
+ // modules/osm/tags.js
+ function osmIsInterestingTag(key) {
+ return key !== "attribution" && key !== "created_by" && key !== "source" && key !== "odbl" && key.indexOf("source:") !== 0 && key.indexOf("source_ref") !== 0 && // purposely exclude colon
+ key.indexOf("tiger:") !== 0;
+ }
+ var osmLifecyclePrefixes = {
+ // nonexistent, might be built
+ proposed: true,
+ planned: true,
+ // under maintentance or between groundbreaking and opening
+ construction: true,
+ // existent but not functional
+ disused: true,
+ // dilapidated to nonexistent
+ abandoned: true,
+ was: true,
+ // nonexistent, still may appear in imagery
+ dismantled: true,
+ razed: true,
+ demolished: true,
+ destroyed: true,
+ removed: true,
+ obliterated: true,
+ // existent occasionally, e.g. stormwater drainage basin
+ intermittent: true
+ };
+ function osmRemoveLifecyclePrefix(key) {
+ const keySegments = key.split(":");
+ if (keySegments.length === 1)
+ return key;
+ if (keySegments[0] in osmLifecyclePrefixes) {
+ return key.slice(keySegments[0].length + 1);
+ }
+ return key;
+ }
+ var osmAreaKeys = {};
+ function osmSetAreaKeys(value) {
+ osmAreaKeys = value;
+ }
+ var osmAreaKeysExceptions = {
+ highway: {
+ elevator: true,
+ rest_area: true,
+ services: true
+ },
+ public_transport: {
+ platform: true
+ },
+ railway: {
+ platform: true,
+ roundhouse: true,
+ station: true,
+ traverser: true,
+ turntable: true,
+ wash: true
+ },
+ traffic_calming: {
+ island: true
+ },
+ waterway: {
+ dam: true
+ }
+ };
+ function osmTagSuggestingArea(tags) {
+ if (tags.area === "yes")
+ return { area: "yes" };
+ if (tags.area === "no")
+ return null;
+ var returnTags = {};
+ for (var realKey in tags) {
+ const key = osmRemoveLifecyclePrefix(realKey);
+ if (key in osmAreaKeys && !(tags[key] in osmAreaKeys[key])) {
+ returnTags[realKey] = tags[realKey];
+ return returnTags;
+ }
+ if (key in osmAreaKeysExceptions && tags[key] in osmAreaKeysExceptions[key]) {
+ returnTags[realKey] = tags[realKey];
+ return returnTags;
+ }
+ }
+ return null;
+ }
+ var osmLineTags = {};
+ function osmSetLineTags(value) {
+ osmLineTags = value;
+ }
+ var osmPointTags = {};
+ function osmSetPointTags(value) {
+ osmPointTags = value;
+ }
+ var osmVertexTags = {};
+ function osmSetVertexTags(value) {
+ osmVertexTags = value;
+ }
+ function osmNodeGeometriesForTags(nodeTags) {
+ var geometries = {};
+ for (var key in nodeTags) {
+ if (osmPointTags[key] && (osmPointTags[key]["*"] || osmPointTags[key][nodeTags[key]])) {
+ geometries.point = true;
+ }
+ if (osmVertexTags[key] && (osmVertexTags[key]["*"] || osmVertexTags[key][nodeTags[key]])) {
+ geometries.vertex = true;
+ }
+ if (geometries.point && geometries.vertex)
+ break;
+ }
+ return geometries;
+ }
+ var osmOneWayTags = {
+ "aerialway": {
+ "chair_lift": true,
+ "drag_lift": true,
+ "j-bar": true,
+ "magic_carpet": true,
+ "mixed_lift": true,
+ "platter": true,
+ "rope_tow": true,
+ "t-bar": true,
+ "zip_line": true
+ },
+ "highway": {
+ "motorway": true
+ },
+ "junction": {
+ "circular": true,
+ "roundabout": true
+ },
+ "man_made": {
+ "goods_conveyor": true,
+ "piste:halfpipe": true
+ },
+ "piste:type": {
+ "downhill": true,
+ "sled": true,
+ "yes": true
+ },
+ "seamark:type": {
+ "separation_lane": true,
+ "separation_roundabout": true
+ },
+ "waterway": {
+ "canal": true,
+ "ditch": true,
+ "drain": true,
+ "fish_pass": true,
+ "river": true,
+ "stream": true,
+ "tidal_channel": true
+ }
+ };
+ var osmPavedTags = {
+ "surface": {
+ "paved": true,
+ "asphalt": true,
+ "concrete": true,
+ "chipseal": true,
+ "concrete:lanes": true,
+ "concrete:plates": true
+ },
+ "tracktype": {
+ "grade1": true
+ }
+ };
+ var osmSemipavedTags = {
+ "surface": {
+ "cobblestone": true,
+ "cobblestone:flattened": true,
+ "unhewn_cobblestone": true,
+ "sett": true,
+ "paving_stones": true,
+ "metal": true,
+ "wood": true
+ }
+ };
+ var osmRightSideIsInsideTags = {
+ "natural": {
+ "cliff": true,
+ "coastline": "coastline"
+ },
+ "barrier": {
+ "retaining_wall": true,
+ "kerb": true,
+ "guard_rail": true,
+ "city_wall": true
+ },
+ "man_made": {
+ "embankment": true
+ },
+ "waterway": {
+ "weir": true
+ }
+ };
+ var osmRoutableHighwayTagValues = {
+ motorway: true,
+ trunk: true,
+ primary: true,
+ secondary: true,
+ tertiary: true,
+ residential: true,
+ motorway_link: true,
+ trunk_link: true,
+ primary_link: true,
+ secondary_link: true,
+ tertiary_link: true,
+ unclassified: true,
+ road: true,
+ service: true,
+ track: true,
+ living_street: true,
+ bus_guideway: true,
+ path: true,
+ footway: true,
+ cycleway: true,
+ bridleway: true,
+ pedestrian: true,
+ corridor: true,
+ steps: true
+ };
+ var osmPathHighwayTagValues = {
+ path: true,
+ footway: true,
+ cycleway: true,
+ bridleway: true,
+ pedestrian: true,
+ corridor: true,
+ steps: true
+ };
+ var osmRailwayTrackTagValues = {
+ rail: true,
+ light_rail: true,
+ tram: true,
+ subway: true,
+ monorail: true,
+ funicular: true,
+ miniature: true,
+ narrow_gauge: true,
+ disused: true,
+ preserved: true
+ };
+ var osmFlowingWaterwayTagValues = {
+ canal: true,
+ ditch: true,
+ drain: true,
+ fish_pass: true,
+ river: true,
+ stream: true,
+ tidal_channel: true
+ };
+
+ // node_modules/d3-array/src/ascending.js
+ function ascending(a, b) {
+ return a == null || b == null ? NaN : a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
+ }
+
+ // node_modules/d3-array/src/descending.js
+ function descending(a, b) {
+ return a == null || b == null ? NaN : b < a ? -1 : b > a ? 1 : b >= a ? 0 : NaN;
+ }
+
+ // node_modules/d3-array/src/bisector.js
+ function bisector(f2) {
+ let compare1, compare2, delta;
+ if (f2.length !== 2) {
+ compare1 = ascending;
+ compare2 = (d, x) => ascending(f2(d), x);
+ delta = (d, x) => f2(d) - x;
+ } else {
+ compare1 = f2 === ascending || f2 === descending ? f2 : zero;
+ compare2 = f2;
+ delta = f2;
+ }
+ function left(a, x, lo = 0, hi = a.length) {
+ if (lo < hi) {
+ if (compare1(x, x) !== 0)
+ return hi;
+ do {
+ const mid = lo + hi >>> 1;
+ if (compare2(a[mid], x) < 0)
+ lo = mid + 1;
+ else
+ hi = mid;
+ } while (lo < hi);
+ }
+ return lo;
+ }
+ function right(a, x, lo = 0, hi = a.length) {
+ if (lo < hi) {
+ if (compare1(x, x) !== 0)
+ return hi;
+ do {
+ const mid = lo + hi >>> 1;
+ if (compare2(a[mid], x) <= 0)
+ lo = mid + 1;
+ else
+ hi = mid;
+ } while (lo < hi);
+ }
+ return lo;
+ }
+ function center(a, x, lo = 0, hi = a.length) {
+ const i2 = left(a, x, lo, hi - 1);
+ return i2 > lo && delta(a[i2 - 1], x) > -delta(a[i2], x) ? i2 - 1 : i2;
+ }
+ return { left, center, right };
+ }
+ function zero() {
+ return 0;
+ }
+
+ // node_modules/d3-array/src/number.js
+ function number(x) {
+ return x === null ? NaN : +x;
+ }
+ function* numbers(values, valueof) {
+ if (valueof === void 0) {
+ for (let value of values) {
+ if (value != null && (value = +value) >= value) {
+ yield value;
+ }
+ }
+ } else {
+ let index = -1;
+ for (let value of values) {
+ if ((value = valueof(value, ++index, values)) != null && (value = +value) >= value) {
+ yield value;
+ }
+ }
+ }
+ }
+
+ // node_modules/d3-array/src/bisect.js
+ var ascendingBisect = bisector(ascending);
+ var bisectRight = ascendingBisect.right;
+ var bisectLeft = ascendingBisect.left;
+ var bisectCenter = bisector(number).center;
+ var bisect_default = bisectRight;
+
+ // node_modules/d3-array/src/fsum.js
+ var Adder = class {
+ constructor() {
+ this._partials = new Float64Array(32);
+ this._n = 0;
+ }
+ add(x) {
+ const p = this._partials;
+ let i2 = 0;
+ for (let j2 = 0; j2 < this._n && j2 < 32; j2++) {
+ const y = p[j2], hi = x + y, lo = Math.abs(x) < Math.abs(y) ? x - (hi - y) : y - (hi - x);
+ if (lo)
+ p[i2++] = lo;
+ x = hi;
+ }
+ p[i2] = x;
+ this._n = i2 + 1;
+ return this;
+ }
+ valueOf() {
+ const p = this._partials;
+ let n2 = this._n, x, y, lo, hi = 0;
+ if (n2 > 0) {
+ hi = p[--n2];
+ while (n2 > 0) {
+ x = hi;
+ y = p[--n2];
+ hi = x + y;
+ lo = y - (hi - x);
+ if (lo)
+ break;
+ }
+ if (n2 > 0 && (lo < 0 && p[n2 - 1] < 0 || lo > 0 && p[n2 - 1] > 0)) {
+ y = lo * 2;
+ x = hi + y;
+ if (y == x - hi)
+ hi = x;
+ }
+ }
+ return hi;
+ }
+ };
+
+ // node_modules/d3-array/src/sort.js
+ function compareDefined(compare = ascending) {
+ if (compare === ascending)
+ return ascendingDefined;
+ if (typeof compare !== "function")
+ throw new TypeError("compare is not a function");
+ return (a, b) => {
+ const x = compare(a, b);
+ if (x || x === 0)
+ return x;
+ return (compare(b, b) === 0) - (compare(a, a) === 0);
+ };
+ }
+ function ascendingDefined(a, b) {
+ return (a == null || !(a >= a)) - (b == null || !(b >= b)) || (a < b ? -1 : a > b ? 1 : 0);
+ }
+
+ // node_modules/d3-array/src/ticks.js
+ var e10 = Math.sqrt(50);
+ var e5 = Math.sqrt(10);
+ var e2 = Math.sqrt(2);
+ function ticks(start2, stop, count) {
+ var reverse, i2 = -1, n2, ticks2, step;
+ stop = +stop, start2 = +start2, count = +count;
+ if (start2 === stop && count > 0)
+ return [start2];
+ if (reverse = stop < start2)
+ n2 = start2, start2 = stop, stop = n2;
+ if ((step = tickIncrement(start2, stop, count)) === 0 || !isFinite(step))
+ return [];
+ if (step > 0) {
+ let r0 = Math.round(start2 / step), r1 = Math.round(stop / step);
+ if (r0 * step < start2)
+ ++r0;
+ if (r1 * step > stop)
+ --r1;
+ ticks2 = new Array(n2 = r1 - r0 + 1);
+ while (++i2 < n2)
+ ticks2[i2] = (r0 + i2) * step;
+ } else {
+ step = -step;
+ let r0 = Math.round(start2 * step), r1 = Math.round(stop * step);
+ if (r0 / step < start2)
+ ++r0;
+ if (r1 / step > stop)
+ --r1;
+ ticks2 = new Array(n2 = r1 - r0 + 1);
+ while (++i2 < n2)
+ ticks2[i2] = (r0 + i2) / step;
+ }
+ if (reverse)
+ ticks2.reverse();
+ return ticks2;
+ }
+ function tickIncrement(start2, stop, count) {
+ var step = (stop - start2) / Math.max(0, count), power = Math.floor(Math.log(step) / Math.LN10), error = step / Math.pow(10, power);
+ return power >= 0 ? (error >= e10 ? 10 : error >= e5 ? 5 : error >= e2 ? 2 : 1) * Math.pow(10, power) : -Math.pow(10, -power) / (error >= e10 ? 10 : error >= e5 ? 5 : error >= e2 ? 2 : 1);
+ }
+ function tickStep(start2, stop, count) {
+ var step0 = Math.abs(stop - start2) / Math.max(0, count), step1 = Math.pow(10, Math.floor(Math.log(step0) / Math.LN10)), error = step0 / step1;
+ if (error >= e10)
+ step1 *= 10;
+ else if (error >= e5)
+ step1 *= 5;
+ else if (error >= e2)
+ step1 *= 2;
+ return stop < start2 ? -step1 : step1;
+ }
+
+ // node_modules/d3-array/src/max.js
+ function max(values, valueof) {
+ let max3;
+ if (valueof === void 0) {
+ for (const value of values) {
+ if (value != null && (max3 < value || max3 === void 0 && value >= value)) {
+ max3 = value;
+ }
+ }
+ } else {
+ let index = -1;
+ for (let value of values) {
+ if ((value = valueof(value, ++index, values)) != null && (max3 < value || max3 === void 0 && value >= value)) {
+ max3 = value;
+ }
+ }
+ }
+ return max3;
+ }
+
+ // node_modules/d3-array/src/min.js
+ function min(values, valueof) {
+ let min3;
+ if (valueof === void 0) {
+ for (const value of values) {
+ if (value != null && (min3 > value || min3 === void 0 && value >= value)) {
+ min3 = value;
+ }
+ }
+ } else {
+ let index = -1;
+ for (let value of values) {
+ if ((value = valueof(value, ++index, values)) != null && (min3 > value || min3 === void 0 && value >= value)) {
+ min3 = value;
+ }
+ }
+ }
+ return min3;
+ }
+
+ // node_modules/d3-array/src/quickselect.js
+ function quickselect(array2, k, left = 0, right = array2.length - 1, compare) {
+ compare = compare === void 0 ? ascendingDefined : compareDefined(compare);
+ while (right > left) {
+ if (right - left > 600) {
+ const n2 = right - left + 1;
+ const m = k - left + 1;
+ const z = Math.log(n2);
+ const s = 0.5 * Math.exp(2 * z / 3);
+ const sd = 0.5 * Math.sqrt(z * s * (n2 - s) / n2) * (m - n2 / 2 < 0 ? -1 : 1);
+ const newLeft = Math.max(left, Math.floor(k - m * s / n2 + sd));
+ const newRight = Math.min(right, Math.floor(k + (n2 - m) * s / n2 + sd));
+ quickselect(array2, k, newLeft, newRight, compare);
+ }
+ const t = array2[k];
+ let i2 = left;
+ let j2 = right;
+ swap(array2, left, k);
+ if (compare(array2[right], t) > 0)
+ swap(array2, left, right);
+ while (i2 < j2) {
+ swap(array2, i2, j2), ++i2, --j2;
+ while (compare(array2[i2], t) < 0)
+ ++i2;
+ while (compare(array2[j2], t) > 0)
+ --j2;
+ }
+ if (compare(array2[left], t) === 0)
+ swap(array2, left, j2);
+ else
+ ++j2, swap(array2, j2, right);
+ if (j2 <= k)
+ left = j2 + 1;
+ if (k <= j2)
+ right = j2 - 1;
+ }
+ return array2;
+ }
+ function swap(array2, i2, j2) {
+ const t = array2[i2];
+ array2[i2] = array2[j2];
+ array2[j2] = t;
+ }
+
+ // node_modules/d3-array/src/quantile.js
+ function quantile(values, p, valueof) {
+ values = Float64Array.from(numbers(values, valueof));
+ if (!(n2 = values.length))
+ return;
+ if ((p = +p) <= 0 || n2 < 2)
+ return min(values);
+ if (p >= 1)
+ return max(values);
+ var n2, i2 = (n2 - 1) * p, i0 = Math.floor(i2), value0 = max(quickselect(values, i0).subarray(0, i0 + 1)), value1 = min(values.subarray(i0 + 1));
+ return value0 + (value1 - value0) * (i2 - i0);
+ }
+
+ // node_modules/d3-array/src/median.js
+ function median(values, valueof) {
+ return quantile(values, 0.5, valueof);
+ }
+
+ // node_modules/d3-array/src/merge.js
+ function* flatten(arrays) {
+ for (const array2 of arrays) {
+ yield* array2;
+ }
+ }
+ function merge(arrays) {
+ return Array.from(flatten(arrays));
+ }
+
+ // node_modules/d3-array/src/range.js
+ function range(start2, stop, step) {
+ start2 = +start2, stop = +stop, step = (n2 = arguments.length) < 2 ? (stop = start2, start2 = 0, 1) : n2 < 3 ? 1 : +step;
+ var i2 = -1, n2 = Math.max(0, Math.ceil((stop - start2) / step)) | 0, range3 = new Array(n2);
+ while (++i2 < n2) {
+ range3[i2] = start2 + i2 * step;
+ }
+ return range3;
+ }
+
+ // node_modules/d3-geo/src/math.js
+ var epsilon = 1e-6;
+ var epsilon2 = 1e-12;
+ var pi = Math.PI;
+ var halfPi = pi / 2;
+ var quarterPi = pi / 4;
+ var tau = pi * 2;
+ var degrees = 180 / pi;
+ var radians = pi / 180;
+ var abs = Math.abs;
+ var atan = Math.atan;
+ var atan2 = Math.atan2;
+ var cos = Math.cos;
+ var exp = Math.exp;
+ var log = Math.log;
+ var sin = Math.sin;
+ var sign = Math.sign || function(x) {
+ return x > 0 ? 1 : x < 0 ? -1 : 0;
+ };
+ var sqrt = Math.sqrt;
+ var tan = Math.tan;
+ function acos(x) {
+ return x > 1 ? 0 : x < -1 ? pi : Math.acos(x);
+ }
+ function asin(x) {
+ return x > 1 ? halfPi : x < -1 ? -halfPi : Math.asin(x);
+ }
+
+ // node_modules/d3-geo/src/noop.js
+ function noop() {
+ }
+
+ // node_modules/d3-geo/src/stream.js
+ function streamGeometry(geometry, stream) {
+ if (geometry && streamGeometryType.hasOwnProperty(geometry.type)) {
+ streamGeometryType[geometry.type](geometry, stream);
+ }
+ }
+ var streamObjectType = {
+ Feature: function(object, stream) {
+ streamGeometry(object.geometry, stream);
+ },
+ FeatureCollection: function(object, stream) {
+ var features = object.features, i2 = -1, n2 = features.length;
+ while (++i2 < n2)
+ streamGeometry(features[i2].geometry, stream);
+ }
+ };
+ var streamGeometryType = {
+ Sphere: function(object, stream) {
+ stream.sphere();
+ },
+ Point: function(object, stream) {
+ object = object.coordinates;
+ stream.point(object[0], object[1], object[2]);
+ },
+ MultiPoint: function(object, stream) {
+ var coordinates = object.coordinates, i2 = -1, n2 = coordinates.length;
+ while (++i2 < n2)
+ object = coordinates[i2], stream.point(object[0], object[1], object[2]);
+ },
+ LineString: function(object, stream) {
+ streamLine(object.coordinates, stream, 0);
+ },
+ MultiLineString: function(object, stream) {
+ var coordinates = object.coordinates, i2 = -1, n2 = coordinates.length;
+ while (++i2 < n2)
+ streamLine(coordinates[i2], stream, 0);
+ },
+ Polygon: function(object, stream) {
+ streamPolygon(object.coordinates, stream);
+ },
+ MultiPolygon: function(object, stream) {
+ var coordinates = object.coordinates, i2 = -1, n2 = coordinates.length;
+ while (++i2 < n2)
+ streamPolygon(coordinates[i2], stream);
+ },
+ GeometryCollection: function(object, stream) {
+ var geometries = object.geometries, i2 = -1, n2 = geometries.length;
+ while (++i2 < n2)
+ streamGeometry(geometries[i2], stream);
+ }
+ };
+ function streamLine(coordinates, stream, closed) {
+ var i2 = -1, n2 = coordinates.length - closed, coordinate;
+ stream.lineStart();
+ while (++i2 < n2)
+ coordinate = coordinates[i2], stream.point(coordinate[0], coordinate[1], coordinate[2]);
+ stream.lineEnd();
+ }
+ function streamPolygon(coordinates, stream) {
+ var i2 = -1, n2 = coordinates.length;
+ stream.polygonStart();
+ while (++i2 < n2)
+ streamLine(coordinates[i2], stream, 1);
+ stream.polygonEnd();
+ }
+ function stream_default(object, stream) {
+ if (object && streamObjectType.hasOwnProperty(object.type)) {
+ streamObjectType[object.type](object, stream);
+ } else {
+ streamGeometry(object, stream);
+ }
+ }
+
+ // node_modules/d3-geo/src/area.js
+ var areaRingSum = new Adder();
+ var areaSum = new Adder();
+ var lambda00;
+ var phi00;
+ var lambda0;
+ var cosPhi0;
+ var sinPhi0;
+ var areaStream = {
+ point: noop,
+ lineStart: noop,
+ lineEnd: noop,
+ polygonStart: function() {
+ areaRingSum = new Adder();
+ areaStream.lineStart = areaRingStart;
+ areaStream.lineEnd = areaRingEnd;
+ },
+ polygonEnd: function() {
+ var areaRing = +areaRingSum;
+ areaSum.add(areaRing < 0 ? tau + areaRing : areaRing);
+ this.lineStart = this.lineEnd = this.point = noop;
+ },
+ sphere: function() {
+ areaSum.add(tau);
+ }
+ };
+ function areaRingStart() {
+ areaStream.point = areaPointFirst;
+ }
+ function areaRingEnd() {
+ areaPoint(lambda00, phi00);
+ }
+ function areaPointFirst(lambda, phi) {
+ areaStream.point = areaPoint;
+ lambda00 = lambda, phi00 = phi;
+ lambda *= radians, phi *= radians;
+ lambda0 = lambda, cosPhi0 = cos(phi = phi / 2 + quarterPi), sinPhi0 = sin(phi);
+ }
+ function areaPoint(lambda, phi) {
+ lambda *= radians, phi *= radians;
+ phi = phi / 2 + quarterPi;
+ var dLambda = lambda - lambda0, sdLambda = dLambda >= 0 ? 1 : -1, adLambda = sdLambda * dLambda, cosPhi = cos(phi), sinPhi = sin(phi), k = sinPhi0 * sinPhi, u = cosPhi0 * cosPhi + k * cos(adLambda), v = k * sdLambda * sin(adLambda);
+ areaRingSum.add(atan2(v, u));
+ lambda0 = lambda, cosPhi0 = cosPhi, sinPhi0 = sinPhi;
+ }
+ function area_default(object) {
+ areaSum = new Adder();
+ stream_default(object, areaStream);
+ return areaSum * 2;
+ }
+
+ // node_modules/d3-geo/src/cartesian.js
+ function spherical(cartesian2) {
+ return [atan2(cartesian2[1], cartesian2[0]), asin(cartesian2[2])];
+ }
+ function cartesian(spherical2) {
+ var lambda = spherical2[0], phi = spherical2[1], cosPhi = cos(phi);
+ return [cosPhi * cos(lambda), cosPhi * sin(lambda), sin(phi)];
+ }
+ function cartesianDot(a, b) {
+ return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
+ }
+ function cartesianCross(a, b) {
+ return [a[1] * b[2] - a[2] * b[1], a[2] * b[0] - a[0] * b[2], a[0] * b[1] - a[1] * b[0]];
+ }
+ function cartesianAddInPlace(a, b) {
+ a[0] += b[0], a[1] += b[1], a[2] += b[2];
+ }
+ function cartesianScale(vector, k) {
+ return [vector[0] * k, vector[1] * k, vector[2] * k];
+ }
+ function cartesianNormalizeInPlace(d) {
+ var l = sqrt(d[0] * d[0] + d[1] * d[1] + d[2] * d[2]);
+ d[0] /= l, d[1] /= l, d[2] /= l;
+ }
+
+ // node_modules/d3-geo/src/bounds.js
+ var lambda02;
+ var phi0;
+ var lambda1;
+ var phi1;
+ var lambda2;
+ var lambda002;
+ var phi002;
+ var p0;
+ var deltaSum;
+ var ranges;
+ var range2;
+ var boundsStream = {
+ point: boundsPoint,
+ lineStart: boundsLineStart,
+ lineEnd: boundsLineEnd,
+ polygonStart: function() {
+ boundsStream.point = boundsRingPoint;
+ boundsStream.lineStart = boundsRingStart;
+ boundsStream.lineEnd = boundsRingEnd;
+ deltaSum = new Adder();
+ areaStream.polygonStart();
+ },
+ polygonEnd: function() {
+ areaStream.polygonEnd();
+ boundsStream.point = boundsPoint;
+ boundsStream.lineStart = boundsLineStart;
+ boundsStream.lineEnd = boundsLineEnd;
+ if (areaRingSum < 0)
+ lambda02 = -(lambda1 = 180), phi0 = -(phi1 = 90);
+ else if (deltaSum > epsilon)
+ phi1 = 90;
+ else if (deltaSum < -epsilon)
+ phi0 = -90;
+ range2[0] = lambda02, range2[1] = lambda1;
+ },
+ sphere: function() {
+ lambda02 = -(lambda1 = 180), phi0 = -(phi1 = 90);
+ }
+ };
+ function boundsPoint(lambda, phi) {
+ ranges.push(range2 = [lambda02 = lambda, lambda1 = lambda]);
+ if (phi < phi0)
+ phi0 = phi;
+ if (phi > phi1)
+ phi1 = phi;
+ }
+ function linePoint(lambda, phi) {
+ var p = cartesian([lambda * radians, phi * radians]);
+ if (p0) {
+ var normal = cartesianCross(p0, p), equatorial = [normal[1], -normal[0], 0], inflection = cartesianCross(equatorial, normal);
+ cartesianNormalizeInPlace(inflection);
+ inflection = spherical(inflection);
+ var delta = lambda - lambda2, sign2 = delta > 0 ? 1 : -1, lambdai = inflection[0] * degrees * sign2, phii, antimeridian = abs(delta) > 180;
+ if (antimeridian ^ (sign2 * lambda2 < lambdai && lambdai < sign2 * lambda)) {
+ phii = inflection[1] * degrees;
+ if (phii > phi1)
+ phi1 = phii;
+ } else if (lambdai = (lambdai + 360) % 360 - 180, antimeridian ^ (sign2 * lambda2 < lambdai && lambdai < sign2 * lambda)) {
+ phii = -inflection[1] * degrees;
+ if (phii < phi0)
+ phi0 = phii;
+ } else {
+ if (phi < phi0)
+ phi0 = phi;
+ if (phi > phi1)
+ phi1 = phi;
+ }
+ if (antimeridian) {
+ if (lambda < lambda2) {
+ if (angle(lambda02, lambda) > angle(lambda02, lambda1))
+ lambda1 = lambda;
+ } else {
+ if (angle(lambda, lambda1) > angle(lambda02, lambda1))
+ lambda02 = lambda;
+ }
+ } else {
+ if (lambda1 >= lambda02) {
+ if (lambda < lambda02)
+ lambda02 = lambda;
+ if (lambda > lambda1)
+ lambda1 = lambda;
+ } else {
+ if (lambda > lambda2) {
+ if (angle(lambda02, lambda) > angle(lambda02, lambda1))
+ lambda1 = lambda;
+ } else {
+ if (angle(lambda, lambda1) > angle(lambda02, lambda1))
+ lambda02 = lambda;
+ }
+ }
+ }
+ } else {
+ ranges.push(range2 = [lambda02 = lambda, lambda1 = lambda]);
+ }
+ if (phi < phi0)
+ phi0 = phi;
+ if (phi > phi1)
+ phi1 = phi;
+ p0 = p, lambda2 = lambda;
+ }
+ function boundsLineStart() {
+ boundsStream.point = linePoint;
+ }
+ function boundsLineEnd() {
+ range2[0] = lambda02, range2[1] = lambda1;
+ boundsStream.point = boundsPoint;
+ p0 = null;
+ }
+ function boundsRingPoint(lambda, phi) {
+ if (p0) {
+ var delta = lambda - lambda2;
+ deltaSum.add(abs(delta) > 180 ? delta + (delta > 0 ? 360 : -360) : delta);
+ } else {
+ lambda002 = lambda, phi002 = phi;
+ }
+ areaStream.point(lambda, phi);
+ linePoint(lambda, phi);
+ }
+ function boundsRingStart() {
+ areaStream.lineStart();
+ }
+ function boundsRingEnd() {
+ boundsRingPoint(lambda002, phi002);
+ areaStream.lineEnd();
+ if (abs(deltaSum) > epsilon)
+ lambda02 = -(lambda1 = 180);
+ range2[0] = lambda02, range2[1] = lambda1;
+ p0 = null;
+ }
+ function angle(lambda04, lambda12) {
+ return (lambda12 -= lambda04) < 0 ? lambda12 + 360 : lambda12;
+ }
+ function rangeCompare(a, b) {
+ return a[0] - b[0];
+ }
+ function rangeContains(range3, x) {
+ return range3[0] <= range3[1] ? range3[0] <= x && x <= range3[1] : x < range3[0] || range3[1] < x;
+ }
+ function bounds_default(feature3) {
+ var i2, n2, a, b, merged, deltaMax, delta;
+ phi1 = lambda1 = -(lambda02 = phi0 = Infinity);
+ ranges = [];
+ stream_default(feature3, boundsStream);
+ if (n2 = ranges.length) {
+ ranges.sort(rangeCompare);
+ for (i2 = 1, a = ranges[0], merged = [a]; i2 < n2; ++i2) {
+ b = ranges[i2];
+ if (rangeContains(a, b[0]) || rangeContains(a, b[1])) {
+ if (angle(a[0], b[1]) > angle(a[0], a[1]))
+ a[1] = b[1];
+ if (angle(b[0], a[1]) > angle(a[0], a[1]))
+ a[0] = b[0];
+ } else {
+ merged.push(a = b);
+ }
+ }
+ for (deltaMax = -Infinity, n2 = merged.length - 1, i2 = 0, a = merged[n2]; i2 <= n2; a = b, ++i2) {
+ b = merged[i2];
+ if ((delta = angle(a[1], b[0])) > deltaMax)
+ deltaMax = delta, lambda02 = b[0], lambda1 = a[1];
+ }
+ }
+ ranges = range2 = null;
+ return lambda02 === Infinity || phi0 === Infinity ? [[NaN, NaN], [NaN, NaN]] : [[lambda02, phi0], [lambda1, phi1]];
+ }
+
+ // node_modules/d3-geo/src/compose.js
+ function compose_default(a, b) {
+ function compose(x, y) {
+ return x = a(x, y), b(x[0], x[1]);
+ }
+ if (a.invert && b.invert)
+ compose.invert = function(x, y) {
+ return x = b.invert(x, y), x && a.invert(x[0], x[1]);
+ };
+ return compose;
+ }
+
+ // node_modules/d3-geo/src/rotation.js
+ function rotationIdentity(lambda, phi) {
+ return [abs(lambda) > pi ? lambda + Math.round(-lambda / tau) * tau : lambda, phi];
+ }
+ rotationIdentity.invert = rotationIdentity;
+ function rotateRadians(deltaLambda, deltaPhi, deltaGamma) {
+ return (deltaLambda %= tau) ? deltaPhi || deltaGamma ? compose_default(rotationLambda(deltaLambda), rotationPhiGamma(deltaPhi, deltaGamma)) : rotationLambda(deltaLambda) : deltaPhi || deltaGamma ? rotationPhiGamma(deltaPhi, deltaGamma) : rotationIdentity;
+ }
+ function forwardRotationLambda(deltaLambda) {
+ return function(lambda, phi) {
+ return lambda += deltaLambda, [lambda > pi ? lambda - tau : lambda < -pi ? lambda + tau : lambda, phi];
+ };
+ }
+ function rotationLambda(deltaLambda) {
+ var rotation = forwardRotationLambda(deltaLambda);
+ rotation.invert = forwardRotationLambda(-deltaLambda);
+ return rotation;
+ }
+ function rotationPhiGamma(deltaPhi, deltaGamma) {
+ var cosDeltaPhi = cos(deltaPhi), sinDeltaPhi = sin(deltaPhi), cosDeltaGamma = cos(deltaGamma), sinDeltaGamma = sin(deltaGamma);
+ function rotation(lambda, phi) {
+ var cosPhi = cos(phi), x = cos(lambda) * cosPhi, y = sin(lambda) * cosPhi, z = sin(phi), k = z * cosDeltaPhi + x * sinDeltaPhi;
+ return [
+ atan2(y * cosDeltaGamma - k * sinDeltaGamma, x * cosDeltaPhi - z * sinDeltaPhi),
+ asin(k * cosDeltaGamma + y * sinDeltaGamma)
+ ];
+ }
+ rotation.invert = function(lambda, phi) {
+ var cosPhi = cos(phi), x = cos(lambda) * cosPhi, y = sin(lambda) * cosPhi, z = sin(phi), k = z * cosDeltaGamma - y * sinDeltaGamma;
+ return [
+ atan2(y * cosDeltaGamma + z * sinDeltaGamma, x * cosDeltaPhi + k * sinDeltaPhi),
+ asin(k * cosDeltaPhi - x * sinDeltaPhi)
+ ];
+ };
+ return rotation;
+ }
+ function rotation_default(rotate) {
+ rotate = rotateRadians(rotate[0] * radians, rotate[1] * radians, rotate.length > 2 ? rotate[2] * radians : 0);
+ function forward(coordinates) {
+ coordinates = rotate(coordinates[0] * radians, coordinates[1] * radians);
+ return coordinates[0] *= degrees, coordinates[1] *= degrees, coordinates;
+ }
+ forward.invert = function(coordinates) {
+ coordinates = rotate.invert(coordinates[0] * radians, coordinates[1] * radians);
+ return coordinates[0] *= degrees, coordinates[1] *= degrees, coordinates;
+ };
+ return forward;
+ }
+
+ // node_modules/d3-geo/src/circle.js
+ function circleStream(stream, radius, delta, direction, t0, t1) {
+ if (!delta)
+ return;
+ var cosRadius = cos(radius), sinRadius = sin(radius), step = direction * delta;
+ if (t0 == null) {
+ t0 = radius + direction * tau;
+ t1 = radius - step / 2;
+ } else {
+ t0 = circleRadius(cosRadius, t0);
+ t1 = circleRadius(cosRadius, t1);
+ if (direction > 0 ? t0 < t1 : t0 > t1)
+ t0 += direction * tau;
+ }
+ for (var point2, t = t0; direction > 0 ? t > t1 : t < t1; t -= step) {
+ point2 = spherical([cosRadius, -sinRadius * cos(t), -sinRadius * sin(t)]);
+ stream.point(point2[0], point2[1]);
+ }
+ }
+ function circleRadius(cosRadius, point2) {
+ point2 = cartesian(point2), point2[0] -= cosRadius;
+ cartesianNormalizeInPlace(point2);
+ var radius = acos(-point2[1]);
+ return ((-point2[2] < 0 ? -radius : radius) + tau - epsilon) % tau;
+ }
+
+ // node_modules/d3-geo/src/clip/buffer.js
+ function buffer_default() {
+ var lines = [], line;
+ return {
+ point: function(x, y, m) {
+ line.push([x, y, m]);
+ },
+ lineStart: function() {
+ lines.push(line = []);
+ },
+ lineEnd: noop,
+ rejoin: function() {
+ if (lines.length > 1)
+ lines.push(lines.pop().concat(lines.shift()));
+ },
+ result: function() {
+ var result = lines;
+ lines = [];
+ line = null;
+ return result;
+ }
+ };
+ }
+
+ // node_modules/d3-geo/src/pointEqual.js
+ function pointEqual_default(a, b) {
+ return abs(a[0] - b[0]) < epsilon && abs(a[1] - b[1]) < epsilon;
+ }
+
+ // node_modules/d3-geo/src/clip/rejoin.js
+ function Intersection(point2, points, other, entry) {
+ this.x = point2;
+ this.z = points;
+ this.o = other;
+ this.e = entry;
+ this.v = false;
+ this.n = this.p = null;
+ }
+ function rejoin_default(segments, compareIntersection2, startInside, interpolate, stream) {
+ var subject = [], clip = [], i2, n2;
+ segments.forEach(function(segment) {
+ if ((n3 = segment.length - 1) <= 0)
+ return;
+ var n3, p02 = segment[0], p1 = segment[n3], x;
+ if (pointEqual_default(p02, p1)) {
+ if (!p02[2] && !p1[2]) {
+ stream.lineStart();
+ for (i2 = 0; i2 < n3; ++i2)
+ stream.point((p02 = segment[i2])[0], p02[1]);
+ stream.lineEnd();
+ return;
+ }
+ p1[0] += 2 * epsilon;
+ }
+ subject.push(x = new Intersection(p02, segment, null, true));
+ clip.push(x.o = new Intersection(p02, null, x, false));
+ subject.push(x = new Intersection(p1, segment, null, false));
+ clip.push(x.o = new Intersection(p1, null, x, true));
+ });
+ if (!subject.length)
+ return;
+ clip.sort(compareIntersection2);
+ link(subject);
+ link(clip);
+ for (i2 = 0, n2 = clip.length; i2 < n2; ++i2) {
+ clip[i2].e = startInside = !startInside;
+ }
+ var start2 = subject[0], points, point2;
+ while (1) {
+ var current = start2, isSubject = true;
+ while (current.v)
+ if ((current = current.n) === start2)
+ return;
+ points = current.z;
+ stream.lineStart();
+ do {
+ current.v = current.o.v = true;
+ if (current.e) {
+ if (isSubject) {
+ for (i2 = 0, n2 = points.length; i2 < n2; ++i2)
+ stream.point((point2 = points[i2])[0], point2[1]);
+ } else {
+ interpolate(current.x, current.n.x, 1, stream);
+ }
+ current = current.n;
+ } else {
+ if (isSubject) {
+ points = current.p.z;
+ for (i2 = points.length - 1; i2 >= 0; --i2)
+ stream.point((point2 = points[i2])[0], point2[1]);
+ } else {
+ interpolate(current.x, current.p.x, -1, stream);
+ }
+ current = current.p;
+ }
+ current = current.o;
+ points = current.z;
+ isSubject = !isSubject;
+ } while (!current.v);
+ stream.lineEnd();
+ }
+ }
+ function link(array2) {
+ if (!(n2 = array2.length))
+ return;
+ var n2, i2 = 0, a = array2[0], b;
+ while (++i2 < n2) {
+ a.n = b = array2[i2];
+ b.p = a;
+ a = b;
+ }
+ a.n = b = array2[0];
+ b.p = a;
+ }
+
+ // node_modules/d3-geo/src/polygonContains.js
+ function longitude(point2) {
+ return abs(point2[0]) <= pi ? point2[0] : sign(point2[0]) * ((abs(point2[0]) + pi) % tau - pi);
+ }
+ function polygonContains_default(polygon2, point2) {
+ var lambda = longitude(point2), phi = point2[1], sinPhi = sin(phi), normal = [sin(lambda), -cos(lambda), 0], angle2 = 0, winding = 0;
+ var sum = new Adder();
+ if (sinPhi === 1)
+ phi = halfPi + epsilon;
+ else if (sinPhi === -1)
+ phi = -halfPi - epsilon;
+ for (var i2 = 0, n2 = polygon2.length; i2 < n2; ++i2) {
+ if (!(m = (ring = polygon2[i2]).length))
+ continue;
+ var ring, m, point0 = ring[m - 1], lambda04 = longitude(point0), phi02 = point0[1] / 2 + quarterPi, sinPhi03 = sin(phi02), cosPhi03 = cos(phi02);
+ for (var j2 = 0; j2 < m; ++j2, lambda04 = lambda12, sinPhi03 = sinPhi1, cosPhi03 = cosPhi1, point0 = point1) {
+ var point1 = ring[j2], lambda12 = longitude(point1), phi12 = point1[1] / 2 + quarterPi, sinPhi1 = sin(phi12), cosPhi1 = cos(phi12), delta = lambda12 - lambda04, sign2 = delta >= 0 ? 1 : -1, absDelta = sign2 * delta, antimeridian = absDelta > pi, k = sinPhi03 * sinPhi1;
+ sum.add(atan2(k * sign2 * sin(absDelta), cosPhi03 * cosPhi1 + k * cos(absDelta)));
+ angle2 += antimeridian ? delta + sign2 * tau : delta;
+ if (antimeridian ^ lambda04 >= lambda ^ lambda12 >= lambda) {
+ var arc = cartesianCross(cartesian(point0), cartesian(point1));
+ cartesianNormalizeInPlace(arc);
+ var intersection = cartesianCross(normal, arc);
+ cartesianNormalizeInPlace(intersection);
+ var phiArc = (antimeridian ^ delta >= 0 ? -1 : 1) * asin(intersection[2]);
+ if (phi > phiArc || phi === phiArc && (arc[0] || arc[1])) {
+ winding += antimeridian ^ delta >= 0 ? 1 : -1;
+ }
+ }
+ }
+ }
+ return (angle2 < -epsilon || angle2 < epsilon && sum < -epsilon2) ^ winding & 1;
+ }
+
+ // node_modules/d3-geo/src/clip/index.js
+ function clip_default(pointVisible, clipLine, interpolate, start2) {
+ return function(sink) {
+ var line = clipLine(sink), ringBuffer = buffer_default(), ringSink = clipLine(ringBuffer), polygonStarted = false, polygon2, segments, ring;
+ var clip = {
+ point: point2,
+ lineStart,
+ lineEnd,
+ polygonStart: function() {
+ clip.point = pointRing;
+ clip.lineStart = ringStart;
+ clip.lineEnd = ringEnd;
+ segments = [];
+ polygon2 = [];
+ },
+ polygonEnd: function() {
+ clip.point = point2;
+ clip.lineStart = lineStart;
+ clip.lineEnd = lineEnd;
+ segments = merge(segments);
+ var startInside = polygonContains_default(polygon2, start2);
+ if (segments.length) {
+ if (!polygonStarted)
+ sink.polygonStart(), polygonStarted = true;
+ rejoin_default(segments, compareIntersection, startInside, interpolate, sink);
+ } else if (startInside) {
+ if (!polygonStarted)
+ sink.polygonStart(), polygonStarted = true;
+ sink.lineStart();
+ interpolate(null, null, 1, sink);
+ sink.lineEnd();
+ }
+ if (polygonStarted)
+ sink.polygonEnd(), polygonStarted = false;
+ segments = polygon2 = null;
+ },
+ sphere: function() {
+ sink.polygonStart();
+ sink.lineStart();
+ interpolate(null, null, 1, sink);
+ sink.lineEnd();
+ sink.polygonEnd();
+ }
+ };
+ function point2(lambda, phi) {
+ if (pointVisible(lambda, phi))
+ sink.point(lambda, phi);
+ }
+ function pointLine(lambda, phi) {
+ line.point(lambda, phi);
+ }
+ function lineStart() {
+ clip.point = pointLine;
+ line.lineStart();
+ }
+ function lineEnd() {
+ clip.point = point2;
+ line.lineEnd();
+ }
+ function pointRing(lambda, phi) {
+ ring.push([lambda, phi]);
+ ringSink.point(lambda, phi);
+ }
+ function ringStart() {
+ ringSink.lineStart();
+ ring = [];
+ }
+ function ringEnd() {
+ pointRing(ring[0][0], ring[0][1]);
+ ringSink.lineEnd();
+ var clean2 = ringSink.clean(), ringSegments = ringBuffer.result(), i2, n2 = ringSegments.length, m, segment, point3;
+ ring.pop();
+ polygon2.push(ring);
+ ring = null;
+ if (!n2)
+ return;
+ if (clean2 & 1) {
+ segment = ringSegments[0];
+ if ((m = segment.length - 1) > 0) {
+ if (!polygonStarted)
+ sink.polygonStart(), polygonStarted = true;
+ sink.lineStart();
+ for (i2 = 0; i2 < m; ++i2)
+ sink.point((point3 = segment[i2])[0], point3[1]);
+ sink.lineEnd();
+ }
+ return;
+ }
+ if (n2 > 1 && clean2 & 2)
+ ringSegments.push(ringSegments.pop().concat(ringSegments.shift()));
+ segments.push(ringSegments.filter(validSegment));
+ }
+ return clip;
+ };
+ }
+ function validSegment(segment) {
+ return segment.length > 1;
+ }
+ 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]);
+ }
+
+ // node_modules/d3-geo/src/clip/antimeridian.js
+ var antimeridian_default = clip_default(
+ function() {
+ return true;
+ },
+ clipAntimeridianLine,
+ clipAntimeridianInterpolate,
+ [-pi, -halfPi]
+ );
+ function clipAntimeridianLine(stream) {
+ var lambda04 = NaN, phi02 = NaN, sign0 = NaN, clean2;
+ return {
+ lineStart: function() {
+ stream.lineStart();
+ clean2 = 1;
+ },
+ point: function(lambda12, phi12) {
+ var sign1 = lambda12 > 0 ? pi : -pi, delta = abs(lambda12 - lambda04);
+ if (abs(delta - pi) < epsilon) {
+ stream.point(lambda04, phi02 = (phi02 + phi12) / 2 > 0 ? halfPi : -halfPi);
+ stream.point(sign0, phi02);
+ stream.lineEnd();
+ stream.lineStart();
+ stream.point(sign1, phi02);
+ stream.point(lambda12, phi02);
+ clean2 = 0;
+ } else if (sign0 !== sign1 && delta >= pi) {
+ if (abs(lambda04 - sign0) < epsilon)
+ lambda04 -= sign0 * epsilon;
+ if (abs(lambda12 - sign1) < epsilon)
+ lambda12 -= sign1 * epsilon;
+ phi02 = clipAntimeridianIntersect(lambda04, phi02, lambda12, phi12);
+ stream.point(sign0, phi02);
+ stream.lineEnd();
+ stream.lineStart();
+ stream.point(sign1, phi02);
+ clean2 = 0;
+ }
+ stream.point(lambda04 = lambda12, phi02 = phi12);
+ sign0 = sign1;
+ },
+ lineEnd: function() {
+ stream.lineEnd();
+ lambda04 = phi02 = NaN;
+ },
+ clean: function() {
+ return 2 - clean2;
+ }
+ };
+ }
+ function clipAntimeridianIntersect(lambda04, phi02, lambda12, phi12) {
+ var cosPhi03, cosPhi1, sinLambda0Lambda1 = sin(lambda04 - lambda12);
+ return abs(sinLambda0Lambda1) > epsilon ? atan((sin(phi02) * (cosPhi1 = cos(phi12)) * sin(lambda12) - sin(phi12) * (cosPhi03 = cos(phi02)) * sin(lambda04)) / (cosPhi03 * cosPhi1 * sinLambda0Lambda1)) : (phi02 + phi12) / 2;
+ }
+ function clipAntimeridianInterpolate(from, to, direction, stream) {
+ var phi;
+ if (from == null) {
+ phi = direction * halfPi;
+ stream.point(-pi, phi);
+ stream.point(0, phi);
+ stream.point(pi, phi);
+ stream.point(pi, 0);
+ stream.point(pi, -phi);
+ stream.point(0, -phi);
+ stream.point(-pi, -phi);
+ stream.point(-pi, 0);
+ stream.point(-pi, phi);
+ } else if (abs(from[0] - to[0]) > epsilon) {
+ var lambda = from[0] < to[0] ? pi : -pi;
+ phi = direction * lambda / 2;
+ stream.point(-lambda, phi);
+ stream.point(0, phi);
+ stream.point(lambda, phi);
+ } else {
+ stream.point(to[0], to[1]);
+ }
+ }
+
+ // node_modules/d3-geo/src/clip/circle.js
+ function circle_default(radius) {
+ var cr = cos(radius), delta = 6 * radians, smallRadius = cr > 0, notHemisphere = abs(cr) > epsilon;
+ function interpolate(from, to, direction, stream) {
+ circleStream(stream, radius, delta, direction, from, to);
+ }
+ function visible(lambda, phi) {
+ return cos(lambda) * cos(phi) > cr;
+ }
+ function clipLine(stream) {
+ var point0, c0, v0, v00, clean2;
+ return {
+ lineStart: function() {
+ v00 = v0 = false;
+ clean2 = 1;
+ },
+ point: function(lambda, phi) {
+ var point1 = [lambda, phi], point2, v = visible(lambda, phi), c = smallRadius ? v ? 0 : code(lambda, phi) : v ? code(lambda + (lambda < 0 ? pi : -pi), phi) : 0;
+ if (!point0 && (v00 = v0 = v))
+ stream.lineStart();
+ if (v !== v0) {
+ point2 = intersect2(point0, point1);
+ if (!point2 || pointEqual_default(point0, point2) || pointEqual_default(point1, point2))
+ point1[2] = 1;
+ }
+ if (v !== v0) {
+ clean2 = 0;
+ if (v) {
+ stream.lineStart();
+ point2 = intersect2(point1, point0);
+ stream.point(point2[0], point2[1]);
+ } else {
+ point2 = intersect2(point0, point1);
+ stream.point(point2[0], point2[1], 2);
+ stream.lineEnd();
+ }
+ point0 = point2;
+ } else if (notHemisphere && point0 && smallRadius ^ v) {
+ var t;
+ if (!(c & c0) && (t = intersect2(point1, point0, true))) {
+ clean2 = 0;
+ if (smallRadius) {
+ stream.lineStart();
+ stream.point(t[0][0], t[0][1]);
+ stream.point(t[1][0], t[1][1]);
+ stream.lineEnd();
+ } else {
+ stream.point(t[1][0], t[1][1]);
+ stream.lineEnd();
+ stream.lineStart();
+ stream.point(t[0][0], t[0][1], 3);
+ }
+ }
+ }
+ if (v && (!point0 || !pointEqual_default(point0, point1))) {
+ stream.point(point1[0], point1[1]);
+ }
+ point0 = point1, v0 = v, c0 = c;
+ },
+ lineEnd: function() {
+ if (v0)
+ stream.lineEnd();
+ point0 = null;
+ },
+ // Rejoin first and last segments if there were intersections and the first
+ // and last points were visible.
+ clean: function() {
+ return clean2 | (v00 && v0) << 1;
+ }
+ };
+ }
+ function intersect2(a, b, two) {
+ var pa = cartesian(a), pb = cartesian(b);
+ var n1 = [1, 0, 0], n2 = cartesianCross(pa, pb), n2n2 = cartesianDot(n2, n2), n1n2 = n2[0], determinant = n2n2 - n1n2 * n1n2;
+ if (!determinant)
+ return !two && a;
+ var c1 = cr * n2n2 / determinant, c2 = -cr * n1n2 / determinant, n1xn2 = cartesianCross(n1, n2), A = cartesianScale(n1, c1), B = cartesianScale(n2, c2);
+ cartesianAddInPlace(A, B);
+ var u = n1xn2, w = cartesianDot(A, u), uu = cartesianDot(u, u), t2 = w * w - uu * (cartesianDot(A, A) - 1);
+ if (t2 < 0)
+ return;
+ var t = sqrt(t2), q = cartesianScale(u, (-w - t) / uu);
+ cartesianAddInPlace(q, A);
+ q = spherical(q);
+ if (!two)
+ return q;
+ var lambda04 = a[0], lambda12 = b[0], phi02 = a[1], phi12 = b[1], z;
+ if (lambda12 < lambda04)
+ z = lambda04, lambda04 = lambda12, lambda12 = z;
+ var delta2 = lambda12 - lambda04, polar = abs(delta2 - pi) < epsilon, meridian = polar || delta2 < epsilon;
+ if (!polar && phi12 < phi02)
+ z = phi02, phi02 = phi12, phi12 = z;
+ if (meridian ? polar ? phi02 + phi12 > 0 ^ q[1] < (abs(q[0] - lambda04) < epsilon ? phi02 : phi12) : phi02 <= q[1] && q[1] <= phi12 : delta2 > pi ^ (lambda04 <= q[0] && q[0] <= lambda12)) {
+ var q1 = cartesianScale(u, (-w + t) / uu);
+ cartesianAddInPlace(q1, A);
+ return [q, spherical(q1)];
+ }
+ }
+ function code(lambda, phi) {
+ var r = smallRadius ? radius : pi - radius, code2 = 0;
+ if (lambda < -r)
+ code2 |= 1;
+ else if (lambda > r)
+ code2 |= 2;
+ if (phi < -r)
+ code2 |= 4;
+ else if (phi > r)
+ code2 |= 8;
+ return code2;
+ }
+ return clip_default(visible, clipLine, interpolate, smallRadius ? [0, -radius] : [-pi, radius - pi]);
+ }
+
+ // node_modules/d3-geo/src/clip/line.js
+ function line_default(a, b, x05, y05, x12, y12) {
+ var ax = a[0], ay = a[1], bx = b[0], by = b[1], t0 = 0, t1 = 1, dx = bx - ax, dy = by - ay, r;
+ r = x05 - ax;
+ if (!dx && r > 0)
+ return;
+ r /= dx;
+ if (dx < 0) {
+ if (r < t0)
+ return;
+ if (r < t1)
+ t1 = r;
+ } else if (dx > 0) {
+ if (r > t1)
+ return;
+ if (r > t0)
+ t0 = r;
+ }
+ r = x12 - ax;
+ if (!dx && r < 0)
+ return;
+ r /= dx;
+ if (dx < 0) {
+ if (r > t1)
+ return;
+ if (r > t0)
+ t0 = r;
+ } else if (dx > 0) {
+ if (r < t0)
+ return;
+ if (r < t1)
+ t1 = r;
+ }
+ r = y05 - ay;
+ if (!dy && r > 0)
+ return;
+ r /= dy;
+ if (dy < 0) {
+ if (r < t0)
+ return;
+ if (r < t1)
+ t1 = r;
+ } else if (dy > 0) {
+ if (r > t1)
+ return;
+ if (r > t0)
+ t0 = r;
+ }
+ r = y12 - ay;
+ if (!dy && r < 0)
+ return;
+ r /= dy;
+ if (dy < 0) {
+ if (r > t1)
+ return;
+ if (r > t0)
+ t0 = r;
+ } else if (dy > 0) {
+ if (r < t0)
+ return;
+ if (r < t1)
+ t1 = r;
+ }
+ if (t0 > 0)
+ a[0] = ax + t0 * dx, a[1] = ay + t0 * dy;
+ if (t1 < 1)
+ b[0] = ax + t1 * dx, b[1] = ay + t1 * dy;
+ return true;
+ }
+
+ // node_modules/d3-geo/src/clip/rectangle.js
+ var clipMax = 1e9;
+ var clipMin = -clipMax;
+ function clipRectangle(x05, y05, x12, y12) {
+ function visible(x, y) {
+ return x05 <= x && x <= x12 && y05 <= y && y <= y12;
+ }
+ function interpolate(from, to, direction, stream) {
+ var a = 0, a1 = 0;
+ if (from == null || (a = corner(from, direction)) !== (a1 = corner(to, direction)) || comparePoint(from, to) < 0 ^ direction > 0) {
+ do
+ stream.point(a === 0 || a === 3 ? x05 : x12, a > 1 ? y12 : y05);
+ while ((a = (a + direction + 4) % 4) !== a1);
+ } else {
+ stream.point(to[0], to[1]);
+ }
+ }
+ function corner(p, direction) {
+ return abs(p[0] - x05) < epsilon ? direction > 0 ? 0 : 3 : abs(p[0] - x12) < epsilon ? direction > 0 ? 2 : 1 : abs(p[1] - y05) < epsilon ? direction > 0 ? 1 : 0 : direction > 0 ? 3 : 2;
+ }
+ function compareIntersection2(a, b) {
+ return comparePoint(a.x, b.x);
+ }
+ function comparePoint(a, b) {
+ var ca = corner(a, 1), cb = corner(b, 1);
+ return ca !== cb ? ca - cb : ca === 0 ? b[1] - a[1] : ca === 1 ? a[0] - b[0] : ca === 2 ? a[1] - b[1] : b[0] - a[0];
+ }
+ return function(stream) {
+ var activeStream = stream, bufferStream = buffer_default(), segments, polygon2, ring, x__, y__, v__, x_, y_, v_, first, clean2;
+ var clipStream = {
+ point: point2,
+ lineStart,
+ lineEnd,
+ polygonStart,
+ polygonEnd
+ };
+ function point2(x, y) {
+ if (visible(x, y))
+ activeStream.point(x, y);
+ }
+ function polygonInside() {
+ var winding = 0;
+ for (var i2 = 0, n2 = polygon2.length; i2 < n2; ++i2) {
+ for (var ring2 = polygon2[i2], j2 = 1, m = ring2.length, point3 = ring2[0], a0, a1, b0 = point3[0], b1 = point3[1]; j2 < m; ++j2) {
+ a0 = b0, a1 = b1, point3 = ring2[j2], b0 = point3[0], b1 = point3[1];
+ if (a1 <= y12) {
+ if (b1 > y12 && (b0 - a0) * (y12 - a1) > (b1 - a1) * (x05 - a0))
+ ++winding;
+ } else {
+ if (b1 <= y12 && (b0 - a0) * (y12 - a1) < (b1 - a1) * (x05 - a0))
+ --winding;
+ }
+ }
+ }
+ return winding;
+ }
+ function polygonStart() {
+ activeStream = bufferStream, segments = [], polygon2 = [], clean2 = true;
+ }
+ function polygonEnd() {
+ var startInside = polygonInside(), cleanInside = clean2 && startInside, visible2 = (segments = merge(segments)).length;
+ if (cleanInside || visible2) {
+ stream.polygonStart();
+ if (cleanInside) {
+ stream.lineStart();
+ interpolate(null, null, 1, stream);
+ stream.lineEnd();
+ }
+ if (visible2) {
+ rejoin_default(segments, compareIntersection2, startInside, interpolate, stream);
+ }
+ stream.polygonEnd();
+ }
+ activeStream = stream, segments = polygon2 = ring = null;
+ }
+ function lineStart() {
+ clipStream.point = linePoint2;
+ if (polygon2)
+ polygon2.push(ring = []);
+ first = true;
+ v_ = false;
+ x_ = y_ = NaN;
+ }
+ function lineEnd() {
+ if (segments) {
+ linePoint2(x__, y__);
+ if (v__ && v_)
+ bufferStream.rejoin();
+ segments.push(bufferStream.result());
+ }
+ clipStream.point = point2;
+ if (v_)
+ activeStream.lineEnd();
+ }
+ function linePoint2(x, y) {
+ var v = visible(x, y);
+ if (polygon2)
+ ring.push([x, y]);
+ if (first) {
+ x__ = x, y__ = y, v__ = v;
+ first = false;
+ if (v) {
+ activeStream.lineStart();
+ activeStream.point(x, y);
+ }
+ } else {
+ if (v && v_)
+ activeStream.point(x, y);
+ else {
+ var a = [x_ = Math.max(clipMin, Math.min(clipMax, x_)), y_ = Math.max(clipMin, Math.min(clipMax, y_))], b = [x = Math.max(clipMin, Math.min(clipMax, x)), y = Math.max(clipMin, Math.min(clipMax, y))];
+ if (line_default(a, b, x05, y05, x12, y12)) {
+ if (!v_) {
+ activeStream.lineStart();
+ activeStream.point(a[0], a[1]);
+ }
+ activeStream.point(b[0], b[1]);
+ if (!v)
+ activeStream.lineEnd();
+ clean2 = false;
+ } else if (v) {
+ activeStream.lineStart();
+ activeStream.point(x, y);
+ clean2 = false;
+ }
+ }
+ }
+ x_ = x, y_ = y, v_ = v;
+ }
+ return clipStream;
+ };
+ }
+
+ // node_modules/d3-geo/src/length.js
+ var lengthSum;
+ var lambda03;
+ var sinPhi02;
+ var cosPhi02;
+ var lengthStream = {
+ sphere: noop,
+ point: noop,
+ lineStart: lengthLineStart,
+ lineEnd: noop,
+ polygonStart: noop,
+ polygonEnd: noop
+ };
+ function lengthLineStart() {
+ lengthStream.point = lengthPointFirst;
+ lengthStream.lineEnd = lengthLineEnd;
+ }
+ function lengthLineEnd() {
+ lengthStream.point = lengthStream.lineEnd = noop;
+ }
+ function lengthPointFirst(lambda, phi) {
+ lambda *= radians, phi *= radians;
+ lambda03 = lambda, sinPhi02 = sin(phi), cosPhi02 = cos(phi);
+ lengthStream.point = lengthPoint;
+ }
+ function lengthPoint(lambda, phi) {
+ lambda *= radians, phi *= radians;
+ var sinPhi = sin(phi), cosPhi = cos(phi), delta = abs(lambda - lambda03), cosDelta = cos(delta), sinDelta = sin(delta), x = cosPhi * sinDelta, y = cosPhi02 * sinPhi - sinPhi02 * cosPhi * cosDelta, z = sinPhi02 * sinPhi + cosPhi02 * cosPhi * cosDelta;
+ lengthSum.add(atan2(sqrt(x * x + y * y), z));
+ lambda03 = lambda, sinPhi02 = sinPhi, cosPhi02 = cosPhi;
+ }
+ function length_default(object) {
+ lengthSum = new Adder();
+ stream_default(object, lengthStream);
+ return +lengthSum;
+ }
+
+ // node_modules/d3-geo/src/identity.js
+ var identity_default = (x) => x;
+
+ // node_modules/d3-geo/src/path/area.js
+ var areaSum2 = new Adder();
+ var areaRingSum2 = new Adder();
+ var x00;
+ var y00;
+ var x0;
+ var y0;
+ var areaStream2 = {
+ point: noop,
+ lineStart: noop,
+ lineEnd: noop,
+ polygonStart: function() {
+ areaStream2.lineStart = areaRingStart2;
+ areaStream2.lineEnd = areaRingEnd2;
+ },
+ polygonEnd: function() {
+ areaStream2.lineStart = areaStream2.lineEnd = areaStream2.point = noop;
+ areaSum2.add(abs(areaRingSum2));
+ areaRingSum2 = new Adder();
+ },
+ result: function() {
+ var area = areaSum2 / 2;
+ areaSum2 = new Adder();
+ return area;
+ }
+ };
+ function areaRingStart2() {
+ areaStream2.point = areaPointFirst2;
+ }
+ function areaPointFirst2(x, y) {
+ areaStream2.point = areaPoint2;
+ x00 = x0 = x, y00 = y0 = y;
+ }
+ function areaPoint2(x, y) {
+ areaRingSum2.add(y0 * x - x0 * y);
+ x0 = x, y0 = y;
+ }
+ function areaRingEnd2() {
+ areaPoint2(x00, y00);
+ }
+ var area_default2 = areaStream2;
+
+ // node_modules/d3-geo/src/path/bounds.js
+ var x02 = Infinity;
+ var y02 = x02;
+ var x1 = -x02;
+ var y1 = x1;
+ var boundsStream2 = {
+ point: boundsPoint2,
+ lineStart: noop,
+ lineEnd: noop,
+ polygonStart: noop,
+ polygonEnd: noop,
+ result: function() {
+ var bounds = [[x02, y02], [x1, y1]];
+ x1 = y1 = -(y02 = x02 = Infinity);
+ return bounds;
+ }
+ };
+ function boundsPoint2(x, y) {
+ if (x < x02)
+ x02 = x;
+ if (x > x1)
+ x1 = x;
+ if (y < y02)
+ y02 = y;
+ if (y > y1)
+ y1 = y;
+ }
+ var bounds_default2 = boundsStream2;
+
+ // node_modules/d3-geo/src/path/centroid.js
+ var X0 = 0;
+ var Y0 = 0;
+ var Z0 = 0;
+ var X1 = 0;
+ var Y1 = 0;
+ var Z1 = 0;
+ var X2 = 0;
+ var Y2 = 0;
+ var Z2 = 0;
+ var x002;
+ var y002;
+ var x03;
+ var y03;
+ var centroidStream = {
+ point: centroidPoint,
+ lineStart: centroidLineStart,
+ lineEnd: centroidLineEnd,
+ polygonStart: function() {
+ centroidStream.lineStart = centroidRingStart;
+ centroidStream.lineEnd = centroidRingEnd;
+ },
+ polygonEnd: function() {
+ centroidStream.point = centroidPoint;
+ centroidStream.lineStart = centroidLineStart;
+ centroidStream.lineEnd = centroidLineEnd;
+ },
+ result: function() {
+ 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(x, y) {
+ X0 += x;
+ Y0 += y;
+ ++Z0;
+ }
+ function centroidLineStart() {
+ centroidStream.point = centroidPointFirstLine;
+ }
+ function centroidPointFirstLine(x, y) {
+ centroidStream.point = centroidPointLine;
+ centroidPoint(x03 = x, y03 = y);
+ }
+ function centroidPointLine(x, y) {
+ var dx = x - x03, dy = y - y03, z = sqrt(dx * dx + dy * dy);
+ X1 += z * (x03 + x) / 2;
+ Y1 += z * (y03 + y) / 2;
+ Z1 += z;
+ centroidPoint(x03 = x, y03 = y);
+ }
+ function centroidLineEnd() {
+ centroidStream.point = centroidPoint;
+ }
+ function centroidRingStart() {
+ centroidStream.point = centroidPointFirstRing;
+ }
+ function centroidRingEnd() {
+ centroidPointRing(x002, y002);
+ }
+ function centroidPointFirstRing(x, y) {
+ centroidStream.point = centroidPointRing;
+ centroidPoint(x002 = x03 = x, y002 = y03 = y);
+ }
+ function centroidPointRing(x, y) {
+ var dx = x - x03, dy = y - y03, z = sqrt(dx * dx + dy * dy);
+ X1 += z * (x03 + x) / 2;
+ Y1 += z * (y03 + y) / 2;
+ Z1 += z;
+ z = y03 * x - x03 * y;
+ X2 += z * (x03 + x);
+ Y2 += z * (y03 + y);
+ Z2 += z * 3;
+ centroidPoint(x03 = x, y03 = y);
+ }
+ var centroid_default = centroidStream;
+
+ // node_modules/d3-geo/src/path/context.js
+ function PathContext(context) {
+ this._context = context;
+ }
+ PathContext.prototype = {
+ _radius: 4.5,
+ pointRadius: function(_) {
+ return this._radius = _, this;
+ },
+ polygonStart: function() {
+ this._line = 0;
+ },
+ polygonEnd: function() {
+ this._line = NaN;
+ },
+ lineStart: function() {
+ this._point = 0;
+ },
+ lineEnd: function() {
+ if (this._line === 0)
+ this._context.closePath();
+ this._point = NaN;
+ },
+ point: function(x, y) {
+ switch (this._point) {
+ case 0: {
+ this._context.moveTo(x, y);
+ this._point = 1;
+ break;
+ }
+ case 1: {
+ this._context.lineTo(x, y);
+ break;
+ }
+ default: {
+ this._context.moveTo(x + this._radius, y);
+ this._context.arc(x, y, this._radius, 0, tau);
+ break;
+ }
+ }
+ },
+ result: noop
+ };
+
+ // node_modules/d3-geo/src/path/measure.js
+ var lengthSum2 = new Adder();
+ var lengthRing;
+ var x003;
+ var y003;
+ var x04;
+ var y04;
+ var lengthStream2 = {
+ point: noop,
+ lineStart: function() {
+ lengthStream2.point = lengthPointFirst2;
+ },
+ lineEnd: function() {
+ if (lengthRing)
+ lengthPoint2(x003, y003);
+ lengthStream2.point = noop;
+ },
+ polygonStart: function() {
+ lengthRing = true;
+ },
+ polygonEnd: function() {
+ lengthRing = null;
+ },
+ result: function() {
+ var length = +lengthSum2;
+ lengthSum2 = new Adder();
+ return length;
+ }
+ };
+ function lengthPointFirst2(x, y) {
+ lengthStream2.point = lengthPoint2;
+ x003 = x04 = x, y003 = y04 = y;
+ }
+ function lengthPoint2(x, y) {
+ x04 -= x, y04 -= y;
+ lengthSum2.add(sqrt(x04 * x04 + y04 * y04));
+ x04 = x, y04 = y;
+ }
+ var measure_default = lengthStream2;
+
+ // node_modules/d3-geo/src/path/string.js
+ function PathString() {
+ this._string = [];
+ }
+ PathString.prototype = {
+ _radius: 4.5,
+ _circle: circle(4.5),
+ pointRadius: function(_) {
+ if ((_ = +_) !== this._radius)
+ this._radius = _, this._circle = null;
+ return this;
+ },
+ polygonStart: function() {
+ this._line = 0;
+ },
+ polygonEnd: function() {
+ this._line = NaN;
+ },
+ lineStart: function() {
+ this._point = 0;
+ },
+ lineEnd: function() {
+ if (this._line === 0)
+ this._string.push("Z");
+ this._point = NaN;
+ },
+ point: function(x, y) {
+ switch (this._point) {
+ case 0: {
+ this._string.push("M", x, ",", y);
+ this._point = 1;
+ break;
+ }
+ case 1: {
+ this._string.push("L", x, ",", y);
+ break;
+ }
+ default: {
+ if (this._circle == null)
+ this._circle = circle(this._radius);
+ this._string.push("M", x, ",", y, this._circle);
+ break;
+ }
+ }
+ },
+ result: function() {
+ if (this._string.length) {
+ var result = this._string.join("");
+ this._string = [];
+ return result;
+ } else {
+ return null;
+ }
+ }
+ };
+ function circle(radius) {
+ return "m0," + radius + "a" + radius + "," + radius + " 0 1,1 0," + -2 * radius + "a" + radius + "," + radius + " 0 1,1 0," + 2 * radius + "z";
+ }
+
+ // node_modules/d3-geo/src/path/index.js
+ function path_default(projection2, context) {
+ var pointRadius = 4.5, projectionStream, contextStream;
+ function path(object) {
+ if (object) {
+ if (typeof pointRadius === "function")
+ contextStream.pointRadius(+pointRadius.apply(this, arguments));
+ stream_default(object, projectionStream(contextStream));
+ }
+ return contextStream.result();
+ }
+ path.area = function(object) {
+ stream_default(object, projectionStream(area_default2));
+ return area_default2.result();
+ };
+ path.measure = function(object) {
+ stream_default(object, projectionStream(measure_default));
+ return measure_default.result();
+ };
+ path.bounds = function(object) {
+ stream_default(object, projectionStream(bounds_default2));
+ return bounds_default2.result();
+ };
+ path.centroid = function(object) {
+ stream_default(object, projectionStream(centroid_default));
+ return centroid_default.result();
+ };
+ path.projection = function(_) {
+ return arguments.length ? (projectionStream = _ == null ? (projection2 = null, identity_default) : (projection2 = _).stream, path) : projection2;
+ };
+ path.context = function(_) {
+ if (!arguments.length)
+ return context;
+ contextStream = _ == null ? (context = null, new PathString()) : new PathContext(context = _);
+ if (typeof pointRadius !== "function")
+ contextStream.pointRadius(pointRadius);
+ return path;
+ };
+ path.pointRadius = function(_) {
+ if (!arguments.length)
+ return pointRadius;
+ pointRadius = typeof _ === "function" ? _ : (contextStream.pointRadius(+_), +_);
+ return path;
+ };
+ return path.projection(projection2).context(context);
+ }
+
+ // node_modules/d3-geo/src/transform.js
+ function transform_default(methods2) {
+ return {
+ stream: transformer(methods2)
+ };
+ }
+ function transformer(methods2) {
+ return function(stream) {
+ var s = new TransformStream();
+ for (var key in methods2)
+ s[key] = methods2[key];
+ s.stream = stream;
+ return s;
+ };
+ }
+ function TransformStream() {
+ }
+ TransformStream.prototype = {
+ constructor: TransformStream,
+ point: function(x, y) {
+ this.stream.point(x, y);
+ },
+ sphere: function() {
+ this.stream.sphere();
+ },
+ lineStart: function() {
+ this.stream.lineStart();
+ },
+ lineEnd: function() {
+ this.stream.lineEnd();
+ },
+ polygonStart: function() {
+ this.stream.polygonStart();
+ },
+ polygonEnd: function() {
+ this.stream.polygonEnd();
+ }
+ };
+
+ // node_modules/d3-geo/src/projection/fit.js
+ function fit(projection2, fitBounds, object) {
+ var clip = projection2.clipExtent && projection2.clipExtent();
+ projection2.scale(150).translate([0, 0]);
+ if (clip != null)
+ projection2.clipExtent(null);
+ stream_default(object, projection2.stream(bounds_default2));
+ fitBounds(bounds_default2.result());
+ if (clip != null)
+ projection2.clipExtent(clip);
+ return projection2;
+ }
+ function fitExtent(projection2, extent, object) {
+ return fit(projection2, function(b) {
+ var w = extent[1][0] - extent[0][0], h = extent[1][1] - extent[0][1], k = Math.min(w / (b[1][0] - b[0][0]), h / (b[1][1] - b[0][1])), x = +extent[0][0] + (w - k * (b[1][0] + b[0][0])) / 2, y = +extent[0][1] + (h - k * (b[1][1] + b[0][1])) / 2;
+ projection2.scale(150 * k).translate([x, y]);
+ }, object);
+ }
+ function fitSize(projection2, size, object) {
+ return fitExtent(projection2, [[0, 0], size], object);
+ }
+ function fitWidth(projection2, width, object) {
+ return fit(projection2, function(b) {
+ var w = +width, k = w / (b[1][0] - b[0][0]), x = (w - k * (b[1][0] + b[0][0])) / 2, y = -k * b[0][1];
+ projection2.scale(150 * k).translate([x, y]);
+ }, object);
+ }
+ function fitHeight(projection2, height, object) {
+ return fit(projection2, function(b) {
+ var h = +height, k = h / (b[1][1] - b[0][1]), x = -k * b[0][0], y = (h - k * (b[1][1] + b[0][1])) / 2;
+ projection2.scale(150 * k).translate([x, y]);
+ }, object);
+ }
+
+ // node_modules/d3-geo/src/projection/resample.js
+ var maxDepth = 16;
+ var cosMinDistance = cos(30 * radians);
+ function resample_default(project, delta2) {
+ return +delta2 ? resample(project, delta2) : resampleNone(project);
+ }
+ function resampleNone(project) {
+ return transformer({
+ point: function(x, y) {
+ x = project(x, y);
+ this.stream.point(x[0], x[1]);
+ }
+ });
+ }
+ function resample(project, delta2) {
+ function resampleLineTo(x05, y05, lambda04, a0, b0, c0, x12, y12, lambda12, a1, b1, c1, depth, stream) {
+ var dx = x12 - x05, dy = y12 - y05, d2 = dx * dx + dy * dy;
+ if (d2 > 4 * delta2 && depth--) {
+ var a = a0 + a1, b = b0 + b1, c = c0 + c1, m = sqrt(a * a + b * b + c * c), phi2 = asin(c /= m), lambda22 = abs(abs(c) - 1) < epsilon || abs(lambda04 - lambda12) < epsilon ? (lambda04 + lambda12) / 2 : atan2(b, a), p = project(lambda22, phi2), x2 = p[0], y2 = p[1], dx2 = x2 - x05, dy2 = y2 - y05, dz = dy * dx2 - dx * dy2;
+ if (dz * dz / d2 > delta2 || abs((dx * dx2 + dy * dy2) / d2 - 0.5) > 0.3 || a0 * a1 + b0 * b1 + c0 * c1 < cosMinDistance) {
+ resampleLineTo(x05, y05, lambda04, a0, b0, c0, x2, y2, lambda22, a /= m, b /= m, c, depth, stream);
+ stream.point(x2, y2);
+ resampleLineTo(x2, y2, lambda22, a, b, c, x12, y12, lambda12, a1, b1, c1, depth, stream);
+ }
+ }
+ }
+ return function(stream) {
+ var lambda003, x004, y004, a00, b00, c00, lambda04, x05, y05, a0, b0, c0;
+ var resampleStream = {
+ point: point2,
+ lineStart,
+ lineEnd,
+ polygonStart: function() {
+ stream.polygonStart();
+ resampleStream.lineStart = ringStart;
+ },
+ polygonEnd: function() {
+ stream.polygonEnd();
+ resampleStream.lineStart = lineStart;
+ }
+ };
+ function point2(x, y) {
+ x = project(x, y);
+ stream.point(x[0], x[1]);
+ }
+ function lineStart() {
+ x05 = NaN;
+ resampleStream.point = linePoint2;
+ stream.lineStart();
+ }
+ function linePoint2(lambda, phi) {
+ var c = cartesian([lambda, phi]), p = project(lambda, phi);
+ resampleLineTo(x05, y05, lambda04, a0, b0, c0, x05 = p[0], y05 = p[1], lambda04 = lambda, a0 = c[0], b0 = c[1], c0 = c[2], maxDepth, stream);
+ stream.point(x05, y05);
+ }
+ function lineEnd() {
+ resampleStream.point = point2;
+ stream.lineEnd();
+ }
+ function ringStart() {
+ lineStart();
+ resampleStream.point = ringPoint;
+ resampleStream.lineEnd = ringEnd;
+ }
+ function ringPoint(lambda, phi) {
+ linePoint2(lambda003 = lambda, phi), x004 = x05, y004 = y05, a00 = a0, b00 = b0, c00 = c0;
+ resampleStream.point = linePoint2;
+ }
+ function ringEnd() {
+ resampleLineTo(x05, y05, lambda04, a0, b0, c0, x004, y004, lambda003, a00, b00, c00, maxDepth, stream);
+ resampleStream.lineEnd = lineEnd;
+ lineEnd();
+ }
+ return resampleStream;
+ };
+ }
+
+ // node_modules/d3-geo/src/projection/index.js
+ var transformRadians = transformer({
+ point: function(x, y) {
+ this.stream.point(x * radians, y * radians);
+ }
+ });
+ function transformRotate(rotate) {
+ return transformer({
+ point: function(x, y) {
+ var r = rotate(x, y);
+ return this.stream.point(r[0], r[1]);
+ }
+ });
+ }
+ function scaleTranslate(k, dx, dy, sx, sy) {
+ function transform2(x, y) {
+ x *= sx;
+ y *= sy;
+ return [dx + k * x, dy - k * y];
+ }
+ transform2.invert = function(x, y) {
+ return [(x - dx) / k * sx, (dy - y) / k * sy];
+ };
+ return transform2;
+ }
+ function scaleTranslateRotate(k, dx, dy, sx, sy, alpha) {
+ if (!alpha)
+ return scaleTranslate(k, dx, dy, sx, sy);
+ var cosAlpha = cos(alpha), sinAlpha = sin(alpha), a = cosAlpha * k, b = sinAlpha * k, ai = cosAlpha / k, bi = sinAlpha / k, ci = (sinAlpha * dy - cosAlpha * dx) / k, fi = (sinAlpha * dx + cosAlpha * dy) / k;
+ function transform2(x, y) {
+ x *= sx;
+ y *= sy;
+ return [a * x - b * y + dx, dy - b * x - a * y];
+ }
+ transform2.invert = function(x, y) {
+ return [sx * (ai * x - bi * y + ci), sy * (fi - bi * x - ai * y)];
+ };
+ return transform2;
+ }
+ function projection(project) {
+ return projectionMutator(function() {
+ return project;
+ })();
+ }
+ function projectionMutator(projectAt) {
+ var project, k = 150, x = 480, y = 250, lambda = 0, phi = 0, deltaLambda = 0, deltaPhi = 0, deltaGamma = 0, rotate, alpha = 0, sx = 1, sy = 1, theta = null, preclip = antimeridian_default, x05 = null, y05, x12, y12, postclip = identity_default, delta2 = 0.5, projectResample, projectTransform, projectRotateTransform, cache, cacheStream;
+ function projection2(point2) {
+ return projectRotateTransform(point2[0] * radians, point2[1] * radians);
+ }
+ function invert(point2) {
+ point2 = projectRotateTransform.invert(point2[0], point2[1]);
+ return point2 && [point2[0] * degrees, point2[1] * degrees];
+ }
+ projection2.stream = function(stream) {
+ return cache && cacheStream === stream ? cache : cache = transformRadians(transformRotate(rotate)(preclip(projectResample(postclip(cacheStream = stream)))));
+ };
+ projection2.preclip = function(_) {
+ return arguments.length ? (preclip = _, theta = void 0, reset()) : preclip;
+ };
+ projection2.postclip = function(_) {
+ return arguments.length ? (postclip = _, x05 = y05 = x12 = y12 = null, reset()) : postclip;
+ };
+ projection2.clipAngle = function(_) {
+ return arguments.length ? (preclip = +_ ? circle_default(theta = _ * radians) : (theta = null, antimeridian_default), reset()) : theta * degrees;
+ };
+ projection2.clipExtent = function(_) {
+ return arguments.length ? (postclip = _ == null ? (x05 = y05 = x12 = y12 = null, identity_default) : clipRectangle(x05 = +_[0][0], y05 = +_[0][1], x12 = +_[1][0], y12 = +_[1][1]), reset()) : x05 == null ? null : [[x05, y05], [x12, y12]];
+ };
+ projection2.scale = function(_) {
+ return arguments.length ? (k = +_, recenter()) : k;
+ };
+ projection2.translate = function(_) {
+ return arguments.length ? (x = +_[0], y = +_[1], recenter()) : [x, y];
+ };
+ projection2.center = function(_) {
+ return arguments.length ? (lambda = _[0] % 360 * radians, phi = _[1] % 360 * radians, recenter()) : [lambda * degrees, phi * degrees];
+ };
+ projection2.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];
+ };
+ projection2.angle = function(_) {
+ return arguments.length ? (alpha = _ % 360 * radians, recenter()) : alpha * degrees;
+ };
+ projection2.reflectX = function(_) {
+ return arguments.length ? (sx = _ ? -1 : 1, recenter()) : sx < 0;
+ };
+ projection2.reflectY = function(_) {
+ return arguments.length ? (sy = _ ? -1 : 1, recenter()) : sy < 0;
+ };
+ projection2.precision = function(_) {
+ return arguments.length ? (projectResample = resample_default(projectTransform, delta2 = _ * _), reset()) : sqrt(delta2);
+ };
+ projection2.fitExtent = function(extent, object) {
+ return fitExtent(projection2, extent, object);
+ };
+ projection2.fitSize = function(size, object) {
+ return fitSize(projection2, size, object);
+ };
+ projection2.fitWidth = function(width, object) {
+ return fitWidth(projection2, width, object);
+ };
+ projection2.fitHeight = function(height, object) {
+ return fitHeight(projection2, height, object);
+ };
+ function recenter() {
+ var center = scaleTranslateRotate(k, 0, 0, sx, sy, alpha).apply(null, project(lambda, phi)), transform2 = scaleTranslateRotate(k, x - center[0], y - center[1], sx, sy, alpha);
+ rotate = rotateRadians(deltaLambda, deltaPhi, deltaGamma);
+ projectTransform = compose_default(project, transform2);
+ projectRotateTransform = compose_default(rotate, projectTransform);
+ projectResample = resample_default(projectTransform, delta2);
+ return reset();
+ }
+ function reset() {
+ cache = cacheStream = null;
+ return projection2;
+ }
+ return function() {
+ project = projectAt.apply(this, arguments);
+ projection2.invert = project.invert && invert;
+ return recenter();
+ };
+ }
+
+ // node_modules/d3-geo/src/projection/mercator.js
+ function mercatorRaw(lambda, phi) {
+ return [lambda, log(tan((halfPi + phi) / 2))];
+ }
+ mercatorRaw.invert = function(x, y) {
+ return [x, 2 * atan(exp(y)) - halfPi];
+ };
+ function mercator_default() {
+ return mercatorProjection(mercatorRaw).scale(961 / tau);
+ }
+ function mercatorProjection(project) {
+ var m = projection(project), center = m.center, scale = m.scale, translate = m.translate, clipExtent = m.clipExtent, x05 = null, y05, x12, y12;
+ m.scale = function(_) {
+ return arguments.length ? (scale(_), reclip()) : scale();
+ };
+ m.translate = function(_) {
+ return arguments.length ? (translate(_), reclip()) : translate();
+ };
+ m.center = function(_) {
+ return arguments.length ? (center(_), reclip()) : center();
+ };
+ m.clipExtent = function(_) {
+ return arguments.length ? (_ == null ? x05 = y05 = x12 = y12 = null : (x05 = +_[0][0], y05 = +_[0][1], x12 = +_[1][0], y12 = +_[1][1]), reclip()) : x05 == null ? null : [[x05, y05], [x12, y12]];
+ };
+ function reclip() {
+ var k = pi * scale(), t = m(rotation_default(m.rotate()).invert([0, 0]));
+ return clipExtent(x05 == null ? [[t[0] - k, t[1] - k], [t[0] + k, t[1] + k]] : project === mercatorRaw ? [[Math.max(t[0] - k, x05), y05], [Math.min(t[0] + k, x12), y12]] : [[x05, Math.max(t[1] - k, y05)], [x12, Math.min(t[1] + k, y12)]]);
+ }
+ return reclip();
+ }
+
+ // node_modules/d3-geo/src/projection/identity.js
+ function identity_default2() {
+ var k = 1, tx = 0, ty = 0, sx = 1, sy = 1, alpha = 0, ca, sa, x05 = null, y05, x12, y12, kx = 1, ky = 1, transform2 = transformer({
+ point: function(x, y) {
+ var p = projection2([x, y]);
+ this.stream.point(p[0], p[1]);
+ }
+ }), postclip = identity_default, cache, cacheStream;
+ function reset() {
+ kx = k * sx;
+ ky = k * sy;
+ cache = cacheStream = null;
+ return projection2;
+ }
+ function projection2(p) {
+ var x = p[0] * kx, y = p[1] * ky;
+ if (alpha) {
+ var t = y * ca - x * sa;
+ x = x * ca + y * sa;
+ y = t;
+ }
+ return [x + tx, y + ty];
+ }
+ projection2.invert = function(p) {
+ var x = p[0] - tx, y = p[1] - ty;
+ if (alpha) {
+ var t = y * ca + x * sa;
+ x = x * ca - y * sa;
+ y = t;
+ }
+ return [x / kx, y / ky];
+ };
+ projection2.stream = function(stream) {
+ return cache && cacheStream === stream ? cache : cache = transform2(postclip(cacheStream = stream));
+ };
+ projection2.postclip = function(_) {
+ return arguments.length ? (postclip = _, x05 = y05 = x12 = y12 = null, reset()) : postclip;
+ };
+ projection2.clipExtent = function(_) {
+ return arguments.length ? (postclip = _ == null ? (x05 = y05 = x12 = y12 = null, identity_default) : clipRectangle(x05 = +_[0][0], y05 = +_[0][1], x12 = +_[1][0], y12 = +_[1][1]), reset()) : x05 == null ? null : [[x05, y05], [x12, y12]];
+ };
+ projection2.scale = function(_) {
+ return arguments.length ? (k = +_, reset()) : k;
+ };
+ projection2.translate = function(_) {
+ return arguments.length ? (tx = +_[0], ty = +_[1], reset()) : [tx, ty];
+ };
+ projection2.angle = function(_) {
+ return arguments.length ? (alpha = _ % 360 * radians, sa = sin(alpha), ca = cos(alpha), reset()) : alpha * degrees;
+ };
+ projection2.reflectX = function(_) {
+ return arguments.length ? (sx = _ ? -1 : 1, reset()) : sx < 0;
+ };
+ projection2.reflectY = function(_) {
+ return arguments.length ? (sy = _ ? -1 : 1, reset()) : sy < 0;
+ };
+ projection2.fitExtent = function(extent, object) {
+ return fitExtent(projection2, extent, object);
+ };
+ projection2.fitSize = function(size, object) {
+ return fitSize(projection2, size, object);
+ };
+ projection2.fitWidth = function(width, object) {
+ return fitWidth(projection2, width, object);
+ };
+ projection2.fitHeight = function(height, object) {
+ return fitHeight(projection2, height, object);
+ };
+ return projection2;
+ }
+
+ // modules/geo/geo.js
+ var TAU = 2 * Math.PI;
+ var EQUATORIAL_RADIUS = 6356752314245179e-9;
+ var POLAR_RADIUS = 6378137;
+ function geoLatToMeters(dLat) {
+ return dLat * (TAU * POLAR_RADIUS / 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);
+ }
+ 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)
+ ];
+ }
+ function geoOffsetToMeters(offset, tileSize) {
+ tileSize = tileSize || 256;
+ return [
+ offset[0] * TAU * EQUATORIAL_RADIUS / tileSize,
+ -offset[1] * TAU * POLAR_RADIUS / tileSize
+ ];
+ }
+ function geoSphericalDistance(a, b) {
+ var x = geoLonToMeters(a[0] - b[0], (a[1] + b[1]) / 2);
+ var y = geoLatToMeters(a[1] - b[1]);
+ return Math.sqrt(x * x + y * y);
+ }
+ function geoScaleToZoom(k, tileSize) {
+ tileSize = tileSize || 256;
+ var log2ts = Math.log(tileSize) * Math.LOG2E;
+ return Math.log(k * TAU) / Math.LN2 - log2ts;
+ }
+ function geoZoomToScale(z, tileSize) {
+ tileSize = tileSize || 256;
+ return tileSize * Math.pow(2, z) / TAU;
+ }
+ function geoSphericalClosestNode(nodes, point2) {
+ var minDistance = Infinity, distance;
+ var indexOfMin;
+ for (var i2 in nodes) {
+ distance = geoSphericalDistance(nodes[i2].loc, point2);
+ if (distance < minDistance) {
+ minDistance = distance;
+ indexOfMin = i2;
+ }
+ }
+ if (indexOfMin !== void 0) {
+ return { index: indexOfMin, distance: minDistance, node: nodes[indexOfMin] };
+ } else {
+ return null;
+ }
+ }
+
+ // modules/geo/extent.js
+ function geoExtent(min3, max3) {
+ if (!(this instanceof geoExtent)) {
+ return new geoExtent(min3, max3);
+ } else if (min3 instanceof geoExtent) {
+ return min3;
+ } else if (min3 && min3.length === 2 && min3[0].length === 2 && min3[1].length === 2) {
+ this[0] = min3[0];
+ this[1] = min3[1];
+ } else {
+ this[0] = min3 || [Infinity, Infinity];
+ this[1] = max3 || min3 || [-Infinity, -Infinity];
+ }
+ }
+ geoExtent.prototype = new Array(2);
+ Object.assign(geoExtent.prototype, {
+ equals: function(obj) {
+ return this[0][0] === obj[0][0] && this[0][1] === obj[0][1] && this[1][0] === obj[1][0] && this[1][1] === obj[1][1];
+ },
+ extend: function(obj) {
+ if (!(obj instanceof geoExtent))
+ obj = new geoExtent(obj);
+ return geoExtent(
+ [Math.min(obj[0][0], this[0][0]), Math.min(obj[0][1], this[0][1])],
+ [Math.max(obj[1][0], this[1][0]), Math.max(obj[1][1], this[1][1])]
+ );
+ },
+ _extend: function(extent) {
+ this[0][0] = Math.min(extent[0][0], this[0][0]);
+ this[0][1] = Math.min(extent[0][1], this[0][1]);
+ this[1][0] = Math.max(extent[1][0], this[1][0]);
+ this[1][1] = Math.max(extent[1][1], this[1][1]);
+ },
+ area: function() {
+ return Math.abs((this[1][0] - this[0][0]) * (this[1][1] - this[0][1]));
+ },
+ center: function() {
+ return [(this[0][0] + this[1][0]) / 2, (this[0][1] + this[1][1]) / 2];
+ },
+ rectangle: function() {
+ return [this[0][0], this[0][1], this[1][0], this[1][1]];
+ },
+ bbox: function() {
+ return { minX: this[0][0], minY: this[0][1], maxX: this[1][0], maxY: this[1][1] };
+ },
+ polygon: function() {
+ return [
+ [this[0][0], this[0][1]],
+ [this[0][0], this[1][1]],
+ [this[1][0], this[1][1]],
+ [this[1][0], this[0][1]],
+ [this[0][0], this[0][1]]
+ ];
+ },
+ contains: function(obj) {
+ if (!(obj instanceof geoExtent))
+ obj = new geoExtent(obj);
+ return obj[0][0] >= this[0][0] && obj[0][1] >= this[0][1] && obj[1][0] <= this[1][0] && obj[1][1] <= this[1][1];
+ },
+ intersects: function(obj) {
+ if (!(obj instanceof geoExtent))
+ obj = new geoExtent(obj);
+ return obj[0][0] <= this[1][0] && obj[0][1] <= this[1][1] && obj[1][0] >= this[0][0] && obj[1][1] >= this[0][1];
+ },
+ intersection: function(obj) {
+ if (!this.intersects(obj))
+ return new geoExtent();
+ return new geoExtent(
+ [Math.max(obj[0][0], this[0][0]), Math.max(obj[0][1], this[0][1])],
+ [Math.min(obj[1][0], this[1][0]), Math.min(obj[1][1], this[1][1])]
+ );
+ },
+ percentContainedIn: function(obj) {
+ if (!(obj instanceof geoExtent))
+ obj = new geoExtent(obj);
+ var a1 = this.intersection(obj).area();
+ var a2 = this.area();
+ if (a1 === Infinity || a2 === Infinity) {
+ return 0;
+ } else if (a1 === 0 || a2 === 0) {
+ if (obj.contains(this)) {
+ return 1;
+ }
+ return 0;
+ } else {
+ return a1 / a2;
+ }
+ },
+ padByMeters: function(meters) {
+ var dLat = geoMetersToLat(meters);
+ var dLon = geoMetersToLon(meters, this.center()[1]);
+ return geoExtent(
+ [this[0][0] - dLon, this[0][1] - dLat],
+ [this[1][0] + dLon, this[1][1] + dLat]
+ );
+ },
+ toParam: function() {
+ return this.rectangle().join(",");
+ }
+ });
+
+ // node_modules/d3-polygon/src/area.js
+ function area_default3(polygon2) {
+ var i2 = -1, n2 = polygon2.length, a, b = polygon2[n2 - 1], area = 0;
+ while (++i2 < n2) {
+ a = b;
+ b = polygon2[i2];
+ area += a[1] * b[0] - a[0] * b[1];
+ }
+ return area / 2;
+ }
+
+ // node_modules/d3-polygon/src/centroid.js
+ function centroid_default2(polygon2) {
+ var i2 = -1, n2 = polygon2.length, x = 0, y = 0, a, b = polygon2[n2 - 1], c, k = 0;
+ while (++i2 < n2) {
+ a = b;
+ b = polygon2[i2];
+ k += c = a[0] * b[1] - b[0] * a[1];
+ x += (a[0] + b[0]) * c;
+ y += (a[1] + b[1]) * c;
+ }
+ return k *= 3, [x / k, y / k];
+ }
+
+ // node_modules/d3-polygon/src/cross.js
+ function cross_default(a, b, c) {
+ return (b[0] - a[0]) * (c[1] - a[1]) - (b[1] - a[1]) * (c[0] - a[0]);
+ }
+
+ // node_modules/d3-polygon/src/hull.js
+ function lexicographicOrder(a, b) {
+ return a[0] - b[0] || a[1] - b[1];
+ }
+ function computeUpperHullIndexes(points) {
+ const n2 = points.length, indexes = [0, 1];
+ let size = 2, i2;
+ for (i2 = 2; i2 < n2; ++i2) {
+ while (size > 1 && cross_default(points[indexes[size - 2]], points[indexes[size - 1]], points[i2]) <= 0)
+ --size;
+ indexes[size++] = i2;
+ }
+ return indexes.slice(0, size);
+ }
+ function hull_default(points) {
+ if ((n2 = points.length) < 3)
+ return null;
+ var i2, n2, sortedPoints = new Array(n2), flippedPoints = new Array(n2);
+ for (i2 = 0; i2 < n2; ++i2)
+ sortedPoints[i2] = [+points[i2][0], +points[i2][1], i2];
+ sortedPoints.sort(lexicographicOrder);
+ for (i2 = 0; i2 < n2; ++i2)
+ flippedPoints[i2] = [sortedPoints[i2][0], -sortedPoints[i2][1]];
+ var upperIndexes = computeUpperHullIndexes(sortedPoints), lowerIndexes = computeUpperHullIndexes(flippedPoints);
+ var skipLeft = lowerIndexes[0] === upperIndexes[0], skipRight = lowerIndexes[lowerIndexes.length - 1] === upperIndexes[upperIndexes.length - 1], hull = [];
+ for (i2 = upperIndexes.length - 1; i2 >= 0; --i2)
+ hull.push(points[sortedPoints[upperIndexes[i2]][2]]);
+ for (i2 = +skipLeft; i2 < lowerIndexes.length - skipRight; ++i2)
+ hull.push(points[sortedPoints[lowerIndexes[i2]][2]]);
+ return hull;
+ }
+
+ // modules/geo/vector.js
+ function geoVecEqual(a, b, epsilon3) {
+ if (epsilon3) {
+ return Math.abs(a[0] - b[0]) <= epsilon3 && Math.abs(a[1] - b[1]) <= epsilon3;
+ } else {
+ return a[0] === b[0] && a[1] === b[1];
+ }
+ }
+ function geoVecAdd(a, b) {
+ return [a[0] + b[0], a[1] + b[1]];
+ }
+ function geoVecSubtract(a, b) {
+ return [a[0] - b[0], a[1] - b[1]];
+ }
+ function geoVecScale(a, mag) {
+ return [a[0] * mag, a[1] * mag];
+ }
+ function geoVecFloor(a) {
+ return [Math.floor(a[0]), Math.floor(a[1])];
+ }
+ function geoVecInterp(a, b, t) {
+ return [
+ a[0] + (b[0] - a[0]) * t,
+ a[1] + (b[1] - a[1]) * t
+ ];
+ }
+ function geoVecLength(a, b) {
+ return Math.sqrt(geoVecLengthSquare(a, b));
+ }
+ function geoVecLengthSquare(a, b) {
+ b = b || [0, 0];
+ var x = a[0] - b[0];
+ var y = a[1] - b[1];
+ return x * x + y * y;
+ }
+ function geoVecNormalize(a) {
+ var length = Math.sqrt(a[0] * a[0] + a[1] * a[1]);
+ if (length !== 0) {
+ return geoVecScale(a, 1 / length);
+ }
+ return [0, 0];
+ }
+ function geoVecAngle(a, b) {
+ return Math.atan2(b[1] - a[1], b[0] - a[0]);
+ }
+ function geoVecDot(a, b, origin) {
+ origin = origin || [0, 0];
+ var p = geoVecSubtract(a, origin);
+ var q = geoVecSubtract(b, origin);
+ return p[0] * q[0] + p[1] * q[1];
+ }
+ function geoVecNormalizedDot(a, b, origin) {
+ origin = origin || [0, 0];
+ var p = geoVecNormalize(geoVecSubtract(a, origin));
+ var q = geoVecNormalize(geoVecSubtract(b, origin));
+ return geoVecDot(p, q);
+ }
+ function geoVecCross(a, b, origin) {
+ origin = origin || [0, 0];
+ var p = geoVecSubtract(a, origin);
+ var q = geoVecSubtract(b, origin);
+ return p[0] * q[1] - p[1] * q[0];
+ }
+ function geoVecProject(a, points) {
+ var min3 = Infinity;
+ var idx;
+ var target;
+ for (var i2 = 0; i2 < points.length - 1; i2++) {
+ var o = points[i2];
+ var s = geoVecSubtract(points[i2 + 1], o);
+ var v = geoVecSubtract(a, o);
+ var proj = geoVecDot(v, s) / geoVecDot(s, s);
+ var p;
+ if (proj < 0) {
+ p = o;
+ } else if (proj > 1) {
+ p = points[i2 + 1];
+ } else {
+ p = [o[0] + proj * s[0], o[1] + proj * s[1]];
+ }
+ var dist = geoVecLength(p, a);
+ if (dist < min3) {
+ min3 = dist;
+ idx = i2 + 1;
+ target = p;
+ }
+ }
+ if (idx !== void 0) {
+ return { index: idx, distance: min3, target };
+ } else {
+ return null;
+ }
+ }
+
+ // modules/geo/geom.js
+ function geoAngle(a, b, projection2) {
+ return geoVecAngle(projection2(a.loc), projection2(b.loc));
+ }
+ function geoEdgeEqual(a, b) {
+ return a[0] === b[0] && a[1] === b[1] || a[0] === b[1] && a[1] === b[0];
+ }
+ function geoRotate(points, angle2, around) {
+ return points.map(function(point2) {
+ var radial = geoVecSubtract(point2, around);
+ return [
+ radial[0] * Math.cos(angle2) - radial[1] * Math.sin(angle2) + around[0],
+ radial[0] * Math.sin(angle2) + radial[1] * Math.cos(angle2) + around[1]
+ ];
+ });
+ }
+ function geoChooseEdge(nodes, point2, projection2, activeID) {
+ var dist = geoVecLength;
+ var points = nodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var ids = nodes.map(function(n2) {
+ return n2.id;
+ });
+ var min3 = Infinity;
+ var idx;
+ var loc;
+ for (var i2 = 0; i2 < points.length - 1; i2++) {
+ if (ids[i2] === activeID || ids[i2 + 1] === activeID)
+ continue;
+ var o = points[i2];
+ var s = geoVecSubtract(points[i2 + 1], o);
+ var v = geoVecSubtract(point2, o);
+ var proj = geoVecDot(v, s) / geoVecDot(s, s);
+ var p;
+ if (proj < 0) {
+ p = o;
+ } else if (proj > 1) {
+ p = points[i2 + 1];
+ } else {
+ p = [o[0] + proj * s[0], o[1] + proj * s[1]];
+ }
+ var d = dist(p, point2);
+ if (d < min3) {
+ min3 = d;
+ idx = i2 + 1;
+ loc = projection2.invert(p);
+ }
+ }
+ if (idx !== void 0) {
+ return { index: idx, distance: min3, loc };
+ } else {
+ return null;
+ }
+ }
+ function geoHasLineIntersections(activeNodes, inactiveNodes, activeID) {
+ var actives = [];
+ var inactives = [];
+ var j2, k, n1, n2, segment;
+ for (j2 = 0; j2 < activeNodes.length - 1; j2++) {
+ n1 = activeNodes[j2];
+ n2 = activeNodes[j2 + 1];
+ segment = [n1.loc, n2.loc];
+ if (n1.id === activeID || n2.id === activeID) {
+ actives.push(segment);
+ }
+ }
+ for (j2 = 0; j2 < inactiveNodes.length - 1; j2++) {
+ n1 = inactiveNodes[j2];
+ n2 = inactiveNodes[j2 + 1];
+ segment = [n1.loc, n2.loc];
+ inactives.push(segment);
+ }
+ for (j2 = 0; j2 < actives.length; j2++) {
+ for (k = 0; k < inactives.length; k++) {
+ var p = actives[j2];
+ var q = inactives[k];
+ var hit = geoLineIntersection(p, q);
+ if (hit) {
+ return true;
+ }
+ }
+ }
+ return false;
+ }
+ function geoHasSelfIntersections(nodes, activeID) {
+ var actives = [];
+ var inactives = [];
+ var j2, k;
+ for (j2 = 0; j2 < nodes.length - 1; j2++) {
+ var n1 = nodes[j2];
+ var n2 = nodes[j2 + 1];
+ var segment = [n1.loc, n2.loc];
+ if (n1.id === activeID || n2.id === activeID) {
+ actives.push(segment);
+ } else {
+ inactives.push(segment);
+ }
+ }
+ for (j2 = 0; j2 < actives.length; j2++) {
+ for (k = 0; k < inactives.length; k++) {
+ var p = actives[j2];
+ var q = inactives[k];
+ if (geoVecEqual(p[1], q[0]) || geoVecEqual(p[0], q[1]) || geoVecEqual(p[0], q[0]) || geoVecEqual(p[1], q[1])) {
+ continue;
+ }
+ var hit = geoLineIntersection(p, q);
+ if (hit) {
+ var epsilon3 = 1e-8;
+ if (geoVecEqual(p[1], hit, epsilon3) || geoVecEqual(p[0], hit, epsilon3) || geoVecEqual(q[1], hit, epsilon3) || geoVecEqual(q[0], hit, epsilon3)) {
+ continue;
+ } else {
+ return true;
+ }
+ }
+ }
+ }
+ return false;
+ }
+ function geoLineIntersection(a, b) {
+ var p = [a[0][0], a[0][1]];
+ var p2 = [a[1][0], a[1][1]];
+ var q = [b[0][0], b[0][1]];
+ var q2 = [b[1][0], b[1][1]];
+ var r = geoVecSubtract(p2, p);
+ var s = geoVecSubtract(q2, q);
+ var uNumerator = geoVecCross(geoVecSubtract(q, p), r);
+ var denominator = geoVecCross(r, s);
+ if (uNumerator && denominator) {
+ var u = uNumerator / denominator;
+ var t = geoVecCross(geoVecSubtract(q, p), s) / denominator;
+ if (t >= 0 && t <= 1 && u >= 0 && u <= 1) {
+ return geoVecInterp(p, p2, t);
+ }
+ }
+ return null;
+ }
+ function geoPathIntersections(path1, path2) {
+ var intersections = [];
+ for (var i2 = 0; i2 < path1.length - 1; i2++) {
+ for (var j2 = 0; j2 < path2.length - 1; j2++) {
+ var a = [path1[i2], path1[i2 + 1]];
+ var b = [path2[j2], path2[j2 + 1]];
+ var hit = geoLineIntersection(a, b);
+ if (hit) {
+ intersections.push(hit);
+ }
+ }
+ }
+ return intersections;
+ }
+ function geoPathHasIntersections(path1, path2) {
+ for (var i2 = 0; i2 < path1.length - 1; i2++) {
+ for (var j2 = 0; j2 < path2.length - 1; j2++) {
+ var a = [path1[i2], path1[i2 + 1]];
+ var b = [path2[j2], path2[j2 + 1]];
+ var hit = geoLineIntersection(a, b);
+ if (hit) {
+ return true;
+ }
+ }
+ }
+ return false;
+ }
+ function geoPointInPolygon(point2, polygon2) {
+ var x = point2[0];
+ var y = point2[1];
+ var inside = false;
+ for (var i2 = 0, j2 = polygon2.length - 1; i2 < polygon2.length; j2 = i2++) {
+ var xi = polygon2[i2][0];
+ var yi = polygon2[i2][1];
+ var xj = polygon2[j2][0];
+ var yj = polygon2[j2][1];
+ var intersect2 = yi > y !== yj > y && x < (xj - xi) * (y - yi) / (yj - yi) + xi;
+ if (intersect2)
+ inside = !inside;
+ }
+ return inside;
+ }
+ function geoPolygonContainsPolygon(outer, inner) {
+ return inner.every(function(point2) {
+ return geoPointInPolygon(point2, outer);
+ });
+ }
+ function geoPolygonIntersectsPolygon(outer, inner, checkSegments) {
+ function testPoints(outer2, inner2) {
+ return inner2.some(function(point2) {
+ return geoPointInPolygon(point2, outer2);
+ });
+ }
+ return testPoints(outer, inner) || !!checkSegments && geoPathHasIntersections(outer, inner);
+ }
+ function geoGetSmallestSurroundingRectangle(points) {
+ var hull = hull_default(points);
+ var centroid = centroid_default2(hull);
+ var minArea = Infinity;
+ var ssrExtent = [];
+ var ssrAngle = 0;
+ var c1 = hull[0];
+ for (var i2 = 0; i2 <= hull.length - 1; i2++) {
+ var c2 = i2 === hull.length - 1 ? hull[0] : hull[i2 + 1];
+ var angle2 = Math.atan2(c2[1] - c1[1], c2[0] - c1[0]);
+ var poly = geoRotate(hull, -angle2, centroid);
+ var extent = poly.reduce(function(extent2, point2) {
+ return extent2.extend(geoExtent(point2));
+ }, geoExtent());
+ var area = extent.area();
+ if (area < minArea) {
+ minArea = area;
+ ssrExtent = extent;
+ ssrAngle = angle2;
+ }
+ c1 = c2;
+ }
+ return {
+ poly: geoRotate(ssrExtent.polygon(), ssrAngle, centroid),
+ angle: ssrAngle
+ };
+ }
+ function geoPathLength(path) {
+ var length = 0;
+ for (var i2 = 0; i2 < path.length - 1; i2++) {
+ length += geoVecLength(path[i2], path[i2 + 1]);
+ }
+ return length;
+ }
+ function geoViewportEdge(point2, dimensions) {
+ var pad2 = [80, 20, 50, 20];
+ var x = 0;
+ var y = 0;
+ if (point2[0] > dimensions[0] - pad2[1]) {
+ x = -10;
+ }
+ if (point2[0] < pad2[3]) {
+ x = 10;
+ }
+ if (point2[1] > dimensions[1] - pad2[2]) {
+ y = -10;
+ }
+ if (point2[1] < pad2[0]) {
+ y = 10;
+ }
+ if (x || y) {
+ return [x, y];
+ } else {
+ return null;
+ }
+ }
+
+ // node_modules/d3-dispatch/src/dispatch.js
+ var noop2 = { value: () => {
+ } };
+ function dispatch() {
+ for (var i2 = 0, n2 = arguments.length, _ = {}, t; i2 < n2; ++i2) {
+ if (!(t = arguments[i2] + "") || t in _ || /[\s.]/.test(t))
+ throw new Error("illegal type: " + t);
+ _[t] = [];
+ }
+ return new Dispatch(_);
+ }
+ function Dispatch(_) {
+ this._ = _;
+ }
+ function parseTypenames(typenames, types) {
+ return typenames.trim().split(/^|\s+/).map(function(t) {
+ var name = "", i2 = t.indexOf(".");
+ if (i2 >= 0)
+ name = t.slice(i2 + 1), t = t.slice(0, i2);
+ if (t && !types.hasOwnProperty(t))
+ throw new Error("unknown type: " + t);
+ return { type: t, name };
+ });
+ }
+ Dispatch.prototype = dispatch.prototype = {
+ constructor: Dispatch,
+ on: function(typename, callback) {
+ var _ = this._, T = parseTypenames(typename + "", _), t, i2 = -1, n2 = T.length;
+ if (arguments.length < 2) {
+ while (++i2 < n2)
+ if ((t = (typename = T[i2]).type) && (t = get(_[t], typename.name)))
+ return t;
+ return;
+ }
+ if (callback != null && typeof callback !== "function")
+ throw new Error("invalid callback: " + callback);
+ while (++i2 < n2) {
+ if (t = (typename = T[i2]).type)
+ _[t] = set(_[t], typename.name, callback);
+ else if (callback == null)
+ for (t in _)
+ _[t] = set(_[t], typename.name, null);
+ }
+ return this;
+ },
+ copy: function() {
+ var copy2 = {}, _ = this._;
+ for (var t in _)
+ copy2[t] = _[t].slice();
+ return new Dispatch(copy2);
+ },
+ call: function(type2, that) {
+ if ((n2 = arguments.length - 2) > 0)
+ for (var args = new Array(n2), i2 = 0, n2, t; i2 < n2; ++i2)
+ args[i2] = arguments[i2 + 2];
+ if (!this._.hasOwnProperty(type2))
+ throw new Error("unknown type: " + type2);
+ for (t = this._[type2], i2 = 0, n2 = t.length; i2 < n2; ++i2)
+ t[i2].value.apply(that, args);
+ },
+ apply: function(type2, that, args) {
+ if (!this._.hasOwnProperty(type2))
+ throw new Error("unknown type: " + type2);
+ for (var t = this._[type2], i2 = 0, n2 = t.length; i2 < n2; ++i2)
+ t[i2].value.apply(that, args);
+ }
+ };
+ function get(type2, name) {
+ for (var i2 = 0, n2 = type2.length, c; i2 < n2; ++i2) {
+ if ((c = type2[i2]).name === name) {
+ return c.value;
+ }
+ }
+ }
+ function set(type2, name, callback) {
+ for (var i2 = 0, n2 = type2.length; i2 < n2; ++i2) {
+ if (type2[i2].name === name) {
+ type2[i2] = noop2, type2 = type2.slice(0, i2).concat(type2.slice(i2 + 1));
+ break;
+ }
+ }
+ if (callback != null)
+ type2.push({ name, value: callback });
+ return type2;
+ }
+ var dispatch_default = dispatch;
+
+ // node_modules/d3-selection/src/namespaces.js
+ var xhtml = "http://www.w3.org/1999/xhtml";
+ var namespaces_default = {
+ svg: "http://www.w3.org/2000/svg",
+ xhtml,
+ xlink: "http://www.w3.org/1999/xlink",
+ xml: "http://www.w3.org/XML/1998/namespace",
+ xmlns: "http://www.w3.org/2000/xmlns/"
+ };
+
+ // node_modules/d3-selection/src/namespace.js
+ function namespace_default(name) {
+ var prefix = name += "", i2 = prefix.indexOf(":");
+ if (i2 >= 0 && (prefix = name.slice(0, i2)) !== "xmlns")
+ name = name.slice(i2 + 1);
+ return namespaces_default.hasOwnProperty(prefix) ? { space: namespaces_default[prefix], local: name } : name;
+ }
+
+ // node_modules/d3-selection/src/creator.js
+ function creatorInherit(name) {
+ return function() {
+ var document2 = this.ownerDocument, uri = this.namespaceURI;
+ return uri === xhtml && document2.documentElement.namespaceURI === xhtml ? document2.createElement(name) : document2.createElementNS(uri, name);
+ };
+ }
+ function creatorFixed(fullname) {
+ return function() {
+ return this.ownerDocument.createElementNS(fullname.space, fullname.local);
+ };
+ }
+ function creator_default(name) {
+ var fullname = namespace_default(name);
+ return (fullname.local ? creatorFixed : creatorInherit)(fullname);
+ }
+
+ // node_modules/d3-selection/src/selector.js
+ function none() {
+ }
+ function selector_default(selector) {
+ return selector == null ? none : function() {
+ return this.querySelector(selector);
+ };
+ }
+
+ // node_modules/d3-selection/src/selection/select.js
+ function select_default(select) {
+ if (typeof select !== "function")
+ select = selector_default(select);
+ for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, subgroup = subgroups[j2] = new Array(n2), node, subnode, i2 = 0; i2 < n2; ++i2) {
+ if ((node = group[i2]) && (subnode = select.call(node, node.__data__, i2, group))) {
+ if ("__data__" in node)
+ subnode.__data__ = node.__data__;
+ subgroup[i2] = subnode;
+ }
+ }
+ }
+ return new Selection(subgroups, this._parents);
+ }
+
+ // node_modules/d3-selection/src/array.js
+ function array(x) {
+ return x == null ? [] : Array.isArray(x) ? x : Array.from(x);
+ }
+
+ // node_modules/d3-selection/src/selectorAll.js
+ function empty() {
+ return [];
+ }
+ function selectorAll_default(selector) {
+ return selector == null ? empty : function() {
+ return this.querySelectorAll(selector);
+ };
+ }
+
+ // node_modules/d3-selection/src/selection/selectAll.js
+ function arrayAll(select) {
+ return function() {
+ return array(select.apply(this, arguments));
+ };
+ }
+ function selectAll_default(select) {
+ if (typeof select === "function")
+ select = arrayAll(select);
+ else
+ select = selectorAll_default(select);
+ for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, node, i2 = 0; i2 < n2; ++i2) {
+ if (node = group[i2]) {
+ subgroups.push(select.call(node, node.__data__, i2, group));
+ parents.push(node);
+ }
+ }
+ }
+ return new Selection(subgroups, parents);
+ }
+
+ // node_modules/d3-selection/src/matcher.js
+ function matcher_default(selector) {
+ return function() {
+ return this.matches(selector);
+ };
+ }
+ function childMatcher(selector) {
+ return function(node) {
+ return node.matches(selector);
+ };
+ }
+
+ // node_modules/d3-selection/src/selection/selectChild.js
+ var find = Array.prototype.find;
+ function childFind(match) {
+ return function() {
+ return find.call(this.children, match);
+ };
+ }
+ function childFirst() {
+ return this.firstElementChild;
+ }
+ function selectChild_default(match) {
+ return this.select(match == null ? childFirst : childFind(typeof match === "function" ? match : childMatcher(match)));
+ }
+
+ // node_modules/d3-selection/src/selection/selectChildren.js
+ var filter = Array.prototype.filter;
+ function children() {
+ return Array.from(this.children);
+ }
+ function childrenFilter(match) {
+ return function() {
+ return filter.call(this.children, match);
+ };
+ }
+ function selectChildren_default(match) {
+ return this.selectAll(match == null ? children : childrenFilter(typeof match === "function" ? match : childMatcher(match)));
+ }
+
+ // node_modules/d3-selection/src/selection/filter.js
+ function filter_default(match) {
+ if (typeof match !== "function")
+ match = matcher_default(match);
+ for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, subgroup = subgroups[j2] = [], node, i2 = 0; i2 < n2; ++i2) {
+ if ((node = group[i2]) && match.call(node, node.__data__, i2, group)) {
+ subgroup.push(node);
+ }
+ }
+ }
+ return new Selection(subgroups, this._parents);
+ }
+
+ // node_modules/d3-selection/src/selection/sparse.js
+ function sparse_default(update) {
+ return new Array(update.length);
+ }
+
+ // node_modules/d3-selection/src/selection/enter.js
+ function enter_default() {
+ return new Selection(this._enter || this._groups.map(sparse_default), this._parents);
+ }
+ function EnterNode(parent, datum2) {
+ this.ownerDocument = parent.ownerDocument;
+ this.namespaceURI = parent.namespaceURI;
+ this._next = null;
+ this._parent = parent;
+ this.__data__ = datum2;
+ }
+ EnterNode.prototype = {
+ constructor: EnterNode,
+ appendChild: function(child) {
+ return this._parent.insertBefore(child, this._next);
+ },
+ insertBefore: function(child, next) {
+ return this._parent.insertBefore(child, next);
+ },
+ querySelector: function(selector) {
+ return this._parent.querySelector(selector);
+ },
+ querySelectorAll: function(selector) {
+ return this._parent.querySelectorAll(selector);
+ }
+ };
+
+ // node_modules/d3-selection/src/constant.js
+ function constant_default(x) {
+ return function() {
+ return x;
+ };
+ }
+
+ // node_modules/d3-selection/src/selection/data.js
+ function bindIndex(parent, group, enter, update, exit, data) {
+ var i2 = 0, node, groupLength = group.length, dataLength = data.length;
+ for (; i2 < dataLength; ++i2) {
+ if (node = group[i2]) {
+ node.__data__ = data[i2];
+ update[i2] = node;
+ } else {
+ enter[i2] = new EnterNode(parent, data[i2]);
+ }
+ }
+ for (; i2 < groupLength; ++i2) {
+ if (node = group[i2]) {
+ exit[i2] = node;
+ }
+ }
+ }
+ function bindKey(parent, group, enter, update, exit, data, key) {
+ var i2, node, nodeByKeyValue = /* @__PURE__ */ new Map(), groupLength = group.length, dataLength = data.length, keyValues = new Array(groupLength), keyValue;
+ for (i2 = 0; i2 < groupLength; ++i2) {
+ if (node = group[i2]) {
+ keyValues[i2] = keyValue = key.call(node, node.__data__, i2, group) + "";
+ if (nodeByKeyValue.has(keyValue)) {
+ exit[i2] = node;
+ } else {
+ nodeByKeyValue.set(keyValue, node);
+ }
+ }
+ }
+ for (i2 = 0; i2 < dataLength; ++i2) {
+ keyValue = key.call(parent, data[i2], i2, data) + "";
+ if (node = nodeByKeyValue.get(keyValue)) {
+ update[i2] = node;
+ node.__data__ = data[i2];
+ nodeByKeyValue.delete(keyValue);
+ } else {
+ enter[i2] = new EnterNode(parent, data[i2]);
+ }
+ }
+ for (i2 = 0; i2 < groupLength; ++i2) {
+ if ((node = group[i2]) && nodeByKeyValue.get(keyValues[i2]) === node) {
+ exit[i2] = node;
+ }
+ }
+ }
+ function datum(node) {
+ return node.__data__;
+ }
+ function data_default(value, key) {
+ if (!arguments.length)
+ return Array.from(this, datum);
+ var bind = key ? bindKey : bindIndex, parents = this._parents, groups = this._groups;
+ if (typeof value !== "function")
+ value = constant_default(value);
+ for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j2 = 0; j2 < m; ++j2) {
+ var parent = parents[j2], group = groups[j2], groupLength = group.length, data = arraylike(value.call(parent, parent && parent.__data__, j2, parents)), dataLength = data.length, enterGroup = enter[j2] = new Array(dataLength), updateGroup = update[j2] = new Array(dataLength), exitGroup = exit[j2] = new Array(groupLength);
+ bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);
+ for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
+ if (previous = enterGroup[i0]) {
+ if (i0 >= i1)
+ i1 = i0 + 1;
+ while (!(next = updateGroup[i1]) && ++i1 < dataLength)
+ ;
+ previous._next = next || null;
+ }
+ }
+ }
+ update = new Selection(update, parents);
+ update._enter = enter;
+ update._exit = exit;
+ return update;
+ }
+ function arraylike(data) {
+ return typeof data === "object" && "length" in data ? data : Array.from(data);
+ }
+
+ // node_modules/d3-selection/src/selection/exit.js
+ function exit_default() {
+ return new Selection(this._exit || this._groups.map(sparse_default), this._parents);
+ }
+
+ // node_modules/d3-selection/src/selection/join.js
+ function join_default(onenter, onupdate, onexit) {
+ var enter = this.enter(), update = this, exit = this.exit();
+ if (typeof onenter === "function") {
+ enter = onenter(enter);
+ if (enter)
+ enter = enter.selection();
+ } else {
+ enter = enter.append(onenter + "");
+ }
+ if (onupdate != null) {
+ update = onupdate(update);
+ if (update)
+ update = update.selection();
+ }
+ if (onexit == null)
+ exit.remove();
+ else
+ onexit(exit);
+ return enter && update ? enter.merge(update).order() : update;
+ }
+
+ // node_modules/d3-selection/src/selection/merge.js
+ function merge_default(context) {
+ var selection2 = context.selection ? context.selection() : context;
+ for (var groups0 = this._groups, groups1 = selection2._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j2 = 0; j2 < m; ++j2) {
+ for (var group0 = groups0[j2], group1 = groups1[j2], n2 = group0.length, merge3 = merges[j2] = new Array(n2), node, i2 = 0; i2 < n2; ++i2) {
+ if (node = group0[i2] || group1[i2]) {
+ merge3[i2] = node;
+ }
+ }
+ }
+ for (; j2 < m0; ++j2) {
+ merges[j2] = groups0[j2];
+ }
+ return new Selection(merges, this._parents);
+ }
+
+ // node_modules/d3-selection/src/selection/order.js
+ function order_default() {
+ for (var groups = this._groups, j2 = -1, m = groups.length; ++j2 < m; ) {
+ for (var group = groups[j2], i2 = group.length - 1, next = group[i2], node; --i2 >= 0; ) {
+ if (node = group[i2]) {
+ if (next && node.compareDocumentPosition(next) ^ 4)
+ next.parentNode.insertBefore(node, next);
+ next = node;
+ }
+ }
+ }
+ return this;
+ }
+
+ // node_modules/d3-selection/src/selection/sort.js
+ function sort_default(compare) {
+ if (!compare)
+ compare = ascending2;
+ function compareNode(a, b) {
+ return a && b ? compare(a.__data__, b.__data__) : !a - !b;
+ }
+ for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, sortgroup = sortgroups[j2] = new Array(n2), node, i2 = 0; i2 < n2; ++i2) {
+ if (node = group[i2]) {
+ sortgroup[i2] = node;
+ }
+ }
+ sortgroup.sort(compareNode);
+ }
+ return new Selection(sortgroups, this._parents).order();
+ }
+ function ascending2(a, b) {
+ return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
+ }
+
+ // node_modules/d3-selection/src/selection/call.js
+ function call_default() {
+ var callback = arguments[0];
+ arguments[0] = this;
+ callback.apply(null, arguments);
+ return this;
+ }
+
+ // node_modules/d3-selection/src/selection/nodes.js
+ function nodes_default() {
+ return Array.from(this);
+ }
+
+ // node_modules/d3-selection/src/selection/node.js
+ function node_default() {
+ for (var groups = this._groups, j2 = 0, m = groups.length; j2 < m; ++j2) {
+ for (var group = groups[j2], i2 = 0, n2 = group.length; i2 < n2; ++i2) {
+ var node = group[i2];
+ if (node)
+ return node;
+ }
+ }
+ return null;
+ }
+
+ // node_modules/d3-selection/src/selection/size.js
+ function size_default() {
+ let size = 0;
+ for (const node of this)
+ ++size;
+ return size;
+ }
+
+ // node_modules/d3-selection/src/selection/empty.js
+ function empty_default() {
+ return !this.node();
+ }
+
+ // node_modules/d3-selection/src/selection/each.js
+ function each_default(callback) {
+ for (var groups = this._groups, j2 = 0, m = groups.length; j2 < m; ++j2) {
+ for (var group = groups[j2], i2 = 0, n2 = group.length, node; i2 < n2; ++i2) {
+ if (node = group[i2])
+ callback.call(node, node.__data__, i2, group);
+ }
+ }
+ return this;
+ }
+
+ // node_modules/d3-selection/src/selection/attr.js
+ function attrRemove(name) {
+ return function() {
+ this.removeAttribute(name);
+ };
+ }
+ function attrRemoveNS(fullname) {
+ return function() {
+ this.removeAttributeNS(fullname.space, fullname.local);
+ };
+ }
+ function attrConstant(name, value) {
+ return function() {
+ this.setAttribute(name, value);
+ };
+ }
+ function attrConstantNS(fullname, value) {
+ return function() {
+ this.setAttributeNS(fullname.space, fullname.local, value);
+ };
+ }
+ function attrFunction(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) {
+ 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);
+ };
+ }
+ function attr_default(name, value) {
+ var fullname = namespace_default(name);
+ if (arguments.length < 2) {
+ var node = this.node();
+ 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));
+ }
+
+ // node_modules/d3-selection/src/window.js
+ function window_default(node) {
+ return node.ownerDocument && node.ownerDocument.defaultView || node.document && node || node.defaultView;
+ }
+
+ // node_modules/d3-selection/src/selection/style.js
+ function styleRemove(name) {
+ return function() {
+ this.style.removeProperty(name);
+ };
+ }
+ function styleConstant(name, value, priority) {
+ return function() {
+ this.style.setProperty(name, value, priority);
+ };
+ }
+ function styleFunction(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 style_default(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);
+ }
+ function styleValue(node, name) {
+ return node.style.getPropertyValue(name) || window_default(node).getComputedStyle(node, null).getPropertyValue(name);
+ }
+
+ // node_modules/d3-selection/src/selection/property.js
+ function propertyRemove(name) {
+ return function() {
+ delete this[name];
+ };
+ }
+ function propertyConstant(name, value) {
+ return function() {
+ this[name] = value;
+ };
+ }
+ function propertyFunction(name, value) {
+ return function() {
+ var v = value.apply(this, arguments);
+ if (v == null)
+ delete this[name];
+ else
+ this[name] = v;
+ };
+ }
+ function property_default(name, value) {
+ return arguments.length > 1 ? this.each((value == null ? propertyRemove : typeof value === "function" ? propertyFunction : propertyConstant)(name, value)) : this.node()[name];
+ }
+
+ // node_modules/d3-selection/src/selection/classed.js
+ function classArray(string) {
+ return string.trim().split(/^|\s+/);
+ }
+ function classList(node) {
+ return node.classList || new ClassList(node);
+ }
+ function ClassList(node) {
+ this._node = node;
+ this._names = classArray(node.getAttribute("class") || "");
+ }
+ ClassList.prototype = {
+ add: function(name) {
+ var i2 = this._names.indexOf(name);
+ if (i2 < 0) {
+ this._names.push(name);
+ this._node.setAttribute("class", this._names.join(" "));
+ }
+ },
+ remove: function(name) {
+ var i2 = this._names.indexOf(name);
+ if (i2 >= 0) {
+ this._names.splice(i2, 1);
+ this._node.setAttribute("class", this._names.join(" "));
+ }
+ },
+ contains: function(name) {
+ return this._names.indexOf(name) >= 0;
+ }
+ };
+ function classedAdd(node, names) {
+ var list = classList(node), i2 = -1, n2 = names.length;
+ while (++i2 < n2)
+ list.add(names[i2]);
+ }
+ function classedRemove(node, names) {
+ var list = classList(node), i2 = -1, n2 = names.length;
+ while (++i2 < n2)
+ list.remove(names[i2]);
+ }
+ function classedTrue(names) {
+ return function() {
+ classedAdd(this, names);
+ };
+ }
+ function classedFalse(names) {
+ return function() {
+ classedRemove(this, names);
+ };
+ }
+ function classedFunction(names, value) {
+ return function() {
+ (value.apply(this, arguments) ? classedAdd : classedRemove)(this, names);
+ };
+ }
+ function classed_default(name, value) {
+ var names = classArray(name + "");
+ if (arguments.length < 2) {
+ var list = classList(this.node()), i2 = -1, n2 = names.length;
+ while (++i2 < n2)
+ if (!list.contains(names[i2]))
+ return false;
+ return true;
+ }
+ return this.each((typeof value === "function" ? classedFunction : value ? classedTrue : classedFalse)(names, value));
+ }
+
+ // node_modules/d3-selection/src/selection/text.js
+ function textRemove() {
+ this.textContent = "";
+ }
+ function textConstant(value) {
+ return function() {
+ this.textContent = value;
+ };
+ }
+ function textFunction(value) {
+ return function() {
+ var v = value.apply(this, arguments);
+ this.textContent = v == null ? "" : v;
+ };
+ }
+ function text_default(value) {
+ return arguments.length ? this.each(value == null ? textRemove : (typeof value === "function" ? textFunction : textConstant)(value)) : this.node().textContent;
+ }
+
+ // node_modules/d3-selection/src/selection/html.js
+ function htmlRemove() {
+ this.innerHTML = "";
+ }
+ function htmlConstant(value) {
+ return function() {
+ this.innerHTML = value;
+ };
+ }
+ function htmlFunction(value) {
+ return function() {
+ var v = value.apply(this, arguments);
+ this.innerHTML = v == null ? "" : v;
+ };
+ }
+ function html_default(value) {
+ return arguments.length ? this.each(value == null ? htmlRemove : (typeof value === "function" ? htmlFunction : htmlConstant)(value)) : this.node().innerHTML;
+ }
+
+ // node_modules/d3-selection/src/selection/raise.js
+ function raise() {
+ if (this.nextSibling)
+ this.parentNode.appendChild(this);
+ }
+ function raise_default() {
+ return this.each(raise);
+ }
+
+ // node_modules/d3-selection/src/selection/lower.js
+ function lower() {
+ if (this.previousSibling)
+ this.parentNode.insertBefore(this, this.parentNode.firstChild);
+ }
+ function lower_default() {
+ return this.each(lower);
+ }
+
+ // node_modules/d3-selection/src/selection/append.js
+ function append_default(name) {
+ var create2 = typeof name === "function" ? name : creator_default(name);
+ return this.select(function() {
+ return this.appendChild(create2.apply(this, arguments));
+ });
+ }
+
+ // node_modules/d3-selection/src/selection/insert.js
+ function constantNull() {
+ return null;
+ }
+ function insert_default(name, before) {
+ var create2 = typeof name === "function" ? name : creator_default(name), select = before == null ? constantNull : typeof before === "function" ? before : selector_default(before);
+ return this.select(function() {
+ return this.insertBefore(create2.apply(this, arguments), select.apply(this, arguments) || null);
+ });
+ }
+
+ // node_modules/d3-selection/src/selection/remove.js
+ function remove() {
+ var parent = this.parentNode;
+ if (parent)
+ parent.removeChild(this);
+ }
+ function remove_default() {
+ return this.each(remove);
+ }
+
+ // node_modules/d3-selection/src/selection/clone.js
+ function selection_cloneShallow() {
+ var clone = this.cloneNode(false), parent = this.parentNode;
+ return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
+ }
+ function selection_cloneDeep() {
+ var clone = this.cloneNode(true), parent = this.parentNode;
+ return parent ? parent.insertBefore(clone, this.nextSibling) : clone;
+ }
+ function clone_default(deep) {
+ return this.select(deep ? selection_cloneDeep : selection_cloneShallow);
+ }
+
+ // node_modules/d3-selection/src/selection/datum.js
+ function datum_default(value) {
+ return arguments.length ? this.property("__data__", value) : this.node().__data__;
+ }
+
+ // node_modules/d3-selection/src/selection/on.js
+ function contextListener(listener) {
+ return function(event) {
+ listener.call(this, event, this.__data__);
+ };
+ }
+ function parseTypenames2(typenames) {
+ return typenames.trim().split(/^|\s+/).map(function(t) {
+ var name = "", i2 = t.indexOf(".");
+ if (i2 >= 0)
+ name = t.slice(i2 + 1), t = t.slice(0, i2);
+ return { type: t, name };
+ });
+ }
+ function onRemove(typename) {
+ return function() {
+ var on = this.__on;
+ if (!on)
+ return;
+ for (var j2 = 0, i2 = -1, m = on.length, o; j2 < m; ++j2) {
+ if (o = on[j2], (!typename.type || o.type === typename.type) && o.name === typename.name) {
+ this.removeEventListener(o.type, o.listener, o.options);
+ } else {
+ on[++i2] = o;
+ }
+ }
+ if (++i2)
+ on.length = i2;
+ else
+ delete this.__on;
+ };
+ }
+ function onAdd(typename, value, options2) {
+ return function() {
+ var on = this.__on, o, listener = contextListener(value);
+ if (on)
+ for (var j2 = 0, m = on.length; j2 < m; ++j2) {
+ if ((o = on[j2]).type === typename.type && o.name === typename.name) {
+ this.removeEventListener(o.type, o.listener, o.options);
+ this.addEventListener(o.type, o.listener = listener, o.options = options2);
+ o.value = value;
+ return;
+ }
+ }
+ this.addEventListener(typename.type, listener, options2);
+ o = { type: typename.type, name: typename.name, value, listener, options: options2 };
+ if (!on)
+ this.__on = [o];
+ else
+ on.push(o);
+ };
+ }
+ function on_default(typename, value, options2) {
+ var typenames = parseTypenames2(typename + ""), i2, n2 = typenames.length, t;
+ if (arguments.length < 2) {
+ var on = this.node().__on;
+ if (on)
+ for (var j2 = 0, m = on.length, o; j2 < m; ++j2) {
+ for (i2 = 0, o = on[j2]; i2 < n2; ++i2) {
+ if ((t = typenames[i2]).type === o.type && t.name === o.name) {
+ return o.value;
+ }
+ }
+ }
+ return;
+ }
+ on = value ? onAdd : onRemove;
+ for (i2 = 0; i2 < n2; ++i2)
+ this.each(on(typenames[i2], value, options2));
+ return this;
+ }
+
+ // node_modules/d3-selection/src/selection/dispatch.js
+ function dispatchEvent(node, type2, params) {
+ var window2 = window_default(node), event = window2.CustomEvent;
+ if (typeof event === "function") {
+ event = new event(type2, params);
+ } else {
+ event = window2.document.createEvent("Event");
+ if (params)
+ event.initEvent(type2, params.bubbles, params.cancelable), event.detail = params.detail;
+ else
+ event.initEvent(type2, false, false);
+ }
+ node.dispatchEvent(event);
+ }
+ function dispatchConstant(type2, params) {
+ return function() {
+ return dispatchEvent(this, type2, params);
+ };
+ }
+ function dispatchFunction(type2, params) {
+ return function() {
+ return dispatchEvent(this, type2, params.apply(this, arguments));
+ };
+ }
+ function dispatch_default2(type2, params) {
+ return this.each((typeof params === "function" ? dispatchFunction : dispatchConstant)(type2, params));
+ }
+
+ // node_modules/d3-selection/src/selection/iterator.js
+ function* iterator_default() {
+ for (var groups = this._groups, j2 = 0, m = groups.length; j2 < m; ++j2) {
+ for (var group = groups[j2], i2 = 0, n2 = group.length, node; i2 < n2; ++i2) {
+ if (node = group[i2])
+ yield node;
+ }
+ }
+ }
+
+ // node_modules/d3-selection/src/selection/index.js
+ var root = [null];
+ function Selection(groups, parents) {
+ this._groups = groups;
+ this._parents = parents;
+ }
+ function selection() {
+ return new Selection([[document.documentElement]], root);
+ }
+ function selection_selection() {
+ return this;
+ }
+ Selection.prototype = selection.prototype = {
+ constructor: Selection,
+ select: select_default,
+ selectAll: selectAll_default,
+ selectChild: selectChild_default,
+ selectChildren: selectChildren_default,
+ filter: filter_default,
+ data: data_default,
+ enter: enter_default,
+ exit: exit_default,
+ join: join_default,
+ merge: merge_default,
+ selection: selection_selection,
+ order: order_default,
+ sort: sort_default,
+ call: call_default,
+ nodes: nodes_default,
+ node: node_default,
+ size: size_default,
+ empty: empty_default,
+ each: each_default,
+ attr: attr_default,
+ style: style_default,
+ property: property_default,
+ classed: classed_default,
+ text: text_default,
+ html: html_default,
+ raise: raise_default,
+ lower: lower_default,
+ append: append_default,
+ insert: insert_default,
+ remove: remove_default,
+ clone: clone_default,
+ datum: datum_default,
+ on: on_default,
+ dispatch: dispatch_default2,
+ [Symbol.iterator]: iterator_default
+ };
+ var selection_default = selection;
+
+ // node_modules/d3-selection/src/select.js
+ function select_default2(selector) {
+ return typeof selector === "string" ? new Selection([[document.querySelector(selector)]], [document.documentElement]) : new Selection([[selector]], root);
+ }
+
+ // node_modules/d3-selection/src/sourceEvent.js
+ function sourceEvent_default(event) {
+ let sourceEvent;
+ while (sourceEvent = event.sourceEvent)
+ event = sourceEvent;
+ return event;
+ }
+
+ // node_modules/d3-selection/src/pointer.js
+ function pointer_default(event, node) {
+ event = sourceEvent_default(event);
+ if (node === void 0)
+ node = event.currentTarget;
+ if (node) {
+ var svg2 = node.ownerSVGElement || node;
+ if (svg2.createSVGPoint) {
+ var point2 = svg2.createSVGPoint();
+ point2.x = event.clientX, point2.y = event.clientY;
+ point2 = point2.matrixTransform(node.getScreenCTM().inverse());
+ return [point2.x, point2.y];
+ }
+ if (node.getBoundingClientRect) {
+ var rect = node.getBoundingClientRect();
+ return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
+ }
+ }
+ return [event.pageX, event.pageY];
+ }
+
+ // node_modules/d3-selection/src/selectAll.js
+ function selectAll_default2(selector) {
+ return typeof selector === "string" ? new Selection([document.querySelectorAll(selector)], [document.documentElement]) : new Selection([array(selector)], root);
+ }
+
+ // node_modules/d3-drag/src/noevent.js
+ var nonpassive = { passive: false };
+ var nonpassivecapture = { capture: true, passive: false };
+ function nopropagation(event) {
+ event.stopImmediatePropagation();
+ }
+ function noevent_default(event) {
+ event.preventDefault();
+ event.stopImmediatePropagation();
+ }
+
+ // node_modules/d3-drag/src/nodrag.js
+ function nodrag_default(view) {
+ var root3 = view.document.documentElement, selection2 = select_default2(view).on("dragstart.drag", noevent_default, nonpassivecapture);
+ if ("onselectstart" in root3) {
+ selection2.on("selectstart.drag", noevent_default, nonpassivecapture);
+ } else {
+ root3.__noselect = root3.style.MozUserSelect;
+ root3.style.MozUserSelect = "none";
+ }
+ }
+ function yesdrag(view, noclick) {
+ var root3 = view.document.documentElement, selection2 = select_default2(view).on("dragstart.drag", null);
+ if (noclick) {
+ selection2.on("click.drag", noevent_default, nonpassivecapture);
+ setTimeout(function() {
+ selection2.on("click.drag", null);
+ }, 0);
+ }
+ if ("onselectstart" in root3) {
+ selection2.on("selectstart.drag", null);
+ } else {
+ root3.style.MozUserSelect = root3.__noselect;
+ delete root3.__noselect;
+ }
+ }
+
+ // node_modules/d3-drag/src/constant.js
+ var constant_default2 = (x) => () => x;
+
+ // node_modules/d3-drag/src/event.js
+ function DragEvent(type2, {
+ sourceEvent,
+ subject,
+ target,
+ identifier,
+ active,
+ x,
+ y,
+ dx,
+ dy,
+ dispatch: dispatch10
+ }) {
+ Object.defineProperties(this, {
+ type: { value: type2, enumerable: true, configurable: true },
+ sourceEvent: { value: sourceEvent, enumerable: true, configurable: true },
+ subject: { value: subject, enumerable: true, configurable: true },
+ target: { value: target, enumerable: true, configurable: true },
+ identifier: { value: identifier, enumerable: true, configurable: true },
+ active: { value: active, enumerable: true, configurable: true },
+ x: { value: x, enumerable: true, configurable: true },
+ y: { value: y, enumerable: true, configurable: true },
+ dx: { value: dx, enumerable: true, configurable: true },
+ dy: { value: dy, enumerable: true, configurable: true },
+ _: { value: dispatch10 }
+ });
+ }
+ DragEvent.prototype.on = function() {
+ var value = this._.on.apply(this._, arguments);
+ return value === this._ ? this : value;
+ };
+
+ // node_modules/d3-drag/src/drag.js
+ function defaultFilter(event) {
+ return !event.ctrlKey && !event.button;
+ }
+ function defaultContainer() {
+ return this.parentNode;
+ }
+ function defaultSubject(event, d) {
+ return d == null ? { x: event.x, y: event.y } : d;
+ }
+ function defaultTouchable() {
+ return navigator.maxTouchPoints || "ontouchstart" in this;
+ }
+ function drag_default() {
+ var filter2 = defaultFilter, container = defaultContainer, subject = defaultSubject, touchable = defaultTouchable, gestures = {}, listeners = dispatch_default("start", "drag", "end"), active = 0, mousedownx, mousedowny, mousemoving, touchending, clickDistance2 = 0;
+ function drag(selection2) {
+ selection2.on("mousedown.drag", mousedowned).filter(touchable).on("touchstart.drag", touchstarted).on("touchmove.drag", touchmoved, nonpassive).on("touchend.drag touchcancel.drag", touchended).style("touch-action", "none").style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
+ }
+ function mousedowned(event, d) {
+ if (touchending || !filter2.call(this, event, d))
+ return;
+ var gesture = beforestart(this, container.call(this, event, d), event, d, "mouse");
+ if (!gesture)
+ return;
+ select_default2(event.view).on("mousemove.drag", mousemoved, nonpassivecapture).on("mouseup.drag", mouseupped, nonpassivecapture);
+ nodrag_default(event.view);
+ nopropagation(event);
+ mousemoving = false;
+ mousedownx = event.clientX;
+ mousedowny = event.clientY;
+ gesture("start", event);
+ }
+ function mousemoved(event) {
+ noevent_default(event);
+ if (!mousemoving) {
+ var dx = event.clientX - mousedownx, dy = event.clientY - mousedowny;
+ mousemoving = dx * dx + dy * dy > clickDistance2;
+ }
+ gestures.mouse("drag", event);
+ }
+ function mouseupped(event) {
+ select_default2(event.view).on("mousemove.drag mouseup.drag", null);
+ yesdrag(event.view, mousemoving);
+ noevent_default(event);
+ gestures.mouse("end", event);
+ }
+ function touchstarted(event, d) {
+ if (!filter2.call(this, event, d))
+ return;
+ var touches = event.changedTouches, c = container.call(this, event, d), n2 = touches.length, i2, gesture;
+ for (i2 = 0; i2 < n2; ++i2) {
+ if (gesture = beforestart(this, c, event, d, touches[i2].identifier, touches[i2])) {
+ nopropagation(event);
+ gesture("start", event, touches[i2]);
+ }
+ }
+ }
+ function touchmoved(event) {
+ var touches = event.changedTouches, n2 = touches.length, i2, gesture;
+ for (i2 = 0; i2 < n2; ++i2) {
+ if (gesture = gestures[touches[i2].identifier]) {
+ noevent_default(event);
+ gesture("drag", event, touches[i2]);
+ }
+ }
+ }
+ function touchended(event) {
+ var touches = event.changedTouches, n2 = touches.length, i2, gesture;
+ if (touchending)
+ clearTimeout(touchending);
+ touchending = setTimeout(function() {
+ touchending = null;
+ }, 500);
+ for (i2 = 0; i2 < n2; ++i2) {
+ if (gesture = gestures[touches[i2].identifier]) {
+ nopropagation(event);
+ gesture("end", event, touches[i2]);
+ }
+ }
+ }
+ function beforestart(that, container2, event, d, identifier, touch) {
+ var dispatch10 = listeners.copy(), p = pointer_default(touch || event, container2), dx, dy, s;
+ if ((s = subject.call(that, new DragEvent("beforestart", {
+ sourceEvent: event,
+ target: drag,
+ identifier,
+ active,
+ x: p[0],
+ y: p[1],
+ dx: 0,
+ dy: 0,
+ dispatch: dispatch10
+ }), d)) == null)
+ return;
+ dx = s.x - p[0] || 0;
+ dy = s.y - p[1] || 0;
+ return function gesture(type2, event2, touch2) {
+ var p02 = p, n2;
+ switch (type2) {
+ case "start":
+ gestures[identifier] = gesture, n2 = active++;
+ break;
+ case "end":
+ delete gestures[identifier], --active;
+ case "drag":
+ p = pointer_default(touch2 || event2, container2), n2 = active;
+ break;
+ }
+ dispatch10.call(
+ type2,
+ that,
+ new DragEvent(type2, {
+ sourceEvent: event2,
+ subject: s,
+ target: drag,
+ identifier,
+ active: n2,
+ x: p[0] + dx,
+ y: p[1] + dy,
+ dx: p[0] - p02[0],
+ dy: p[1] - p02[1],
+ dispatch: dispatch10
+ }),
+ d
+ );
+ };
+ }
+ drag.filter = function(_) {
+ return arguments.length ? (filter2 = typeof _ === "function" ? _ : constant_default2(!!_), drag) : filter2;
+ };
+ drag.container = function(_) {
+ return arguments.length ? (container = typeof _ === "function" ? _ : constant_default2(_), drag) : container;
+ };
+ drag.subject = function(_) {
+ return arguments.length ? (subject = typeof _ === "function" ? _ : constant_default2(_), drag) : subject;
+ };
+ drag.touchable = function(_) {
+ return arguments.length ? (touchable = typeof _ === "function" ? _ : constant_default2(!!_), drag) : touchable;
+ };
+ drag.on = function() {
+ var value = listeners.on.apply(listeners, arguments);
+ return value === listeners ? drag : value;
+ };
+ drag.clickDistance = function(_) {
+ return arguments.length ? (clickDistance2 = (_ = +_) * _, drag) : Math.sqrt(clickDistance2);
+ };
+ return drag;
+ }
+
+ // node_modules/d3-color/src/define.js
+ function define_default(constructor, factory, prototype) {
+ constructor.prototype = factory.prototype = prototype;
+ prototype.constructor = constructor;
+ }
+ function extend(parent, definition) {
+ var prototype = Object.create(parent.prototype);
+ for (var key in definition)
+ prototype[key] = definition[key];
+ return prototype;
+ }
+
+ // node_modules/d3-color/src/color.js
+ function Color() {
+ }
+ var darker = 0.7;
+ var brighter = 1 / darker;
+ var reI = "\\s*([+-]?\\d+)\\s*";
+ var reN = "\\s*([+-]?(?:\\d*\\.)?\\d+(?:[eE][+-]?\\d+)?)\\s*";
+ var reP = "\\s*([+-]?(?:\\d*\\.)?\\d+(?:[eE][+-]?\\d+)?)%\\s*";
+ var reHex = /^#([0-9a-f]{3,8})$/;
+ var reRgbInteger = new RegExp(`^rgb\\(${reI},${reI},${reI}\\)$`);
+ var reRgbPercent = new RegExp(`^rgb\\(${reP},${reP},${reP}\\)$`);
+ var reRgbaInteger = new RegExp(`^rgba\\(${reI},${reI},${reI},${reN}\\)$`);
+ var reRgbaPercent = new RegExp(`^rgba\\(${reP},${reP},${reP},${reN}\\)$`);
+ var reHslPercent = new RegExp(`^hsl\\(${reN},${reP},${reP}\\)$`);
+ var reHslaPercent = new RegExp(`^hsla\\(${reN},${reP},${reP},${reN}\\)$`);
+ var named = {
+ aliceblue: 15792383,
+ antiquewhite: 16444375,
+ aqua: 65535,
+ aquamarine: 8388564,
+ azure: 15794175,
+ beige: 16119260,
+ bisque: 16770244,
+ black: 0,
+ blanchedalmond: 16772045,
+ blue: 255,
+ blueviolet: 9055202,
+ brown: 10824234,
+ burlywood: 14596231,
+ cadetblue: 6266528,
+ chartreuse: 8388352,
+ chocolate: 13789470,
+ coral: 16744272,
+ cornflowerblue: 6591981,
+ cornsilk: 16775388,
+ crimson: 14423100,
+ cyan: 65535,
+ darkblue: 139,
+ darkcyan: 35723,
+ darkgoldenrod: 12092939,
+ darkgray: 11119017,
+ darkgreen: 25600,
+ darkgrey: 11119017,
+ darkkhaki: 12433259,
+ darkmagenta: 9109643,
+ darkolivegreen: 5597999,
+ darkorange: 16747520,
+ darkorchid: 10040012,
+ darkred: 9109504,
+ darksalmon: 15308410,
+ darkseagreen: 9419919,
+ darkslateblue: 4734347,
+ darkslategray: 3100495,
+ darkslategrey: 3100495,
+ darkturquoise: 52945,
+ darkviolet: 9699539,
+ deeppink: 16716947,
+ deepskyblue: 49151,
+ dimgray: 6908265,
+ dimgrey: 6908265,
+ dodgerblue: 2003199,
+ firebrick: 11674146,
+ floralwhite: 16775920,
+ forestgreen: 2263842,
+ fuchsia: 16711935,
+ gainsboro: 14474460,
+ ghostwhite: 16316671,
+ gold: 16766720,
+ goldenrod: 14329120,
+ gray: 8421504,
+ green: 32768,
+ greenyellow: 11403055,
+ grey: 8421504,
+ honeydew: 15794160,
+ hotpink: 16738740,
+ indianred: 13458524,
+ indigo: 4915330,
+ ivory: 16777200,
+ khaki: 15787660,
+ lavender: 15132410,
+ lavenderblush: 16773365,
+ lawngreen: 8190976,
+ lemonchiffon: 16775885,
+ lightblue: 11393254,
+ lightcoral: 15761536,
+ lightcyan: 14745599,
+ lightgoldenrodyellow: 16448210,
+ lightgray: 13882323,
+ lightgreen: 9498256,
+ lightgrey: 13882323,
+ lightpink: 16758465,
+ lightsalmon: 16752762,
+ lightseagreen: 2142890,
+ lightskyblue: 8900346,
+ lightslategray: 7833753,
+ lightslategrey: 7833753,
+ lightsteelblue: 11584734,
+ lightyellow: 16777184,
+ lime: 65280,
+ limegreen: 3329330,
+ linen: 16445670,
+ magenta: 16711935,
+ maroon: 8388608,
+ mediumaquamarine: 6737322,
+ mediumblue: 205,
+ mediumorchid: 12211667,
+ mediumpurple: 9662683,
+ mediumseagreen: 3978097,
+ mediumslateblue: 8087790,
+ mediumspringgreen: 64154,
+ mediumturquoise: 4772300,
+ mediumvioletred: 13047173,
+ midnightblue: 1644912,
+ mintcream: 16121850,
+ mistyrose: 16770273,
+ moccasin: 16770229,
+ navajowhite: 16768685,
+ navy: 128,
+ oldlace: 16643558,
+ olive: 8421376,
+ olivedrab: 7048739,
+ orange: 16753920,
+ orangered: 16729344,
+ orchid: 14315734,
+ palegoldenrod: 15657130,
+ palegreen: 10025880,
+ paleturquoise: 11529966,
+ palevioletred: 14381203,
+ papayawhip: 16773077,
+ peachpuff: 16767673,
+ peru: 13468991,
+ pink: 16761035,
+ plum: 14524637,
+ powderblue: 11591910,
+ purple: 8388736,
+ rebeccapurple: 6697881,
+ red: 16711680,
+ rosybrown: 12357519,
+ royalblue: 4286945,
+ saddlebrown: 9127187,
+ salmon: 16416882,
+ sandybrown: 16032864,
+ seagreen: 3050327,
+ seashell: 16774638,
+ sienna: 10506797,
+ silver: 12632256,
+ skyblue: 8900331,
+ slateblue: 6970061,
+ slategray: 7372944,
+ slategrey: 7372944,
+ snow: 16775930,
+ springgreen: 65407,
+ steelblue: 4620980,
+ tan: 13808780,
+ teal: 32896,
+ thistle: 14204888,
+ tomato: 16737095,
+ turquoise: 4251856,
+ violet: 15631086,
+ wheat: 16113331,
+ white: 16777215,
+ whitesmoke: 16119285,
+ yellow: 16776960,
+ yellowgreen: 10145074
+ };
+ define_default(Color, color, {
+ copy(channels) {
+ return Object.assign(new this.constructor(), this, channels);
+ },
+ displayable() {
+ return this.rgb().displayable();
+ },
+ hex: color_formatHex,
+ // Deprecated! Use color.formatHex.
+ formatHex: color_formatHex,
+ formatHex8: color_formatHex8,
+ formatHsl: color_formatHsl,
+ formatRgb: color_formatRgb,
+ toString: color_formatRgb
+ });
+ function color_formatHex() {
+ return this.rgb().formatHex();
+ }
+ function color_formatHex8() {
+ return this.rgb().formatHex8();
+ }
+ function color_formatHsl() {
+ return hslConvert(this).formatHsl();
+ }
+ function color_formatRgb() {
+ return this.rgb().formatRgb();
+ }
+ function color(format2) {
+ var m, l;
+ format2 = (format2 + "").trim().toLowerCase();
+ return (m = reHex.exec(format2)) ? (l = m[1].length, m = parseInt(m[1], 16), l === 6 ? rgbn(m) : l === 3 ? new Rgb(m >> 8 & 15 | m >> 4 & 240, m >> 4 & 15 | m & 240, (m & 15) << 4 | m & 15, 1) : l === 8 ? rgba(m >> 24 & 255, m >> 16 & 255, m >> 8 & 255, (m & 255) / 255) : l === 4 ? rgba(m >> 12 & 15 | m >> 8 & 240, m >> 8 & 15 | m >> 4 & 240, m >> 4 & 15 | m & 240, ((m & 15) << 4 | m & 15) / 255) : null) : (m = reRgbInteger.exec(format2)) ? new Rgb(m[1], m[2], m[3], 1) : (m = reRgbPercent.exec(format2)) ? new Rgb(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) : (m = reRgbaInteger.exec(format2)) ? rgba(m[1], m[2], m[3], m[4]) : (m = reRgbaPercent.exec(format2)) ? rgba(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) : (m = reHslPercent.exec(format2)) ? hsla(m[1], m[2] / 100, m[3] / 100, 1) : (m = reHslaPercent.exec(format2)) ? hsla(m[1], m[2] / 100, m[3] / 100, m[4]) : named.hasOwnProperty(format2) ? rgbn(named[format2]) : format2 === "transparent" ? new Rgb(NaN, NaN, NaN, 0) : null;
+ }
+ function rgbn(n2) {
+ return new Rgb(n2 >> 16 & 255, n2 >> 8 & 255, n2 & 255, 1);
+ }
+ function rgba(r, g, b, a) {
+ if (a <= 0)
+ r = g = b = NaN;
+ return new Rgb(r, g, b, a);
+ }
+ function rgbConvert(o) {
+ if (!(o instanceof Color))
+ o = color(o);
+ if (!o)
+ return new Rgb();
+ o = o.rgb();
+ return new Rgb(o.r, o.g, o.b, o.opacity);
+ }
+ function rgb(r, g, b, opacity) {
+ return arguments.length === 1 ? rgbConvert(r) : new Rgb(r, g, b, opacity == null ? 1 : opacity);
+ }
+ function Rgb(r, g, b, opacity) {
+ this.r = +r;
+ this.g = +g;
+ this.b = +b;
+ this.opacity = +opacity;
+ }
+ define_default(Rgb, rgb, extend(Color, {
+ brighter(k) {
+ k = k == null ? brighter : Math.pow(brighter, k);
+ return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
+ },
+ darker(k) {
+ k = k == null ? darker : Math.pow(darker, k);
+ return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
+ },
+ rgb() {
+ return this;
+ },
+ clamp() {
+ return new Rgb(clampi(this.r), clampi(this.g), clampi(this.b), clampa(this.opacity));
+ },
+ displayable() {
+ return -0.5 <= this.r && this.r < 255.5 && (-0.5 <= this.g && this.g < 255.5) && (-0.5 <= this.b && this.b < 255.5) && (0 <= this.opacity && this.opacity <= 1);
+ },
+ hex: rgb_formatHex,
+ // Deprecated! Use color.formatHex.
+ formatHex: rgb_formatHex,
+ formatHex8: rgb_formatHex8,
+ formatRgb: rgb_formatRgb,
+ toString: rgb_formatRgb
+ }));
+ function rgb_formatHex() {
+ return `#${hex(this.r)}${hex(this.g)}${hex(this.b)}`;
+ }
+ function rgb_formatHex8() {
+ return `#${hex(this.r)}${hex(this.g)}${hex(this.b)}${hex((isNaN(this.opacity) ? 1 : this.opacity) * 255)}`;
+ }
+ function rgb_formatRgb() {
+ const a = clampa(this.opacity);
+ return `${a === 1 ? "rgb(" : "rgba("}${clampi(this.r)}, ${clampi(this.g)}, ${clampi(this.b)}${a === 1 ? ")" : `, ${a})`}`;
+ }
+ function clampa(opacity) {
+ return isNaN(opacity) ? 1 : Math.max(0, Math.min(1, opacity));
+ }
+ function clampi(value) {
+ return Math.max(0, Math.min(255, Math.round(value) || 0));
+ }
+ function hex(value) {
+ value = clampi(value);
+ return (value < 16 ? "0" : "") + value.toString(16);
+ }
+ function hsla(h, s, l, a) {
+ if (a <= 0)
+ h = s = l = NaN;
+ else if (l <= 0 || l >= 1)
+ h = s = NaN;
+ else if (s <= 0)
+ h = NaN;
+ return new Hsl(h, s, l, a);
+ }
+ function hslConvert(o) {
+ if (o instanceof Hsl)
+ return new Hsl(o.h, o.s, o.l, o.opacity);
+ if (!(o instanceof Color))
+ o = color(o);
+ if (!o)
+ return new Hsl();
+ if (o instanceof Hsl)
+ return o;
+ o = o.rgb();
+ var r = o.r / 255, g = o.g / 255, b = o.b / 255, min3 = Math.min(r, g, b), max3 = Math.max(r, g, b), h = NaN, s = max3 - min3, l = (max3 + min3) / 2;
+ if (s) {
+ if (r === max3)
+ h = (g - b) / s + (g < b) * 6;
+ else if (g === max3)
+ h = (b - r) / s + 2;
+ else
+ h = (r - g) / s + 4;
+ s /= l < 0.5 ? max3 + min3 : 2 - max3 - min3;
+ h *= 60;
+ } else {
+ s = l > 0 && l < 1 ? 0 : h;
+ }
+ return new Hsl(h, s, l, o.opacity);
+ }
+ function hsl(h, s, l, opacity) {
+ return arguments.length === 1 ? hslConvert(h) : new Hsl(h, s, l, opacity == null ? 1 : opacity);
+ }
+ function Hsl(h, s, l, opacity) {
+ this.h = +h;
+ this.s = +s;
+ this.l = +l;
+ this.opacity = +opacity;
+ }
+ define_default(Hsl, hsl, extend(Color, {
+ brighter(k) {
+ k = k == null ? brighter : Math.pow(brighter, k);
+ return new Hsl(this.h, this.s, this.l * k, this.opacity);
+ },
+ darker(k) {
+ k = k == null ? darker : Math.pow(darker, k);
+ return new Hsl(this.h, this.s, this.l * k, this.opacity);
+ },
+ rgb() {
+ var h = this.h % 360 + (this.h < 0) * 360, s = isNaN(h) || isNaN(this.s) ? 0 : this.s, l = this.l, m2 = l + (l < 0.5 ? l : 1 - l) * s, m1 = 2 * l - m2;
+ return new Rgb(
+ hsl2rgb(h >= 240 ? h - 240 : h + 120, m1, m2),
+ hsl2rgb(h, m1, m2),
+ hsl2rgb(h < 120 ? h + 240 : h - 120, m1, m2),
+ this.opacity
+ );
+ },
+ clamp() {
+ return new Hsl(clamph(this.h), clampt(this.s), clampt(this.l), clampa(this.opacity));
+ },
+ displayable() {
+ return (0 <= this.s && this.s <= 1 || isNaN(this.s)) && (0 <= this.l && this.l <= 1) && (0 <= this.opacity && this.opacity <= 1);
+ },
+ formatHsl() {
+ const a = clampa(this.opacity);
+ return `${a === 1 ? "hsl(" : "hsla("}${clamph(this.h)}, ${clampt(this.s) * 100}%, ${clampt(this.l) * 100}%${a === 1 ? ")" : `, ${a})`}`;
+ }
+ }));
+ function clamph(value) {
+ value = (value || 0) % 360;
+ return value < 0 ? value + 360 : value;
+ }
+ function clampt(value) {
+ return Math.max(0, Math.min(1, value || 0));
+ }
+ function hsl2rgb(h, m1, m2) {
+ return (h < 60 ? m1 + (m2 - m1) * h / 60 : h < 180 ? m2 : h < 240 ? m1 + (m2 - m1) * (240 - h) / 60 : m1) * 255;
+ }
+
+ // node_modules/d3-interpolate/src/basis.js
+ function basis(t1, v0, v1, v2, v3) {
+ var t2 = t1 * t1, t3 = t2 * t1;
+ return ((1 - 3 * t1 + 3 * t2 - t3) * v0 + (4 - 6 * t2 + 3 * t3) * v1 + (1 + 3 * t1 + 3 * t2 - 3 * t3) * v2 + t3 * v3) / 6;
+ }
+ function basis_default(values) {
+ var n2 = values.length - 1;
+ return function(t) {
+ var i2 = t <= 0 ? t = 0 : t >= 1 ? (t = 1, n2 - 1) : Math.floor(t * n2), v1 = values[i2], v2 = values[i2 + 1], v0 = i2 > 0 ? values[i2 - 1] : 2 * v1 - v2, v3 = i2 < n2 - 1 ? values[i2 + 2] : 2 * v2 - v1;
+ return basis((t - i2 / n2) * n2, v0, v1, v2, v3);
+ };
+ }
+
+ // node_modules/d3-interpolate/src/basisClosed.js
+ function basisClosed_default(values) {
+ var n2 = values.length;
+ return function(t) {
+ var i2 = Math.floor(((t %= 1) < 0 ? ++t : t) * n2), v0 = values[(i2 + n2 - 1) % n2], v1 = values[i2 % n2], v2 = values[(i2 + 1) % n2], v3 = values[(i2 + 2) % n2];
+ return basis((t - i2 / n2) * n2, v0, v1, v2, v3);
+ };
+ }
+
+ // node_modules/d3-interpolate/src/constant.js
+ var constant_default3 = (x) => () => x;
+
+ // node_modules/d3-interpolate/src/color.js
+ function linear(a, d) {
+ return function(t) {
+ return a + t * d;
+ };
+ }
+ function exponential(a, b, y) {
+ return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
+ return Math.pow(a + t * b, y);
+ };
+ }
+ function gamma(y) {
+ return (y = +y) === 1 ? nogamma : function(a, b) {
+ return b - a ? exponential(a, b, y) : constant_default3(isNaN(a) ? b : a);
+ };
+ }
+ function nogamma(a, b) {
+ var d = b - a;
+ return d ? linear(a, d) : constant_default3(isNaN(a) ? b : a);
+ }
+
+ // node_modules/d3-interpolate/src/rgb.js
+ var rgb_default = function rgbGamma(y) {
+ var color2 = gamma(y);
+ function rgb2(start2, end) {
+ var r = color2((start2 = rgb(start2)).r, (end = rgb(end)).r), g = color2(start2.g, end.g), b = color2(start2.b, end.b), opacity = nogamma(start2.opacity, end.opacity);
+ return function(t) {
+ start2.r = r(t);
+ start2.g = g(t);
+ start2.b = b(t);
+ start2.opacity = opacity(t);
+ return start2 + "";
+ };
+ }
+ rgb2.gamma = rgbGamma;
+ return rgb2;
+ }(1);
+ function rgbSpline(spline) {
+ return function(colors) {
+ var n2 = colors.length, r = new Array(n2), g = new Array(n2), b = new Array(n2), i2, color2;
+ for (i2 = 0; i2 < n2; ++i2) {
+ color2 = rgb(colors[i2]);
+ r[i2] = color2.r || 0;
+ g[i2] = color2.g || 0;
+ b[i2] = color2.b || 0;
+ }
+ r = spline(r);
+ g = spline(g);
+ b = spline(b);
+ color2.opacity = 1;
+ return function(t) {
+ color2.r = r(t);
+ color2.g = g(t);
+ color2.b = b(t);
+ return color2 + "";
+ };
+ };
+ }
+ var rgbBasis = rgbSpline(basis_default);
+ var rgbBasisClosed = rgbSpline(basisClosed_default);
+
+ // node_modules/d3-interpolate/src/numberArray.js
+ function numberArray_default(a, b) {
+ if (!b)
+ b = [];
+ var n2 = a ? Math.min(b.length, a.length) : 0, c = b.slice(), i2;
+ return function(t) {
+ for (i2 = 0; i2 < n2; ++i2)
+ c[i2] = a[i2] * (1 - t) + b[i2] * t;
+ return c;
+ };
+ }
+ function isNumberArray(x) {
+ return ArrayBuffer.isView(x) && !(x instanceof DataView);
+ }
+
+ // node_modules/d3-interpolate/src/array.js
+ function genericArray(a, b) {
+ var nb = b ? b.length : 0, na = a ? Math.min(nb, a.length) : 0, x = new Array(na), c = new Array(nb), i2;
+ for (i2 = 0; i2 < na; ++i2)
+ x[i2] = value_default(a[i2], b[i2]);
+ for (; i2 < nb; ++i2)
+ c[i2] = b[i2];
+ return function(t) {
+ for (i2 = 0; i2 < na; ++i2)
+ c[i2] = x[i2](t);
+ return c;
+ };
+ }
+
+ // node_modules/d3-interpolate/src/date.js
+ function date_default(a, b) {
+ var d = new Date();
+ return a = +a, b = +b, function(t) {
+ return d.setTime(a * (1 - t) + b * t), d;
+ };
+ }
+
+ // node_modules/d3-interpolate/src/number.js
+ function number_default(a, b) {
+ return a = +a, b = +b, function(t) {
+ return a * (1 - t) + b * t;
+ };
+ }
+
+ // node_modules/d3-interpolate/src/object.js
+ function object_default(a, b) {
+ var i2 = {}, c = {}, k;
+ if (a === null || typeof a !== "object")
+ a = {};
+ if (b === null || typeof b !== "object")
+ b = {};
+ for (k in b) {
+ if (k in a) {
+ i2[k] = value_default(a[k], b[k]);
+ } else {
+ c[k] = b[k];
+ }
+ }
+ return function(t) {
+ for (k in i2)
+ c[k] = i2[k](t);
+ return c;
+ };
+ }
+
+ // node_modules/d3-interpolate/src/string.js
+ var reA = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g;
+ var reB = new RegExp(reA.source, "g");
+ function zero2(b) {
+ return function() {
+ return b;
+ };
+ }
+ function one(b) {
+ return function(t) {
+ return b(t) + "";
+ };
+ }
+ function string_default(a, b) {
+ var bi = reA.lastIndex = reB.lastIndex = 0, am, bm, bs, i2 = -1, s = [], q = [];
+ a = a + "", b = b + "";
+ while ((am = reA.exec(a)) && (bm = reB.exec(b))) {
+ if ((bs = bm.index) > bi) {
+ bs = b.slice(bi, bs);
+ if (s[i2])
+ s[i2] += bs;
+ else
+ s[++i2] = bs;
+ }
+ if ((am = am[0]) === (bm = bm[0])) {
+ if (s[i2])
+ s[i2] += bm;
+ else
+ s[++i2] = bm;
+ } else {
+ s[++i2] = null;
+ q.push({ i: i2, x: number_default(am, bm) });
+ }
+ bi = reB.lastIndex;
+ }
+ if (bi < b.length) {
+ bs = b.slice(bi);
+ if (s[i2])
+ s[i2] += bs;
+ else
+ s[++i2] = bs;
+ }
+ return s.length < 2 ? q[0] ? one(q[0].x) : zero2(b) : (b = q.length, function(t) {
+ for (var i3 = 0, o; i3 < b; ++i3)
+ s[(o = q[i3]).i] = o.x(t);
+ return s.join("");
+ });
+ }
+
+ // node_modules/d3-interpolate/src/value.js
+ function value_default(a, b) {
+ var t = typeof b, c;
+ return b == null || t === "boolean" ? constant_default3(b) : (t === "number" ? number_default : t === "string" ? (c = color(b)) ? (b = c, rgb_default) : string_default : b instanceof color ? rgb_default : b instanceof Date ? date_default : isNumberArray(b) ? numberArray_default : Array.isArray(b) ? genericArray : typeof b.valueOf !== "function" && typeof b.toString !== "function" || isNaN(b) ? object_default : number_default)(a, b);
+ }
+
+ // node_modules/d3-interpolate/src/round.js
+ function round_default(a, b) {
+ return a = +a, b = +b, function(t) {
+ return Math.round(a * (1 - t) + b * t);
+ };
+ }
+
+ // node_modules/d3-interpolate/src/transform/decompose.js
+ var degrees2 = 180 / Math.PI;
+ var identity = {
+ translateX: 0,
+ translateY: 0,
+ rotate: 0,
+ skewX: 0,
+ scaleX: 1,
+ scaleY: 1
+ };
+ function decompose_default(a, b, c, d, e, f2) {
+ var scaleX, scaleY, skewX;
+ if (scaleX = Math.sqrt(a * a + b * b))
+ a /= scaleX, b /= scaleX;
+ if (skewX = a * c + b * d)
+ c -= a * skewX, d -= b * skewX;
+ if (scaleY = Math.sqrt(c * c + d * d))
+ c /= scaleY, d /= scaleY, skewX /= scaleY;
+ if (a * d < b * c)
+ a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
+ return {
+ translateX: e,
+ translateY: f2,
+ rotate: Math.atan2(b, a) * degrees2,
+ skewX: Math.atan(skewX) * degrees2,
+ scaleX,
+ scaleY
+ };
+ }
+
+ // node_modules/d3-interpolate/src/transform/parse.js
+ var svgNode;
+ function parseCss(value) {
+ const m = new (typeof DOMMatrix === "function" ? DOMMatrix : WebKitCSSMatrix)(value + "");
+ return m.isIdentity ? identity : decompose_default(m.a, m.b, m.c, m.d, m.e, m.f);
+ }
+ function parseSvg(value) {
+ if (value == null)
+ return identity;
+ if (!svgNode)
+ svgNode = document.createElementNS("http://www.w3.org/2000/svg", "g");
+ svgNode.setAttribute("transform", value);
+ if (!(value = svgNode.transform.baseVal.consolidate()))
+ return identity;
+ value = value.matrix;
+ return decompose_default(value.a, value.b, value.c, value.d, value.e, value.f);
+ }
+
+ // node_modules/d3-interpolate/src/transform/index.js
+ function interpolateTransform(parse, pxComma, pxParen, degParen) {
+ function pop(s) {
+ return s.length ? s.pop() + " " : "";
+ }
+ function translate(xa, ya, xb, yb, s, q) {
+ if (xa !== xb || ya !== yb) {
+ var i2 = s.push("translate(", null, pxComma, null, pxParen);
+ q.push({ i: i2 - 4, x: number_default(xa, xb) }, { i: i2 - 2, x: number_default(ya, yb) });
+ } else if (xb || yb) {
+ s.push("translate(" + xb + pxComma + yb + pxParen);
+ }
+ }
+ function rotate(a, b, s, q) {
+ if (a !== b) {
+ if (a - b > 180)
+ b += 360;
+ else if (b - a > 180)
+ a += 360;
+ q.push({ i: s.push(pop(s) + "rotate(", null, degParen) - 2, x: number_default(a, b) });
+ } else if (b) {
+ s.push(pop(s) + "rotate(" + b + degParen);
+ }
+ }
+ function skewX(a, b, s, q) {
+ if (a !== b) {
+ q.push({ i: s.push(pop(s) + "skewX(", null, degParen) - 2, x: number_default(a, b) });
+ } else if (b) {
+ s.push(pop(s) + "skewX(" + b + degParen);
+ }
+ }
+ function scale(xa, ya, xb, yb, s, q) {
+ if (xa !== xb || ya !== yb) {
+ var i2 = s.push(pop(s) + "scale(", null, ",", null, ")");
+ q.push({ i: i2 - 4, x: number_default(xa, xb) }, { i: i2 - 2, x: number_default(ya, yb) });
+ } else if (xb !== 1 || yb !== 1) {
+ s.push(pop(s) + "scale(" + xb + "," + yb + ")");
+ }
+ }
+ return function(a, b) {
+ var s = [], q = [];
+ a = parse(a), b = parse(b);
+ translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
+ rotate(a.rotate, b.rotate, s, q);
+ skewX(a.skewX, b.skewX, s, q);
+ scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
+ a = b = null;
+ return function(t) {
+ var i2 = -1, n2 = q.length, o;
+ while (++i2 < n2)
+ s[(o = q[i2]).i] = o.x(t);
+ return s.join("");
+ };
+ };
+ }
+ var interpolateTransformCss = interpolateTransform(parseCss, "px, ", "px)", "deg)");
+ var interpolateTransformSvg = interpolateTransform(parseSvg, ", ", ")", ")");
+
+ // node_modules/d3-interpolate/src/zoom.js
+ var epsilon22 = 1e-12;
+ function cosh(x) {
+ return ((x = Math.exp(x)) + 1 / x) / 2;
+ }
+ function sinh(x) {
+ return ((x = Math.exp(x)) - 1 / x) / 2;
+ }
+ function tanh(x) {
+ return ((x = Math.exp(2 * x)) - 1) / (x + 1);
+ }
+ var zoom_default = function zoomRho(rho, rho2, rho4) {
+ function zoom(p02, p1) {
+ var ux0 = p02[0], uy0 = p02[1], w0 = p02[2], ux1 = p1[0], uy1 = p1[1], w1 = p1[2], dx = ux1 - ux0, dy = uy1 - uy0, d2 = dx * dx + dy * dy, i2, S;
+ if (d2 < epsilon22) {
+ S = Math.log(w1 / w0) / rho;
+ i2 = function(t) {
+ return [
+ ux0 + t * dx,
+ uy0 + t * dy,
+ w0 * Math.exp(rho * t * S)
+ ];
+ };
+ } else {
+ var d1 = Math.sqrt(d2), b0 = (w1 * w1 - w0 * w0 + rho4 * d2) / (2 * w0 * rho2 * d1), b1 = (w1 * w1 - w0 * w0 - rho4 * d2) / (2 * w1 * rho2 * d1), r0 = Math.log(Math.sqrt(b0 * b0 + 1) - b0), r1 = Math.log(Math.sqrt(b1 * b1 + 1) - b1);
+ S = (r1 - r0) / rho;
+ i2 = function(t) {
+ var s = t * S, coshr0 = cosh(r0), u = w0 / (rho2 * d1) * (coshr0 * tanh(rho * s + r0) - sinh(r0));
+ return [
+ ux0 + u * dx,
+ uy0 + u * dy,
+ w0 * coshr0 / cosh(rho * s + r0)
+ ];
+ };
+ }
+ i2.duration = S * 1e3 * rho / Math.SQRT2;
+ return i2;
+ }
+ zoom.rho = function(_) {
+ var _1 = Math.max(1e-3, +_), _2 = _1 * _1, _4 = _2 * _2;
+ return zoomRho(_1, _2, _4);
+ };
+ return zoom;
+ }(Math.SQRT2, 2, 4);
+
+ // node_modules/d3-interpolate/src/quantize.js
+ function quantize_default(interpolator, n2) {
+ var samples = new Array(n2);
+ for (var i2 = 0; i2 < n2; ++i2)
+ samples[i2] = interpolator(i2 / (n2 - 1));
+ return samples;
+ }
+
+ // node_modules/d3-timer/src/timer.js
+ var frame = 0;
+ var timeout = 0;
+ var interval = 0;
+ var pokeDelay = 1e3;
+ var taskHead;
+ var taskTail;
+ var clockLast = 0;
+ var clockNow = 0;
+ var clockSkew = 0;
+ var clock = typeof performance === "object" && performance.now ? performance : Date;
+ var setFrame = typeof window === "object" && window.requestAnimationFrame ? window.requestAnimationFrame.bind(window) : function(f2) {
+ setTimeout(f2, 17);
+ };
+ function now() {
+ return clockNow || (setFrame(clearNow), clockNow = clock.now() + clockSkew);
+ }
+ function clearNow() {
+ clockNow = 0;
+ }
+ function Timer() {
+ this._call = this._time = this._next = null;
+ }
+ Timer.prototype = timer.prototype = {
+ constructor: Timer,
+ restart: function(callback, delay, time) {
+ if (typeof callback !== "function")
+ throw new TypeError("callback is not a function");
+ time = (time == null ? now() : +time) + (delay == null ? 0 : +delay);
+ if (!this._next && taskTail !== this) {
+ if (taskTail)
+ taskTail._next = this;
+ else
+ taskHead = this;
+ taskTail = this;
+ }
+ this._call = callback;
+ this._time = time;
+ sleep();
+ },
+ stop: function() {
+ if (this._call) {
+ this._call = null;
+ this._time = Infinity;
+ sleep();
+ }
+ }
+ };
+ function timer(callback, delay, time) {
+ var t = new Timer();
+ t.restart(callback, delay, time);
+ return t;
+ }
+ function timerFlush() {
+ now();
+ ++frame;
+ var t = taskHead, e;
+ while (t) {
+ if ((e = clockNow - t._time) >= 0)
+ t._call.call(void 0, e);
+ t = t._next;
+ }
+ --frame;
+ }
+ function wake() {
+ clockNow = (clockLast = clock.now()) + clockSkew;
+ frame = timeout = 0;
+ try {
+ timerFlush();
+ } finally {
+ frame = 0;
+ nap();
+ clockNow = 0;
+ }
+ }
+ function poke() {
+ var now3 = clock.now(), delay = now3 - clockLast;
+ if (delay > pokeDelay)
+ clockSkew -= delay, clockLast = now3;
+ }
+ function nap() {
+ var t0, t1 = taskHead, t2, time = Infinity;
+ while (t1) {
+ if (t1._call) {
+ if (time > t1._time)
+ time = t1._time;
+ t0 = t1, t1 = t1._next;
+ } else {
+ t2 = t1._next, t1._next = null;
+ t1 = t0 ? t0._next = t2 : taskHead = t2;
+ }
+ }
+ taskTail = t0;
+ sleep(time);
+ }
+ function sleep(time) {
+ if (frame)
+ return;
+ if (timeout)
+ timeout = clearTimeout(timeout);
+ var delay = time - clockNow;
+ if (delay > 24) {
+ if (time < Infinity)
+ timeout = setTimeout(wake, time - clock.now() - clockSkew);
+ if (interval)
+ interval = clearInterval(interval);
+ } else {
+ if (!interval)
+ clockLast = clock.now(), interval = setInterval(poke, pokeDelay);
+ frame = 1, setFrame(wake);
+ }
+ }
+
+ // node_modules/d3-timer/src/timeout.js
+ function timeout_default(callback, delay, time) {
+ var t = new Timer();
+ delay = delay == null ? 0 : +delay;
+ t.restart((elapsed) => {
+ t.stop();
+ callback(elapsed + delay);
+ }, delay, time);
+ return t;
+ }
+
+ // node_modules/d3-transition/src/transition/schedule.js
+ var emptyOn = dispatch_default("start", "end", "cancel", "interrupt");
+ var emptyTween = [];
+ var CREATED = 0;
+ var SCHEDULED = 1;
+ var STARTING = 2;
+ var STARTED = 3;
+ var RUNNING = 4;
+ var ENDING = 5;
+ var ENDED = 6;
+ function schedule_default(node, name, id2, index, group, timing) {
+ var schedules = node.__transition;
+ if (!schedules)
+ node.__transition = {};
+ else if (id2 in schedules)
+ return;
+ create(node, id2, {
+ name,
+ index,
+ // For context during callback.
+ group,
+ // For context during callback.
+ on: emptyOn,
+ tween: emptyTween,
+ time: timing.time,
+ delay: timing.delay,
+ duration: timing.duration,
+ ease: timing.ease,
+ timer: null,
+ state: CREATED
+ });
+ }
+ function init(node, id2) {
+ var schedule = get2(node, id2);
+ if (schedule.state > CREATED)
+ throw new Error("too late; already scheduled");
+ return schedule;
+ }
+ function set2(node, id2) {
+ var schedule = get2(node, id2);
+ if (schedule.state > STARTED)
+ throw new Error("too late; already running");
+ return schedule;
+ }
+ function get2(node, id2) {
+ var schedule = node.__transition;
+ if (!schedule || !(schedule = schedule[id2]))
+ throw new Error("transition not found");
+ return schedule;
+ }
+ function create(node, id2, self2) {
+ var schedules = node.__transition, tween;
+ schedules[id2] = self2;
+ self2.timer = timer(schedule, 0, self2.time);
+ function schedule(elapsed) {
+ self2.state = SCHEDULED;
+ self2.timer.restart(start2, self2.delay, self2.time);
+ if (self2.delay <= elapsed)
+ start2(elapsed - self2.delay);
+ }
+ function start2(elapsed) {
+ var i2, j2, n2, o;
+ if (self2.state !== SCHEDULED)
+ return stop();
+ for (i2 in schedules) {
+ o = schedules[i2];
+ if (o.name !== self2.name)
+ continue;
+ if (o.state === STARTED)
+ return timeout_default(start2);
+ if (o.state === RUNNING) {
+ o.state = ENDED;
+ o.timer.stop();
+ o.on.call("interrupt", node, node.__data__, o.index, o.group);
+ delete schedules[i2];
+ } else if (+i2 < id2) {
+ o.state = ENDED;
+ o.timer.stop();
+ o.on.call("cancel", node, node.__data__, o.index, o.group);
+ delete schedules[i2];
+ }
+ }
+ timeout_default(function() {
+ if (self2.state === STARTED) {
+ self2.state = RUNNING;
+ self2.timer.restart(tick, self2.delay, self2.time);
+ tick(elapsed);
+ }
+ });
+ self2.state = STARTING;
+ self2.on.call("start", node, node.__data__, self2.index, self2.group);
+ if (self2.state !== STARTING)
+ return;
+ self2.state = STARTED;
+ tween = new Array(n2 = self2.tween.length);
+ for (i2 = 0, j2 = -1; i2 < n2; ++i2) {
+ if (o = self2.tween[i2].value.call(node, node.__data__, self2.index, self2.group)) {
+ tween[++j2] = o;
+ }
+ }
+ tween.length = j2 + 1;
+ }
+ function tick(elapsed) {
+ var t = elapsed < self2.duration ? self2.ease.call(null, elapsed / self2.duration) : (self2.timer.restart(stop), self2.state = ENDING, 1), i2 = -1, n2 = tween.length;
+ while (++i2 < n2) {
+ tween[i2].call(node, t);
+ }
+ if (self2.state === ENDING) {
+ self2.on.call("end", node, node.__data__, self2.index, self2.group);
+ stop();
+ }
+ }
+ function stop() {
+ self2.state = ENDED;
+ self2.timer.stop();
+ delete schedules[id2];
+ for (var i2 in schedules)
+ return;
+ delete node.__transition;
+ }
+ }
+
+ // node_modules/d3-transition/src/interrupt.js
+ function interrupt_default(node, name) {
+ var schedules = node.__transition, schedule, active, empty2 = true, i2;
+ if (!schedules)
+ return;
+ name = name == null ? null : name + "";
+ for (i2 in schedules) {
+ if ((schedule = schedules[i2]).name !== name) {
+ empty2 = false;
+ continue;
+ }
+ active = schedule.state > STARTING && schedule.state < ENDING;
+ schedule.state = ENDED;
+ schedule.timer.stop();
+ schedule.on.call(active ? "interrupt" : "cancel", node, node.__data__, schedule.index, schedule.group);
+ delete schedules[i2];
+ }
+ if (empty2)
+ delete node.__transition;
+ }
+
+ // node_modules/d3-transition/src/selection/interrupt.js
+ function interrupt_default2(name) {
+ return this.each(function() {
+ interrupt_default(this, name);
+ });
+ }
+
+ // node_modules/d3-transition/src/transition/tween.js
+ function tweenRemove(id2, name) {
+ var tween0, tween1;
+ return function() {
+ var schedule = set2(this, id2), tween = schedule.tween;
+ if (tween !== tween0) {
+ tween1 = tween0 = tween;
+ for (var i2 = 0, n2 = tween1.length; i2 < n2; ++i2) {
+ if (tween1[i2].name === name) {
+ tween1 = tween1.slice();
+ tween1.splice(i2, 1);
+ break;
+ }
+ }
+ }
+ schedule.tween = tween1;
+ };
+ }
+ function tweenFunction(id2, name, value) {
+ var tween0, tween1;
+ if (typeof value !== "function")
+ throw new Error();
+ return function() {
+ var schedule = set2(this, id2), tween = schedule.tween;
+ if (tween !== tween0) {
+ tween1 = (tween0 = tween).slice();
+ for (var t = { name, value }, i2 = 0, n2 = tween1.length; i2 < n2; ++i2) {
+ if (tween1[i2].name === name) {
+ tween1[i2] = t;
+ break;
+ }
+ }
+ if (i2 === n2)
+ tween1.push(t);
+ }
+ schedule.tween = tween1;
+ };
+ }
+ function tween_default(name, value) {
+ var id2 = this._id;
+ name += "";
+ if (arguments.length < 2) {
+ var tween = get2(this.node(), id2).tween;
+ for (var i2 = 0, n2 = tween.length, t; i2 < n2; ++i2) {
+ if ((t = tween[i2]).name === name) {
+ return t.value;
+ }
+ }
+ return null;
+ }
+ return this.each((value == null ? tweenRemove : tweenFunction)(id2, name, value));
+ }
+ function tweenValue(transition2, name, value) {
+ var id2 = transition2._id;
+ transition2.each(function() {
+ var schedule = set2(this, id2);
+ (schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
+ });
+ return function(node) {
+ return get2(node, id2).value[name];
+ };
+ }
+
+ // node_modules/d3-transition/src/transition/interpolate.js
+ function interpolate_default(a, b) {
+ var c;
+ return (typeof b === "number" ? number_default : b instanceof color ? rgb_default : (c = color(b)) ? (b = c, rgb_default) : string_default)(a, b);
+ }
+
+ // node_modules/d3-transition/src/transition/attr.js
+ function attrRemove2(name) {
+ return function() {
+ this.removeAttribute(name);
+ };
+ }
+ function attrRemoveNS2(fullname) {
+ return function() {
+ this.removeAttributeNS(fullname.space, fullname.local);
+ };
+ }
+ function attrConstant2(name, interpolate, value1) {
+ var string00, string1 = value1 + "", interpolate0;
+ return function() {
+ var string0 = this.getAttribute(name);
+ return string0 === string1 ? null : string0 === string00 ? interpolate0 : interpolate0 = interpolate(string00 = string0, value1);
+ };
+ }
+ function attrConstantNS2(fullname, interpolate, value1) {
+ var string00, string1 = value1 + "", interpolate0;
+ return function() {
+ var string0 = this.getAttributeNS(fullname.space, fullname.local);
+ return string0 === string1 ? null : string0 === string00 ? interpolate0 : interpolate0 = interpolate(string00 = string0, value1);
+ };
+ }
+ function attrFunction2(name, interpolate, value) {
+ var string00, string10, interpolate0;
+ return function() {
+ var string0, value1 = value(this), string1;
+ if (value1 == null)
+ return void this.removeAttribute(name);
+ string0 = this.getAttribute(name);
+ string1 = value1 + "";
+ return string0 === string1 ? null : string0 === string00 && string1 === string10 ? interpolate0 : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
+ };
+ }
+ function attrFunctionNS2(fullname, interpolate, value) {
+ var string00, string10, interpolate0;
+ return function() {
+ var string0, value1 = value(this), string1;
+ if (value1 == null)
+ return void this.removeAttributeNS(fullname.space, fullname.local);
+ string0 = this.getAttributeNS(fullname.space, fullname.local);
+ string1 = value1 + "";
+ return string0 === string1 ? null : string0 === string00 && string1 === string10 ? interpolate0 : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
+ };
+ }
+ function attr_default2(name, value) {
+ var fullname = namespace_default(name), i2 = fullname === "transform" ? interpolateTransformSvg : interpolate_default;
+ return this.attrTween(name, typeof value === "function" ? (fullname.local ? attrFunctionNS2 : attrFunction2)(fullname, i2, tweenValue(this, "attr." + name, value)) : value == null ? (fullname.local ? attrRemoveNS2 : attrRemove2)(fullname) : (fullname.local ? attrConstantNS2 : attrConstant2)(fullname, i2, value));
+ }
+
+ // node_modules/d3-transition/src/transition/attrTween.js
+ function attrInterpolate(name, i2) {
+ return function(t) {
+ this.setAttribute(name, i2.call(this, t));
+ };
+ }
+ function attrInterpolateNS(fullname, i2) {
+ return function(t) {
+ this.setAttributeNS(fullname.space, fullname.local, i2.call(this, t));
+ };
+ }
+ function attrTweenNS(fullname, value) {
+ var t0, i0;
+ function tween() {
+ var i2 = value.apply(this, arguments);
+ if (i2 !== i0)
+ t0 = (i0 = i2) && attrInterpolateNS(fullname, i2);
+ return t0;
+ }
+ tween._value = value;
+ return tween;
+ }
+ function attrTween(name, value) {
+ var t0, i0;
+ function tween() {
+ var i2 = value.apply(this, arguments);
+ if (i2 !== i0)
+ t0 = (i0 = i2) && attrInterpolate(name, i2);
+ return t0;
+ }
+ tween._value = value;
+ return tween;
+ }
+ function attrTween_default(name, value) {
+ var key = "attr." + name;
+ if (arguments.length < 2)
+ return (key = this.tween(key)) && key._value;
+ if (value == null)
+ return this.tween(key, null);
+ if (typeof value !== "function")
+ throw new Error();
+ var fullname = namespace_default(name);
+ return this.tween(key, (fullname.local ? attrTweenNS : attrTween)(fullname, value));
+ }
+
+ // node_modules/d3-transition/src/transition/delay.js
+ function delayFunction(id2, value) {
+ return function() {
+ init(this, id2).delay = +value.apply(this, arguments);
+ };
+ }
+ function delayConstant(id2, value) {
+ return value = +value, function() {
+ init(this, id2).delay = value;
+ };
+ }
+ function delay_default(value) {
+ var id2 = this._id;
+ return arguments.length ? this.each((typeof value === "function" ? delayFunction : delayConstant)(id2, value)) : get2(this.node(), id2).delay;
+ }
+
+ // node_modules/d3-transition/src/transition/duration.js
+ function durationFunction(id2, value) {
+ return function() {
+ set2(this, id2).duration = +value.apply(this, arguments);
+ };
+ }
+ function durationConstant(id2, value) {
+ return value = +value, function() {
+ set2(this, id2).duration = value;
+ };
+ }
+ function duration_default(value) {
+ var id2 = this._id;
+ return arguments.length ? this.each((typeof value === "function" ? durationFunction : durationConstant)(id2, value)) : get2(this.node(), id2).duration;
+ }
+
+ // node_modules/d3-transition/src/transition/ease.js
+ function easeConstant(id2, value) {
+ if (typeof value !== "function")
+ throw new Error();
+ return function() {
+ set2(this, id2).ease = value;
+ };
+ }
+ function ease_default(value) {
+ var id2 = this._id;
+ return arguments.length ? this.each(easeConstant(id2, value)) : get2(this.node(), id2).ease;
+ }
+
+ // node_modules/d3-transition/src/transition/easeVarying.js
+ function easeVarying(id2, value) {
+ return function() {
+ var v = value.apply(this, arguments);
+ if (typeof v !== "function")
+ throw new Error();
+ set2(this, id2).ease = v;
+ };
+ }
+ function easeVarying_default(value) {
+ if (typeof value !== "function")
+ throw new Error();
+ return this.each(easeVarying(this._id, value));
+ }
+
+ // node_modules/d3-transition/src/transition/filter.js
+ function filter_default2(match) {
+ if (typeof match !== "function")
+ match = matcher_default(match);
+ for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, subgroup = subgroups[j2] = [], node, i2 = 0; i2 < n2; ++i2) {
+ if ((node = group[i2]) && match.call(node, node.__data__, i2, group)) {
+ subgroup.push(node);
+ }
+ }
+ }
+ return new Transition(subgroups, this._parents, this._name, this._id);
+ }
+
+ // node_modules/d3-transition/src/transition/merge.js
+ function merge_default2(transition2) {
+ if (transition2._id !== this._id)
+ throw new Error();
+ for (var groups0 = this._groups, groups1 = transition2._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j2 = 0; j2 < m; ++j2) {
+ for (var group0 = groups0[j2], group1 = groups1[j2], n2 = group0.length, merge3 = merges[j2] = new Array(n2), node, i2 = 0; i2 < n2; ++i2) {
+ if (node = group0[i2] || group1[i2]) {
+ merge3[i2] = node;
+ }
+ }
+ }
+ for (; j2 < m0; ++j2) {
+ merges[j2] = groups0[j2];
+ }
+ return new Transition(merges, this._parents, this._name, this._id);
+ }
+
+ // node_modules/d3-transition/src/transition/on.js
+ function start(name) {
+ return (name + "").trim().split(/^|\s+/).every(function(t) {
+ var i2 = t.indexOf(".");
+ if (i2 >= 0)
+ t = t.slice(0, i2);
+ return !t || t === "start";
+ });
+ }
+ function onFunction(id2, name, listener) {
+ var on0, on1, sit = start(name) ? init : set2;
+ return function() {
+ var schedule = sit(this, id2), on = schedule.on;
+ if (on !== on0)
+ (on1 = (on0 = on).copy()).on(name, listener);
+ schedule.on = on1;
+ };
+ }
+ function on_default2(name, listener) {
+ var id2 = this._id;
+ return arguments.length < 2 ? get2(this.node(), id2).on.on(name) : this.each(onFunction(id2, name, listener));
+ }
+
+ // node_modules/d3-transition/src/transition/remove.js
+ function removeFunction(id2) {
+ return function() {
+ var parent = this.parentNode;
+ for (var i2 in this.__transition)
+ if (+i2 !== id2)
+ return;
+ if (parent)
+ parent.removeChild(this);
+ };
+ }
+ function remove_default2() {
+ return this.on("end.remove", removeFunction(this._id));
+ }
+
+ // node_modules/d3-transition/src/transition/select.js
+ function select_default3(select) {
+ var name = this._name, id2 = this._id;
+ if (typeof select !== "function")
+ select = selector_default(select);
+ for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, subgroup = subgroups[j2] = new Array(n2), node, subnode, i2 = 0; i2 < n2; ++i2) {
+ if ((node = group[i2]) && (subnode = select.call(node, node.__data__, i2, group))) {
+ if ("__data__" in node)
+ subnode.__data__ = node.__data__;
+ subgroup[i2] = subnode;
+ schedule_default(subgroup[i2], name, id2, i2, subgroup, get2(node, id2));
+ }
+ }
+ }
+ return new Transition(subgroups, this._parents, name, id2);
+ }
+
+ // node_modules/d3-transition/src/transition/selectAll.js
+ function selectAll_default3(select) {
+ var name = this._name, id2 = this._id;
+ if (typeof select !== "function")
+ select = selectorAll_default(select);
+ for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, node, i2 = 0; i2 < n2; ++i2) {
+ if (node = group[i2]) {
+ for (var children2 = select.call(node, node.__data__, i2, group), child, inherit2 = get2(node, id2), k = 0, l = children2.length; k < l; ++k) {
+ if (child = children2[k]) {
+ schedule_default(child, name, id2, k, children2, inherit2);
+ }
+ }
+ subgroups.push(children2);
+ parents.push(node);
+ }
+ }
+ }
+ return new Transition(subgroups, parents, name, id2);
+ }
+
+ // node_modules/d3-transition/src/transition/selection.js
+ var Selection2 = selection_default.prototype.constructor;
+ function selection_default2() {
+ return new Selection2(this._groups, this._parents);
+ }
+
+ // node_modules/d3-transition/src/transition/style.js
+ function styleNull(name, interpolate) {
+ var string00, string10, interpolate0;
+ return function() {
+ var string0 = styleValue(this, name), string1 = (this.style.removeProperty(name), styleValue(this, name));
+ return string0 === string1 ? null : string0 === string00 && string1 === string10 ? interpolate0 : interpolate0 = interpolate(string00 = string0, string10 = string1);
+ };
+ }
+ function styleRemove2(name) {
+ return function() {
+ this.style.removeProperty(name);
+ };
+ }
+ function styleConstant2(name, interpolate, value1) {
+ var string00, string1 = value1 + "", interpolate0;
+ return function() {
+ var string0 = styleValue(this, name);
+ return string0 === string1 ? null : string0 === string00 ? interpolate0 : interpolate0 = interpolate(string00 = string0, value1);
+ };
+ }
+ function styleFunction2(name, interpolate, value) {
+ var string00, string10, interpolate0;
+ return function() {
+ var string0 = styleValue(this, name), value1 = value(this), string1 = value1 + "";
+ if (value1 == null)
+ string1 = value1 = (this.style.removeProperty(name), styleValue(this, name));
+ return string0 === string1 ? null : string0 === string00 && string1 === string10 ? interpolate0 : (string10 = string1, interpolate0 = interpolate(string00 = string0, value1));
+ };
+ }
+ function styleMaybeRemove(id2, name) {
+ var on0, on1, listener0, key = "style." + name, event = "end." + key, remove2;
+ return function() {
+ var schedule = set2(this, id2), on = schedule.on, listener = schedule.value[key] == null ? remove2 || (remove2 = styleRemove2(name)) : void 0;
+ if (on !== on0 || listener0 !== listener)
+ (on1 = (on0 = on).copy()).on(event, listener0 = listener);
+ schedule.on = on1;
+ };
+ }
+ function style_default2(name, value, priority) {
+ var i2 = (name += "") === "transform" ? interpolateTransformCss : interpolate_default;
+ return value == null ? this.styleTween(name, styleNull(name, i2)).on("end.style." + name, styleRemove2(name)) : typeof value === "function" ? this.styleTween(name, styleFunction2(name, i2, tweenValue(this, "style." + name, value))).each(styleMaybeRemove(this._id, name)) : this.styleTween(name, styleConstant2(name, i2, value), priority).on("end.style." + name, null);
+ }
+
+ // node_modules/d3-transition/src/transition/styleTween.js
+ function styleInterpolate(name, i2, priority) {
+ return function(t) {
+ this.style.setProperty(name, i2.call(this, t), priority);
+ };
+ }
+ function styleTween(name, value, priority) {
+ var t, i0;
+ function tween() {
+ var i2 = value.apply(this, arguments);
+ if (i2 !== i0)
+ t = (i0 = i2) && styleInterpolate(name, i2, priority);
+ return t;
+ }
+ tween._value = value;
+ return tween;
+ }
+ function styleTween_default(name, value, priority) {
+ var key = "style." + (name += "");
+ if (arguments.length < 2)
+ return (key = this.tween(key)) && key._value;
+ if (value == null)
+ return this.tween(key, null);
+ if (typeof value !== "function")
+ throw new Error();
+ return this.tween(key, styleTween(name, value, priority == null ? "" : priority));
+ }
+
+ // node_modules/d3-transition/src/transition/text.js
+ function textConstant2(value) {
+ return function() {
+ this.textContent = value;
+ };
+ }
+ function textFunction2(value) {
+ return function() {
+ var value1 = value(this);
+ this.textContent = value1 == null ? "" : value1;
+ };
+ }
+ function text_default2(value) {
+ return this.tween("text", typeof value === "function" ? textFunction2(tweenValue(this, "text", value)) : textConstant2(value == null ? "" : value + ""));
+ }
+
+ // node_modules/d3-transition/src/transition/textTween.js
+ function textInterpolate(i2) {
+ return function(t) {
+ this.textContent = i2.call(this, t);
+ };
+ }
+ function textTween(value) {
+ var t0, i0;
+ function tween() {
+ var i2 = value.apply(this, arguments);
+ if (i2 !== i0)
+ t0 = (i0 = i2) && textInterpolate(i2);
+ return t0;
+ }
+ tween._value = value;
+ return tween;
+ }
+ function textTween_default(value) {
+ var key = "text";
+ if (arguments.length < 1)
+ return (key = this.tween(key)) && key._value;
+ if (value == null)
+ return this.tween(key, null);
+ if (typeof value !== "function")
+ throw new Error();
+ return this.tween(key, textTween(value));
+ }
+
+ // node_modules/d3-transition/src/transition/transition.js
+ function transition_default() {
+ var name = this._name, id0 = this._id, id1 = newId();
+ for (var groups = this._groups, m = groups.length, j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, node, i2 = 0; i2 < n2; ++i2) {
+ if (node = group[i2]) {
+ var inherit2 = get2(node, id0);
+ schedule_default(node, name, id1, i2, group, {
+ time: inherit2.time + inherit2.delay + inherit2.duration,
+ delay: 0,
+ duration: inherit2.duration,
+ ease: inherit2.ease
+ });
+ }
+ }
+ }
+ return new Transition(groups, this._parents, name, id1);
+ }
+
+ // node_modules/d3-transition/src/transition/end.js
+ function end_default() {
+ var on0, on1, that = this, id2 = that._id, size = that.size();
+ return new Promise(function(resolve, reject) {
+ var cancel = { value: reject }, end = { value: function() {
+ if (--size === 0)
+ resolve();
+ } };
+ that.each(function() {
+ var schedule = set2(this, id2), on = schedule.on;
+ if (on !== on0) {
+ on1 = (on0 = on).copy();
+ on1._.cancel.push(cancel);
+ on1._.interrupt.push(cancel);
+ on1._.end.push(end);
+ }
+ schedule.on = on1;
+ });
+ if (size === 0)
+ resolve();
+ });
+ }
+
+ // node_modules/d3-transition/src/transition/index.js
+ var id = 0;
+ function Transition(groups, parents, name, id2) {
+ this._groups = groups;
+ this._parents = parents;
+ this._name = name;
+ this._id = id2;
+ }
+ function transition(name) {
+ return selection_default().transition(name);
+ }
+ function newId() {
+ return ++id;
+ }
+ var selection_prototype = selection_default.prototype;
+ Transition.prototype = transition.prototype = {
+ constructor: Transition,
+ select: select_default3,
+ selectAll: selectAll_default3,
+ selectChild: selection_prototype.selectChild,
+ selectChildren: selection_prototype.selectChildren,
+ filter: filter_default2,
+ merge: merge_default2,
+ selection: selection_default2,
+ transition: transition_default,
+ call: selection_prototype.call,
+ nodes: selection_prototype.nodes,
+ node: selection_prototype.node,
+ size: selection_prototype.size,
+ empty: selection_prototype.empty,
+ each: selection_prototype.each,
+ on: on_default2,
+ attr: attr_default2,
+ attrTween: attrTween_default,
+ style: style_default2,
+ styleTween: styleTween_default,
+ text: text_default2,
+ textTween: textTween_default,
+ remove: remove_default2,
+ tween: tween_default,
+ delay: delay_default,
+ duration: duration_default,
+ ease: ease_default,
+ easeVarying: easeVarying_default,
+ end: end_default,
+ [Symbol.iterator]: selection_prototype[Symbol.iterator]
+ };
+
+ // node_modules/d3-ease/src/linear.js
+ var linear2 = (t) => +t;
+
+ // node_modules/d3-ease/src/cubic.js
+ function cubicInOut(t) {
+ return ((t *= 2) <= 1 ? t * t * t : (t -= 2) * t * t + 2) / 2;
+ }
+
+ // node_modules/d3-transition/src/selection/transition.js
+ var defaultTiming = {
+ time: null,
+ // Set on use.
+ delay: 0,
+ duration: 250,
+ ease: cubicInOut
+ };
+ function inherit(node, id2) {
+ var timing;
+ while (!(timing = node.__transition) || !(timing = timing[id2])) {
+ if (!(node = node.parentNode)) {
+ throw new Error(`transition ${id2} not found`);
+ }
+ }
+ return timing;
+ }
+ function transition_default2(name) {
+ var id2, timing;
+ if (name instanceof Transition) {
+ id2 = name._id, name = name._name;
+ } else {
+ id2 = newId(), (timing = defaultTiming).time = now(), name = name == null ? null : name + "";
+ }
+ for (var groups = this._groups, m = groups.length, j2 = 0; j2 < m; ++j2) {
+ for (var group = groups[j2], n2 = group.length, node, i2 = 0; i2 < n2; ++i2) {
+ if (node = group[i2]) {
+ schedule_default(node, name, id2, i2, group, timing || inherit(node, id2));
+ }
+ }
+ }
+ return new Transition(groups, this._parents, name, id2);
+ }
+
+ // node_modules/d3-transition/src/selection/index.js
+ selection_default.prototype.interrupt = interrupt_default2;
+ selection_default.prototype.transition = transition_default2;
+
+ // node_modules/d3-zoom/src/constant.js
+ var constant_default4 = (x) => () => x;
+
+ // node_modules/d3-zoom/src/event.js
+ function ZoomEvent(type2, {
+ sourceEvent,
+ target,
+ transform: transform2,
+ dispatch: dispatch10
+ }) {
+ Object.defineProperties(this, {
+ type: { value: type2, enumerable: true, configurable: true },
+ sourceEvent: { value: sourceEvent, enumerable: true, configurable: true },
+ target: { value: target, enumerable: true, configurable: true },
+ transform: { value: transform2, enumerable: true, configurable: true },
+ _: { value: dispatch10 }
+ });
+ }
+
+ // node_modules/d3-zoom/src/transform.js
+ function Transform(k, x, y) {
+ this.k = k;
+ this.x = x;
+ this.y = y;
+ }
+ Transform.prototype = {
+ constructor: Transform,
+ scale: function(k) {
+ return k === 1 ? this : new Transform(this.k * k, this.x, this.y);
+ },
+ translate: function(x, y) {
+ return x === 0 & y === 0 ? this : new Transform(this.k, this.x + this.k * x, this.y + this.k * y);
+ },
+ apply: function(point2) {
+ return [point2[0] * this.k + this.x, point2[1] * this.k + this.y];
+ },
+ applyX: function(x) {
+ return x * this.k + this.x;
+ },
+ applyY: function(y) {
+ return y * this.k + this.y;
+ },
+ invert: function(location) {
+ return [(location[0] - this.x) / this.k, (location[1] - this.y) / this.k];
+ },
+ invertX: function(x) {
+ return (x - this.x) / this.k;
+ },
+ invertY: function(y) {
+ return (y - this.y) / this.k;
+ },
+ rescaleX: function(x) {
+ return x.copy().domain(x.range().map(this.invertX, this).map(x.invert, x));
+ },
+ rescaleY: function(y) {
+ return y.copy().domain(y.range().map(this.invertY, this).map(y.invert, y));
+ },
+ toString: function() {
+ return "translate(" + this.x + "," + this.y + ") scale(" + this.k + ")";
+ }
+ };
+ var identity2 = new Transform(1, 0, 0);
+ transform.prototype = Transform.prototype;
+ function transform(node) {
+ while (!node.__zoom)
+ if (!(node = node.parentNode))
+ return identity2;
+ return node.__zoom;
+ }
+
+ // node_modules/d3-zoom/src/noevent.js
+ function nopropagation2(event) {
+ event.stopImmediatePropagation();
+ }
+ function noevent_default2(event) {
+ event.preventDefault();
+ event.stopImmediatePropagation();
+ }
+
+ // node_modules/d3-zoom/src/zoom.js
+ function defaultFilter2(event) {
+ return (!event.ctrlKey || event.type === "wheel") && !event.button;
+ }
+ function defaultExtent() {
+ var e = this;
+ if (e instanceof SVGElement) {
+ e = e.ownerSVGElement || e;
+ if (e.hasAttribute("viewBox")) {
+ e = e.viewBox.baseVal;
+ return [[e.x, e.y], [e.x + e.width, e.y + e.height]];
+ }
+ return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];
+ }
+ return [[0, 0], [e.clientWidth, e.clientHeight]];
+ }
+ function defaultTransform() {
+ return this.__zoom || identity2;
+ }
+ function defaultWheelDelta(event) {
+ return -event.deltaY * (event.deltaMode === 1 ? 0.05 : event.deltaMode ? 1 : 2e-3) * (event.ctrlKey ? 10 : 1);
+ }
+ function defaultTouchable2() {
+ return navigator.maxTouchPoints || "ontouchstart" in this;
+ }
+ function defaultConstrain(transform2, extent, translateExtent) {
+ var dx0 = transform2.invertX(extent[0][0]) - translateExtent[0][0], dx1 = transform2.invertX(extent[1][0]) - translateExtent[1][0], dy0 = transform2.invertY(extent[0][1]) - translateExtent[0][1], dy1 = transform2.invertY(extent[1][1]) - translateExtent[1][1];
+ return transform2.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)
+ );
+ }
+ function zoom_default2() {
+ var filter2 = defaultFilter2, extent = defaultExtent, constrain = defaultConstrain, wheelDelta = defaultWheelDelta, touchable = defaultTouchable2, scaleExtent = [0, Infinity], translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]], duration = 250, interpolate = zoom_default, listeners = dispatch_default("start", "zoom", "end"), touchstarting, touchfirst, touchending, touchDelay = 500, wheelDelay = 150, clickDistance2 = 0, tapDistance = 10;
+ function zoom(selection2) {
+ selection2.property("__zoom", defaultTransform).on("wheel.zoom", wheeled, { passive: false }).on("mousedown.zoom", mousedowned).on("dblclick.zoom", dblclicked).filter(touchable).on("touchstart.zoom", touchstarted).on("touchmove.zoom", touchmoved).on("touchend.zoom touchcancel.zoom", touchended).style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
+ }
+ zoom.transform = function(collection, transform2, point2, event) {
+ var selection2 = collection.selection ? collection.selection() : collection;
+ selection2.property("__zoom", defaultTransform);
+ if (collection !== selection2) {
+ schedule(collection, transform2, point2, event);
+ } else {
+ selection2.interrupt().each(function() {
+ gesture(this, arguments).event(event).start().zoom(null, typeof transform2 === "function" ? transform2.apply(this, arguments) : transform2).end();
+ });
+ }
+ };
+ zoom.scaleBy = function(selection2, k, p, event) {
+ zoom.scaleTo(selection2, function() {
+ var k0 = this.__zoom.k, k1 = typeof k === "function" ? k.apply(this, arguments) : k;
+ return k0 * k1;
+ }, p, event);
+ };
+ zoom.scaleTo = function(selection2, k, p, event) {
+ zoom.transform(selection2, function() {
+ var e = extent.apply(this, arguments), t0 = this.__zoom, p02 = p == null ? centroid(e) : typeof p === "function" ? p.apply(this, arguments) : p, p1 = t0.invert(p02), k1 = typeof k === "function" ? k.apply(this, arguments) : k;
+ return constrain(translate(scale(t0, k1), p02, p1), e, translateExtent);
+ }, p, event);
+ };
+ zoom.translateBy = function(selection2, x, y, event) {
+ zoom.transform(selection2, function() {
+ return constrain(this.__zoom.translate(
+ typeof x === "function" ? x.apply(this, arguments) : x,
+ typeof y === "function" ? y.apply(this, arguments) : y
+ ), extent.apply(this, arguments), translateExtent);
+ }, null, event);
+ };
+ zoom.translateTo = function(selection2, x, y, p, event) {
+ zoom.transform(selection2, function() {
+ var e = extent.apply(this, arguments), t = this.__zoom, p02 = p == null ? centroid(e) : typeof p === "function" ? p.apply(this, arguments) : p;
+ return constrain(identity2.translate(p02[0], p02[1]).scale(t.k).translate(
+ typeof x === "function" ? -x.apply(this, arguments) : -x,
+ typeof y === "function" ? -y.apply(this, arguments) : -y
+ ), e, translateExtent);
+ }, p, event);
+ };
+ function scale(transform2, k) {
+ k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k));
+ return k === transform2.k ? transform2 : new Transform(k, transform2.x, transform2.y);
+ }
+ function translate(transform2, p02, p1) {
+ var x = p02[0] - p1[0] * transform2.k, y = p02[1] - p1[1] * transform2.k;
+ return x === transform2.x && y === transform2.y ? transform2 : new Transform(transform2.k, x, y);
+ }
+ function centroid(extent2) {
+ return [(+extent2[0][0] + +extent2[1][0]) / 2, (+extent2[0][1] + +extent2[1][1]) / 2];
+ }
+ function schedule(transition2, transform2, point2, event) {
+ transition2.on("start.zoom", function() {
+ gesture(this, arguments).event(event).start();
+ }).on("interrupt.zoom end.zoom", function() {
+ gesture(this, arguments).event(event).end();
+ }).tween("zoom", function() {
+ var that = this, args = arguments, g = gesture(that, args).event(event), e = extent.apply(that, args), p = point2 == null ? centroid(e) : typeof point2 === "function" ? point2.apply(that, args) : point2, w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]), a = that.__zoom, b = typeof transform2 === "function" ? transform2.apply(that, args) : transform2, i2 = interpolate(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));
+ return function(t) {
+ if (t === 1)
+ t = b;
+ else {
+ var l = i2(t), k = w / l[2];
+ t = new Transform(k, p[0] - l[0] * k, p[1] - l[1] * k);
+ }
+ g.zoom(null, t);
+ };
+ });
+ }
+ function gesture(that, args, clean2) {
+ return !clean2 && that.__zooming || new Gesture(that, args);
+ }
+ function Gesture(that, args) {
+ this.that = that;
+ this.args = args;
+ this.active = 0;
+ this.sourceEvent = null;
+ this.extent = extent.apply(that, args);
+ this.taps = 0;
+ }
+ Gesture.prototype = {
+ event: function(event) {
+ if (event)
+ this.sourceEvent = event;
+ return this;
+ },
+ start: function() {
+ if (++this.active === 1) {
+ this.that.__zooming = this;
+ this.emit("start");
+ }
+ return this;
+ },
+ zoom: function(key, transform2) {
+ if (this.mouse && key !== "mouse")
+ this.mouse[1] = transform2.invert(this.mouse[0]);
+ if (this.touch0 && key !== "touch")
+ this.touch0[1] = transform2.invert(this.touch0[0]);
+ if (this.touch1 && key !== "touch")
+ this.touch1[1] = transform2.invert(this.touch1[0]);
+ this.that.__zoom = transform2;
+ this.emit("zoom");
+ return this;
+ },
+ end: function() {
+ if (--this.active === 0) {
+ delete this.that.__zooming;
+ this.emit("end");
+ }
+ return this;
+ },
+ emit: function(type2) {
+ var d = select_default2(this.that).datum();
+ listeners.call(
+ type2,
+ this.that,
+ new ZoomEvent(type2, {
+ sourceEvent: this.sourceEvent,
+ target: zoom,
+ type: type2,
+ transform: this.that.__zoom,
+ dispatch: listeners
+ }),
+ d
+ );
+ }
+ };
+ function wheeled(event, ...args) {
+ if (!filter2.apply(this, arguments))
+ return;
+ var g = gesture(this, args).event(event), t = this.__zoom, k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t.k * Math.pow(2, wheelDelta.apply(this, arguments)))), p = pointer_default(event);
+ 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);
+ }
+ clearTimeout(g.wheel);
+ } else if (t.k === k)
+ return;
+ else {
+ g.mouse = [p, t.invert(p)];
+ interrupt_default(this);
+ g.start();
+ }
+ noevent_default2(event);
+ g.wheel = setTimeout(wheelidled, wheelDelay);
+ g.zoom("mouse", constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent, translateExtent));
+ function wheelidled() {
+ g.wheel = null;
+ g.end();
+ }
+ }
+ function mousedowned(event, ...args) {
+ if (touchending || !filter2.apply(this, arguments))
+ return;
+ var currentTarget = event.currentTarget, g = gesture(this, args, true).event(event), v = select_default2(event.view).on("mousemove.zoom", mousemoved, true).on("mouseup.zoom", mouseupped, true), p = pointer_default(event, currentTarget), x05 = event.clientX, y05 = event.clientY;
+ nodrag_default(event.view);
+ nopropagation2(event);
+ g.mouse = [p, this.__zoom.invert(p)];
+ interrupt_default(this);
+ g.start();
+ function mousemoved(event2) {
+ noevent_default2(event2);
+ if (!g.moved) {
+ var dx = event2.clientX - x05, dy = event2.clientY - y05;
+ g.moved = dx * dx + dy * dy > clickDistance2;
+ }
+ g.event(event2).zoom("mouse", constrain(translate(g.that.__zoom, g.mouse[0] = pointer_default(event2, currentTarget), g.mouse[1]), g.extent, translateExtent));
+ }
+ function mouseupped(event2) {
+ v.on("mousemove.zoom mouseup.zoom", null);
+ yesdrag(event2.view, g.moved);
+ noevent_default2(event2);
+ g.event(event2).end();
+ }
+ }
+ function dblclicked(event, ...args) {
+ if (!filter2.apply(this, arguments))
+ return;
+ var t0 = this.__zoom, p02 = pointer_default(event.changedTouches ? event.changedTouches[0] : event, this), p1 = t0.invert(p02), k1 = t0.k * (event.shiftKey ? 0.5 : 2), t1 = constrain(translate(scale(t0, k1), p02, p1), extent.apply(this, args), translateExtent);
+ noevent_default2(event);
+ if (duration > 0)
+ select_default2(this).transition().duration(duration).call(schedule, t1, p02, event);
+ else
+ select_default2(this).call(zoom.transform, t1, p02, event);
+ }
+ function touchstarted(event, ...args) {
+ if (!filter2.apply(this, arguments))
+ return;
+ var touches = event.touches, n2 = touches.length, g = gesture(this, args, event.changedTouches.length === n2).event(event), started, i2, t, p;
+ nopropagation2(event);
+ for (i2 = 0; i2 < n2; ++i2) {
+ t = touches[i2], p = pointer_default(t, this);
+ p = [p, this.__zoom.invert(p), t.identifier];
+ if (!g.touch0)
+ g.touch0 = p, started = true, g.taps = 1 + !!touchstarting;
+ else if (!g.touch1 && g.touch0[2] !== p[2])
+ g.touch1 = p, g.taps = 0;
+ }
+ if (touchstarting)
+ touchstarting = clearTimeout(touchstarting);
+ if (started) {
+ if (g.taps < 2)
+ touchfirst = p[0], touchstarting = setTimeout(function() {
+ touchstarting = null;
+ }, touchDelay);
+ interrupt_default(this);
+ g.start();
+ }
+ }
+ function touchmoved(event, ...args) {
+ if (!this.__zooming)
+ return;
+ var g = gesture(this, args).event(event), touches = event.changedTouches, n2 = touches.length, i2, t, p, l;
+ noevent_default2(event);
+ for (i2 = 0; i2 < n2; ++i2) {
+ t = touches[i2], p = pointer_default(t, this);
+ if (g.touch0 && g.touch0[2] === t.identifier)
+ g.touch0[0] = p;
+ else if (g.touch1 && g.touch1[2] === t.identifier)
+ g.touch1[0] = p;
+ }
+ t = g.that.__zoom;
+ if (g.touch1) {
+ var p02 = g.touch0[0], l0 = g.touch0[1], p1 = g.touch1[0], l1 = g.touch1[1], dp = (dp = p1[0] - p02[0]) * dp + (dp = p1[1] - p02[1]) * dp, dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;
+ t = scale(t, Math.sqrt(dp / dl));
+ p = [(p02[0] + p1[0]) / 2, (p02[1] + p1[1]) / 2];
+ l = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];
+ } else if (g.touch0)
+ p = g.touch0[0], l = g.touch0[1];
+ else
+ return;
+ g.zoom("touch", constrain(translate(t, p, l), g.extent, translateExtent));
+ }
+ function touchended(event, ...args) {
+ if (!this.__zooming)
+ return;
+ var g = gesture(this, args).event(event), touches = event.changedTouches, n2 = touches.length, i2, t;
+ nopropagation2(event);
+ if (touchending)
+ clearTimeout(touchending);
+ touchending = setTimeout(function() {
+ touchending = null;
+ }, touchDelay);
+ for (i2 = 0; i2 < n2; ++i2) {
+ t = touches[i2];
+ if (g.touch0 && g.touch0[2] === t.identifier)
+ delete g.touch0;
+ else if (g.touch1 && g.touch1[2] === t.identifier)
+ delete g.touch1;
+ }
+ if (g.touch1 && !g.touch0)
+ g.touch0 = g.touch1, delete g.touch1;
+ if (g.touch0)
+ g.touch0[1] = this.__zoom.invert(g.touch0[0]);
+ else {
+ g.end();
+ if (g.taps === 2) {
+ t = pointer_default(t, this);
+ if (Math.hypot(touchfirst[0] - t[0], touchfirst[1] - t[1]) < tapDistance) {
+ var p = select_default2(this).on("dblclick.zoom");
+ if (p)
+ p.apply(this, arguments);
+ }
+ }
+ }
+ }
+ zoom.wheelDelta = function(_) {
+ return arguments.length ? (wheelDelta = typeof _ === "function" ? _ : constant_default4(+_), zoom) : wheelDelta;
+ };
+ zoom.filter = function(_) {
+ return arguments.length ? (filter2 = typeof _ === "function" ? _ : constant_default4(!!_), zoom) : filter2;
+ };
+ zoom.touchable = function(_) {
+ return arguments.length ? (touchable = typeof _ === "function" ? _ : constant_default4(!!_), zoom) : touchable;
+ };
+ zoom.extent = function(_) {
+ return arguments.length ? (extent = typeof _ === "function" ? _ : constant_default4([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
+ };
+ zoom.scaleExtent = function(_) {
+ return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];
+ };
+ 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]]];
+ };
+ zoom.constrain = function(_) {
+ return arguments.length ? (constrain = _, zoom) : constrain;
+ };
+ zoom.duration = function(_) {
+ return arguments.length ? (duration = +_, zoom) : duration;
+ };
+ zoom.interpolate = function(_) {
+ return arguments.length ? (interpolate = _, zoom) : interpolate;
+ };
+ zoom.on = function() {
+ var value = listeners.on.apply(listeners, arguments);
+ return value === listeners ? zoom : value;
+ };
+ zoom.clickDistance = function(_) {
+ return arguments.length ? (clickDistance2 = (_ = +_) * _, zoom) : Math.sqrt(clickDistance2);
+ };
+ zoom.tapDistance = function(_) {
+ return arguments.length ? (tapDistance = +_, zoom) : tapDistance;
+ };
+ return zoom;
+ }
+
+ // modules/geo/raw_mercator.js
+ function geoRawMercator() {
+ var project = mercatorRaw;
+ var k = 512 / Math.PI;
+ var x = 0;
+ var y = 0;
+ var clipExtent = [[0, 0], [0, 0]];
+ function projection2(point2) {
+ point2 = project(point2[0] * Math.PI / 180, point2[1] * Math.PI / 180);
+ return [point2[0] * k + x, y - point2[1] * k];
+ }
+ projection2.invert = function(point2) {
+ point2 = project.invert((point2[0] - x) / k, (y - point2[1]) / k);
+ return point2 && [point2[0] * 180 / Math.PI, point2[1] * 180 / Math.PI];
+ };
+ projection2.scale = function(_) {
+ if (!arguments.length)
+ return k;
+ k = +_;
+ return projection2;
+ };
+ projection2.translate = function(_) {
+ if (!arguments.length)
+ return [x, y];
+ x = +_[0];
+ y = +_[1];
+ return projection2;
+ };
+ projection2.clipExtent = function(_) {
+ if (!arguments.length)
+ return clipExtent;
+ clipExtent = _;
+ return projection2;
+ };
+ projection2.transform = function(obj) {
+ if (!arguments.length)
+ return identity2.translate(x, y).scale(k);
+ x = +obj.x;
+ y = +obj.y;
+ k = +obj.k;
+ return projection2;
+ };
+ projection2.stream = transform_default({
+ point: function(x2, y2) {
+ var vec = projection2([x2, y2]);
+ this.stream.point(vec[0], vec[1]);
+ }
+ }).stream;
+ return projection2;
+ }
+
+ // modules/geo/ortho.js
+ function geoOrthoNormalizedDotProduct(a, b, origin) {
+ if (geoVecEqual(origin, a) || geoVecEqual(origin, b)) {
+ return 1;
+ }
+ return geoVecNormalizedDot(a, b, origin);
+ }
+ function geoOrthoFilterDotProduct(dotp, epsilon3, lowerThreshold, upperThreshold, allowStraightAngles) {
+ var val = Math.abs(dotp);
+ if (val < epsilon3) {
+ return 0;
+ } else if (allowStraightAngles && Math.abs(val - 1) < epsilon3) {
+ return 0;
+ } else if (val < lowerThreshold || val > upperThreshold) {
+ return dotp;
+ } else {
+ return null;
+ }
+ }
+ function geoOrthoCalcScore(points, isClosed, epsilon3, threshold) {
+ var score = 0;
+ var first = isClosed ? 0 : 1;
+ var last = isClosed ? points.length : points.length - 1;
+ var coords = points.map(function(p) {
+ return p.coord;
+ });
+ var lowerThreshold = Math.cos((90 - threshold) * Math.PI / 180);
+ var upperThreshold = Math.cos(threshold * Math.PI / 180);
+ for (var i2 = first; i2 < last; i2++) {
+ var a = coords[(i2 - 1 + coords.length) % coords.length];
+ var origin = coords[i2];
+ var b = coords[(i2 + 1) % coords.length];
+ var dotp = geoOrthoFilterDotProduct(geoOrthoNormalizedDotProduct(a, b, origin), epsilon3, lowerThreshold, upperThreshold);
+ if (dotp === null)
+ continue;
+ score = score + 2 * Math.min(Math.abs(dotp - 1), Math.min(Math.abs(dotp), Math.abs(dotp + 1)));
+ }
+ return score;
+ }
+ function geoOrthoMaxOffsetAngle(coords, isClosed, lessThan) {
+ var max3 = -Infinity;
+ var first = isClosed ? 0 : 1;
+ var last = isClosed ? coords.length : coords.length - 1;
+ for (var i2 = first; i2 < last; i2++) {
+ var a = coords[(i2 - 1 + coords.length) % coords.length];
+ var origin = coords[i2];
+ var b = coords[(i2 + 1) % coords.length];
+ var normalizedDotP = geoOrthoNormalizedDotProduct(a, b, origin);
+ var angle2 = Math.acos(Math.abs(normalizedDotP)) * 180 / Math.PI;
+ if (angle2 > 45)
+ angle2 = 90 - angle2;
+ if (angle2 >= lessThan)
+ continue;
+ if (angle2 > max3)
+ max3 = angle2;
+ }
+ if (max3 === -Infinity)
+ return null;
+ return max3;
+ }
+ function geoOrthoCanOrthogonalize(coords, isClosed, epsilon3, threshold, allowStraightAngles) {
+ var score = null;
+ var first = isClosed ? 0 : 1;
+ var last = isClosed ? coords.length : coords.length - 1;
+ var lowerThreshold = Math.cos((90 - threshold) * Math.PI / 180);
+ var upperThreshold = Math.cos(threshold * Math.PI / 180);
+ for (var i2 = first; i2 < last; i2++) {
+ var a = coords[(i2 - 1 + coords.length) % coords.length];
+ var origin = coords[i2];
+ var b = coords[(i2 + 1) % coords.length];
+ var dotp = geoOrthoFilterDotProduct(geoOrthoNormalizedDotProduct(a, b, origin), epsilon3, lowerThreshold, upperThreshold, allowStraightAngles);
+ if (dotp === null)
+ continue;
+ if (Math.abs(dotp) > 0)
+ return 1;
+ score = 0;
+ }
+ return score;
+ }
+
+ // modules/util/array.js
+ function utilArrayIdentical(a, b) {
+ if (a === b)
+ return true;
+ var i2 = a.length;
+ if (i2 !== b.length)
+ return false;
+ while (i2--) {
+ if (a[i2] !== b[i2])
+ return false;
+ }
+ return true;
+ }
+ function utilArrayDifference(a, b) {
+ var other = new Set(b);
+ return Array.from(new Set(a)).filter(function(v) {
+ return !other.has(v);
+ });
+ }
+ function utilArrayIntersection(a, b) {
+ var other = new Set(b);
+ return Array.from(new Set(a)).filter(function(v) {
+ return other.has(v);
+ });
+ }
+ function utilArrayUnion(a, b) {
+ var result = new Set(a);
+ b.forEach(function(v) {
+ result.add(v);
+ });
+ return Array.from(result);
+ }
+ function utilArrayUniq(a) {
+ return Array.from(new Set(a));
+ }
+ function utilArrayChunk(a, chunkSize) {
+ if (!chunkSize || chunkSize < 0)
+ return [a.slice()];
+ var result = new Array(Math.ceil(a.length / chunkSize));
+ return Array.from(result, function(item, i2) {
+ return a.slice(i2 * chunkSize, i2 * chunkSize + chunkSize);
+ });
+ }
+ function utilArrayFlatten(a) {
+ return a.reduce(function(acc, val) {
+ return acc.concat(val);
+ }, []);
+ }
+ function utilArrayGroupBy(a, key) {
+ return a.reduce(function(acc, item) {
+ var group = typeof key === "function" ? key(item) : item[key];
+ (acc[group] = acc[group] || []).push(item);
+ return acc;
+ }, {});
+ }
+ function utilArrayUniqBy(a, key) {
+ var seen = /* @__PURE__ */ new Set();
+ return a.reduce(function(acc, item) {
+ var val = typeof key === "function" ? key(item) : item[key];
+ if (val && !seen.has(val)) {
+ seen.add(val);
+ acc.push(item);
+ }
+ return acc;
+ }, []);
+ }
+
+ // modules/util/util.js
+ var import_diacritics = __toESM(require_diacritics());
+
+ // modules/util/svg_paths_rtl_fix.js
+ var import_alif_toolkit = __toESM(require_lib());
+ var rtlRegex = /[\u0590-\u05FF\u0600-\u06FF\u0750-\u07BF\u08A0–\u08BF]/;
+ function fixRTLTextForSvg(inputText) {
+ var ret = "", rtlBuffer = [];
+ var arabicRegex = /[\u0600-\u06FF]/g;
+ var arabicDiacritics = /[\u0610-\u061A\u064B-\u065F\u0670\u06D6-\u06ED]/g;
+ var arabicMath = /[\u0660-\u066C\u06F0-\u06F9]+/g;
+ var thaanaVowel = /[\u07A6-\u07B0]/;
+ var hebrewSign = /[\u0591-\u05bd\u05bf\u05c1-\u05c5\u05c7]/;
+ if (arabicRegex.test(inputText)) {
+ inputText = (0, import_alif_toolkit.WordShaper)(inputText);
+ }
+ for (var n2 = 0; n2 < inputText.length; n2++) {
+ var c = inputText[n2];
+ if (arabicMath.test(c)) {
+ ret += rtlBuffer.reverse().join("");
+ rtlBuffer = [c];
+ } else {
+ if (rtlBuffer.length && arabicMath.test(rtlBuffer[rtlBuffer.length - 1])) {
+ ret += rtlBuffer.reverse().join("");
+ rtlBuffer = [];
+ }
+ if ((thaanaVowel.test(c) || hebrewSign.test(c) || arabicDiacritics.test(c)) && rtlBuffer.length) {
+ rtlBuffer[rtlBuffer.length - 1] += c;
+ } else if (rtlRegex.test(c) || c.charCodeAt(0) >= 64336 && c.charCodeAt(0) <= 65023 || c.charCodeAt(0) >= 65136 && c.charCodeAt(0) <= 65279) {
+ rtlBuffer.push(c);
+ } else if (c === " " && rtlBuffer.length) {
+ rtlBuffer = [rtlBuffer.reverse().join("") + " "];
+ } else {
+ ret += rtlBuffer.reverse().join("") + c;
+ rtlBuffer = [];
+ }
+ }
+ }
+ ret += rtlBuffer.reverse().join("");
+ return ret;
+ }
+
+ // modules/core/preferences.js
+ var _storage;
+ try {
+ _storage = localStorage;
+ } catch (e) {
+ }
+ _storage = _storage || (() => {
+ let s = {};
+ return {
+ getItem: (k) => s[k],
+ setItem: (k, v) => s[k] = v,
+ removeItem: (k) => delete s[k]
+ };
+ })();
+ var _listeners = {};
+ function corePreferences(k, v) {
+ try {
+ if (v === void 0)
+ return _storage.getItem(k);
+ else if (v === null)
+ _storage.removeItem(k);
+ else
+ _storage.setItem(k, v);
+ if (_listeners[k]) {
+ _listeners[k].forEach((handler) => handler(v));
+ }
+ return true;
+ } catch (e) {
+ if (typeof console !== "undefined") {
+ console.error("localStorage quota exceeded");
+ }
+ return false;
+ }
+ }
+ corePreferences.onChange = function(k, handler) {
+ _listeners[k] = _listeners[k] || [];
+ _listeners[k].push(handler);
+ };
+
+ // modules/core/file_fetcher.js
+ var import_vparse = __toESM(require_vparse());
+
+ // config/id.js
+ var presetsCdnUrl = "https://cdn.jsdelivr.net/npm/@openstreetmap/id-tagging-schema@{presets_version}/";
+ var ociCdnUrl = "https://cdn.jsdelivr.net/npm/osm-community-index@{version}/";
+ var wmfSitematrixCdnUrl = "https://cdn.jsdelivr.net/npm/wmf-sitematrix@{version}/";
+ var nsiCdnUrl = "https://cdn.jsdelivr.net/npm/name-suggestion-index@{version}/";
+ var osmApiConnections = [
+ {
+ // "live" db
+ url: "https://www.openstreetmap.org",
+ client_id: "0tmNTmd0Jo1dQp4AUmMBLtGiD9YpMuXzHefitcuVStc",
+ client_secret: "BTlNrNxIPitHdL4sP2clHw5KLoee9aKkA7dQbc0Bj7Q"
+ },
+ {
+ // "dev" db
+ url: "https://api06.dev.openstreetmap.org",
+ client_id: "Ee1wWJ6UlpERbF6BfTNOpwn0R8k_06mvMXdDUkeHMgw",
+ client_secret: "OnfWFC-JkZNHyYdr_viNn_h_RTZXRslKcUxllOXqf5g"
+ }
+ ];
+ var taginfoApiUrl = "https://taginfo.openstreetmap.org/api/4/";
+ var nominatimApiUrl = "https://nominatim.openstreetmap.org/";
+
+ // package.json
+ var package_default = {
+ name: "iD",
+ version: "2.24.2",
+ description: "A friendly editor for OpenStreetMap",
+ main: "dist/iD.min.js",
+ repository: "github:openstreetmap/iD",
+ homepage: "https://github.com/openstreetmap/iD",
+ bugs: "https://github.com/openstreetmap/iD/issues",
+ keywords: [
+ "editor",
+ "openstreetmap"
+ ],
+ license: "ISC",
+ scripts: {
+ all: "run-s clean build dist",
+ build: "run-s build:css build:data build:js",
+ "build:css": "node scripts/build_css.js",
+ "build:data": "shx mkdir -p dist/data && node scripts/build_data.js",
+ "build:stats": "node config/esbuild.config.mjs --stats && esbuild-visualizer --metadata dist/esbuild.json --exclude *.png --filename docs/statistics.html && shx rm dist/esbuild.json",
+ "build:js": "node config/esbuild.config.mjs",
+ "build:js:watch": "node config/esbuild.config.mjs --watch",
+ clean: "shx rm -f dist/esbuild.json dist/*.js dist/*.map dist/*.css dist/img/*.svg",
+ dist: "run-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": "node config/esbuild.config.min.mjs",
+ "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: "run-s build:js start:server",
+ "start:watch": "run-p build:js:watch start:server",
+ "start:server": "node scripts/server.js",
+ test: "npm-run-all -s lint build test:spec",
+ "test:spec": "karma start config/karma.conf.js",
+ translations: "node scripts/update_locales.js"
+ },
+ dependencies: {
+ "@ideditor/country-coder": "~5.1.0",
+ "@ideditor/location-conflation": "~1.1.0",
+ "@mapbox/geojson-area": "^0.2.2",
+ "@mapbox/sexagesimal": "1.2.0",
+ "@mapbox/vector-tile": "^1.3.1",
+ "@tmcw/togeojson": "^5.2.1",
+ "@turf/bbox": "^6.0.0",
+ "@turf/bbox-clip": "^6.0.0",
+ "abortcontroller-polyfill": "^1.4.0",
+ "aes-js": "^3.1.2",
+ "alif-toolkit": "^1.2.9",
+ "core-js-bundle": "^3.19.0",
+ diacritics: "1.3.0",
+ "fast-deep-equal": "~3.1.1",
+ "fast-json-stable-stringify": "2.1.0",
+ "lodash-es": "~4.17.15",
+ marked: "~4.2.2",
+ "node-diff3": "~3.1.0",
+ "osm-auth": "~2.0.0",
+ pannellum: "2.5.6",
+ pbf: "^3.2.1",
+ "polygon-clipping": "~0.15.1",
+ rbush: "3.0.1",
+ "whatwg-fetch": "^3.4.1",
+ "which-polygon": "2.2.0"
+ },
+ devDependencies: {
+ "@fortawesome/fontawesome-svg-core": "~6.2.0",
+ "@fortawesome/free-brands-svg-icons": "~6.2.0",
+ "@fortawesome/free-regular-svg-icons": "~6.2.0",
+ "@fortawesome/free-solid-svg-icons": "~6.2.0",
+ "@ideditor/temaki": "~5.2.0",
+ "@mapbox/maki": "^8.0.0",
+ "@openstreetmap/id-tagging-schema": "^5.0.1",
+ "@transifex/api": "^5.0.1",
+ autoprefixer: "^10.0.1",
+ chai: "^4.3.4",
+ chalk: "^4.1.2",
+ "cldr-core": "^41.0.0",
+ "cldr-localenames-full": "^41.0.0",
+ "concat-files": "^0.1.1",
+ d3: "~7.8.1",
+ "editor-layer-index": "github:osmlab/editor-layer-index#gh-pages",
+ esbuild: "^0.17.3",
+ "esbuild-visualizer": "^0.4.0",
+ eslint: "^8.8.0",
+ "fetch-mock": "^9.11.0",
+ gaze: "^1.1.3",
+ glob: "^8.0.3",
+ happen: "^0.3.2",
+ "js-yaml": "^4.0.0",
+ "json-stringify-pretty-compact": "^3.0.0",
+ karma: "^6.3.5",
+ "karma-chrome-launcher": "^3.1.0",
+ "karma-coverage": "2.1.1",
+ "karma-mocha": "^2.0.1",
+ "karma-remap-istanbul": "^0.6.0",
+ mapillary_sprite_source: "^1.8.0",
+ "mapillary-js": "4.1.1",
+ minimist: "^1.2.3",
+ mocha: "^10.0.0",
+ "name-suggestion-index": "~6.0",
+ "node-fetch": "^2.6.1",
+ "npm-run-all": "^4.0.0",
+ "osm-community-index": "~5.5.0",
+ postcss: "^8.1.1",
+ "postcss-selector-prepend": "^0.5.0",
+ shelljs: "^0.8.0",
+ shx: "^0.3.0",
+ sinon: "^11.1.2",
+ "sinon-chai": "^3.7.0",
+ smash: "0.0",
+ "static-server": "^2.2.1",
+ "svg-sprite": "2.0.2",
+ vparse: "~1.1.0"
+ },
+ engines: {
+ node: ">=16.14"
+ },
+ browserslist: [
+ "> 0.2%, last 6 major versions, Firefox ESR, maintained node versions"
+ ]
+ };
+
+ // modules/core/file_fetcher.js
+ var _mainFileFetcher = coreFileFetcher();
+ function coreFileFetcher() {
+ const ociVersion = package_default.dependencies["osm-community-index"] || package_default.devDependencies["osm-community-index"];
+ const v = (0, import_vparse.default)(ociVersion);
+ const ociVersionMinor = `${v.major}.${v.minor}`;
+ const presetsVersion = package_default.devDependencies["@openstreetmap/id-tagging-schema"];
+ let _this = {};
+ let _inflight4 = {};
+ let _fileMap = {
+ "address_formats": "data/address_formats.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": "locales/index.min.json",
+ "phone_formats": "data/phone_formats.min.json",
+ "qa_data": "data/qa_data.min.json",
+ "shortcuts": "data/shortcuts.min.json",
+ "territory_languages": "data/territory_languages.min.json",
+ "oci_defaults": ociCdnUrl.replace("{version}", ociVersionMinor) + "dist/defaults.min.json",
+ "oci_features": ociCdnUrl.replace("{version}", ociVersionMinor) + "dist/featureCollection.min.json",
+ "oci_resources": ociCdnUrl.replace("{version}", ociVersionMinor) + "dist/resources.min.json",
+ "presets_package": presetsCdnUrl.replace("{presets_version}", presetsVersion) + "package.json",
+ "deprecated": presetsCdnUrl + "dist/deprecated.min.json",
+ "discarded": presetsCdnUrl + "dist/discarded.min.json",
+ "preset_categories": presetsCdnUrl + "dist/preset_categories.min.json",
+ "preset_defaults": presetsCdnUrl + "dist/preset_defaults.min.json",
+ "preset_fields": presetsCdnUrl + "dist/fields.min.json",
+ "preset_presets": presetsCdnUrl + "dist/presets.min.json",
+ "wmf_sitematrix": wmfSitematrixCdnUrl.replace("{version}", "0.1") + "wikipedia.min.json"
+ };
+ let _cachedData = {};
+ _this.cache = () => _cachedData;
+ _this.get = (which) => {
+ if (_cachedData[which]) {
+ return Promise.resolve(_cachedData[which]);
+ }
+ const file = _fileMap[which];
+ const url = file && _this.asset(file);
+ if (!url) {
+ return Promise.reject(`Unknown data file for "${which}"`);
+ }
+ if (url.includes("{presets_version}")) {
+ return _this.get("presets_package").then((result) => {
+ const presetsVersion2 = result.version;
+ return getUrl(url.replace("{presets_version}", presetsVersion2), which);
+ });
+ } else {
+ return getUrl(url);
+ }
+ };
+ function getUrl(url, which) {
+ let prom = _inflight4[url];
+ if (!prom) {
+ _inflight4[url] = prom = fetch(url).then((response) => {
+ if (!response.ok || !response.json) {
+ throw new Error(response.status + " " + response.statusText);
+ }
+ if (response.status === 204 || response.status === 205)
+ return;
+ return response.json();
+ }).then((result) => {
+ delete _inflight4[url];
+ if (!result) {
+ throw new Error(`No data loaded for "${which}"`);
+ }
+ _cachedData[which] = result;
+ return result;
+ }).catch((err) => {
+ delete _inflight4[url];
+ throw err;
+ });
+ }
+ return prom;
+ }
+ _this.fileMap = function(val) {
+ if (!arguments.length)
+ return _fileMap;
+ _fileMap = val;
+ return _this;
+ };
+ let _assetPath = "";
+ _this.assetPath = function(val) {
+ if (!arguments.length)
+ return _assetPath;
+ _assetPath = val;
+ return _this;
+ };
+ let _assetMap = {};
+ _this.assetMap = function(val) {
+ if (!arguments.length)
+ return _assetMap;
+ _assetMap = val;
+ return _this;
+ };
+ _this.asset = (val) => {
+ if (/^http(s)?:\/\//i.test(val))
+ return val;
+ const filename = _assetPath + val;
+ return _assetMap[filename] || filename;
+ };
+ return _this;
+ }
+
+ // node_modules/@ideditor/country-coder/dist/country-coder.mjs
+ var import_which_polygon = __toESM(require_which_polygon(), 1);
+ var borders_default = { type: "FeatureCollection", 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]]]] } },
+ { 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]]]] } },
+ { 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]]]] } },
+ { 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]]]] } },
+ { 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: ["EU", "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: ["EU", "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: ["EU", "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: ["EU", "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", aliases: ["GB-UKM"], 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: ["EU", "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", 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.35512, 47.57014], [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]]]] }, properties: { iso1A2: "CH", iso1A3: "CHE", iso1N3: "756", wikidata: "Q39", nameEn: "Switzerland", groups: ["155", "150", "UN"], callingCodes: ["41"] } },
+ { 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: ["EU", "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]]]] } },
+ { 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", 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], [-11.75, 54], [-11, 51], [-6.03913, 51.13217], [-5.37267, 53.63269], [-6.26218, 54.09785]]]] }, properties: { iso1A2: "IE", iso1A3: "IRL", iso1N3: "372", wikidata: "Q27", nameEn: "Republic of Ireland", groups: ["EU", "Q22890", "154", "150", "UN"], driveSide: "left", callingCodes: ["353"] } },
+ { 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: ["EU", "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: ["EU", "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: ["EU", "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]]]] } },
+ { 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]]]] } },
+ { 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]]]] } },
+ { 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", groups: ["EU", "UN"] }, 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: ["EU", "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 = borders_default;
+ var whichPolygonGetter = {};
+ var featuresByCode = {};
+ var idFilterRegex = /(?=(?!^(and|the|of|el|la|de)$))(\b(and|the|of|el|la|de)\b)|[-_ .,'()&[\]/]/gi;
+ function canonicalID(id2) {
+ const s = id2 || "";
+ if (s.charAt(0) === ".") {
+ return s.toUpperCase();
+ } else {
+ return s.replace(idFilterRegex, "").toUpperCase();
+ }
+ }
+ var levels = [
+ "subterritory",
+ "territory",
+ "subcountryGroup",
+ "country",
+ "sharedLandform",
+ "intermediateRegion",
+ "subregion",
+ "region",
+ "subunion",
+ "union",
+ "unitedNations",
+ "world"
+ ];
+ loadDerivedDataAndCaches(borders);
+ function loadDerivedDataAndCaches(borders2) {
+ const identifierProps = ["iso1A2", "iso1A3", "m49", "wikidata", "emojiFlag", "ccTLD", "nameEn"];
+ let geometryFeatures = [];
+ borders2.features.forEach((feature22) => {
+ feature22.properties.id = feature22.properties.iso1A2 || feature22.properties.m49 || feature22.properties.wikidata;
+ loadM49(feature22);
+ loadTLD(feature22);
+ loadIsoStatus(feature22);
+ loadLevel(feature22);
+ loadGroups(feature22);
+ loadFlag(feature22);
+ cacheFeatureByIDs(feature22);
+ if (feature22.geometry)
+ geometryFeatures.push(feature22);
+ });
+ borders2.features.forEach((feature22) => {
+ feature22.properties.groups = feature22.properties.groups.map((groupID) => {
+ return featuresByCode[groupID].properties.id;
+ });
+ loadMembersForGroupsOf(feature22);
+ });
+ borders2.features.forEach((feature22) => {
+ loadRoadSpeedUnit(feature22);
+ loadRoadHeightUnit(feature22);
+ loadDriveSide(feature22);
+ loadCallingCodes(feature22);
+ loadGroupGroups(feature22);
+ });
+ borders2.features.forEach((feature22) => {
+ feature22.properties.groups.sort((groupID1, groupID2) => {
+ return levels.indexOf(featuresByCode[groupID1].properties.level) - levels.indexOf(featuresByCode[groupID2].properties.level);
+ });
+ if (feature22.properties.members)
+ feature22.properties.members.sort((id1, id2) => {
+ const diff = levels.indexOf(featuresByCode[id1].properties.level) - levels.indexOf(featuresByCode[id2].properties.level);
+ if (diff === 0) {
+ return borders2.features.indexOf(featuresByCode[id1]) - borders2.features.indexOf(featuresByCode[id2]);
+ }
+ return diff;
+ });
+ });
+ const geometryOnlyCollection = {
+ type: "FeatureCollection",
+ features: geometryFeatures
+ };
+ whichPolygonGetter = (0, import_which_polygon.default)(geometryOnlyCollection);
+ function loadGroups(feature22) {
+ const props = feature22.properties;
+ if (!props.groups) {
+ props.groups = [];
+ }
+ if (feature22.geometry && props.country) {
+ props.groups.push(props.country);
+ }
+ if (props.m49 !== "001") {
+ props.groups.push("001");
+ }
+ }
+ function loadM49(feature22) {
+ const props = feature22.properties;
+ if (!props.m49 && props.iso1N3) {
+ props.m49 = props.iso1N3;
+ }
+ }
+ function loadTLD(feature22) {
+ const props = feature22.properties;
+ if (props.level === "unitedNations")
+ return;
+ if (!props.ccTLD && props.iso1A2) {
+ props.ccTLD = "." + props.iso1A2.toLowerCase();
+ }
+ }
+ function loadIsoStatus(feature22) {
+ const props = feature22.properties;
+ if (!props.isoStatus && props.iso1A2) {
+ props.isoStatus = "official";
+ }
+ }
+ function loadLevel(feature22) {
+ const props = feature22.properties;
+ if (props.level)
+ return;
+ if (!props.country) {
+ props.level = "country";
+ } else if (!props.iso1A2 || props.isoStatus === "official") {
+ props.level = "territory";
+ } else {
+ props.level = "subterritory";
+ }
+ }
+ function loadGroupGroups(feature22) {
+ const props = feature22.properties;
+ if (feature22.geometry || !props.members)
+ return;
+ const featureLevelIndex = levels.indexOf(props.level);
+ let sharedGroups = [];
+ props.members.forEach((memberID, index) => {
+ const member = featuresByCode[memberID];
+ const memberGroups = member.properties.groups.filter((groupID) => {
+ return groupID !== feature22.properties.id && featureLevelIndex < levels.indexOf(featuresByCode[groupID].properties.level);
+ });
+ if (index === 0) {
+ sharedGroups = memberGroups;
+ } else {
+ sharedGroups = sharedGroups.filter((groupID) => memberGroups.indexOf(groupID) !== -1);
+ }
+ });
+ props.groups = props.groups.concat(
+ sharedGroups.filter((groupID) => props.groups.indexOf(groupID) === -1)
+ );
+ sharedGroups.forEach((groupID) => {
+ const groupFeature = featuresByCode[groupID];
+ if (groupFeature.properties.members.indexOf(props.id) === -1) {
+ groupFeature.properties.members.push(props.id);
+ }
+ });
+ }
+ function loadRoadSpeedUnit(feature22) {
+ const props = feature22.properties;
+ if (feature22.geometry) {
+ if (!props.roadSpeedUnit)
+ props.roadSpeedUnit = "km/h";
+ } else if (props.members) {
+ const vals = Array.from(
+ new Set(
+ props.members.map((id2) => {
+ const member = featuresByCode[id2];
+ if (member.geometry)
+ return member.properties.roadSpeedUnit || "km/h";
+ }).filter(Boolean)
+ )
+ );
+ if (vals.length === 1)
+ props.roadSpeedUnit = vals[0];
+ }
+ }
+ function loadRoadHeightUnit(feature22) {
+ const props = feature22.properties;
+ if (feature22.geometry) {
+ if (!props.roadHeightUnit)
+ props.roadHeightUnit = "m";
+ } else if (props.members) {
+ const vals = Array.from(
+ new Set(
+ props.members.map((id2) => {
+ const member = featuresByCode[id2];
+ if (member.geometry)
+ return member.properties.roadHeightUnit || "m";
+ }).filter(Boolean)
+ )
+ );
+ if (vals.length === 1)
+ props.roadHeightUnit = vals[0];
+ }
+ }
+ function loadDriveSide(feature22) {
+ const props = feature22.properties;
+ if (feature22.geometry) {
+ if (!props.driveSide)
+ props.driveSide = "right";
+ } else if (props.members) {
+ const vals = Array.from(
+ new Set(
+ props.members.map((id2) => {
+ const member = featuresByCode[id2];
+ if (member.geometry)
+ return member.properties.driveSide || "right";
+ }).filter(Boolean)
+ )
+ );
+ if (vals.length === 1)
+ props.driveSide = vals[0];
+ }
+ }
+ function loadCallingCodes(feature22) {
+ const props = feature22.properties;
+ if (!feature22.geometry && props.members) {
+ props.callingCodes = Array.from(
+ new Set(
+ props.members.reduce((array2, id2) => {
+ const member = featuresByCode[id2];
+ if (member.geometry && member.properties.callingCodes) {
+ return array2.concat(member.properties.callingCodes);
+ }
+ return array2;
+ }, [])
+ )
+ );
+ }
+ }
+ function loadFlag(feature22) {
+ if (!feature22.properties.iso1A2)
+ return;
+ const flag = feature22.properties.iso1A2.replace(/./g, function(char) {
+ return String.fromCodePoint(char.charCodeAt(0) + 127397);
+ });
+ feature22.properties.emojiFlag = flag;
+ }
+ function loadMembersForGroupsOf(feature22) {
+ feature22.properties.groups.forEach((groupID) => {
+ const groupFeature = featuresByCode[groupID];
+ if (!groupFeature.properties.members) {
+ groupFeature.properties.members = [];
+ }
+ groupFeature.properties.members.push(feature22.properties.id);
+ });
+ }
+ function cacheFeatureByIDs(feature22) {
+ let ids = [];
+ identifierProps.forEach((prop) => {
+ const id2 = feature22.properties[prop];
+ if (id2)
+ ids.push(id2);
+ });
+ (feature22.properties.aliases || []).forEach((alias) => {
+ ids.push(alias);
+ });
+ ids.forEach((id2) => {
+ let cid = canonicalID(id2);
+ featuresByCode[cid] = feature22;
+ });
+ }
+ }
+ function locArray(loc) {
+ if (Array.isArray(loc)) {
+ return loc;
+ } else if (loc.coordinates) {
+ return loc.coordinates;
+ }
+ return loc.geometry.coordinates;
+ }
+ function smallestFeature(loc) {
+ const query = locArray(loc);
+ const featureProperties = whichPolygonGetter(query);
+ if (!featureProperties)
+ return null;
+ return featuresByCode[featureProperties.id];
+ }
+ function countryFeature(loc) {
+ const feature22 = smallestFeature(loc);
+ if (!feature22)
+ return null;
+ const countryCode = feature22.properties.country || feature22.properties.iso1A2;
+ return featuresByCode[countryCode] || null;
+ }
+ var defaultOpts = {
+ level: void 0,
+ maxLevel: void 0,
+ withProp: void 0
+ };
+ function featureForLoc(loc, opts) {
+ const targetLevel = opts.level || "country";
+ const maxLevel = opts.maxLevel || "world";
+ const withProp = opts.withProp;
+ const targetLevelIndex = levels.indexOf(targetLevel);
+ if (targetLevelIndex === -1)
+ return null;
+ const maxLevelIndex = levels.indexOf(maxLevel);
+ if (maxLevelIndex === -1)
+ return null;
+ if (maxLevelIndex < targetLevelIndex)
+ return null;
+ if (targetLevel === "country") {
+ const fastFeature = countryFeature(loc);
+ if (fastFeature) {
+ if (!withProp || fastFeature.properties[withProp]) {
+ return fastFeature;
+ }
+ }
+ }
+ const features = featuresContaining(loc);
+ const match = features.find((feature22) => {
+ let levelIndex = levels.indexOf(feature22.properties.level);
+ if (feature22.properties.level === targetLevel || levelIndex > targetLevelIndex && levelIndex <= maxLevelIndex) {
+ if (!withProp || feature22.properties[withProp]) {
+ return feature22;
+ }
+ }
+ return false;
+ });
+ return match || null;
+ }
+ function featureForID(id2) {
+ let stringID;
+ if (typeof id2 === "number") {
+ stringID = id2.toString();
+ if (stringID.length === 1) {
+ stringID = "00" + stringID;
+ } else if (stringID.length === 2) {
+ stringID = "0" + stringID;
+ }
+ } else {
+ stringID = canonicalID(id2);
+ }
+ return featuresByCode[stringID] || null;
+ }
+ function smallestFeaturesForBbox(bbox2) {
+ return whichPolygonGetter.bbox(bbox2).map((props) => featuresByCode[props.id]);
+ }
+ function smallestOrMatchingFeature(query) {
+ if (typeof query === "object") {
+ return smallestFeature(query);
+ }
+ return featureForID(query);
+ }
+ function feature(query, opts = defaultOpts) {
+ if (typeof query === "object") {
+ return featureForLoc(query, opts);
+ }
+ return featureForID(query);
+ }
+ function iso1A2Code(query, opts = defaultOpts) {
+ opts.withProp = "iso1A2";
+ const match = feature(query, opts);
+ if (!match)
+ return null;
+ return match.properties.iso1A2 || null;
+ }
+ function featuresContaining(query, strict) {
+ let matchingFeatures;
+ if (Array.isArray(query) && query.length === 4) {
+ matchingFeatures = smallestFeaturesForBbox(query);
+ } else {
+ const smallestOrMatching = smallestOrMatchingFeature(query);
+ matchingFeatures = smallestOrMatching ? [smallestOrMatching] : [];
+ }
+ if (!matchingFeatures.length)
+ return [];
+ let returnFeatures;
+ if (!strict || typeof query === "object") {
+ returnFeatures = matchingFeatures.slice();
+ } else {
+ returnFeatures = [];
+ }
+ matchingFeatures.forEach((feature22) => {
+ const properties = feature22.properties;
+ properties.groups.forEach((groupID) => {
+ const groupFeature = featuresByCode[groupID];
+ if (returnFeatures.indexOf(groupFeature) === -1) {
+ returnFeatures.push(groupFeature);
+ }
+ });
+ });
+ return returnFeatures;
+ }
+ function featuresIn(id2, strict) {
+ const feature22 = featureForID(id2);
+ if (!feature22)
+ return [];
+ let features = [];
+ if (!strict) {
+ features.push(feature22);
+ }
+ const properties = feature22.properties;
+ (properties.members || []).forEach((memberID) => {
+ features.push(featuresByCode[memberID]);
+ });
+ return features;
+ }
+ function aggregateFeature(id2) {
+ const features = featuresIn(id2, false);
+ if (features.length === 0)
+ return null;
+ let aggregateCoordinates = [];
+ features.forEach((feature22) => {
+ if (feature22.geometry && feature22.geometry.type === "MultiPolygon" && feature22.geometry.coordinates) {
+ aggregateCoordinates = aggregateCoordinates.concat(feature22.geometry.coordinates);
+ }
+ });
+ return {
+ type: "Feature",
+ properties: features[0].properties,
+ geometry: {
+ type: "MultiPolygon",
+ coordinates: aggregateCoordinates
+ }
+ };
+ }
+ function roadSpeedUnit(query) {
+ const feature22 = smallestOrMatchingFeature(query);
+ return feature22 && feature22.properties.roadSpeedUnit || null;
+ }
+ function roadHeightUnit(query) {
+ const feature22 = smallestOrMatchingFeature(query);
+ return feature22 && feature22.properties.roadHeightUnit || null;
+ }
+
+ // node_modules/@ideditor/location-conflation/index.mjs
+ var import_geojson_area = __toESM(require_geojson_area(), 1);
+ var import_circle_to_polygon = __toESM(require_circle_to_polygon(), 1);
+ var import_polygon_clipping = __toESM(require_polygon_clipping_umd(), 1);
+ var import_geojson_precision = __toESM(require_geojson_precision(), 1);
+ var import_json_stringify_pretty_compact = __toESM(require_json_stringify_pretty_compact(), 1);
+ var location_conflation_default = class {
+ // 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": { … }
+ // }
+ // ]
+ // }
+ constructor(fc) {
+ this._cache = {};
+ this._strict = true;
+ if (fc && fc.type === "FeatureCollection" && Array.isArray(fc.features)) {
+ fc.features.forEach((feature3) => {
+ feature3.properties = feature3.properties || {};
+ let props = feature3.properties;
+ let id2 = feature3.id || props.id;
+ if (!id2 || !/^\S+\.geojson$/i.test(id2))
+ return;
+ id2 = id2.toLowerCase();
+ feature3.id = id2;
+ props.id = id2;
+ if (!props.area) {
+ const area = import_geojson_area.default.geometry(feature3.geometry) / 1e6;
+ props.area = Number(area.toFixed(2));
+ }
+ this._cache[id2] = feature3;
+ });
+ }
+ let world = _cloneDeep(feature("Q2"));
+ 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 = import_geojson_area.default.geometry(world.geometry) / 1e6;
+ 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
+ //
+ validateLocation(location) {
+ if (Array.isArray(location) && (location.length === 2 || location.length === 3)) {
+ const lon = location[0];
+ const lat = location[1];
+ const radius = location[2];
+ if (Number.isFinite(lon) && lon >= -180 && lon <= 180 && Number.isFinite(lat) && lat >= -90 && lat <= 90 && (location.length === 2 || Number.isFinite(radius) && radius > 0)) {
+ const id2 = "[" + location.toString() + "]";
+ return { type: "point", location, id: id2 };
+ }
+ } else if (typeof location === "string" && /^\S+\.geojson$/i.test(location)) {
+ const id2 = location.toLowerCase();
+ if (this._cache[id2]) {
+ return { type: "geojson", location, id: id2 };
+ }
+ } else if (typeof location === "string" || typeof location === "number") {
+ const feature3 = feature(location);
+ if (feature3) {
+ const id2 = feature3.properties.wikidata;
+ return { type: "countrycoder", location, id: id2 };
+ }
+ }
+ if (this._strict) {
+ throw new Error(`validateLocation: Invalid location: "${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
+ //
+ resolveLocation(location) {
+ const valid = this.validateLocation(location);
+ if (!valid)
+ return null;
+ const id2 = valid.id;
+ if (this._cache[id2]) {
+ return Object.assign(valid, { feature: this._cache[id2] });
+ }
+ if (valid.type === "point") {
+ const lon = location[0];
+ const lat = location[1];
+ const radius = location[2] || 25;
+ const EDGES = 10;
+ const PRECISION = 3;
+ const area = Math.PI * radius * radius;
+ const feature3 = this._cache[id2] = (0, import_geojson_precision.default)({
+ type: "Feature",
+ id: id2,
+ properties: { id: id2, area: Number(area.toFixed(2)) },
+ geometry: (0, import_circle_to_polygon.default)([lon, lat], radius * 1e3, EDGES)
+ // km to m
+ }, PRECISION);
+ return Object.assign(valid, { feature: feature3 });
+ } else if (valid.type === "geojson") {
+ } else if (valid.type === "countrycoder") {
+ let feature3 = _cloneDeep(feature(id2));
+ let props = feature3.properties;
+ if (Array.isArray(props.members)) {
+ let aggregate = aggregateFeature(id2);
+ aggregate.geometry.coordinates = _clip([aggregate], "UNION").geometry.coordinates;
+ feature3.geometry = aggregate.geometry;
+ }
+ if (!props.area) {
+ const area = import_geojson_area.default.geometry(feature3.geometry) / 1e6;
+ props.area = Number(area.toFixed(2));
+ }
+ feature3.id = id2;
+ props.id = id2;
+ this._cache[id2] = feature3;
+ return Object.assign(valid, { feature: feature3 });
+ }
+ if (this._strict) {
+ throw new Error(`resolveLocation: Couldn't resolve location "${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
+ //
+ validateLocationSet(locationSet) {
+ locationSet = locationSet || {};
+ const validator = this.validateLocation.bind(this);
+ let include = (locationSet.include || []).map(validator).filter(Boolean);
+ let exclude = (locationSet.exclude || []).map(validator).filter(Boolean);
+ if (!include.length) {
+ if (this._strict) {
+ throw new Error(`validateLocationSet: LocationSet includes nothing.`);
+ } else {
+ locationSet.include = ["Q2"];
+ include = [{ type: "countrycoder", location: "Q2", id: "Q2" }];
+ }
+ }
+ include.sort(_sortLocations);
+ let id2 = "+[" + include.map((d) => d.id).join(",") + "]";
+ if (exclude.length) {
+ exclude.sort(_sortLocations);
+ id2 += "-[" + exclude.map((d) => d.id).join(",") + "]";
+ }
+ return { type: "locationset", locationSet, id: id2 };
+ }
+ // 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
+ //
+ resolveLocationSet(locationSet) {
+ locationSet = locationSet || {};
+ const valid = this.validateLocationSet(locationSet);
+ if (!valid)
+ return null;
+ const id2 = valid.id;
+ if (this._cache[id2]) {
+ return Object.assign(valid, { feature: this._cache[id2] });
+ }
+ const resolver = this.resolveLocation.bind(this);
+ const includes = (locationSet.include || []).map(resolver).filter(Boolean);
+ const excludes = (locationSet.exclude || []).map(resolver).filter(Boolean);
+ if (includes.length === 1 && excludes.length === 0) {
+ return Object.assign(valid, { feature: includes[0].feature });
+ }
+ const includeGeoJSON = _clip(includes.map((d) => d.feature), "UNION");
+ const excludeGeoJSON = _clip(excludes.map((d) => d.feature), "UNION");
+ let resultGeoJSON = excludeGeoJSON ? _clip([includeGeoJSON, excludeGeoJSON], "DIFFERENCE") : includeGeoJSON;
+ const area = import_geojson_area.default.geometry(resultGeoJSON.geometry) / 1e6;
+ resultGeoJSON.id = id2;
+ resultGeoJSON.properties = { id: id2, area: Number(area.toFixed(2)) };
+ this._cache[id2] = resultGeoJSON;
+ return Object.assign(valid, { feature: resultGeoJSON });
+ }
+ // strict
+ //
+ strict(val) {
+ if (val === void 0) {
+ return this._strict;
+ } else {
+ this._strict = val;
+ return this;
+ }
+ }
+ // cache
+ // convenience method to access the internal cache
+ cache() {
+ return this._cache;
+ }
+ // stringify
+ // convenience method to prettyStringify the given object
+ stringify(obj, options2) {
+ return (0, import_json_stringify_pretty_compact.default)(obj, options2);
+ }
+ };
+ function _clip(features, which) {
+ if (!Array.isArray(features) || !features.length)
+ return null;
+ const fn = { UNION: import_polygon_clipping.default.union, DIFFERENCE: import_polygon_clipping.default.difference }[which];
+ const args = features.map((feature3) => feature3.geometry.coordinates);
+ const coords = fn.apply(null, args);
+ return {
+ type: "Feature",
+ properties: {},
+ geometry: {
+ type: whichType(coords),
+ coordinates: coords
+ }
+ };
+ function whichType(coords2) {
+ const a = Array.isArray(coords2);
+ const b = a && Array.isArray(coords2[0]);
+ const c = b && Array.isArray(coords2[0][0]);
+ const d = c && Array.isArray(coords2[0][0][0]);
+ return d ? "MultiPolygon" : "Polygon";
+ }
+ }
+ function _cloneDeep(obj) {
+ return JSON.parse(JSON.stringify(obj));
+ }
+ function _sortLocations(a, b) {
+ const rank = { countrycoder: 1, geojson: 2, point: 3 };
+ const aRank = rank[a.type];
+ const bRank = rank[b.type];
+ return aRank > bRank ? 1 : aRank < bRank ? -1 : a.id.localeCompare(b.id);
+ }
+
+ // modules/core/LocationManager.js
+ var import_which_polygon2 = __toESM(require_which_polygon());
+ var import_geojson_area2 = __toESM(require_geojson_area());
+ var _loco = new location_conflation_default();
+ var LocationManager = class {
+ /**
+ * @constructor
+ */
+ constructor() {
+ this._wp = null;
+ this._resolved = /* @__PURE__ */ new Map();
+ this._knownLocationSets = /* @__PURE__ */ new Map();
+ this._locationIncludedIn = /* @__PURE__ */ new Map();
+ this._locationExcludedIn = /* @__PURE__ */ new Map();
+ const world = { locationSet: { include: ["Q2"] } };
+ this._resolveLocationSet(world);
+ this._rebuildIndex();
+ }
+ /**
+ * _validateLocationSet
+ * Pass an Object with a `locationSet` property.
+ * Validates the `locationSet` and sets a `locationSetID` property on the object.
+ * To avoid so much computation we only resolve the include and exclude regions, but not the locationSet itself.
+ *
+ * Use `_resolveLocationSet()` instead if you need to resolve geojson of locationSet, for example to render it.
+ * Note: You need to call `_rebuildIndex()` after you're all finished validating the locationSets.
+ *
+ * @param `obj` Object to check, it should have `locationSet` property
+ */
+ _validateLocationSet(obj) {
+ if (obj.locationSetID)
+ return;
+ try {
+ let locationSet = obj.locationSet;
+ if (!locationSet) {
+ throw new Error("object missing locationSet property");
+ }
+ if (!locationSet.include) {
+ locationSet.include = ["Q2"];
+ }
+ const locationSetID = _loco.validateLocationSet(locationSet).id;
+ obj.locationSetID = locationSetID;
+ if (this._knownLocationSets.has(locationSetID))
+ return;
+ let area = 0;
+ (locationSet.include || []).forEach((location) => {
+ const locationID = _loco.validateLocation(location).id;
+ let geojson = this._resolved.get(locationID);
+ if (!geojson) {
+ geojson = _loco.resolveLocation(location).feature;
+ this._resolved.set(locationID, geojson);
+ }
+ area += geojson.properties.area;
+ let s = this._locationIncludedIn.get(locationID);
+ if (!s) {
+ s = /* @__PURE__ */ new Set();
+ this._locationIncludedIn.set(locationID, s);
+ }
+ s.add(locationSetID);
+ });
+ (locationSet.exclude || []).forEach((location) => {
+ const locationID = _loco.validateLocation(location).id;
+ let geojson = this._resolved.get(locationID);
+ if (!geojson) {
+ geojson = _loco.resolveLocation(location).feature;
+ this._resolved.set(locationID, geojson);
+ }
+ area -= geojson.properties.area;
+ let s = this._locationExcludedIn.get(locationID);
+ if (!s) {
+ s = /* @__PURE__ */ new Set();
+ this._locationExcludedIn.set(locationID, s);
+ }
+ s.add(locationSetID);
+ });
+ this._knownLocationSets.set(locationSetID, area);
+ } catch (err) {
+ obj.locationSet = { include: ["Q2"] };
+ obj.locationSetID = "+[Q2]";
+ }
+ }
+ /**
+ * _resolveLocationSet
+ * Does everything that `_validateLocationSet()` does, but then "resolves" the locationSet into GeoJSON.
+ * This step is a bit more computationally expensive, so really only needed if you intend to render the shape.
+ *
+ * Note: You need to call `_rebuildIndex()` after you're all finished validating the locationSets.
+ *
+ * @param `obj` Object to check, it should have `locationSet` property
+ */
+ _resolveLocationSet(obj) {
+ this._validateLocationSet(obj);
+ if (this._resolved.has(obj.locationSetID))
+ return;
+ try {
+ const result = _loco.resolveLocationSet(obj.locationSet);
+ const locationSetID = result.id;
+ obj.locationSetID = locationSetID;
+ if (!result.feature.geometry.coordinates.length || !result.feature.properties.area) {
+ throw new Error(`locationSet ${locationSetID} resolves to an empty feature.`);
+ }
+ let geojson = JSON.parse(JSON.stringify(result.feature));
+ geojson.id = locationSetID;
+ geojson.properties.id = locationSetID;
+ this._resolved.set(locationSetID, geojson);
+ } catch (err) {
+ obj.locationSet = { include: ["Q2"] };
+ obj.locationSetID = "+[Q2]";
+ }
+ }
+ /**
+ * _rebuildIndex
+ * Rebuilds the whichPolygon index with whatever features have been resolved into GeoJSON.
+ */
+ _rebuildIndex() {
+ this._wp = (0, import_which_polygon2.default)({ features: [...this._resolved.values()] });
+ }
+ /**
+ * mergeCustomGeoJSON
+ * Accepts a 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": { … }
+ * }
+ * ]
+ * }
+ *
+ * @param `fc` FeatureCollection-like Object containing custom locations
+ */
+ mergeCustomGeoJSON(fc) {
+ if (!fc || fc.type !== "FeatureCollection" || !Array.isArray(fc.features))
+ return;
+ fc.features.forEach((feature3) => {
+ feature3.properties = feature3.properties || {};
+ let props = feature3.properties;
+ let id2 = feature3.id || props.id;
+ if (!id2 || !/^\S+\.geojson$/i.test(id2))
+ return;
+ id2 = id2.toLowerCase();
+ feature3.id = id2;
+ props.id = id2;
+ if (!props.area) {
+ const area = import_geojson_area2.default.geometry(feature3.geometry) / 1e6;
+ props.area = Number(area.toFixed(2));
+ }
+ _loco._cache[id2] = feature3;
+ });
+ }
+ /**
+ * mergeLocationSets
+ * Accepts an Array of Objects containing `locationSet` properties:
+ * [
+ * { id: 'preset1', locationSet: {…} },
+ * { id: 'preset2', locationSet: {…} },
+ * …
+ * ]
+ * After validating, the Objects will be decorated with a `locationSetID` property:
+ * [
+ * { id: 'preset1', locationSet: {…}, locationSetID: '+[Q2]' },
+ * { id: 'preset2', locationSet: {…}, locationSetID: '+[Q30]' },
+ * …
+ * ]
+ *
+ * @param `objects` Objects to check - they should have `locationSet` property
+ * @return Promise resolved true (this function used to be slow/async, now it's faster and sync)
+ */
+ mergeLocationSets(objects) {
+ if (!Array.isArray(objects))
+ return Promise.reject("nothing to do");
+ objects.forEach((obj) => this._validateLocationSet(obj));
+ this._rebuildIndex();
+ return Promise.resolve(objects);
+ }
+ /**
+ * 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`)
+ *
+ * @param `locationSet` A locationSet Object, e.g. `{ include: ['us'] }`
+ * @return String locationSetID, e.g. `+[Q30]`
+ */
+ locationSetID(locationSet) {
+ let locationSetID;
+ try {
+ locationSetID = _loco.validateLocationSet(locationSet).id;
+ } catch (err) {
+ locationSetID = "+[Q2]";
+ }
+ return locationSetID;
+ }
+ /**
+ * feature
+ * Returns the resolved GeoJSON feature for a given locationSetID (fallback to 'world')
+ * A GeoJSON feature:
+ * {
+ * type: 'Feature',
+ * id: '+[Q30]',
+ * properties: { id: '+[Q30]', area: 21817019.17, … },
+ * geometry: { … }
+ * }
+ *
+ * @param `locationSetID` String identifier, e.g. `+[Q30]`
+ * @return GeoJSON object (fallback to world)
+ */
+ feature(locationSetID = "+[Q2]") {
+ const feature3 = this._resolved.get(locationSetID);
+ return feature3 || this._resolved.get("+[Q2]");
+ }
+ /**
+ * locationSetsAt
+ * Find all the locationSets valid at the given location.
+ * Results include the area (in km²) to facilitate sorting.
+ *
+ * Object of locationSetIDs to areas (in km²)
+ * {
+ * "+[Q2]": 511207893.3958111,
+ * "+[Q30]": 21817019.17,
+ * "+[new_jersey.geojson]": 22390.77,
+ * …
+ * }
+ *
+ * @param `loc` `[lon,lat]` location to query, e.g. `[-74.4813, 40.7967]`
+ * @return Object of locationSetIDs valid at given location
+ */
+ locationSetsAt(loc) {
+ let result = {};
+ const hits = this._wp(loc, true) || [];
+ const thiz = this;
+ hits.forEach((prop) => {
+ if (prop.id[0] !== "+")
+ return;
+ const locationSetID = prop.id;
+ const area = thiz._knownLocationSets.get(locationSetID);
+ if (area) {
+ result[locationSetID] = area;
+ }
+ });
+ hits.forEach((prop) => {
+ if (prop.id[0] === "+")
+ return;
+ const locationID = prop.id;
+ const included = thiz._locationIncludedIn.get(locationID);
+ (included || []).forEach((locationSetID) => {
+ const area = thiz._knownLocationSets.get(locationSetID);
+ if (area) {
+ result[locationSetID] = area;
+ }
+ });
+ });
+ hits.forEach((prop) => {
+ if (prop.id[0] === "+")
+ return;
+ const locationID = prop.id;
+ const excluded = thiz._locationExcludedIn.get(locationID);
+ (excluded || []).forEach((locationSetID) => {
+ delete result[locationSetID];
+ });
+ });
+ return result;
+ }
+ // Direct access to the location-conflation resolver
+ loco() {
+ return _loco;
+ }
+ };
+ var _sharedLocationManager = new LocationManager();
+
+ // node_modules/lodash-es/_freeGlobal.js
+ var freeGlobal = typeof global == "object" && global && global.Object === Object && global;
+ var freeGlobal_default = freeGlobal;
+
+ // node_modules/lodash-es/_root.js
+ var freeSelf = typeof self == "object" && self && self.Object === Object && self;
+ var root2 = freeGlobal_default || freeSelf || Function("return this")();
+ var root_default = root2;
+
+ // node_modules/lodash-es/_Symbol.js
+ var Symbol2 = root_default.Symbol;
+ var Symbol_default = Symbol2;
+
+ // node_modules/lodash-es/_getRawTag.js
+ var objectProto = Object.prototype;
+ var hasOwnProperty = objectProto.hasOwnProperty;
+ var nativeObjectToString = objectProto.toString;
+ var symToStringTag = Symbol_default ? Symbol_default.toStringTag : void 0;
+ function getRawTag(value) {
+ var isOwn = hasOwnProperty.call(value, symToStringTag), tag = value[symToStringTag];
+ try {
+ value[symToStringTag] = void 0;
+ var unmasked = true;
+ } catch (e) {
+ }
+ var result = nativeObjectToString.call(value);
+ if (unmasked) {
+ if (isOwn) {
+ value[symToStringTag] = tag;
+ } else {
+ delete value[symToStringTag];
+ }
+ }
+ return result;
+ }
+ var getRawTag_default = getRawTag;
+
+ // node_modules/lodash-es/_objectToString.js
+ var objectProto2 = Object.prototype;
+ var nativeObjectToString2 = objectProto2.toString;
+ function objectToString(value) {
+ return nativeObjectToString2.call(value);
+ }
+ var objectToString_default = objectToString;
+
+ // node_modules/lodash-es/_baseGetTag.js
+ var nullTag = "[object Null]";
+ var undefinedTag = "[object Undefined]";
+ var symToStringTag2 = Symbol_default ? Symbol_default.toStringTag : void 0;
+ function baseGetTag(value) {
+ if (value == null) {
+ return value === void 0 ? undefinedTag : nullTag;
+ }
+ return symToStringTag2 && symToStringTag2 in Object(value) ? getRawTag_default(value) : objectToString_default(value);
+ }
+ var baseGetTag_default = baseGetTag;
+
+ // node_modules/lodash-es/isObjectLike.js
+ function isObjectLike(value) {
+ return value != null && typeof value == "object";
+ }
+ var isObjectLike_default = isObjectLike;
+
+ // node_modules/lodash-es/isSymbol.js
+ var symbolTag = "[object Symbol]";
+ function isSymbol(value) {
+ return typeof value == "symbol" || isObjectLike_default(value) && baseGetTag_default(value) == symbolTag;
+ }
+ var isSymbol_default = isSymbol;
+
+ // node_modules/lodash-es/_arrayMap.js
+ function arrayMap(array2, iteratee) {
+ var index = -1, length = array2 == null ? 0 : array2.length, result = Array(length);
+ while (++index < length) {
+ result[index] = iteratee(array2[index], index, array2);
+ }
+ return result;
+ }
+ var arrayMap_default = arrayMap;
+
+ // node_modules/lodash-es/isArray.js
+ var isArray = Array.isArray;
+ var isArray_default = isArray;
+
+ // node_modules/lodash-es/_baseToString.js
+ var INFINITY = 1 / 0;
+ var symbolProto = Symbol_default ? Symbol_default.prototype : void 0;
+ var symbolToString = symbolProto ? symbolProto.toString : void 0;
+ function baseToString(value) {
+ if (typeof value == "string") {
+ return value;
+ }
+ if (isArray_default(value)) {
+ return arrayMap_default(value, baseToString) + "";
+ }
+ if (isSymbol_default(value)) {
+ return symbolToString ? symbolToString.call(value) : "";
+ }
+ var result = value + "";
+ return result == "0" && 1 / value == -INFINITY ? "-0" : result;
+ }
+ var baseToString_default = baseToString;
+
+ // node_modules/lodash-es/_trimmedEndIndex.js
+ var reWhitespace = /\s/;
+ function trimmedEndIndex(string) {
+ var index = string.length;
+ while (index-- && reWhitespace.test(string.charAt(index))) {
+ }
+ return index;
+ }
+ var trimmedEndIndex_default = trimmedEndIndex;
+
+ // node_modules/lodash-es/_baseTrim.js
+ var reTrimStart = /^\s+/;
+ function baseTrim(string) {
+ return string ? string.slice(0, trimmedEndIndex_default(string) + 1).replace(reTrimStart, "") : string;
+ }
+ var baseTrim_default = baseTrim;
+
+ // node_modules/lodash-es/isObject.js
+ function isObject(value) {
+ var type2 = typeof value;
+ return value != null && (type2 == "object" || type2 == "function");
+ }
+ var isObject_default = isObject;
+
+ // node_modules/lodash-es/toNumber.js
+ var NAN = 0 / 0;
+ var reIsBadHex = /^[-+]0x[0-9a-f]+$/i;
+ var reIsBinary = /^0b[01]+$/i;
+ var reIsOctal = /^0o[0-7]+$/i;
+ var freeParseInt = parseInt;
+ function toNumber(value) {
+ if (typeof value == "number") {
+ return value;
+ }
+ if (isSymbol_default(value)) {
+ return NAN;
+ }
+ if (isObject_default(value)) {
+ var other = typeof value.valueOf == "function" ? value.valueOf() : value;
+ value = isObject_default(other) ? other + "" : other;
+ }
+ if (typeof value != "string") {
+ return value === 0 ? value : +value;
+ }
+ value = baseTrim_default(value);
+ var isBinary = reIsBinary.test(value);
+ return isBinary || reIsOctal.test(value) ? freeParseInt(value.slice(2), isBinary ? 2 : 8) : reIsBadHex.test(value) ? NAN : +value;
+ }
+ var toNumber_default = toNumber;
+
+ // node_modules/lodash-es/toString.js
+ function toString(value) {
+ return value == null ? "" : baseToString_default(value);
+ }
+ var toString_default = toString;
+
+ // node_modules/lodash-es/_basePropertyOf.js
+ function basePropertyOf(object) {
+ return function(key) {
+ return object == null ? void 0 : object[key];
+ };
+ }
+ var basePropertyOf_default = basePropertyOf;
+
+ // node_modules/lodash-es/now.js
+ var now2 = function() {
+ return root_default.Date.now();
+ };
+ var now_default = now2;
+
+ // node_modules/lodash-es/debounce.js
+ var FUNC_ERROR_TEXT = "Expected a function";
+ var nativeMax = Math.max;
+ var nativeMin = Math.min;
+ function debounce(func, wait, options2) {
+ var lastArgs, lastThis, maxWait, result, timerId, lastCallTime, lastInvokeTime = 0, leading = false, maxing = false, trailing = true;
+ if (typeof func != "function") {
+ throw new TypeError(FUNC_ERROR_TEXT);
+ }
+ wait = toNumber_default(wait) || 0;
+ if (isObject_default(options2)) {
+ leading = !!options2.leading;
+ maxing = "maxWait" in options2;
+ maxWait = maxing ? nativeMax(toNumber_default(options2.maxWait) || 0, wait) : maxWait;
+ trailing = "trailing" in options2 ? !!options2.trailing : trailing;
+ }
+ function invokeFunc(time) {
+ var args = lastArgs, thisArg = lastThis;
+ lastArgs = lastThis = void 0;
+ lastInvokeTime = time;
+ result = func.apply(thisArg, args);
+ return result;
+ }
+ function leadingEdge(time) {
+ lastInvokeTime = time;
+ timerId = setTimeout(timerExpired, wait);
+ 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;
+ }
+ function shouldInvoke(time) {
+ var timeSinceLastCall = time - lastCallTime, timeSinceLastInvoke = time - lastInvokeTime;
+ return lastCallTime === void 0 || timeSinceLastCall >= wait || timeSinceLastCall < 0 || maxing && timeSinceLastInvoke >= maxWait;
+ }
+ function timerExpired() {
+ var time = now_default();
+ if (shouldInvoke(time)) {
+ return trailingEdge(time);
+ }
+ timerId = setTimeout(timerExpired, remainingWait(time));
+ }
+ function trailingEdge(time) {
+ timerId = void 0;
+ if (trailing && lastArgs) {
+ return invokeFunc(time);
+ }
+ lastArgs = lastThis = void 0;
+ return result;
+ }
+ function cancel() {
+ if (timerId !== void 0) {
+ clearTimeout(timerId);
+ }
+ lastInvokeTime = 0;
+ lastArgs = lastCallTime = lastThis = timerId = void 0;
+ }
+ function flush() {
+ return timerId === void 0 ? result : trailingEdge(now_default());
+ }
+ function debounced() {
+ var time = now_default(), isInvoking = shouldInvoke(time);
+ lastArgs = arguments;
+ lastThis = this;
+ lastCallTime = time;
+ if (isInvoking) {
+ if (timerId === void 0) {
+ return leadingEdge(lastCallTime);
+ }
+ if (maxing) {
+ clearTimeout(timerId);
+ timerId = setTimeout(timerExpired, wait);
+ return invokeFunc(lastCallTime);
+ }
+ }
+ if (timerId === void 0) {
+ timerId = setTimeout(timerExpired, wait);
+ }
+ return result;
+ }
+ debounced.cancel = cancel;
+ debounced.flush = flush;
+ return debounced;
+ }
+ var debounce_default = debounce;
+
+ // node_modules/lodash-es/_escapeHtmlChar.js
+ var htmlEscapes = {
+ "&": "&",
+ "<": "<",
+ ">": ">",
+ '"': """,
+ "'": "'"
+ };
+ var escapeHtmlChar = basePropertyOf_default(htmlEscapes);
+ var escapeHtmlChar_default = escapeHtmlChar;
+
+ // node_modules/lodash-es/escape.js
+ var reUnescapedHtml = /[&<>"']/g;
+ var reHasUnescapedHtml = RegExp(reUnescapedHtml.source);
+ function escape2(string) {
+ string = toString_default(string);
+ return string && reHasUnescapedHtml.test(string) ? string.replace(reUnescapedHtml, escapeHtmlChar_default) : string;
+ }
+ var escape_default = escape2;
+
+ // node_modules/lodash-es/throttle.js
+ var FUNC_ERROR_TEXT2 = "Expected a function";
+ function throttle(func, wait, options2) {
+ var leading = true, trailing = true;
+ if (typeof func != "function") {
+ throw new TypeError(FUNC_ERROR_TEXT2);
+ }
+ if (isObject_default(options2)) {
+ leading = "leading" in options2 ? !!options2.leading : leading;
+ trailing = "trailing" in options2 ? !!options2.trailing : trailing;
+ }
+ return debounce_default(func, wait, {
+ "leading": leading,
+ "maxWait": wait,
+ "trailing": trailing
+ });
+ }
+ var throttle_default = throttle;
+
+ // node_modules/lodash-es/_unescapeHtmlChar.js
+ var htmlUnescapes = {
+ "&": "&",
+ "<": "<",
+ ">": ">",
+ """: '"',
+ "'": "'"
+ };
+ var unescapeHtmlChar = basePropertyOf_default(htmlUnescapes);
+ var unescapeHtmlChar_default = unescapeHtmlChar;
+
+ // node_modules/lodash-es/unescape.js
+ var reEscapedHtml = /&(?:amp|lt|gt|quot|#39);/g;
+ var reHasEscapedHtml = RegExp(reEscapedHtml.source);
+ function unescape2(string) {
+ string = toString_default(string);
+ return string && reHasEscapedHtml.test(string) ? string.replace(reEscapedHtml, unescapeHtmlChar_default) : string;
+ }
+ var unescape_default = unescape2;
+
+ // modules/util/detect.js
+ var _detected;
+ function utilDetect(refresh2) {
+ if (_detected && !refresh2)
+ return _detected;
+ _detected = {};
+ const ua = navigator.userAgent;
+ let m = null;
+ m = ua.match(/(edge)\/?\s*(\.?\d+(\.\d+)*)/i);
+ 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);
+ if (m !== null) {
+ _detected.browser = "msie";
+ _detected.version = m[1];
+ }
+ }
+ if (!_detected.browser) {
+ m = ua.match(/(opr)\/?\s*(\.?\d+(\.\d+)*)/i);
+ 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;
+ }
+ _detected.version = _detected.version.split(/\W/).slice(0, 2).join(".");
+ _detected.opera = _detected.browser.toLowerCase() === "opera" && Number(_detected.version) < 15;
+ if (_detected.browser.toLowerCase() === "msie") {
+ _detected.ie = true;
+ _detected.browser = "Internet Explorer";
+ _detected.support = false;
+ } else {
+ _detected.ie = false;
+ _detected.support = true;
+ }
+ _detected.filedrop = window.FileReader && "ondrop" in window;
+ 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;
+ _detected.browserLocales = Array.from(new Set(
+ // remove duplicates
+ [navigator.language].concat(navigator.languages || []).concat([
+ // old property for backwards compatibility
+ navigator.userLanguage
+ ]).filter(Boolean)
+ ));
+ const loc = window.top.location;
+ let origin = loc.origin;
+ if (!origin) {
+ origin = loc.protocol + "//" + loc.hostname + (loc.port ? ":" + loc.port : "");
+ }
+ _detected.host = origin + loc.pathname;
+ return _detected;
+ }
+
+ // modules/util/aes.js
+ var import_aes_js = __toESM(require_aes_js());
+ var DEFAULT_128 = [250, 157, 60, 79, 142, 134, 229, 129, 138, 126, 210, 129, 29, 71, 160, 208];
+ function utilAesEncrypt(text2, key) {
+ key = key || DEFAULT_128;
+ const textBytes = import_aes_js.default.utils.utf8.toBytes(text2);
+ const aesCtr = new import_aes_js.default.ModeOfOperation.ctr(key);
+ const encryptedBytes = aesCtr.encrypt(textBytes);
+ const encryptedHex = import_aes_js.default.utils.hex.fromBytes(encryptedBytes);
+ return encryptedHex;
+ }
+ function utilAesDecrypt(encryptedHex, key) {
+ key = key || DEFAULT_128;
+ const encryptedBytes = import_aes_js.default.utils.hex.toBytes(encryptedHex);
+ const aesCtr = new import_aes_js.default.ModeOfOperation.ctr(key);
+ const decryptedBytes = aesCtr.decrypt(encryptedBytes);
+ const text2 = import_aes_js.default.utils.utf8.fromBytes(decryptedBytes);
+ return text2;
+ }
+
+ // modules/util/clean_tags.js
+ function utilCleanTags(tags) {
+ var out = {};
+ for (var k in tags) {
+ if (!k)
+ continue;
+ var v = tags[k];
+ if (v !== void 0) {
+ out[k] = cleanValue(k, v);
+ }
+ }
+ return out;
+ function cleanValue(k2, v2) {
+ function keepSpaces(k3) {
+ return /_hours|_times|:conditional$/.test(k3);
+ }
+ function skip(k3) {
+ return /^(description|note|fixme)$/.test(k3);
+ }
+ if (skip(k2))
+ return v2;
+ var cleaned = v2.split(";").map(function(s) {
+ return s.trim();
+ }).join(keepSpaces(k2) ? "; " : ";");
+ if (k2.indexOf("website") !== -1 || k2.indexOf("email") !== -1 || cleaned.indexOf("http") === 0) {
+ cleaned = cleaned.replace(/[\u200B-\u200F\uFEFF]/g, "");
+ }
+ return cleaned;
+ }
+ }
+
+ // modules/util/get_set_value.js
+ function utilGetSetValue(selection2, value) {
+ function d3_selection_value(value2) {
+ function valueNull() {
+ delete this.value;
+ }
+ function valueConstant() {
+ if (this.value !== value2) {
+ this.value = value2;
+ }
+ }
+ function valueFunction() {
+ var x = value2.apply(this, arguments);
+ if (x === null || x === void 0) {
+ delete this.value;
+ } else if (this.value !== x) {
+ this.value = x;
+ }
+ }
+ return value2 === null || value2 === void 0 ? valueNull : typeof value2 === "function" ? valueFunction : valueConstant;
+ }
+ if (arguments.length === 1) {
+ return selection2.property("value");
+ }
+ return selection2.each(d3_selection_value(value));
+ }
+
+ // modules/util/keybinding.js
+ function utilKeybinding(namespace) {
+ var _keybindings = {};
+ function testBindings(d3_event, isCapturing) {
+ var didMatch = false;
+ var bindings = Object.keys(_keybindings).map(function(id2) {
+ return _keybindings[id2];
+ });
+ var i2, binding;
+ for (i2 = 0; i2 < bindings.length; i2++) {
+ binding = bindings[i2];
+ if (!binding.event.modifiers.shiftKey)
+ continue;
+ if (!!binding.capture !== isCapturing)
+ continue;
+ if (matches(d3_event, binding, true)) {
+ binding.callback(d3_event);
+ didMatch = true;
+ break;
+ }
+ }
+ if (didMatch)
+ return;
+ for (i2 = 0; i2 < bindings.length; i2++) {
+ binding = bindings[i2];
+ if (binding.event.modifiers.shiftKey)
+ continue;
+ if (!!binding.capture !== isCapturing)
+ continue;
+ if (matches(d3_event, binding, false)) {
+ binding.callback(d3_event);
+ break;
+ }
+ }
+ function matches(d3_event2, binding2, testShift) {
+ var event = d3_event2;
+ var isMatch = false;
+ var tryKeyCode = true;
+ if (event.key !== void 0) {
+ tryKeyCode = event.key.charCodeAt(0) > 127;
+ isMatch = true;
+ if (binding2.event.key === void 0) {
+ isMatch = false;
+ } else if (Array.isArray(binding2.event.key)) {
+ if (binding2.event.key.map(function(s) {
+ return s.toLowerCase();
+ }).indexOf(event.key.toLowerCase()) === -1) {
+ isMatch = false;
+ }
+ } else {
+ if (event.key.toLowerCase() !== binding2.event.key.toLowerCase()) {
+ isMatch = false;
+ }
+ }
+ }
+ if (!isMatch && (tryKeyCode || binding2.event.modifiers.altKey)) {
+ isMatch = event.keyCode === binding2.event.keyCode;
+ }
+ if (!isMatch)
+ return false;
+ if (!(event.ctrlKey && event.altKey)) {
+ if (event.ctrlKey !== binding2.event.modifiers.ctrlKey)
+ return false;
+ if (event.altKey !== binding2.event.modifiers.altKey)
+ return false;
+ }
+ if (event.metaKey !== binding2.event.modifiers.metaKey)
+ return false;
+ if (testShift && event.shiftKey !== binding2.event.modifiers.shiftKey)
+ return false;
+ return true;
+ }
+ }
+ function capture(d3_event) {
+ testBindings(d3_event, true);
+ }
+ function bubble(d3_event) {
+ var tagName = select_default2(d3_event.target).node().tagName;
+ if (tagName === "INPUT" || tagName === "SELECT" || tagName === "TEXTAREA") {
+ return;
+ }
+ testBindings(d3_event, false);
+ }
+ function keybinding(selection2) {
+ selection2 = selection2 || select_default2(document);
+ selection2.on("keydown.capture." + namespace, capture, true);
+ selection2.on("keydown.bubble." + namespace, bubble, false);
+ return keybinding;
+ }
+ keybinding.unbind = function(selection2) {
+ _keybindings = [];
+ selection2 = selection2 || select_default2(document);
+ selection2.on("keydown.capture." + namespace, null);
+ selection2.on("keydown.bubble." + namespace, null);
+ return keybinding;
+ };
+ keybinding.clear = function() {
+ _keybindings = {};
+ return keybinding;
+ };
+ keybinding.off = function(codes, capture2) {
+ var arr = utilArrayUniq([].concat(codes));
+ for (var i2 = 0; i2 < arr.length; i2++) {
+ var id2 = arr[i2] + (capture2 ? "-capture" : "-bubble");
+ delete _keybindings[id2];
+ }
+ return keybinding;
+ };
+ keybinding.on = function(codes, callback, capture2) {
+ if (typeof callback !== "function") {
+ return keybinding.off(codes, capture2);
+ }
+ var arr = utilArrayUniq([].concat(codes));
+ for (var i2 = 0; i2 < arr.length; i2++) {
+ var id2 = arr[i2] + (capture2 ? "-capture" : "-bubble");
+ var binding = {
+ id: id2,
+ capture: capture2,
+ callback,
+ event: {
+ key: void 0,
+ // preferred
+ keyCode: 0,
+ // fallback
+ modifiers: {
+ shiftKey: false,
+ ctrlKey: false,
+ altKey: false,
+ metaKey: false
+ }
+ }
+ };
+ if (_keybindings[id2]) {
+ console.warn('warning: duplicate keybinding for "' + id2 + '"');
+ }
+ _keybindings[id2] = binding;
+ var matches = arr[i2].toLowerCase().match(/(?:(?:[^+⇧⌃⌥⌘])+|[⇧⌃⌥⌘]|\+\+|^\+$)/g);
+ for (var j2 = 0; j2 < matches.length; j2++) {
+ if (matches[j2] === "++")
+ matches[j2] = "+";
+ if (matches[j2] in utilKeybinding.modifierCodes) {
+ var prop = utilKeybinding.modifierProperties[utilKeybinding.modifierCodes[matches[j2]]];
+ binding.event.modifiers[prop] = true;
+ } else {
+ binding.event.key = utilKeybinding.keys[matches[j2]] || matches[j2];
+ if (matches[j2] in utilKeybinding.keyCodes) {
+ binding.event.keyCode = utilKeybinding.keyCodes[matches[j2]];
+ }
+ }
+ }
+ }
+ return keybinding;
+ };
+ return keybinding;
+ }
+ utilKeybinding.modifierCodes = {
+ // Shift key, ⇧
+ "\u21E7": 16,
+ shift: 16,
+ // CTRL key, on Mac: ⌃
+ "\u2303": 17,
+ ctrl: 17,
+ // ALT key, on Mac: ⌥ (Alt)
+ "\u2325": 18,
+ alt: 18,
+ option: 18,
+ // META, on Mac: ⌘ (CMD), on Windows (Win), on Linux (Super)
+ "\u2318": 91,
+ meta: 91,
+ cmd: 91,
+ "super": 91,
+ win: 91
+ };
+ utilKeybinding.modifierProperties = {
+ 16: "shiftKey",
+ 17: "ctrlKey",
+ 18: "altKey",
+ 91: "metaKey"
+ };
+ utilKeybinding.plusKeys = ["plus", "ffplus", "=", "ffequals", "\u2260", "\xB1"];
+ utilKeybinding.minusKeys = ["_", "-", "ffminus", "dash", "\u2013", "\u2014"];
+ utilKeybinding.keys = {
+ // Backspace key, on Mac: ⌫ (Backspace)
+ "\u232B": "Backspace",
+ backspace: "Backspace",
+ // Tab Key, on Mac: ⇥ (Tab), on Windows ⇥⇥
+ "\u21E5": "Tab",
+ "\u21C6": "Tab",
+ tab: "Tab",
+ // Return key, ↩
+ "\u21A9": "Enter",
+ "\u21B5": "Enter",
+ "\u23CE": "Enter",
+ "return": "Enter",
+ enter: "Enter",
+ "\u2305": "Enter",
+ // Pause/Break key
+ "pause": "Pause",
+ "pause-break": "Pause",
+ // Caps Lock key, ⇪
+ "\u21EA": "CapsLock",
+ caps: "CapsLock",
+ "caps-lock": "CapsLock",
+ // Escape key, on Mac: ⎋, on Windows: Esc
+ "\u238B": ["Escape", "Esc"],
+ escape: ["Escape", "Esc"],
+ esc: ["Escape", "Esc"],
+ // Space key
+ space: [" ", "Spacebar"],
+ // Page-Up key, or pgup, on Mac: ↖
+ "\u2196": "PageUp",
+ pgup: "PageUp",
+ "page-up": "PageUp",
+ // Page-Down key, or pgdown, on Mac: ↘
+ "\u2198": "PageDown",
+ pgdown: "PageDown",
+ "page-down": "PageDown",
+ // END key, on Mac: ⇟
+ "\u21DF": "End",
+ end: "End",
+ // HOME key, on Mac: ⇞
+ "\u21DE": "Home",
+ home: "Home",
+ // Insert key, or ins
+ ins: "Insert",
+ insert: "Insert",
+ // Delete key, on Mac: ⌦ (Delete)
+ "\u2326": ["Delete", "Del"],
+ del: ["Delete", "Del"],
+ "delete": ["Delete", "Del"],
+ // Left Arrow Key, or ←
+ "\u2190": ["ArrowLeft", "Left"],
+ left: ["ArrowLeft", "Left"],
+ "arrow-left": ["ArrowLeft", "Left"],
+ // Up Arrow Key, or ↑
+ "\u2191": ["ArrowUp", "Up"],
+ up: ["ArrowUp", "Up"],
+ "arrow-up": ["ArrowUp", "Up"],
+ // Right Arrow Key, or →
+ "\u2192": ["ArrowRight", "Right"],
+ right: ["ArrowRight", "Right"],
+ "arrow-right": ["ArrowRight", "Right"],
+ // Up Arrow Key, or ↓
+ "\u2193": ["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)
+ "\u232B": 8,
+ backspace: 8,
+ // Tab Key, on Mac: ⇥ (Tab), on Windows ⇥⇥
+ "\u21E5": 9,
+ "\u21C6": 9,
+ tab: 9,
+ // Return key, ↩
+ "\u21A9": 13,
+ "\u21B5": 13,
+ "\u23CE": 13,
+ "return": 13,
+ enter: 13,
+ "\u2305": 13,
+ // Pause/Break key
+ "pause": 19,
+ "pause-break": 19,
+ // Caps Lock key, ⇪
+ "\u21EA": 20,
+ caps: 20,
+ "caps-lock": 20,
+ // Escape key, on Mac: ⎋, on Windows: Esc
+ "\u238B": 27,
+ escape: 27,
+ esc: 27,
+ // Space key
+ space: 32,
+ // Page-Up key, or pgup, on Mac: ↖
+ "\u2196": 33,
+ pgup: 33,
+ "page-up": 33,
+ // Page-Down key, or pgdown, on Mac: ↘
+ "\u2198": 34,
+ pgdown: 34,
+ "page-down": 34,
+ // END key, on Mac: ⇟
+ "\u21DF": 35,
+ end: 35,
+ // HOME key, on Mac: ⇞
+ "\u21DE": 36,
+ home: 36,
+ // Insert key, or ins
+ ins: 45,
+ insert: 45,
+ // Delete key, on Mac: ⌦ (Delete)
+ "\u2326": 46,
+ del: 46,
+ "delete": 46,
+ // Left Arrow Key, or ←
+ "\u2190": 37,
+ left: 37,
+ "arrow-left": 37,
+ // Up Arrow Key, or ↑
+ "\u2191": 38,
+ up: 38,
+ "arrow-up": 38,
+ // Right Arrow Key, or →
+ "\u2192": 39,
+ right: 39,
+ "arrow-right": 39,
+ // Up Arrow Key, or ↓
+ "\u2193": 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
+ };
+ var i = 95;
+ var n = 0;
+ while (++i < 106) {
+ utilKeybinding.keyCodes["num-" + n] = i;
+ ++n;
+ }
+ i = 47;
+ n = 0;
+ while (++i < 58) {
+ utilKeybinding.keyCodes[n] = i;
+ ++n;
+ }
+ i = 111;
+ n = 1;
+ while (++i < 136) {
+ utilKeybinding.keyCodes["f" + n] = i;
+ ++n;
+ }
+ i = 64;
+ while (++i < 91) {
+ utilKeybinding.keyCodes[String.fromCharCode(i).toLowerCase()] = i;
+ }
+
+ // modules/util/object.js
+ function utilObjectOmit(obj, omitKeys) {
+ return Object.keys(obj).reduce(function(result, key) {
+ if (omitKeys.indexOf(key) === -1) {
+ result[key] = obj[key];
+ }
+ return result;
+ }, {});
+ }
+
+ // modules/util/rebind.js
+ function utilRebind(target, source) {
+ var i2 = 1, n2 = arguments.length, method;
+ while (++i2 < n2) {
+ target[method = arguments[i2]] = d3_rebind(target, source, source[method]);
+ }
+ return target;
+ }
+ function d3_rebind(target, source, method) {
+ return function() {
+ var value = method.apply(source, arguments);
+ return value === source ? target : value;
+ };
+ }
+
+ // modules/util/session_mutex.js
+ 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";
+ }
+ 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, 4e3);
+ 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;
+ }
+
+ // modules/util/tiler.js
+ 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 clamp3(num, min3, max3) {
+ return Math.max(min3, Math.min(num, max3));
+ }
+ 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 min3 = center - width / 2;
+ var max3 = center + width / 2 - 1;
+ return x >= min3 && x <= max3 && y >= min3 && y <= max3;
+ }
+ return false;
+ }
+ function tiler8() {
+ var z = geoScaleToZoom(_scale / (2 * Math.PI), _tileSize);
+ var z0 = clamp3(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(
+ clamp3(Math.floor(-origin[0]) - _margin, tileMin, tileMax + 1),
+ clamp3(Math.ceil(_size[0] / k - origin[0]) + _margin, tileMin, tileMax + 1)
+ );
+ var rows = range(
+ clamp3(Math.floor(-origin[1]) - _margin, tileMin, tileMax + 1),
+ clamp3(Math.ceil(_size[1] / k - origin[1]) + _margin, tileMin, tileMax + 1)
+ );
+ var tiles = [];
+ for (var i2 = 0; i2 < rows.length; i2++) {
+ var y = rows[i2];
+ for (var j2 = 0; j2 < cols.length; j2++) {
+ var x = cols[j2];
+ if (i2 >= _margin && i2 <= rows.length - _margin && j2 >= _margin && j2 <= cols.length - _margin) {
+ tiles.unshift([x, y, z0]);
+ } else {
+ tiles.push([x, y, z0]);
+ }
+ }
+ }
+ tiles.translate = origin;
+ tiles.scale = k;
+ return tiles;
+ }
+ tiler8.getTiles = function(projection2) {
+ var origin = [
+ projection2.scale() * Math.PI - projection2.translate()[0],
+ projection2.scale() * Math.PI - projection2.translate()[1]
+ ];
+ this.size(projection2.clipExtent()[1]).scale(projection2.scale() * 2 * Math.PI).translate(projection2.translate());
+ var tiles = tiler8();
+ 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(
+ projection2.invert([x, y + ts]),
+ projection2.invert([x + ts, y])
+ )
+ };
+ }).filter(Boolean);
+ };
+ tiler8.getGeoJSON = function(projection2) {
+ var features = tiler8.getTiles(projection2).map(function(tile) {
+ return {
+ type: "Feature",
+ properties: {
+ id: tile.id,
+ name: tile.id
+ },
+ geometry: {
+ type: "Polygon",
+ coordinates: [tile.extent.polygon()]
+ }
+ };
+ });
+ return {
+ type: "FeatureCollection",
+ features
+ };
+ };
+ tiler8.tileSize = function(val) {
+ if (!arguments.length)
+ return _tileSize;
+ _tileSize = val;
+ return tiler8;
+ };
+ tiler8.zoomExtent = function(val) {
+ if (!arguments.length)
+ return _zoomExtent;
+ _zoomExtent = val;
+ return tiler8;
+ };
+ tiler8.size = function(val) {
+ if (!arguments.length)
+ return _size;
+ _size = val;
+ return tiler8;
+ };
+ tiler8.scale = function(val) {
+ if (!arguments.length)
+ return _scale;
+ _scale = val;
+ return tiler8;
+ };
+ tiler8.translate = function(val) {
+ if (!arguments.length)
+ return _translate;
+ _translate = val;
+ return tiler8;
+ };
+ tiler8.margin = function(val) {
+ if (!arguments.length)
+ return _margin;
+ _margin = +val;
+ return tiler8;
+ };
+ tiler8.skipNullIsland = function(val) {
+ if (!arguments.length)
+ return _skipNullIsland;
+ _skipNullIsland = val;
+ return tiler8;
+ };
+ return tiler8;
+ }
+
+ // modules/util/trigger_event.js
+ function utilTriggerEvent(target, type2) {
+ target.each(function() {
+ var evt = document.createEvent("HTMLEvents");
+ evt.initEvent(type2, true, true);
+ this.dispatchEvent(evt);
+ });
+ }
+
+ // modules/core/localizer.js
+ var _mainLocalizer = coreLocalizer();
+ var _t = _mainLocalizer.t;
+ function coreLocalizer() {
+ let localizer = {};
+ let _dataLanguages = {};
+ let _dataLocales = {};
+ let _localeStrings = {};
+ let _localeCode = "en-US";
+ let _localeCodes = ["en-US", "en"];
+ let _languageCode = "en";
+ let _textDirection = "ltr";
+ let _usesMetric = false;
+ let _languageNames = {};
+ let _scriptNames = {};
+ localizer.localeCode = () => _localeCode;
+ localizer.localeCodes = () => _localeCodes;
+ localizer.languageCode = () => _languageCode;
+ localizer.textDirection = () => _textDirection;
+ localizer.usesMetric = () => _usesMetric;
+ localizer.languageNames = () => _languageNames;
+ localizer.scriptNames = () => _scriptNames;
+ let _preferredLocaleCodes = [];
+ localizer.preferredLocaleCodes = function(codes) {
+ if (!arguments.length)
+ return _preferredLocaleCodes;
+ if (typeof codes === "string") {
+ _preferredLocaleCodes = codes.split(/,|;| /gi).filter(Boolean);
+ } else {
+ _preferredLocaleCodes = codes;
+ }
+ return localizer;
+ };
+ var _loadPromise;
+ localizer.ensureLoaded = () => {
+ if (_loadPromise)
+ return _loadPromise;
+ let filesToFetch = [
+ "languages",
+ // load the list of languages
+ "locales"
+ // load the list of supported locales
+ ];
+ const localeDirs = {
+ general: "locales",
+ tagging: presetsCdnUrl + "dist/translations"
+ };
+ let fileMap = _mainFileFetcher.fileMap();
+ for (let scopeId in localeDirs) {
+ const key = `locales_index_${scopeId}`;
+ if (!fileMap[key]) {
+ fileMap[key] = localeDirs[scopeId] + "/index.min.json";
+ }
+ filesToFetch.push(key);
+ }
+ return _loadPromise = Promise.all(filesToFetch.map((key) => _mainFileFetcher.get(key))).then((results) => {
+ _dataLanguages = results[0];
+ _dataLocales = results[1];
+ let indexes = results.slice(2);
+ let requestedLocales = (_preferredLocaleCodes || []).concat(utilDetect().browserLocales).concat(["en"]);
+ _localeCodes = localesToUseFrom(requestedLocales);
+ _localeCode = _localeCodes[0];
+ let loadStringsPromises = [];
+ indexes.forEach((index, i2) => {
+ const fullCoverageIndex = _localeCodes.findIndex(function(locale2) {
+ return index[locale2] && index[locale2].pct === 1;
+ });
+ _localeCodes.slice(0, fullCoverageIndex + 1).forEach(function(code) {
+ let scopeId = Object.keys(localeDirs)[i2];
+ let directory = Object.values(localeDirs)[i2];
+ if (index[code])
+ loadStringsPromises.push(localizer.loadLocale(code, scopeId, directory));
+ });
+ });
+ return Promise.all(loadStringsPromises);
+ }).then(() => {
+ updateForCurrentLocale();
+ }).catch((err) => console.error(err));
+ };
+ function localesToUseFrom(requestedLocales) {
+ let supportedLocales = _dataLocales;
+ let toUse = [];
+ for (let i2 in requestedLocales) {
+ let locale2 = requestedLocales[i2];
+ if (supportedLocales[locale2])
+ toUse.push(locale2);
+ if (locale2.includes("-")) {
+ let langPart = locale2.split("-")[0];
+ if (supportedLocales[langPart])
+ toUse.push(langPart);
+ }
+ }
+ return utilArrayUniq(toUse);
+ }
+ function updateForCurrentLocale() {
+ if (!_localeCode)
+ return;
+ _languageCode = _localeCode.split("-")[0];
+ const currentData = _dataLocales[_localeCode] || _dataLocales[_languageCode];
+ const 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";
+ }
+ let locale2 = _localeCode;
+ if (locale2.toLowerCase() === "en-us")
+ locale2 = "en";
+ _languageNames = _localeStrings.general[locale2].languageNames;
+ _scriptNames = _localeStrings.general[locale2].scriptNames;
+ _usesMetric = _localeCode.slice(-3).toLowerCase() !== "-us";
+ }
+ localizer.loadLocale = (locale2, scopeId, directory) => {
+ if (locale2.toLowerCase() === "en-us")
+ locale2 = "en";
+ if (_localeStrings[scopeId] && _localeStrings[scopeId][locale2]) {
+ return Promise.resolve(locale2);
+ }
+ let fileMap = _mainFileFetcher.fileMap();
+ const key = `locale_${scopeId}_${locale2}`;
+ if (!fileMap[key]) {
+ fileMap[key] = `${directory}/${locale2}.min.json`;
+ }
+ return _mainFileFetcher.get(key).then((d) => {
+ if (!_localeStrings[scopeId])
+ _localeStrings[scopeId] = {};
+ _localeStrings[scopeId][locale2] = d[locale2];
+ return locale2;
+ });
+ };
+ localizer.pluralRule = function(number3) {
+ return pluralRule(number3, _localeCode);
+ };
+ function pluralRule(number3, localeCode) {
+ const rules = "Intl" in window && Intl.PluralRules && new Intl.PluralRules(localeCode);
+ if (rules) {
+ return rules.select(number3);
+ }
+ if (number3 === 1)
+ return "one";
+ return "other";
+ }
+ localizer.tInfo = function(origStringId, replacements, locale2) {
+ let stringId = origStringId.trim();
+ let scopeId = "general";
+ if (stringId[0] === "_") {
+ let split = stringId.split(".");
+ scopeId = split[0].slice(1);
+ stringId = split.slice(1).join(".");
+ }
+ locale2 = locale2 || _localeCode;
+ let path = stringId.split(".").map((s) => s.replace(/<TX_DOT>/g, ".")).reverse();
+ let stringsKey = locale2;
+ if (stringsKey.toLowerCase() === "en-us")
+ stringsKey = "en";
+ let result = _localeStrings && _localeStrings[scopeId] && _localeStrings[scopeId][stringsKey];
+ while (result !== void 0 && path.length) {
+ result = result[path.pop()];
+ }
+ if (result !== void 0) {
+ if (replacements) {
+ if (typeof result === "object" && Object.keys(result).length) {
+ const number3 = Object.values(replacements).find(function(value) {
+ return typeof value === "number";
+ });
+ if (number3 !== void 0) {
+ const rule = pluralRule(number3, locale2);
+ if (result[rule]) {
+ result = result[rule];
+ } else {
+ result = Object.values(result)[0];
+ }
+ }
+ }
+ if (typeof result === "string") {
+ for (let key in replacements) {
+ let value = replacements[key];
+ if (typeof value === "number") {
+ if (value.toLocaleString) {
+ value = value.toLocaleString(locale2, {
+ style: "decimal",
+ useGrouping: true,
+ minimumFractionDigits: 0
+ });
+ } else {
+ value = value.toString();
+ }
+ }
+ const token = `{${key}}`;
+ const regex = new RegExp(token, "g");
+ result = result.replace(regex, value);
+ }
+ }
+ }
+ if (typeof result === "string") {
+ return {
+ text: result,
+ locale: locale2
+ };
+ }
+ }
+ let index = _localeCodes.indexOf(locale2);
+ if (index >= 0 && index < _localeCodes.length - 1) {
+ let fallback = _localeCodes[index + 1];
+ return localizer.tInfo(origStringId, replacements, fallback);
+ }
+ if (replacements && "default" in replacements) {
+ return {
+ text: replacements.default,
+ locale: null
+ };
+ }
+ const missing = `Missing ${locale2} translation: ${origStringId}`;
+ if (typeof console !== "undefined")
+ console.error(missing);
+ return {
+ text: missing,
+ locale: "en"
+ };
+ };
+ localizer.hasTextForStringId = function(stringId) {
+ return !!localizer.tInfo(stringId, { default: "nothing found" }).locale;
+ };
+ localizer.t = function(stringId, replacements, locale2) {
+ return localizer.tInfo(stringId, replacements, locale2).text;
+ };
+ localizer.t.html = function(stringId, replacements, locale2) {
+ replacements = Object.assign({}, replacements);
+ for (var k in replacements) {
+ if (typeof replacements[k] === "string") {
+ replacements[k] = escape_default(replacements[k]);
+ }
+ if (typeof replacements[k] === "object" && typeof replacements[k].html === "string") {
+ replacements[k] = replacements[k].html;
+ }
+ }
+ const info = localizer.tInfo(stringId, replacements, locale2);
+ if (info.text) {
+ return `<span class="localized-text" lang="${info.locale || "und"}">${info.text}</span>`;
+ } else {
+ return "";
+ }
+ };
+ localizer.t.append = function(stringId, replacements, locale2) {
+ const ret = function(selection2) {
+ const info = localizer.tInfo(stringId, replacements, locale2);
+ return selection2.append("span").attr("class", "localized-text").attr("lang", info.locale || "und").text((replacements && replacements.prefix || "") + info.text + (replacements && replacements.suffix || ""));
+ };
+ ret.stringId = stringId;
+ return ret;
+ };
+ localizer.languageName = (code, options2) => {
+ if (_languageNames[code]) {
+ return _languageNames[code];
+ }
+ if (options2 && options2.localOnly)
+ return null;
+ const langInfo = _dataLanguages[code];
+ if (langInfo) {
+ if (langInfo.nativeName) {
+ return localizer.t("translate.language_and_code", { language: langInfo.nativeName, code });
+ } else if (langInfo.base && langInfo.script) {
+ const base = langInfo.base;
+ if (_languageNames[base]) {
+ const scriptCode = langInfo.script;
+ const script = _scriptNames[scriptCode] || scriptCode;
+ return localizer.t("translate.language_and_code", { language: _languageNames[base], code: script });
+ } else if (_dataLanguages[base] && _dataLanguages[base].nativeName) {
+ return localizer.t("translate.language_and_code", { language: _dataLanguages[base].nativeName, code });
+ }
+ }
+ }
+ return code;
+ };
+ return localizer;
+ }
+
+ // modules/presets/collection.js
+ function presetCollection(collection) {
+ const MAXRESULTS = 50;
+ let _this = {};
+ let _memo = {};
+ _this.collection = collection;
+ _this.item = (id2) => {
+ if (_memo[id2])
+ return _memo[id2];
+ const found = _this.collection.find((d) => d.id === id2);
+ if (found)
+ _memo[id2] = found;
+ return found;
+ };
+ _this.index = (id2) => _this.collection.findIndex((d) => d.id === id2);
+ _this.matchGeometry = (geometry) => {
+ return presetCollection(
+ _this.collection.filter((d) => d.matchGeometry(geometry))
+ );
+ };
+ _this.matchAllGeometry = (geometries) => {
+ return presetCollection(
+ _this.collection.filter((d) => d && d.matchAllGeometry(geometries))
+ );
+ };
+ _this.matchAnyGeometry = (geometries) => {
+ return presetCollection(
+ _this.collection.filter((d) => geometries.some((geom) => d.matchGeometry(geom)))
+ );
+ };
+ _this.fallback = (geometry) => {
+ let id2 = geometry;
+ if (id2 === "vertex")
+ id2 = "point";
+ return _this.item(id2);
+ };
+ _this.search = (value, geometry, loc) => {
+ if (!value)
+ return _this;
+ value = value.toLowerCase().trim();
+ function leading(a) {
+ const index = a.indexOf(value);
+ return index === 0 || a[index - 1] === " ";
+ }
+ function leadingStrict(a) {
+ const index = a.indexOf(value);
+ return index === 0;
+ }
+ function sortPresets(nameProp, aliasesProp) {
+ return function sortNames(a, b) {
+ let aCompare = a[nameProp]();
+ let bCompare = b[nameProp]();
+ if (aliasesProp) {
+ const findMatchingAlias = (strings) => {
+ if (strings.some((s) => s === value)) {
+ return strings.find((s) => s === value);
+ } else {
+ return strings.filter((s) => s.includes(value)).sort((a2, b2) => a2.length - b2.length)[0];
+ }
+ };
+ aCompare = findMatchingAlias([aCompare].concat(a[aliasesProp]()));
+ bCompare = findMatchingAlias([bCompare].concat(b[aliasesProp]()));
+ }
+ if (value === aCompare)
+ return -1;
+ if (value === bCompare)
+ return 1;
+ let i2 = b.originalScore - a.originalScore;
+ if (i2 !== 0)
+ return i2;
+ i2 = aCompare.indexOf(value) - bCompare.indexOf(value);
+ if (i2 !== 0)
+ return i2;
+ return aCompare.length - bCompare.length;
+ };
+ }
+ let pool = _this.collection;
+ if (Array.isArray(loc)) {
+ const validHere = _sharedLocationManager.locationSetsAt(loc);
+ pool = pool.filter((a) => !a.locationSetID || validHere[a.locationSetID]);
+ }
+ const searchable = pool.filter((a) => a.searchable !== false && a.suggestion !== true);
+ const suggestions = pool.filter((a) => a.suggestion === true);
+ const leadingNames = searchable.filter((a) => leading(a.searchName()) || a.searchAliases().some(leading)).sort(sortPresets("searchName", "searchAliases"));
+ const leadingSuggestions = suggestions.filter((a) => leadingStrict(a.searchName())).sort(sortPresets("searchName"));
+ const leadingNamesStripped = searchable.filter((a) => leading(a.searchNameStripped()) || a.searchAliasesStripped().some(leading)).sort(sortPresets("searchNameStripped", "searchAliasesStripped"));
+ const leadingSuggestionsStripped = suggestions.filter((a) => leadingStrict(a.searchNameStripped())).sort(sortPresets("searchNameStripped"));
+ const leadingTerms = searchable.filter((a) => (a.terms() || []).some(leading));
+ const leadingSuggestionTerms = suggestions.filter((a) => (a.terms() || []).some(leading));
+ const leadingTagValues = searchable.filter((a) => Object.values(a.tags || {}).filter((val) => val !== "*").some(leading));
+ const similarName = searchable.map((a) => ({ preset: a, dist: utilEditDistance(value, a.searchName()) })).filter((a) => a.dist + Math.min(value.length - a.preset.searchName().length, 0) < 3).sort((a, b) => a.dist - b.dist).map((a) => a.preset);
+ const similarSuggestions = suggestions.map((a) => ({ preset: a, dist: utilEditDistance(value, a.searchName()) })).filter((a) => a.dist + Math.min(value.length - a.preset.searchName().length, 0) < 1).sort((a, b) => a.dist - b.dist).map((a) => a.preset);
+ const similarTerms = searchable.filter((a) => {
+ return (a.terms() || []).some((b) => {
+ return utilEditDistance(value, b) + Math.min(value.length - b.length, 0) < 3;
+ });
+ });
+ let leadingTagKeyValues = [];
+ if (value.includes("=")) {
+ leadingTagKeyValues = searchable.filter((a) => a.tags && Object.keys(a.tags).some((key) => key + "=" + a.tags[key] === value)).concat(searchable.filter((a) => a.tags && Object.keys(a.tags).some((key) => leading(key + "=" + a.tags[key]))));
+ }
+ let results = leadingNames.concat(
+ leadingSuggestions,
+ leadingNamesStripped,
+ leadingSuggestionsStripped,
+ leadingTerms,
+ leadingSuggestionTerms,
+ leadingTagValues,
+ similarName,
+ similarSuggestions,
+ similarTerms,
+ leadingTagKeyValues
+ ).slice(0, MAXRESULTS - 1);
+ if (geometry) {
+ if (typeof geometry === "string") {
+ results.push(_this.fallback(geometry));
+ } else {
+ geometry.forEach((geom) => results.push(_this.fallback(geom)));
+ }
+ }
+ return presetCollection(utilArrayUniq(results));
+ };
+ return _this;
+ }
+
+ // modules/presets/category.js
+ function presetCategory(categoryID, category, allPresets) {
+ let _this = Object.assign({}, category);
+ let _searchName;
+ let _searchNameStripped;
+ _this.id = categoryID;
+ _this.members = presetCollection(
+ (category.members || []).map((presetID) => allPresets[presetID]).filter(Boolean)
+ );
+ _this.geometry = _this.members.collection.reduce((acc, preset) => {
+ for (let i2 in preset.geometry) {
+ const geometry = preset.geometry[i2];
+ if (acc.indexOf(geometry) === -1) {
+ acc.push(geometry);
+ }
+ }
+ return acc;
+ }, []);
+ _this.matchGeometry = (geom) => _this.geometry.indexOf(geom) >= 0;
+ _this.matchAllGeometry = (geometries) => _this.members.collection.some((preset) => preset.matchAllGeometry(geometries));
+ _this.matchScore = () => -1;
+ _this.name = () => _t(`_tagging.presets.categories.${categoryID}.name`, { "default": categoryID });
+ _this.nameLabel = () => _t.append(`_tagging.presets.categories.${categoryID}.name`, { "default": categoryID });
+ _this.terms = () => [];
+ _this.searchName = () => {
+ if (!_searchName) {
+ _searchName = (_this.suggestion ? _this.originalName : _this.name()).toLowerCase();
+ }
+ return _searchName;
+ };
+ _this.searchNameStripped = () => {
+ if (!_searchNameStripped) {
+ _searchNameStripped = _this.searchName();
+ if (_searchNameStripped.normalize)
+ _searchNameStripped = _searchNameStripped.normalize("NFD");
+ _searchNameStripped = _searchNameStripped.replace(/[\u0300-\u036f]/g, "");
+ }
+ return _searchNameStripped;
+ };
+ _this.searchAliases = () => [];
+ _this.searchAliasesStripped = () => [];
+ return _this;
+ }
+
+ // modules/presets/field.js
+ function presetField(fieldID, field, allFields) {
+ allFields = allFields || {};
+ let _this = Object.assign({}, field);
+ _this.id = fieldID;
+ _this.safeid = utilSafeClassName(fieldID);
+ _this.matchGeometry = (geom) => !_this.geometry || _this.geometry.indexOf(geom) !== -1;
+ _this.matchAllGeometry = (geometries) => {
+ return !_this.geometry || geometries.every((geom) => _this.geometry.indexOf(geom) !== -1);
+ };
+ _this.t = (scope, options2) => _t(`_tagging.presets.fields.${fieldID}.${scope}`, options2);
+ _this.t.html = (scope, options2) => _t.html(`_tagging.presets.fields.${fieldID}.${scope}`, options2);
+ _this.t.append = (scope, options2) => _t.append(`_tagging.presets.fields.${fieldID}.${scope}`, options2);
+ _this.hasTextForStringId = (scope) => _mainLocalizer.hasTextForStringId(`_tagging.presets.fields.${fieldID}.${scope}`);
+ _this.resolveReference = (which) => {
+ const referenceRegex = /^\{(.*)\}$/;
+ const match = (field[which] || "").match(referenceRegex);
+ if (match) {
+ const field2 = allFields[match[1]];
+ if (field2) {
+ return field2;
+ }
+ console.error(`Unable to resolve referenced field: ${match[1]}`);
+ }
+ return _this;
+ };
+ _this.title = () => _this.overrideLabel || _this.resolveReference("label").t("label", { "default": fieldID });
+ _this.label = () => _this.overrideLabel ? (selection2) => selection2.text(_this.overrideLabel) : _this.resolveReference("label").t.append("label", { "default": fieldID });
+ _this.placeholder = () => _this.resolveReference("placeholder").t("placeholder", { "default": "" });
+ _this.originalTerms = (_this.terms || []).join();
+ _this.terms = () => _this.resolveReference("label").t("terms", { "default": _this.originalTerms }).toLowerCase().trim().split(/\s*,+\s*/);
+ _this.increment = _this.type === "number" ? _this.increment || 1 : void 0;
+ return _this;
+ }
+
+ // modules/presets/preset.js
+ function presetPreset(presetID, preset, addable, allFields, allPresets) {
+ allFields = allFields || {};
+ allPresets = allPresets || {};
+ let _this = Object.assign({}, preset);
+ let _addable = addable || false;
+ let _resolvedFields;
+ let _resolvedMoreFields;
+ let _searchName;
+ let _searchNameStripped;
+ let _searchAliases;
+ let _searchAliasesStripped;
+ const referenceRegex = /^\{(.*)\}$/;
+ _this.id = presetID;
+ _this.safeid = utilSafeClassName(presetID);
+ _this.originalTerms = (_this.terms || []).join();
+ _this.originalName = _this.name || "";
+ _this.originalAliases = (_this.aliases || []).join("\n");
+ _this.originalScore = _this.matchScore || 1;
+ _this.originalReference = _this.reference || {};
+ _this.originalFields = _this.fields || [];
+ _this.originalMoreFields = _this.moreFields || [];
+ _this.fields = () => _resolvedFields || (_resolvedFields = resolveFields("fields"));
+ _this.moreFields = () => _resolvedMoreFields || (_resolvedMoreFields = resolveFields("moreFields"));
+ _this.resetFields = () => _resolvedFields = _resolvedMoreFields = null;
+ _this.tags = _this.tags || {};
+ _this.addTags = _this.addTags || _this.tags;
+ _this.removeTags = _this.removeTags || _this.addTags;
+ _this.geometry = _this.geometry || [];
+ _this.matchGeometry = (geom) => _this.geometry.indexOf(geom) >= 0;
+ _this.matchAllGeometry = (geoms) => geoms.every(_this.matchGeometry);
+ _this.matchScore = (entityTags) => {
+ const tags = _this.tags;
+ let seen = {};
+ let score = 0;
+ for (let k in tags) {
+ seen[k] = true;
+ if (entityTags[k] === tags[k]) {
+ score += _this.originalScore;
+ } else if (tags[k] === "*" && k in entityTags) {
+ score += _this.originalScore / 2;
+ } else {
+ return -1;
+ }
+ }
+ const addTags = _this.addTags;
+ for (let k in addTags) {
+ if (!seen[k] && entityTags[k] === addTags[k]) {
+ score += _this.originalScore;
+ }
+ }
+ if (_this.searchable === false) {
+ score *= 0.999;
+ }
+ return score;
+ };
+ _this.t = (scope, options2) => {
+ const textID = `_tagging.presets.presets.${presetID}.${scope}`;
+ return _t(textID, options2);
+ };
+ _this.t.append = (scope, options2) => {
+ const textID = `_tagging.presets.presets.${presetID}.${scope}`;
+ return _t.append(textID, options2);
+ };
+ function resolveReference(which) {
+ const match = (_this[which] || "").match(referenceRegex);
+ if (match) {
+ const preset2 = allPresets[match[1]];
+ if (preset2) {
+ return preset2;
+ }
+ console.error(`Unable to resolve referenced preset: ${match[1]}`);
+ }
+ return _this;
+ }
+ _this.name = () => {
+ return resolveReference("originalName").t("name", { "default": _this.originalName || presetID });
+ };
+ _this.nameLabel = () => {
+ return resolveReference("originalName").t.append("name", { "default": _this.originalName || presetID });
+ };
+ _this.subtitle = () => {
+ if (_this.suggestion) {
+ let path = presetID.split("/");
+ path.pop();
+ return _t("_tagging.presets.presets." + path.join("/") + ".name");
+ }
+ return null;
+ };
+ _this.subtitleLabel = () => {
+ if (_this.suggestion) {
+ let path = presetID.split("/");
+ path.pop();
+ return _t.append("_tagging.presets.presets." + path.join("/") + ".name");
+ }
+ return null;
+ };
+ _this.aliases = () => {
+ return resolveReference("originalName").t("aliases", { "default": _this.originalAliases }).trim().split(/\s*[\r\n]+\s*/);
+ };
+ _this.terms = () => {
+ return resolveReference("originalName").t("terms", { "default": _this.originalTerms }).toLowerCase().trim().split(/\s*,+\s*/);
+ };
+ _this.searchName = () => {
+ if (!_searchName) {
+ _searchName = (_this.suggestion ? _this.originalName : _this.name()).toLowerCase();
+ }
+ return _searchName;
+ };
+ _this.searchNameStripped = () => {
+ if (!_searchNameStripped) {
+ _searchNameStripped = stripDiacritics(_this.searchName());
+ }
+ return _searchNameStripped;
+ };
+ _this.searchAliases = () => {
+ if (!_searchAliases) {
+ _searchAliases = _this.aliases().map((alias) => alias.toLowerCase());
+ }
+ return _searchAliases;
+ };
+ _this.searchAliasesStripped = () => {
+ if (!_searchAliasesStripped) {
+ _searchAliasesStripped = _this.searchAliases();
+ _searchAliasesStripped = _searchAliasesStripped.map(stripDiacritics);
+ }
+ return _searchAliasesStripped;
+ };
+ _this.isFallback = () => {
+ const tagCount = Object.keys(_this.tags).length;
+ return tagCount === 0 || tagCount === 1 && _this.tags.hasOwnProperty("area");
+ };
+ _this.addable = function(val) {
+ if (!arguments.length)
+ return _addable;
+ _addable = val;
+ return _this;
+ };
+ _this.reference = () => {
+ const qid = _this.tags.wikidata || _this.tags["flag:wikidata"] || _this.tags["brand:wikidata"] || _this.tags["network:wikidata"] || _this.tags["operator:wikidata"];
+ if (qid) {
+ return { qid };
+ }
+ let key = _this.originalReference.key || Object.keys(utilObjectOmit(_this.tags, "name"))[0];
+ let value = _this.originalReference.value || _this.tags[key];
+ if (value === "*") {
+ return { key };
+ } else {
+ return { key, value };
+ }
+ };
+ _this.unsetTags = (tags, geometry, ignoringKeys, skipFieldDefaults) => {
+ let removeTags = ignoringKeys ? utilObjectOmit(_this.removeTags, ignoringKeys) : _this.removeTags;
+ tags = utilObjectOmit(tags, Object.keys(removeTags));
+ if (geometry && !skipFieldDefaults) {
+ _this.fields().forEach((field) => {
+ if (field.matchGeometry(geometry) && field.key && field.default === tags[field.key] && (!ignoringKeys || ignoringKeys.indexOf(field.key) === -1)) {
+ delete tags[field.key];
+ }
+ });
+ }
+ delete tags.area;
+ return tags;
+ };
+ _this.setTags = (tags, geometry, skipFieldDefaults) => {
+ const addTags = _this.addTags;
+ tags = Object.assign({}, tags);
+ for (let k in addTags) {
+ if (addTags[k] === "*") {
+ if (_this.tags[k] || !tags[k]) {
+ tags[k] = "yes";
+ }
+ } else {
+ tags[k] = addTags[k];
+ }
+ }
+ if (!addTags.hasOwnProperty("area")) {
+ delete tags.area;
+ if (geometry === "area") {
+ let needsAreaTag = true;
+ for (let k in addTags) {
+ if (_this.geometry.indexOf("line") === -1 && k in osmAreaKeys || k in osmAreaKeysExceptions && addTags[k] in osmAreaKeysExceptions[k]) {
+ needsAreaTag = false;
+ break;
+ }
+ }
+ if (needsAreaTag) {
+ tags.area = "yes";
+ }
+ }
+ }
+ if (geometry && !skipFieldDefaults) {
+ _this.fields().forEach((field) => {
+ if (field.matchGeometry(geometry) && field.key && !tags[field.key] && field.default) {
+ tags[field.key] = field.default;
+ }
+ });
+ }
+ return tags;
+ };
+ function resolveFields(which) {
+ const fieldIDs = which === "fields" ? _this.originalFields : _this.originalMoreFields;
+ let resolved = [];
+ fieldIDs.forEach((fieldID) => {
+ const match = fieldID.match(referenceRegex);
+ if (match !== null) {
+ resolved = resolved.concat(inheritFields(match[1], which));
+ } else if (allFields[fieldID]) {
+ resolved.push(allFields[fieldID]);
+ } else {
+ console.log(`Cannot resolve "${fieldID}" found in ${_this.id}.${which}`);
+ }
+ });
+ if (!resolved.length) {
+ const endIndex = _this.id.lastIndexOf("/");
+ const parentID = endIndex && _this.id.substring(0, endIndex);
+ if (parentID) {
+ resolved = inheritFields(parentID, which);
+ }
+ }
+ return utilArrayUniq(resolved);
+ function inheritFields(presetID2, which2) {
+ const parent = allPresets[presetID2];
+ if (!parent)
+ return [];
+ if (which2 === "fields") {
+ return parent.fields().filter(shouldInherit);
+ } else if (which2 === "moreFields") {
+ return parent.moreFields();
+ } else {
+ return [];
+ }
+ }
+ function shouldInherit(f2) {
+ if (f2.key && _this.tags[f2.key] !== void 0 && // inherit anyway if multiple values are allowed or just a checkbox
+ f2.type !== "multiCombo" && f2.type !== "semiCombo" && f2.type !== "manyCombo" && f2.type !== "check")
+ return false;
+ return true;
+ }
+ }
+ function stripDiacritics(s) {
+ if (s.normalize)
+ s = s.normalize("NFD");
+ s = s.replace(/[\u0300-\u036f]/g, "");
+ return s;
+ }
+ return _this;
+ }
+
+ // modules/presets/index.js
+ var _mainPresetIndex = presetIndex();
+ function presetIndex() {
+ const dispatch10 = dispatch_default("favoritePreset", "recentsChange");
+ const MAXRECENTS = 30;
+ const POINT = presetPreset("point", { name: "Point", tags: {}, geometry: ["point", "vertex"], matchScore: 0.1 });
+ const LINE = presetPreset("line", { name: "Line", tags: {}, geometry: ["line"], matchScore: 0.1 });
+ const AREA = presetPreset("area", { name: "Area", tags: { area: "yes" }, geometry: ["area"], matchScore: 0.1 });
+ const RELATION = presetPreset("relation", { name: "Relation", tags: {}, geometry: ["relation"], matchScore: 0.1 });
+ let _this = presetCollection([POINT, LINE, AREA, RELATION]);
+ let _presets = { point: POINT, line: LINE, area: AREA, relation: RELATION };
+ let _defaults = {
+ point: presetCollection([POINT]),
+ vertex: presetCollection([POINT]),
+ line: presetCollection([LINE]),
+ area: presetCollection([AREA]),
+ relation: presetCollection([RELATION])
+ };
+ let _fields = {};
+ let _categories = {};
+ let _universal = [];
+ let _addablePresetIDs = null;
+ let _recents;
+ let _favorites;
+ let _geometryIndex = { point: {}, vertex: {}, line: {}, area: {}, relation: {} };
+ let _loadPromise;
+ _this.ensureLoaded = () => {
+ 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((vals) => {
+ _this.merge({
+ categories: vals[0],
+ defaults: vals[1],
+ presets: vals[2],
+ fields: vals[3]
+ });
+ osmSetAreaKeys(_this.areaKeys());
+ osmSetLineTags(_this.lineTags());
+ osmSetPointTags(_this.pointTags());
+ osmSetVertexTags(_this.vertexTags());
+ });
+ };
+ _this.merge = (d) => {
+ let newLocationSets = [];
+ if (d.fields) {
+ Object.keys(d.fields).forEach((fieldID) => {
+ let f2 = d.fields[fieldID];
+ if (f2) {
+ f2 = presetField(fieldID, f2, _fields);
+ if (f2.locationSet)
+ newLocationSets.push(f2);
+ _fields[fieldID] = f2;
+ } else {
+ delete _fields[fieldID];
+ }
+ });
+ }
+ if (d.presets) {
+ Object.keys(d.presets).forEach((presetID) => {
+ let p = d.presets[presetID];
+ if (p) {
+ const isAddable = !_addablePresetIDs || _addablePresetIDs.has(presetID);
+ p = presetPreset(presetID, p, isAddable, _fields, _presets);
+ if (p.locationSet)
+ newLocationSets.push(p);
+ _presets[presetID] = p;
+ } else {
+ const existing = _presets[presetID];
+ if (existing && !existing.isFallback()) {
+ delete _presets[presetID];
+ }
+ }
+ });
+ }
+ if (d.categories) {
+ Object.keys(d.categories).forEach((categoryID) => {
+ let c = d.categories[categoryID];
+ if (c) {
+ c = presetCategory(categoryID, c, _presets);
+ if (c.locationSet)
+ newLocationSets.push(c);
+ _categories[categoryID] = c;
+ } else {
+ delete _categories[categoryID];
+ }
+ });
+ }
+ _this.collection = Object.values(_presets).concat(Object.values(_categories));
+ if (d.defaults) {
+ Object.keys(d.defaults).forEach((geometry) => {
+ const def = d.defaults[geometry];
+ if (Array.isArray(def)) {
+ _defaults[geometry] = presetCollection(
+ def.map((id2) => _presets[id2] || _categories[id2]).filter(Boolean)
+ );
+ } else {
+ delete _defaults[geometry];
+ }
+ });
+ }
+ _universal = Object.values(_fields).filter((field) => field.universal);
+ Object.values(_presets).forEach((preset) => preset.resetFields());
+ _geometryIndex = { point: {}, vertex: {}, line: {}, area: {}, relation: {} };
+ _this.collection.forEach((preset) => {
+ (preset.geometry || []).forEach((geometry) => {
+ let g = _geometryIndex[geometry];
+ for (let key in preset.tags) {
+ g[key] = g[key] || {};
+ let value = preset.tags[key];
+ (g[key][value] = g[key][value] || []).push(preset);
+ }
+ });
+ });
+ if (d.featureCollection && Array.isArray(d.featureCollection.features)) {
+ _sharedLocationManager.mergeCustomGeoJSON(d.featureCollection);
+ }
+ if (newLocationSets.length) {
+ _sharedLocationManager.mergeLocationSets(newLocationSets);
+ }
+ return _this;
+ };
+ _this.match = (entity, resolver) => {
+ return resolver.transient(entity, "presetMatch", () => {
+ let geometry = entity.geometry(resolver);
+ if (geometry === "vertex" && entity.isOnAddressLine(resolver)) {
+ geometry = "point";
+ }
+ const entityExtent = entity.extent(resolver);
+ return _this.matchTags(entity.tags, geometry, entityExtent.center());
+ });
+ };
+ _this.matchTags = (tags, geometry, loc) => {
+ const keyIndex = _geometryIndex[geometry];
+ let bestScore = -1;
+ let bestMatch;
+ let matchCandidates = [];
+ for (let k in tags) {
+ let indexMatches = [];
+ let valueIndex = keyIndex[k];
+ if (!valueIndex)
+ continue;
+ let keyValueMatches = valueIndex[tags[k]];
+ if (keyValueMatches)
+ indexMatches.push(...keyValueMatches);
+ let keyStarMatches = valueIndex["*"];
+ if (keyStarMatches)
+ indexMatches.push(...keyStarMatches);
+ if (indexMatches.length === 0)
+ continue;
+ for (let i2 = 0; i2 < indexMatches.length; i2++) {
+ const candidate = indexMatches[i2];
+ const score = candidate.matchScore(tags);
+ if (score === -1) {
+ continue;
+ }
+ matchCandidates.push({ score, candidate });
+ if (score > bestScore) {
+ bestScore = score;
+ bestMatch = candidate;
+ }
+ }
+ }
+ if (bestMatch && bestMatch.locationSetID && bestMatch.locationSetID !== "+[Q2]" && Array.isArray(loc)) {
+ const validHere = _sharedLocationManager.locationSetsAt(loc);
+ if (!validHere[bestMatch.locationSetID]) {
+ matchCandidates.sort((a, b) => a.score < b.score ? 1 : -1);
+ for (let i2 = 0; i2 < matchCandidates.length; i2++) {
+ const candidateScore = matchCandidates[i2];
+ if (!candidateScore.candidate.locationSetID || validHere[candidateScore.candidate.locationSetID]) {
+ bestMatch = candidateScore.candidate;
+ bestScore = candidateScore.score;
+ break;
+ }
+ }
+ }
+ }
+ if (!bestMatch || bestMatch.isFallback()) {
+ for (let k in tags) {
+ if (/^addr:/.test(k) && keyIndex["addr:*"] && keyIndex["addr:*"]["*"]) {
+ bestMatch = keyIndex["addr:*"]["*"][0];
+ break;
+ }
+ }
+ }
+ return bestMatch || _this.fallback(geometry);
+ };
+ _this.allowsVertex = (entity, resolver) => {
+ if (entity.type !== "node")
+ return false;
+ if (Object.keys(entity.tags).length === 0)
+ return true;
+ return resolver.transient(entity, "vertexMatch", () => {
+ if (entity.isOnAddressLine(resolver))
+ return true;
+ const geometries = osmNodeGeometriesForTags(entity.tags);
+ if (geometries.vertex)
+ return true;
+ if (geometries.point)
+ return false;
+ return true;
+ });
+ };
+ _this.areaKeys = () => {
+ const ignore = {
+ barrier: true,
+ highway: true,
+ footway: true,
+ railway: true,
+ junction: true,
+ traffic_calming: true,
+ type: true
+ };
+ let areaKeys = {};
+ const presets = _this.collection.filter((p) => !p.suggestion && !p.replacement);
+ presets.forEach((p) => {
+ const keys = p.tags && Object.keys(p.tags);
+ const key = keys && keys.length && keys[0];
+ if (!key)
+ return;
+ if (ignore[key])
+ return;
+ if (p.geometry.indexOf("area") !== -1) {
+ areaKeys[key] = areaKeys[key] || {};
+ }
+ });
+ presets.forEach((p) => {
+ let key;
+ for (key in p.addTags) {
+ const 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;
+ };
+ _this.lineTags = () => {
+ return _this.collection.filter((lineTags, d) => {
+ if (d.suggestion || d.replacement || d.searchable === false)
+ return lineTags;
+ const keys = d.tags && Object.keys(d.tags);
+ const key = keys && keys.length && keys[0];
+ if (!key)
+ return lineTags;
+ if (d.geometry.indexOf("line") !== -1) {
+ lineTags[key] = lineTags[key] || [];
+ lineTags[key].push(d.tags);
+ }
+ return lineTags;
+ }, {});
+ };
+ _this.pointTags = () => {
+ return _this.collection.reduce((pointTags, d) => {
+ if (d.suggestion || d.replacement || d.searchable === false)
+ return pointTags;
+ const keys = d.tags && Object.keys(d.tags);
+ const key = keys && keys.length && keys[0];
+ if (!key)
+ return pointTags;
+ if (d.geometry.indexOf("point") !== -1) {
+ pointTags[key] = pointTags[key] || {};
+ pointTags[key][d.tags[key]] = true;
+ }
+ return pointTags;
+ }, {});
+ };
+ _this.vertexTags = () => {
+ return _this.collection.reduce((vertexTags, d) => {
+ if (d.suggestion || d.replacement || d.searchable === false)
+ return vertexTags;
+ const keys = d.tags && Object.keys(d.tags);
+ const key = keys && keys.length && keys[0];
+ if (!key)
+ return vertexTags;
+ if (d.geometry.indexOf("vertex") !== -1) {
+ vertexTags[key] = vertexTags[key] || {};
+ vertexTags[key][d.tags[key]] = true;
+ }
+ return vertexTags;
+ }, {});
+ };
+ _this.field = (id2) => _fields[id2];
+ _this.universal = () => _universal;
+ _this.defaults = (geometry, n2, startWithRecents, loc, extraPresets) => {
+ let recents = [];
+ if (startWithRecents) {
+ recents = _this.recent().matchGeometry(geometry).collection.slice(0, 4);
+ }
+ let defaults2;
+ if (_addablePresetIDs) {
+ defaults2 = Array.from(_addablePresetIDs).map(function(id2) {
+ var preset = _this.item(id2);
+ if (preset && preset.matchGeometry(geometry))
+ return preset;
+ return null;
+ }).filter(Boolean);
+ } else {
+ defaults2 = _defaults[geometry].collection.concat(_this.fallback(geometry));
+ }
+ let result = presetCollection(
+ utilArrayUniq(recents.concat(defaults2).concat(extraPresets || [])).slice(0, n2 - 1)
+ );
+ if (Array.isArray(loc)) {
+ const validHere = _sharedLocationManager.locationSetsAt(loc);
+ result.collection = result.collection.filter((a) => !a.locationSetID || validHere[a.locationSetID]);
+ }
+ return result;
+ };
+ _this.addablePresetIDs = function(val) {
+ if (!arguments.length)
+ return _addablePresetIDs;
+ if (Array.isArray(val))
+ val = new Set(val);
+ _addablePresetIDs = val;
+ if (_addablePresetIDs) {
+ _this.collection.forEach((p) => {
+ if (p.addable)
+ p.addable(_addablePresetIDs.has(p.id));
+ });
+ } else {
+ _this.collection.forEach((p) => {
+ if (p.addable)
+ p.addable(true);
+ });
+ }
+ return _this;
+ };
+ _this.recent = () => {
+ return presetCollection(
+ utilArrayUniq(_this.getRecents().map((d) => d.preset).filter((d) => d.searchable !== false))
+ );
+ };
+ function RibbonItem(preset, source) {
+ let item = {};
+ item.preset = preset;
+ item.source = source;
+ item.isFavorite = () => item.source === "favorite";
+ item.isRecent = () => item.source === "recent";
+ item.matches = (preset2) => item.preset.id === preset2.id;
+ item.minified = () => ({ pID: item.preset.id });
+ return item;
+ }
+ function ribbonItemForMinified(d, source) {
+ if (d && d.pID) {
+ const preset = _this.item(d.pID);
+ if (!preset)
+ return null;
+ return RibbonItem(preset, source);
+ }
+ return null;
+ }
+ _this.getGenericRibbonItems = () => {
+ return ["point", "line", "area"].map((id2) => RibbonItem(_this.item(id2), "generic"));
+ };
+ _this.getAddable = () => {
+ if (!_addablePresetIDs)
+ return [];
+ return _addablePresetIDs.map((id2) => {
+ const preset = _this.item(id2);
+ if (preset)
+ return RibbonItem(preset, "addable");
+ return null;
+ }).filter(Boolean);
+ };
+ function setRecents(items) {
+ _recents = items;
+ const minifiedItems = items.map((d) => d.minified());
+ corePreferences("preset_recents", JSON.stringify(minifiedItems));
+ dispatch10.call("recentsChange");
+ }
+ _this.getRecents = () => {
+ if (!_recents) {
+ _recents = (JSON.parse(corePreferences("preset_recents")) || []).reduce((acc, d) => {
+ let item = ribbonItemForMinified(d, "recent");
+ if (item && item.preset.addable())
+ acc.push(item);
+ return acc;
+ }, []);
+ }
+ return _recents;
+ };
+ _this.addRecent = (preset, besidePreset, after) => {
+ const recents = _this.getRecents();
+ const beforeItem = _this.recentMatching(besidePreset);
+ let toIndex = recents.indexOf(beforeItem);
+ if (after)
+ toIndex += 1;
+ const newItem = RibbonItem(preset, "recent");
+ recents.splice(toIndex, 0, newItem);
+ setRecents(recents);
+ };
+ _this.removeRecent = (preset) => {
+ const item = _this.recentMatching(preset);
+ if (item) {
+ let items = _this.getRecents();
+ items.splice(items.indexOf(item), 1);
+ setRecents(items);
+ }
+ };
+ _this.recentMatching = (preset) => {
+ const items = _this.getRecents();
+ for (let i2 in items) {
+ if (items[i2].matches(preset)) {
+ return items[i2];
+ }
+ }
+ return null;
+ };
+ _this.moveItem = (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;
+ };
+ _this.moveRecent = (item, beforeItem) => {
+ const recents = _this.getRecents();
+ const fromIndex = recents.indexOf(item);
+ const toIndex = recents.indexOf(beforeItem);
+ const items = _this.moveItem(recents, fromIndex, toIndex);
+ if (items)
+ setRecents(items);
+ };
+ _this.setMostRecent = (preset) => {
+ if (preset.searchable === false)
+ return;
+ let items = _this.getRecents();
+ let item = _this.recentMatching(preset);
+ if (item) {
+ items.splice(items.indexOf(item), 1);
+ } else {
+ item = RibbonItem(preset, "recent");
+ }
+ while (items.length >= MAXRECENTS) {
+ items.pop();
+ }
+ items.unshift(item);
+ setRecents(items);
+ };
+ function setFavorites(items) {
+ _favorites = items;
+ const minifiedItems = items.map((d) => d.minified());
+ corePreferences("preset_favorites", JSON.stringify(minifiedItems));
+ dispatch10.call("favoritePreset");
+ }
+ _this.addFavorite = (preset, besidePreset, after) => {
+ const favorites = _this.getFavorites();
+ const beforeItem = _this.favoriteMatching(besidePreset);
+ let toIndex = favorites.indexOf(beforeItem);
+ if (after)
+ toIndex += 1;
+ const newItem = RibbonItem(preset, "favorite");
+ favorites.splice(toIndex, 0, newItem);
+ setFavorites(favorites);
+ };
+ _this.toggleFavorite = (preset) => {
+ const favs = _this.getFavorites();
+ const favorite = _this.favoriteMatching(preset);
+ if (favorite) {
+ favs.splice(favs.indexOf(favorite), 1);
+ } else {
+ if (favs.length === 10) {
+ favs.pop();
+ }
+ favs.push(RibbonItem(preset, "favorite"));
+ }
+ setFavorites(favs);
+ };
+ _this.removeFavorite = (preset) => {
+ const item = _this.favoriteMatching(preset);
+ if (item) {
+ const items = _this.getFavorites();
+ items.splice(items.indexOf(item), 1);
+ setFavorites(items);
+ }
+ };
+ _this.getFavorites = () => {
+ if (!_favorites) {
+ let rawFavorites = JSON.parse(corePreferences("preset_favorites"));
+ if (!rawFavorites) {
+ rawFavorites = [];
+ corePreferences("preset_favorites", JSON.stringify(rawFavorites));
+ }
+ _favorites = rawFavorites.reduce((output, d) => {
+ const item = ribbonItemForMinified(d, "favorite");
+ if (item && item.preset.addable())
+ output.push(item);
+ return output;
+ }, []);
+ }
+ return _favorites;
+ };
+ _this.favoriteMatching = (preset) => {
+ const favs = _this.getFavorites();
+ for (let index in favs) {
+ if (favs[index].matches(preset)) {
+ return favs[index];
+ }
+ }
+ return null;
+ };
+ return utilRebind(_this, dispatch10, "on");
+ }
+
+ // modules/util/util.js
+ function utilTagText(entity) {
+ var obj = entity && entity.tags || {};
+ return Object.keys(obj).map(function(k) {
+ return k + "=" + obj[k];
+ }).join(", ");
+ }
+ function utilTotalExtent(array2, graph) {
+ var extent = geoExtent();
+ var val, entity;
+ for (var i2 = 0; i2 < array2.length; i2++) {
+ val = array2[i2];
+ entity = typeof val === "string" ? graph.hasEntity(val) : val;
+ 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];
+ if ((oldVal || oldVal === "") && (newVal === void 0 || newVal !== oldVal)) {
+ tagDiff.push({
+ type: "-",
+ key: k,
+ oldVal,
+ newVal,
+ display: "- " + k + "=" + oldVal
+ });
+ }
+ if ((newVal || newVal === "") && (oldVal === void 0 || newVal !== oldVal)) {
+ tagDiff.push({
+ type: "+",
+ key: k,
+ oldVal,
+ newVal,
+ display: "+ " + k + "=" + newVal
+ });
+ }
+ });
+ return tagDiff;
+ }
+ function utilEntitySelector(ids) {
+ return ids.length ? "." + ids.join(",.") : "nothing";
+ }
+ function utilEntityOrMemberSelector(ids, graph) {
+ var seen = new Set(ids);
+ ids.forEach(collectShallowDescendants);
+ return utilEntitySelector(Array.from(seen));
+ function collectShallowDescendants(id2) {
+ var entity = graph.hasEntity(id2);
+ if (!entity || entity.type !== "relation")
+ return;
+ entity.members.map(function(member) {
+ return member.id;
+ }).forEach(function(id3) {
+ seen.add(id3);
+ });
+ }
+ }
+ function utilEntityOrDeepMemberSelector(ids, graph) {
+ return utilEntitySelector(utilEntityAndDeepMemberIDs(ids, graph));
+ }
+ function utilEntityAndDeepMemberIDs(ids, graph) {
+ var seen = /* @__PURE__ */ new Set();
+ ids.forEach(collectDeepDescendants);
+ return Array.from(seen);
+ function collectDeepDescendants(id2) {
+ if (seen.has(id2))
+ return;
+ seen.add(id2);
+ var entity = graph.hasEntity(id2);
+ if (!entity || entity.type !== "relation")
+ return;
+ entity.members.map(function(member) {
+ return member.id;
+ }).forEach(collectDeepDescendants);
+ }
+ }
+ function utilDeepMemberSelector(ids, graph, skipMultipolgonMembers) {
+ var idsSet = new Set(ids);
+ var seen = /* @__PURE__ */ new Set();
+ var returners = /* @__PURE__ */ new Set();
+ ids.forEach(collectDeepDescendants);
+ return utilEntitySelector(Array.from(returners));
+ function collectDeepDescendants(id2) {
+ if (seen.has(id2))
+ return;
+ seen.add(id2);
+ if (!idsSet.has(id2)) {
+ returners.add(id2);
+ }
+ var entity = graph.hasEntity(id2);
+ if (!entity || entity.type !== "relation")
+ return;
+ if (skipMultipolgonMembers && entity.isMultipolygon())
+ return;
+ entity.members.map(function(member) {
+ return member.id;
+ }).forEach(collectDeepDescendants);
+ }
+ }
+ function utilHighlightEntities(ids, highlighted, context) {
+ context.surface().selectAll(utilEntityOrDeepMemberSelector(ids, context.graph())).classed("highlighted", highlighted);
+ }
+ function utilGetAllNodes(ids, graph) {
+ var seen = /* @__PURE__ */ new Set();
+ var nodes = /* @__PURE__ */ new Set();
+ ids.forEach(collectNodes);
+ return Array.from(nodes);
+ function collectNodes(id2) {
+ if (seen.has(id2))
+ return;
+ seen.add(id2);
+ var entity = graph.hasEntity(id2);
+ if (!entity)
+ return;
+ 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);
+ }
+ }
+ }
+ 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 = [];
+ if (tags.network) {
+ keyComponents.push("network");
+ }
+ if (tags.ref) {
+ keyComponents.push("ref");
+ }
+ if (entity.tags.route) {
+ if (tags.direction) {
+ keyComponents.push("direction");
+ } else if (tags.from && tags.to) {
+ keyComponents.push("from");
+ keyComponents.push("to");
+ if (tags.via) {
+ keyComponents.push("via");
+ }
+ }
+ }
+ if (keyComponents.length) {
+ name = _t("inspector.display_name." + keyComponents.join("_"), tags);
+ }
+ return name;
+ }
+ function utilDisplayNameForPath(entity) {
+ var name = utilDisplayName(entity);
+ var isFirefox = utilDetect().browser.toLowerCase().indexOf("firefox") > -1;
+ var isNewChromium = Number(utilDetect().version.split(".")[0]) >= 96;
+ if (!isFirefox && !isNewChromium && name && rtlRegex.test(name)) {
+ name = fixRTLTextForSvg(name);
+ }
+ return name;
+ }
+ function utilDisplayType(id2) {
+ return {
+ n: _t("inspector.node"),
+ w: _t("inspector.way"),
+ r: _t("inspector.relation")
+ }[id2.charAt(0)];
+ }
+ 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());
+ if (verbose) {
+ result = [presetName, displayName].filter(Boolean).join(" ");
+ } else {
+ result = displayName || presetName;
+ }
+ return result || utilDisplayType(entity.id);
+ }
+ function utilEntityRoot(entityType) {
+ return {
+ node: "n",
+ way: "w",
+ relation: "r"
+ }[entityType];
+ }
+ function utilCombinedTags(entityIDs, graph) {
+ var tags = {};
+ var tagCounts = {};
+ var allKeys = /* @__PURE__ */ new Set();
+ var entities = entityIDs.map(function(entityID) {
+ return graph.hasEntity(entityID);
+ }).filter(Boolean);
+ entities.forEach(function(entity) {
+ var keys = Object.keys(entity.tags).filter(Boolean);
+ keys.forEach(function(key2) {
+ allKeys.add(key2);
+ });
+ });
+ entities.forEach(function(entity) {
+ allKeys.forEach(function(key2) {
+ var value = entity.tags[key2];
+ if (!tags.hasOwnProperty(key2)) {
+ tags[key2] = value;
+ } else {
+ if (!Array.isArray(tags[key2])) {
+ if (tags[key2] !== value) {
+ tags[key2] = [tags[key2], value];
+ }
+ } else {
+ if (tags[key2].indexOf(value) === -1) {
+ tags[key2].push(value);
+ }
+ }
+ }
+ var tagHash = key2 + "=" + value;
+ if (!tagCounts[tagHash])
+ tagCounts[tagHash] = 0;
+ tagCounts[tagHash] += 1;
+ });
+ });
+ for (var key in tags) {
+ if (!Array.isArray(tags[key]))
+ continue;
+ tags[key] = tags[key].sort(function(val12, val2) {
+ var key2 = key2;
+ var count2 = tagCounts[key2 + "=" + val2];
+ var count1 = tagCounts[key2 + "=" + val12];
+ if (count2 !== count1) {
+ return count2 - count1;
+ }
+ if (val2 && val12) {
+ return val12.localeCompare(val2);
+ }
+ return val12 ? 1 : -1;
+ });
+ }
+ return tags;
+ }
+ function utilStringQs(str2) {
+ var i2 = 0;
+ while (i2 < str2.length && (str2[i2] === "?" || str2[i2] === "#"))
+ i2++;
+ str2 = str2.slice(i2);
+ return str2.split("&").reduce(function(obj, pair2) {
+ var parts = pair2.split("=");
+ if (parts.length === 2) {
+ obj[parts[0]] = null === parts[1] ? "" : decodeURIComponent(parts[1]);
+ }
+ return obj;
+ }, {});
+ }
+ function utilQsString(obj, noencode) {
+ function softEncode(s) {
+ return encodeURIComponent(s).replace(/(%2F|%3A|%2C|%7B|%7D)/g, decodeURIComponent);
+ }
+ return Object.keys(obj).sort().map(function(key) {
+ return encodeURIComponent(key) + "=" + (noencode ? softEncode(obj[key]) : encodeURIComponent(obj[key]));
+ }).join("&");
+ }
+ function utilPrefixDOMProperty(property) {
+ var prefixes2 = ["webkit", "ms", "moz", "o"];
+ var i2 = -1;
+ var n2 = prefixes2.length;
+ var s = document.body;
+ if (property in s)
+ return property;
+ property = property.slice(0, 1).toUpperCase() + property.slice(1);
+ while (++i2 < n2) {
+ if (prefixes2[i2] + property in s) {
+ return prefixes2[i2] + property;
+ }
+ }
+ return false;
+ }
+ function utilPrefixCSSProperty(property) {
+ var prefixes2 = ["webkit", "ms", "Moz", "O"];
+ var i2 = -1;
+ var n2 = prefixes2.length;
+ var s = document.body.style;
+ if (property.toLowerCase() in s) {
+ return property.toLowerCase();
+ }
+ while (++i2 < n2) {
+ if (prefixes2[i2] + property in s) {
+ return "-" + prefixes2[i2].toLowerCase() + property.replace(/([A-Z])/g, "-$1").toLowerCase();
+ }
+ }
+ return false;
+ }
+ 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 + ")" : ""));
+ }
+ function utilEditDistance(a, b) {
+ a = (0, import_diacritics.remove)(a.toLowerCase());
+ b = (0, import_diacritics.remove)(b.toLowerCase());
+ if (a.length === 0)
+ return b.length;
+ if (b.length === 0)
+ return a.length;
+ var matrix = [];
+ var i2, j2;
+ for (i2 = 0; i2 <= b.length; i2++) {
+ matrix[i2] = [i2];
+ }
+ for (j2 = 0; j2 <= a.length; j2++) {
+ matrix[0][j2] = j2;
+ }
+ for (i2 = 1; i2 <= b.length; i2++) {
+ for (j2 = 1; j2 <= a.length; j2++) {
+ if (b.charAt(i2 - 1) === a.charAt(j2 - 1)) {
+ matrix[i2][j2] = matrix[i2 - 1][j2 - 1];
+ } else {
+ matrix[i2][j2] = Math.min(
+ matrix[i2 - 1][j2 - 1] + 1,
+ // substitution
+ Math.min(
+ matrix[i2][j2 - 1] + 1,
+ // insertion
+ matrix[i2 - 1][j2] + 1
+ )
+ );
+ }
+ }
+ }
+ return matrix[b.length][a.length];
+ }
+ 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, i2) {
+ func(d, function done(err, data) {
+ errors[i2] = err;
+ results[i2] = data;
+ remaining--;
+ if (!remaining)
+ callback(errors, results);
+ });
+ });
+ }
+ function utilWrap(index, length) {
+ if (index < 0) {
+ index += Math.ceil(-index / length) * length;
+ }
+ return index % length;
+ }
+ function utilFunctor(value) {
+ if (typeof value === "function")
+ return value;
+ return function() {
+ return value;
+ };
+ }
+ function utilNoAuto(selection2) {
+ var isText = selection2.size() && selection2.node().tagName.toLowerCase() === "textarea";
+ return selection2.attr("autocomplete", "new-password").attr("autocorrect", "off").attr("autocapitalize", "off").attr("spellcheck", isText ? "true" : "false");
+ }
+ function utilHashcode(str2) {
+ var hash = 0;
+ if (str2.length === 0) {
+ return hash;
+ }
+ for (var i2 = 0; i2 < str2.length; i2++) {
+ var char = str2.charCodeAt(i2);
+ hash = (hash << 5) - hash + char;
+ hash = hash & hash;
+ }
+ return hash;
+ }
+ function utilSafeClassName(str2) {
+ return str2.toLowerCase().replace(/[^a-z0-9]+/g, "_");
+ }
+ function utilUniqueDomId(val) {
+ return "ideditor-" + utilSafeClassName(val.toString()) + "-" + new Date().getTime().toString();
+ }
+ function utilUnicodeCharsCount(str2) {
+ return Array.from(str2).length;
+ }
+ function utilUnicodeCharsTruncated(str2, limit) {
+ return Array.from(str2).slice(0, limit).join("");
+ }
+ function toNumericID(id2) {
+ var match = id2.match(/^[cnwr](-?\d+)$/);
+ if (match) {
+ return parseInt(match[1], 10);
+ }
+ return NaN;
+ }
+ function compareNumericIDs(left, right) {
+ if (isNaN(left) && isNaN(right))
+ return -1;
+ if (isNaN(left))
+ return 1;
+ if (isNaN(right))
+ return -1;
+ if (Math.sign(left) !== Math.sign(right))
+ return -Math.sign(left);
+ if (Math.sign(left) < 0)
+ return Math.sign(right - left);
+ return Math.sign(left - right);
+ }
+ function utilCompareIDs(left, right) {
+ return compareNumericIDs(toNumericID(left), toNumericID(right));
+ }
+ function utilOldestID(ids) {
+ if (ids.length === 0) {
+ return void 0;
+ }
+ var oldestIDIndex = 0;
+ var oldestID = toNumericID(ids[0]);
+ for (var i2 = 1; i2 < ids.length; i2++) {
+ var num = toNumericID(ids[i2]);
+ if (compareNumericIDs(oldestID, num) === 1) {
+ oldestIDIndex = i2;
+ oldestID = num;
+ }
+ }
+ return ids[oldestIDIndex];
+ }
+ function utilCleanOsmString(val, maxChars) {
+ if (val === void 0 || val === null) {
+ val = "";
+ } else {
+ val = val.toString();
+ }
+ val = val.trim();
+ if (val.normalize)
+ val = val.normalize("NFC");
+ return utilUnicodeCharsTruncated(val, maxChars);
+ }
+
+ // modules/osm/entity.js
+ function osmEntity(attrs) {
+ if (this instanceof osmEntity)
+ return;
+ 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);
+ }
+ return new osmEntity().initialize(arguments);
+ }
+ osmEntity.id = function(type2) {
+ return osmEntity.id.fromOSM(type2, osmEntity.id.next[type2]--);
+ };
+ osmEntity.id.next = {
+ changeset: -1,
+ node: -1,
+ way: -1,
+ relation: -1
+ };
+ osmEntity.id.fromOSM = function(type2, id2) {
+ return type2[0] + id2;
+ };
+ osmEntity.id.toOSM = function(id2) {
+ var match = id2.match(/^[cnwr](-?\d+)$/);
+ if (match) {
+ return match[1];
+ }
+ return "";
+ };
+ osmEntity.id.type = function(id2) {
+ return { "c": "changeset", "n": "node", "w": "way", "r": "relation" }[id2[0]];
+ };
+ osmEntity.key = function(entity) {
+ return entity.id + "v" + (entity.v || 0);
+ };
+ var _deprecatedTagValuesByKey;
+ osmEntity.deprecatedTagValuesByKey = function(dataDeprecated) {
+ if (!_deprecatedTagValuesByKey) {
+ _deprecatedTagValuesByKey = {};
+ dataDeprecated.forEach(function(d) {
+ var oldKeys = Object.keys(d.old);
+ 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);
+ }
+ }
+ }
+ });
+ }
+ return _deprecatedTagValuesByKey;
+ };
+ osmEntity.prototype = {
+ tags: {},
+ initialize: function(sources) {
+ for (var i2 = 0; i2 < sources.length; ++i2) {
+ var source = sources[i2];
+ for (var prop in source) {
+ if (Object.prototype.hasOwnProperty.call(source, prop)) {
+ if (source[prop] === void 0) {
+ delete this[prop];
+ } else {
+ this[prop] = source[prop];
+ }
+ }
+ }
+ }
+ if (!this.id && this.type) {
+ this.id = osmEntity.id(this.type);
+ }
+ if (!this.hasOwnProperty("visible")) {
+ this.visible = true;
+ }
+ 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);
+ }
+ return this;
+ },
+ copy: function(resolver, copies) {
+ if (copies[this.id])
+ return copies[this.id];
+ var copy2 = osmEntity(this, { id: void 0, user: void 0, version: void 0 });
+ copies[this.id] = copy2;
+ return copy2;
+ },
+ osmId: function() {
+ return osmEntity.id.toOSM(this.id);
+ },
+ isNew: function() {
+ var osmId = osmEntity.id.toOSM(this.id);
+ return osmId.length === 0 || osmId[0] === "-";
+ },
+ update: function(attrs) {
+ return osmEntity(this, attrs, { v: 1 + (this.v || 0) });
+ },
+ mergeTags: function(tags) {
+ var merged = Object.assign({}, this.tags);
+ var changed = false;
+ for (var k in tags) {
+ var t1 = merged[k];
+ var t2 = tags[k];
+ 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 context.maxCharsForTagValue()
+ );
+ }
+ }
+ return changed ? this.update({ tags: merged }) : this;
+ },
+ intersects: function(extent, resolver) {
+ return this.extent(resolver).intersects(extent);
+ },
+ hasNonGeometryTags: function() {
+ return Object.keys(this.tags).some(function(k) {
+ return k !== "area";
+ });
+ },
+ hasParentRelations: function(resolver) {
+ return resolver.parentRelations(this).length > 0;
+ },
+ hasInterestingTags: function() {
+ return Object.keys(this.tags).some(osmIsInterestingTag);
+ },
+ isHighwayIntersection: function() {
+ return false;
+ },
+ isDegenerate: function() {
+ return true;
+ },
+ deprecatedTags: function(dataDeprecated) {
+ var tags = this.tags;
+ 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;
+ });
+ if (hasExistingValues)
+ return;
+ }
+ 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;
+ }
+ }
+ }
+ return false;
+ });
+ if (matchesDeprecatedTags) {
+ deprecated.push(d);
+ }
+ });
+ return deprecated;
+ }
+ };
+
+ // modules/osm/lanes.js
+ 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;
+ 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: []
+ };
+ 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,
+ backward,
+ bothways,
+ turnLanes,
+ maxspeed,
+ maxspeedLanes,
+ psvLanes,
+ busLanes,
+ taxiLanes,
+ hovLanes,
+ hgvLanes,
+ bicyclewayLanes
+ },
+ lanes: lanesObj
+ };
+ }
+ 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;
+ default:
+ count = isOneWay ? 1 : 2;
+ break;
+ }
+ 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 {
+ forward,
+ backward,
+ bothways
+ };
+ }
+ 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 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 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 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 mapToLanesObj(lanesObj, data, key) {
+ if (data.forward) {
+ data.forward.forEach(function(l, i2) {
+ if (!lanesObj.forward[i2])
+ lanesObj.forward[i2] = {};
+ lanesObj.forward[i2][key] = l;
+ });
+ }
+ if (data.backward) {
+ data.backward.forEach(function(l, i2) {
+ if (!lanesObj.backward[i2])
+ lanesObj.backward[i2] = {};
+ lanesObj.backward[i2][key] = l;
+ });
+ }
+ if (data.unspecified) {
+ data.unspecified.forEach(function(l, i2) {
+ if (!lanesObj.unspecified[i2])
+ lanesObj.unspecified[i2] = {};
+ lanesObj.unspecified[i2][key] = l;
+ });
+ }
+ }
+
+ // modules/osm/way.js
+ 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(resolver, copies) {
+ if (copies[this.id])
+ return copies[this.id];
+ var copy2 = osmEntity.prototype.copy.call(this, resolver, copies);
+ var nodes = this.nodes.map(function(id2) {
+ return resolver.entity(id2).copy(resolver, copies).id;
+ });
+ copy2 = copy2.update({ nodes });
+ copies[this.id] = copy2;
+ return copy2;
+ },
+ extent: function(resolver) {
+ return resolver.transient(this, "extent", function() {
+ var extent = geoExtent();
+ for (var i2 = 0; i2 < this.nodes.length; i2++) {
+ var node = resolver.hasEntity(this.nodes[i2]);
+ if (node) {
+ extent._extend(node.extent());
+ }
+ }
+ return extent;
+ });
+ },
+ first: function() {
+ return this.nodes[0];
+ },
+ last: function() {
+ return this.nodes[this.nodes.length - 1];
+ },
+ contains: function(node) {
+ return this.nodes.indexOf(node) >= 0;
+ },
+ affix: function(node) {
+ if (this.nodes[0] === node)
+ return "prefix";
+ if (this.nodes[this.nodes.length - 1] === node)
+ return "suffix";
+ },
+ layer: function() {
+ if (isFinite(this.tags.layer)) {
+ return Math.max(-10, Math.min(+this.tags.layer, 10));
+ }
+ 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() {
+ 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]];
+ if (key === "highway") {
+ var laneCount = this.tags.lanes && parseInt(this.tags.lanes, 10);
+ if (!laneCount)
+ laneCount = this.isOneWay() ? 1 : 2;
+ return width * laneCount;
+ }
+ return width;
+ }
+ }
+ return null;
+ },
+ isOneWay: function() {
+ var values = {
+ "yes": true,
+ "1": true,
+ "-1": true,
+ "reversible": true,
+ "alternating": true,
+ "no": false,
+ "0": false
+ };
+ if (values[this.tags.oneway] !== void 0) {
+ return values[this.tags.oneway];
+ }
+ 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() {
+ 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 {
+ return osmRightSideIsInsideTags[key][value];
+ }
+ }
+ }
+ return null;
+ },
+ isSided: function() {
+ if (this.tags.two_sided === "yes") {
+ return false;
+ }
+ return this.sidednessIdentifier() !== null;
+ },
+ lanes: function() {
+ return osmLanes(this);
+ },
+ isClosed: function() {
+ return this.nodes.length > 1 && this.first() === this.last();
+ },
+ isConvex: function(resolver) {
+ if (!this.isClosed() || this.isDegenerate())
+ return null;
+ var nodes = utilArrayUniq(resolver.childNodes(this));
+ var coords = nodes.map(function(n2) {
+ return n2.loc;
+ });
+ var curr = 0;
+ var prev = 0;
+ for (var i2 = 0; i2 < coords.length; i2++) {
+ var o = coords[(i2 + 1) % coords.length];
+ var a = coords[i2];
+ var b = coords[(i2 + 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;
+ }
+ prev = curr;
+ }
+ return true;
+ },
+ // returns an object with the tag that implies this is an area, if any
+ tagSuggestingArea: function() {
+ return osmTagSuggestingArea(this.tags);
+ },
+ isArea: function() {
+ if (this.tags.area === "yes")
+ return true;
+ if (!this.isClosed() || this.tags.area === "no")
+ return false;
+ return this.tagSuggestingArea() !== null;
+ },
+ isDegenerate: function() {
+ return new Set(this.nodes).size < (this.isArea() ? 3 : 2);
+ },
+ areAdjacent: function(n1, n2) {
+ for (var i2 = 0; i2 < this.nodes.length; i2++) {
+ if (this.nodes[i2] === n1) {
+ if (this.nodes[i2 - 1] === n2)
+ return true;
+ if (this.nodes[i2 + 1] === n2)
+ return true;
+ }
+ }
+ return false;
+ },
+ geometry: function(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(graph) {
+ function segmentExtent(graph2) {
+ var n1 = graph2.hasEntity(this.nodes[0]);
+ var n2 = graph2.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])
+ ]
+ ]);
+ }
+ return graph.transient(this, "segments", function() {
+ var segments = [];
+ for (var i2 = 0; i2 < this.nodes.length - 1; i2++) {
+ segments.push({
+ id: this.id + "-" + i2,
+ wayId: this.id,
+ index: i2,
+ nodes: [this.nodes[i2], this.nodes[i2 + 1]],
+ extent: segmentExtent
+ });
+ }
+ return segments;
+ });
+ },
+ // If this way is not closed, append the beginning node to the end of the nodelist to close it.
+ close: function() {
+ 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 });
+ },
+ // If this way is closed, remove any connector nodes from the end of the nodelist to unclose it.
+ unclose: function() {
+ if (!this.isClosed())
+ return this;
+ var nodes = this.nodes.slice();
+ var connector = this.first();
+ var i2 = nodes.length - 1;
+ while (i2 > 0 && nodes.length > 1 && nodes[i2] === connector) {
+ nodes.splice(i2, 1);
+ i2 = nodes.length - 1;
+ }
+ nodes = nodes.filter(noRepeatNodes);
+ return this.update({ 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(id2, index) {
+ var nodes = this.nodes.slice();
+ var isClosed = this.isClosed();
+ var max3 = isClosed ? nodes.length - 1 : nodes.length;
+ if (index === void 0) {
+ index = max3;
+ }
+ if (index < 0 || index > max3) {
+ throw new RangeError("index " + index + " out of range 0.." + max3);
+ }
+ if (isClosed) {
+ var connector = this.first();
+ var i2 = 1;
+ while (i2 < nodes.length && nodes.length > 2 && nodes[i2] === connector) {
+ nodes.splice(i2, 1);
+ if (index > i2)
+ index--;
+ }
+ i2 = nodes.length - 1;
+ while (i2 > 0 && nodes.length > 1 && nodes[i2] === connector) {
+ nodes.splice(i2, 1);
+ if (index > i2)
+ index--;
+ i2 = nodes.length - 1;
+ }
+ }
+ nodes.splice(index, 0, id2);
+ nodes = nodes.filter(noRepeatNodes);
+ if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+ nodes.push(nodes[0]);
+ }
+ return this.update({ 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(id2, index) {
+ var nodes = this.nodes.slice();
+ var isClosed = this.isClosed();
+ var max3 = nodes.length - 1;
+ if (index === void 0 || index < 0 || index > max3) {
+ throw new RangeError("index " + index + " out of range 0.." + max3);
+ }
+ if (isClosed) {
+ var connector = this.first();
+ var i2 = 1;
+ while (i2 < nodes.length && nodes.length > 2 && nodes[i2] === connector) {
+ nodes.splice(i2, 1);
+ if (index > i2)
+ index--;
+ }
+ i2 = nodes.length - 1;
+ while (i2 > 0 && nodes.length > 1 && nodes[i2] === connector) {
+ nodes.splice(i2, 1);
+ if (index === i2)
+ index = 0;
+ i2 = nodes.length - 1;
+ }
+ }
+ nodes.splice(index, 1, id2);
+ nodes = nodes.filter(noRepeatNodes);
+ if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+ nodes.push(nodes[0]);
+ }
+ return this.update({ nodes });
+ },
+ // Replaces each occurrence of node id needle with replacement.
+ // Consecutive duplicates are eliminated including existing ones.
+ // Circularity is preserved.
+ replaceNode: function(needleID, replacementID) {
+ var nodes = this.nodes.slice();
+ var isClosed = this.isClosed();
+ for (var i2 = 0; i2 < nodes.length; i2++) {
+ if (nodes[i2] === needleID) {
+ nodes[i2] = replacementID;
+ }
+ }
+ nodes = nodes.filter(noRepeatNodes);
+ if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+ nodes.push(nodes[0]);
+ }
+ return this.update({ nodes });
+ },
+ // Removes each occurrence of node id.
+ // Consecutive duplicates are eliminated including existing ones.
+ // Circularity is preserved.
+ removeNode: function(id2) {
+ var nodes = this.nodes.slice();
+ var isClosed = this.isClosed();
+ nodes = nodes.filter(function(node) {
+ return node !== id2;
+ }).filter(noRepeatNodes);
+ if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+ nodes.push(nodes[0]);
+ }
+ return this.update({ nodes });
+ },
+ asJXON: function(changeset_id) {
+ var r = {
+ way: {
+ "@id": this.osmId(),
+ "@version": this.version || 0,
+ nd: this.nodes.map(function(id2) {
+ return { keyAttributes: { ref: osmEntity.id.toOSM(id2) } };
+ }, this),
+ tag: Object.keys(this.tags).map(function(k) {
+ return { keyAttributes: { k, v: this.tags[k] } };
+ }, this)
+ }
+ };
+ if (changeset_id) {
+ r.way["@changeset"] = changeset_id;
+ }
+ return r;
+ },
+ asGeoJSON: function(resolver) {
+ return resolver.transient(this, "GeoJSON", function() {
+ var coordinates = resolver.childNodes(this).map(function(n2) {
+ return n2.loc;
+ });
+ if (this.isArea() && this.isClosed()) {
+ return {
+ type: "Polygon",
+ coordinates: [coordinates]
+ };
+ } else {
+ return {
+ type: "LineString",
+ coordinates
+ };
+ }
+ });
+ },
+ area: function(resolver) {
+ return resolver.transient(this, "area", function() {
+ var nodes = resolver.childNodes(this);
+ var json = {
+ type: "Polygon",
+ coordinates: [nodes.map(function(n2) {
+ return n2.loc;
+ })]
+ };
+ if (!this.isClosed() && nodes.length) {
+ json.coordinates[0].push(nodes[0].loc);
+ }
+ var area = area_default(json);
+ if (area > 2 * Math.PI) {
+ json.coordinates[0] = json.coordinates[0].reverse();
+ area = area_default(json);
+ }
+ return isNaN(area) ? 0 : area;
+ });
+ }
+ });
+ function noRepeatNodes(node, i2, arr) {
+ return i2 === 0 || node !== arr[i2 - 1];
+ }
+
+ // modules/osm/multipolygon.js
+ function osmOldMultipolygonOuterMemberOfRelation(entity, graph) {
+ if (entity.type !== "relation" || !entity.isMultipolygon() || Object.keys(entity.tags).filter(osmIsInterestingTag).length > 1) {
+ return false;
+ }
+ var outerMember;
+ for (var memberIndex in entity.members) {
+ var member = entity.members[memberIndex];
+ 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 (Object.keys(outerMember.tags).filter(osmIsInterestingTag).length === 0) {
+ return false;
+ }
+ }
+ }
+ return outerMember;
+ }
+ 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;
+ for (var i2 = 0; i2 < members.length; i2++) {
+ member = members[i2];
+ if (member.id === entity.id && member.role && member.role !== "outer") {
+ return false;
+ }
+ if (member.id !== entity.id && (!member.role || member.role === "outer")) {
+ return false;
+ }
+ }
+ 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;
+ for (var i2 = 0; i2 < members.length; i2++) {
+ member = members[i2];
+ if (!member.role || member.role === "outer") {
+ if (outerMember)
+ return false;
+ outerMember = member;
+ }
+ }
+ if (!outerMember)
+ return false;
+ var outerEntity = graph.hasEntity(outerMember.id);
+ if (!outerEntity || !Object.keys(outerEntity.tags).filter(osmIsInterestingTag).length) {
+ return false;
+ }
+ return outerEntity;
+ }
+ function osmJoinWays(toJoin, graph) {
+ function resolve(member) {
+ return graph.childNodes(graph.entity(member.id));
+ }
+ function reverse(item2) {
+ var action = actionReverse(item2.id, { reverseOneway: true });
+ sequences.actions.push(action);
+ return item2 instanceof osmWay ? action(graph).entity(item2.id) : item2;
+ }
+ toJoin = toJoin.filter(function(member) {
+ return member.type === "way" && graph.hasEntity(member.id);
+ });
+ var i2;
+ var joinAsMembers = true;
+ for (i2 = 0; i2 < toJoin.length; i2++) {
+ if (toJoin[i2] instanceof osmWay) {
+ joinAsMembers = false;
+ break;
+ }
+ }
+ var sequences = [];
+ sequences.actions = [];
+ while (toJoin.length) {
+ var item = toJoin.shift();
+ var currWays = [item];
+ var currNodes = resolve(item).slice();
+ while (toJoin.length) {
+ var start2 = currNodes[0];
+ var end = currNodes[currNodes.length - 1];
+ var fn = null;
+ var nodes = null;
+ for (i2 = 0; i2 < toJoin.length; i2++) {
+ item = toJoin[i2];
+ nodes = resolve(item);
+ if (joinAsMembers && currWays.length === 1 && nodes[0] !== end && nodes[nodes.length - 1] !== end && (nodes[nodes.length - 1] === start2 || nodes[0] === start2)) {
+ currWays[0] = reverse(currWays[0]);
+ currNodes.reverse();
+ start2 = currNodes[0];
+ end = currNodes[currNodes.length - 1];
+ }
+ if (nodes[0] === end) {
+ fn = currNodes.push;
+ nodes = nodes.slice(1);
+ break;
+ } else if (nodes[nodes.length - 1] === end) {
+ fn = currNodes.push;
+ nodes = nodes.slice(0, -1).reverse();
+ item = reverse(item);
+ break;
+ } else if (nodes[nodes.length - 1] === start2) {
+ fn = currNodes.unshift;
+ nodes = nodes.slice(0, -1);
+ break;
+ } else if (nodes[0] === start2) {
+ fn = currNodes.unshift;
+ nodes = nodes.slice(1).reverse();
+ item = reverse(item);
+ break;
+ } else {
+ fn = nodes = null;
+ }
+ }
+ if (!nodes) {
+ break;
+ }
+ fn.apply(currWays, [item]);
+ fn.apply(currNodes, nodes);
+ toJoin.splice(i2, 1);
+ }
+ currWays.nodes = currNodes;
+ sequences.push(currWays);
+ }
+ return sequences;
+ }
+
+ // modules/actions/add_member.js
+ function actionAddMember(relationId, member, memberIndex, insertPair) {
+ return function action(graph) {
+ var relation = graph.entity(relationId);
+ var isPTv2 = /stop|platform/.test(member.role);
+ if ((isNaN(memberIndex) || insertPair) && member.type === "way" && !isPTv2) {
+ graph = addWayMember(relation, graph);
+ } else {
+ if (isPTv2 && isNaN(memberIndex)) {
+ memberIndex = 0;
+ }
+ graph = graph.replace(relation.addMember(member, memberIndex));
+ }
+ return graph;
+ };
+ function addWayMember(relation, graph) {
+ var groups, tempWay, insertPairIsReversed, item, i2, j2, k;
+ var PTv2members = [];
+ var members = [];
+ for (i2 = 0; i2 < relation.members.length; i2++) {
+ var m = relation.members[i2];
+ if (/stop|platform/.test(m.role)) {
+ PTv2members.push(m);
+ } else {
+ members.push(m);
+ }
+ }
+ relation = relation.update({ members });
+ if (insertPair) {
+ 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 || [];
+ var originalWay = graph.entity(insertPair.originalID);
+ var insertedWay = graph.entity(insertPair.insertedID);
+ insertPairIsReversed = originalWay.nodes.length > 0 && insertedWay.nodes.length > 0 && insertedWay.nodes[insertedWay.nodes.length - 1] === originalWay.nodes[0] && originalWay.nodes[originalWay.nodes.length - 1] !== insertedWay.nodes[0];
+ } else {
+ groups = utilArrayGroupBy(relation.members, "type");
+ groups.way = groups.way || [];
+ groups.way.push(member);
+ }
+ members = withIndex(groups.way);
+ var joined = osmJoinWays(members, graph);
+ for (i2 = 0; i2 < joined.length; i2++) {
+ var segment = joined[i2];
+ var nodes = segment.nodes.slice();
+ var startIndex = segment[0].index;
+ for (j2 = 0; j2 < members.length; j2++) {
+ if (members[j2].index === startIndex) {
+ break;
+ }
+ }
+ for (k = 0; k < segment.length; k++) {
+ item = segment[k];
+ var way = graph.entity(item.id);
+ if (tempWay && item.id === tempWay.id) {
+ var reverse = nodes[0].id !== insertPair.nodes[0] ^ insertPairIsReversed;
+ if (reverse) {
+ item.pair = [
+ { id: insertPair.insertedID, type: "way", role: item.role },
+ { id: insertPair.originalID, type: "way", role: item.role }
+ ];
+ } else {
+ item.pair = [
+ { id: insertPair.originalID, type: "way", role: item.role },
+ { id: insertPair.insertedID, type: "way", role: item.role }
+ ];
+ }
+ }
+ if (k > 0) {
+ if (j2 + k >= members.length || item.index !== members[j2 + k].index) {
+ moveMember(members, item.index, j2 + k);
+ }
+ }
+ nodes.splice(0, way.nodes.length - 1);
+ }
+ }
+ if (tempWay) {
+ graph = graph.remove(tempWay);
+ }
+ var wayMembers = [];
+ for (i2 = 0; i2 < members.length; i2++) {
+ item = members[i2];
+ if (item.index === -1)
+ continue;
+ if (item.pair) {
+ wayMembers.push(item.pair[0]);
+ wayMembers.push(item.pair[1]);
+ } else {
+ wayMembers.push(utilObjectOmit(item, ["index"]));
+ }
+ }
+ var newMembers = PTv2members.concat(groups.node || [], wayMembers, groups.relation || []);
+ return graph.replace(relation.update({ members: newMembers }));
+ function moveMember(arr, findIndex, toIndex) {
+ var i3;
+ for (i3 = 0; i3 < arr.length; i3++) {
+ if (arr[i3].index === findIndex) {
+ break;
+ }
+ }
+ var item2 = Object.assign({}, arr[i3]);
+ arr[i3].index = -1;
+ item2.index = toIndex;
+ arr.splice(toIndex, 0, item2);
+ }
+ function withIndex(arr) {
+ var result = new Array(arr.length);
+ for (var i3 = 0; i3 < arr.length; i3++) {
+ result[i3] = Object.assign({}, arr[i3]);
+ result[i3].index = i3;
+ }
+ return result;
+ }
+ }
+ }
+
+ // modules/actions/add_midpoint.js
+ 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 i2 = 0; i2 < way.nodes.length - 1; i2++) {
+ if (geoEdgeEqual([way.nodes[i2], way.nodes[i2 + 1]], midpoint.edge)) {
+ graph = graph.replace(graph.entity(way.id).addNode(node.id, i2 + 1));
+ return;
+ }
+ }
+ });
+ return graph;
+ };
+ }
+
+ // modules/actions/add_vertex.js
+ function actionAddVertex(wayId, nodeId, index) {
+ return function(graph) {
+ return graph.replace(graph.entity(wayId).addNode(nodeId, index));
+ };
+ }
+
+ // modules/actions/change_member.js
+ function actionChangeMember(relationId, member, memberIndex) {
+ return function(graph) {
+ return graph.replace(graph.entity(relationId).updateMember(member, memberIndex));
+ };
+ }
+
+ // modules/actions/change_preset.js
+ function actionChangePreset(entityID, oldPreset, newPreset, skipFieldDefaults) {
+ return function action(graph) {
+ var entity = graph.entity(entityID);
+ var geometry = entity.geometry(graph);
+ var tags = entity.tags;
+ var preserveKeys;
+ if (newPreset) {
+ preserveKeys = [];
+ if (newPreset.addTags) {
+ preserveKeys = preserveKeys.concat(Object.keys(newPreset.addTags));
+ }
+ if (oldPreset && !oldPreset.id.startsWith(newPreset.id)) {
+ newPreset.fields().concat(newPreset.moreFields()).filter((f2) => f2.matchGeometry(geometry)).map((f2) => f2.key).filter(Boolean).forEach((key) => preserveKeys.push(key));
+ }
+ }
+ if (oldPreset)
+ tags = oldPreset.unsetTags(tags, geometry, preserveKeys);
+ if (newPreset)
+ tags = newPreset.setTags(tags, geometry, skipFieldDefaults);
+ return graph.replace(entity.update({ tags }));
+ };
+ }
+
+ // modules/actions/change_tags.js
+ function actionChangeTags(entityId, tags) {
+ return function(graph) {
+ var entity = graph.entity(entityId);
+ return graph.replace(entity.update({ tags }));
+ };
+ }
+
+ // modules/osm/node.js
+ 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
+ };
+ 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() {
+ return new geoExtent(this.loc);
+ },
+ geometry: function(graph) {
+ return graph.transient(this, "geometry", function() {
+ return graph.isPoi(this) ? "point" : "vertex";
+ });
+ },
+ move: function(loc) {
+ return this.update({ loc });
+ },
+ isDegenerate: function() {
+ 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(resolver, projection2) {
+ var val;
+ var i2;
+ if (this.isHighwayIntersection(resolver) && (this.tags.stop || "").toLowerCase() === "all") {
+ val = "all";
+ } else {
+ val = (this.tags.direction || "").toLowerCase();
+ var re2 = /:direction$/i;
+ var keys = Object.keys(this.tags);
+ for (i2 = 0; i2 < keys.length; i2++) {
+ if (re2.test(keys[i2])) {
+ val = this.tags[keys[i2]].toLowerCase();
+ break;
+ }
+ }
+ }
+ if (val === "")
+ return [];
+ var values = val.split(";");
+ var results = [];
+ values.forEach(function(v) {
+ if (cardinal[v] !== void 0) {
+ v = cardinal[v];
+ }
+ if (v !== "" && !isNaN(+v)) {
+ results.push(+v);
+ return;
+ }
+ 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;
+ for (i2 = 0; i2 < nodes.length; i2++) {
+ if (nodes[i2] === this.id) {
+ if (lookForward && i2 > 0) {
+ nodeIds[nodes[i2 - 1]] = true;
+ }
+ if (lookBackward && i2 < nodes.length - 1) {
+ nodeIds[nodes[i2 + 1]] = true;
+ }
+ }
+ }
+ }, this);
+ Object.keys(nodeIds).forEach(function(nodeId) {
+ results.push(
+ geoAngle(this, resolver.entity(nodeId), projection2) * (180 / Math.PI) + 90
+ );
+ }, this);
+ }, this);
+ return utilArrayUniq(results);
+ },
+ isCrossing: function() {
+ return this.tags.highway === "crossing" || this.tags.railway && this.tags.railway.indexOf("crossing") !== -1;
+ },
+ isEndpoint: function(resolver) {
+ return resolver.transient(this, "isEndpoint", function() {
+ var id2 = this.id;
+ return resolver.parentWays(this).filter(function(parent) {
+ return !parent.isClosed() && !!parent.affix(id2);
+ }).length > 0;
+ });
+ },
+ isConnected: function(resolver) {
+ return resolver.transient(this, "isConnected", function() {
+ var parents = resolver.parentWays(this);
+ if (parents.length > 1) {
+ for (var i2 in parents) {
+ if (parents[i2].geometry(resolver) === "line" && parents[i2].hasInterestingTags())
+ return true;
+ }
+ } else if (parents.length === 1) {
+ var way = parents[0];
+ var nodes = way.nodes.slice();
+ if (way.isClosed()) {
+ nodes.pop();
+ }
+ return nodes.indexOf(this.id) !== nodes.lastIndexOf(this.id);
+ }
+ return false;
+ });
+ },
+ parentIntersectionWays: function(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(resolver) {
+ return this.parentIntersectionWays(resolver).length > 1;
+ },
+ isHighwayIntersection: function(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(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(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, v: this.tags[k] } };
+ }, this)
+ }
+ };
+ if (changeset_id)
+ r.node["@changeset"] = changeset_id;
+ return r;
+ },
+ asGeoJSON: function() {
+ return {
+ type: "Point",
+ coordinates: this.loc
+ };
+ }
+ });
+
+ // modules/actions/circularize.js
+ function actionCircularize(wayId, projection2, maxAngle) {
+ maxAngle = (maxAngle || 20) * Math.PI / 180;
+ var action = function(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(node2) {
+ if (!origNodes[node2.id])
+ origNodes[node2.id] = node2;
+ });
+ if (!way.isConvex(graph)) {
+ graph = action.makeConvex(graph);
+ }
+ var nodes = utilArrayUniq(graph.childNodes(way));
+ var keyNodes = nodes.filter(function(n2) {
+ return graph.parentWays(n2).length !== 1;
+ });
+ var points = nodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var keyPoints = keyNodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var centroid = points.length === 2 ? geoVecInterp(points[0], points[1], 0.5) : centroid_default2(points);
+ var radius = median(points, function(p) {
+ return geoVecLength(centroid, p);
+ });
+ var sign2 = area_default3(points) > 0 ? 1 : -1;
+ var ids, i2, j2, k;
+ if (!keyNodes.length) {
+ keyNodes = [nodes[0]];
+ keyPoints = [points[0]];
+ }
+ 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]);
+ }
+ for (i2 = 0; i2 < keyPoints.length; i2++) {
+ var nextKeyNodeIndex = (i2 + 1) % keyNodes.length;
+ var startNode = keyNodes[i2];
+ 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 angle2, loc, node, origNode;
+ if (indexRange < 0) {
+ indexRange += nodes.length;
+ }
+ var distance = geoVecLength(centroid, keyPoints[i2]) || 1e-4;
+ keyPoints[i2] = [
+ centroid[0] + (keyPoints[i2][0] - centroid[0]) / distance * radius,
+ centroid[1] + (keyPoints[i2][1] - centroid[1]) / distance * radius
+ ];
+ loc = projection2.invert(keyPoints[i2]);
+ node = keyNodes[i2];
+ origNode = origNodes[node.id];
+ node = node.move(geoVecInterp(origNode.loc, loc, t));
+ graph = graph.replace(node);
+ startAngle = Math.atan2(keyPoints[i2][1] - centroid[1], keyPoints[i2][0] - centroid[0]);
+ endAngle = Math.atan2(keyPoints[nextKeyNodeIndex][1] - centroid[1], keyPoints[nextKeyNodeIndex][0] - centroid[0]);
+ totalAngle = endAngle - startAngle;
+ if (totalAngle * sign2 > 0) {
+ totalAngle = -sign2 * (2 * Math.PI - Math.abs(totalAngle));
+ }
+ do {
+ numberNewPoints++;
+ eachAngle = totalAngle / (indexRange + numberNewPoints);
+ } while (Math.abs(eachAngle) > maxAngle);
+ for (j2 = 1; j2 < indexRange; j2++) {
+ angle2 = startAngle + j2 * eachAngle;
+ loc = projection2.invert([
+ centroid[0] + Math.cos(angle2) * radius,
+ centroid[1] + Math.sin(angle2) * radius
+ ]);
+ node = nodes[(j2 + startNodeIndex) % nodes.length];
+ origNode = origNodes[node.id];
+ nearNodes[node.id] = angle2;
+ node = node.move(geoVecInterp(origNode.loc, loc, t));
+ graph = graph.replace(node);
+ }
+ for (j2 = 0; j2 < numberNewPoints; j2++) {
+ angle2 = startAngle + (indexRange + j2) * eachAngle;
+ loc = projection2.invert([
+ centroid[0] + Math.cos(angle2) * radius,
+ centroid[1] + Math.sin(angle2) * radius
+ ]);
+ var min3 = Infinity;
+ for (var nodeId in nearNodes) {
+ var nearAngle = nearNodes[nodeId];
+ var dist = Math.abs(nearAngle - angle2);
+ if (dist < min3) {
+ min3 = dist;
+ origNode = origNodes[nodeId];
+ }
+ }
+ node = osmNode({ loc: geoVecInterp(origNode.loc, loc, t) });
+ graph = graph.replace(node);
+ nodes.splice(endNodeIndex + j2, 0, node);
+ inBetweenNodes.push(node.id);
+ }
+ if (indexRange === 1 && inBetweenNodes.length) {
+ var startIndex1 = way.nodes.lastIndexOf(startNode.id);
+ var endIndex1 = way.nodes.lastIndexOf(endNode.id);
+ var wayDirection1 = endIndex1 - startIndex1;
+ if (wayDirection1 < -1) {
+ wayDirection1 = 1;
+ }
+ var parentWays = graph.parentWays(keyNodes[i2]);
+ for (j2 = 0; j2 < parentWays.length; j2++) {
+ var sharedWay = parentWays[j2];
+ if (sharedWay === way)
+ continue;
+ 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;
+ if (wayDirection2 < -1) {
+ wayDirection2 = 1;
+ }
+ if (wayDirection1 !== wayDirection2) {
+ inBetweenNodes.reverse();
+ insertAt = startIndex2;
+ }
+ for (k = 0; k < inBetweenNodes.length; k++) {
+ sharedWay = sharedWay.addNode(inBetweenNodes[k], insertAt + k);
+ }
+ graph = graph.replace(sharedWay);
+ }
+ }
+ }
+ }
+ ids = nodes.map(function(n2) {
+ return n2.id;
+ });
+ ids.push(ids[0]);
+ way = way.update({ nodes: ids });
+ graph = graph.replace(way);
+ return graph;
+ };
+ action.makeConvex = function(graph) {
+ var way = graph.entity(wayId);
+ var nodes = utilArrayUniq(graph.childNodes(way));
+ var points = nodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var sign2 = area_default3(points) > 0 ? 1 : -1;
+ var hull = hull_default(points);
+ var i2, j2;
+ if (sign2 === -1) {
+ nodes.reverse();
+ points.reverse();
+ }
+ for (i2 = 0; i2 < hull.length - 1; i2++) {
+ var startIndex = points.indexOf(hull[i2]);
+ var endIndex = points.indexOf(hull[i2 + 1]);
+ var indexRange = endIndex - startIndex;
+ if (indexRange < 0) {
+ indexRange += nodes.length;
+ }
+ for (j2 = 1; j2 < indexRange; j2++) {
+ var point2 = geoVecInterp(hull[i2], hull[i2 + 1], j2 / indexRange);
+ var node = nodes[(j2 + startIndex) % nodes.length].move(projection2.invert(point2));
+ graph = graph.replace(node);
+ }
+ }
+ return graph;
+ };
+ action.disabled = function(graph) {
+ if (!graph.entity(wayId).isClosed()) {
+ return "not_closed";
+ }
+ var way = graph.entity(wayId);
+ var nodes = utilArrayUniq(graph.childNodes(way));
+ var points = nodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var hull = hull_default(points);
+ var epsilonAngle = Math.PI / 180;
+ if (hull.length !== points.length || hull.length < 3) {
+ return false;
+ }
+ var centroid = centroid_default2(points);
+ var radius = geoVecLengthSquare(centroid, points[0]);
+ var i2, actualPoint;
+ for (i2 = 0; i2 < hull.length; i2++) {
+ actualPoint = hull[i2];
+ var actualDist = geoVecLengthSquare(actualPoint, centroid);
+ var diff = Math.abs(actualDist - radius);
+ if (diff > 0.05 * radius) {
+ return false;
+ }
+ }
+ for (i2 = 0; i2 < hull.length; i2++) {
+ actualPoint = hull[i2];
+ var nextPoint = hull[(i2 + 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 angle2 = endAngle - startAngle;
+ if (angle2 < 0) {
+ angle2 = -angle2;
+ }
+ if (angle2 > Math.PI) {
+ angle2 = 2 * Math.PI - angle2;
+ }
+ if (angle2 > maxAngle + epsilonAngle) {
+ return false;
+ }
+ }
+ return "already_circular";
+ };
+ action.transitionable = true;
+ return action;
+ }
+
+ // modules/actions/delete_way.js
+ function actionDeleteWay(wayID) {
+ function canDeleteNode(node, graph) {
+ if (graph.parentWays(node).length || graph.parentRelations(node).length)
+ return false;
+ var geometries = osmNodeGeometriesForTags(node.tags);
+ if (geometries.point)
+ return false;
+ if (geometries.vertex)
+ return true;
+ return !node.hasInterestingTags();
+ }
+ var action = function(graph) {
+ var way = graph.entity(wayID);
+ graph.parentRelations(way).forEach(function(parent) {
+ parent = parent.removeMembersWithID(wayID);
+ graph = graph.replace(parent);
+ 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 (canDeleteNode(node, graph)) {
+ graph = graph.remove(node);
+ }
+ });
+ return graph.remove(way);
+ };
+ return action;
+ }
+
+ // modules/actions/delete_multiple.js
+ function actionDeleteMultiple(ids) {
+ var actions = {
+ way: actionDeleteWay,
+ node: actionDeleteNode,
+ relation: actionDeleteRelation
+ };
+ var action = function(graph) {
+ ids.forEach(function(id2) {
+ if (graph.hasEntity(id2)) {
+ graph = actions[graph.entity(id2).type](id2)(graph);
+ }
+ });
+ return graph;
+ };
+ return action;
+ }
+
+ // modules/actions/delete_relation.js
+ function actionDeleteRelation(relationID, allowUntaggedMembers) {
+ function canDeleteEntity(entity, graph) {
+ return !graph.parentWays(entity).length && !graph.parentRelations(entity).length && (!entity.hasInterestingTags() && !allowUntaggedMembers);
+ }
+ var action = function(graph) {
+ var relation = graph.entity(relationID);
+ graph.parentRelations(relation).forEach(function(parent) {
+ parent = parent.removeMembersWithID(relationID);
+ graph = graph.replace(parent);
+ 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);
+ if (canDeleteEntity(entity, graph)) {
+ graph = actionDeleteMultiple([memberID])(graph);
+ }
+ });
+ return graph.remove(relation);
+ };
+ return action;
+ }
+
+ // modules/actions/delete_node.js
+ function actionDeleteNode(nodeId) {
+ var action = function(graph) {
+ var node = graph.entity(nodeId);
+ graph.parentWays(node).forEach(function(parent) {
+ parent = parent.removeNode(nodeId);
+ graph = graph.replace(parent);
+ if (parent.isDegenerate()) {
+ graph = actionDeleteWay(parent.id)(graph);
+ }
+ });
+ graph.parentRelations(node).forEach(function(parent) {
+ parent = parent.removeMembersWithID(nodeId);
+ graph = graph.replace(parent);
+ if (parent.isDegenerate()) {
+ graph = actionDeleteRelation(parent.id)(graph);
+ }
+ });
+ return graph.remove(node);
+ };
+ return action;
+ }
+
+ // modules/actions/connect.js
+ function actionConnect(nodeIDs) {
+ var action = function(graph) {
+ var survivor;
+ var node;
+ var parents;
+ var i2, j2;
+ nodeIDs.reverse();
+ var interestingIDs = [];
+ for (i2 = 0; i2 < nodeIDs.length; i2++) {
+ node = graph.entity(nodeIDs[i2]);
+ if (node.hasInterestingTags()) {
+ if (!node.isNew()) {
+ interestingIDs.push(node.id);
+ }
+ }
+ }
+ survivor = graph.entity(utilOldestID(interestingIDs.length > 0 ? interestingIDs : nodeIDs));
+ for (i2 = 0; i2 < nodeIDs.length; i2++) {
+ node = graph.entity(nodeIDs[i2]);
+ if (node.id === survivor.id)
+ continue;
+ parents = graph.parentWays(node);
+ for (j2 = 0; j2 < parents.length; j2++) {
+ graph = graph.replace(parents[j2].replaceNode(node.id, survivor.id));
+ }
+ parents = graph.parentRelations(node);
+ for (j2 = 0; j2 < parents.length; j2++) {
+ graph = graph.replace(parents[j2].replaceMember(node, survivor));
+ }
+ survivor = survivor.mergeTags(node.tags);
+ graph = actionDeleteNode(node.id)(graph);
+ }
+ graph = graph.replace(survivor);
+ parents = graph.parentWays(survivor);
+ for (i2 = 0; i2 < parents.length; i2++) {
+ if (parents[i2].isDegenerate()) {
+ graph = actionDeleteWay(parents[i2].id)(graph);
+ }
+ }
+ return graph;
+ };
+ action.disabled = function(graph) {
+ var seen = {};
+ var restrictionIDs = [];
+ var survivor;
+ var node, way;
+ var relations, relation, role;
+ var i2, j2, k;
+ survivor = graph.entity(utilOldestID(nodeIDs));
+ for (i2 = 0; i2 < nodeIDs.length; i2++) {
+ node = graph.entity(nodeIDs[i2]);
+ relations = graph.parentRelations(node);
+ for (j2 = 0; j2 < relations.length; j2++) {
+ relation = relations[j2];
+ role = relation.memberById(node.id).role || "";
+ if (relation.hasFromViaTo()) {
+ restrictionIDs.push(relation.id);
+ }
+ if (seen[relation.id] !== void 0 && seen[relation.id] !== role) {
+ return "relation";
+ } else {
+ seen[relation.id] = role;
+ }
+ }
+ }
+ for (i2 = 0; i2 < nodeIDs.length; i2++) {
+ node = graph.entity(nodeIDs[i2]);
+ var parents = graph.parentWays(node);
+ for (j2 = 0; j2 < parents.length; j2++) {
+ var parent = parents[j2];
+ relations = graph.parentRelations(parent);
+ for (k = 0; k < relations.length; k++) {
+ relation = relations[k];
+ if (relation.hasFromViaTo()) {
+ restrictionIDs.push(relation.id);
+ }
+ }
+ }
+ }
+ restrictionIDs = utilArrayUniq(restrictionIDs);
+ for (i2 = 0; i2 < restrictionIDs.length; i2++) {
+ relation = graph.entity(restrictionIDs[i2]);
+ 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 f2 = relation.memberByRole("from");
+ var t = relation.memberByRole("to");
+ var isUturn = f2.id === t.id;
+ var nodes = { from: [], via: [], to: [], keyfrom: [], keyto: [] };
+ for (j2 = 0; j2 < relation.members.length; j2++) {
+ collectNodes(relation.members[j2], nodes);
+ }
+ nodes.keyfrom = utilArrayUniq(nodes.keyfrom.filter(hasDuplicates));
+ nodes.keyto = utilArrayUniq(nodes.keyto.filter(hasDuplicates));
+ var filter2 = keyNodeFilter(nodes.keyfrom, nodes.keyto);
+ nodes.from = nodes.from.filter(filter2);
+ nodes.via = nodes.via.filter(filter2);
+ nodes.to = nodes.to.filter(filter2);
+ var connectFrom = false;
+ var connectVia = false;
+ var connectTo = false;
+ var connectKeyFrom = false;
+ var connectKeyTo = false;
+ for (j2 = 0; j2 < nodeIDs.length; j2++) {
+ var n2 = nodeIDs[j2];
+ if (nodes.from.indexOf(n2) !== -1) {
+ connectFrom = true;
+ }
+ if (nodes.via.indexOf(n2) !== -1) {
+ connectVia = true;
+ }
+ if (nodes.to.indexOf(n2) !== -1) {
+ connectTo = true;
+ }
+ if (nodes.keyfrom.indexOf(n2) !== -1) {
+ connectKeyFrom = true;
+ }
+ if (nodes.keyto.indexOf(n2) !== -1) {
+ connectKeyTo = true;
+ }
+ }
+ if (connectFrom && connectTo && !isUturn) {
+ return "restriction";
+ }
+ if (connectFrom && connectVia) {
+ return "restriction";
+ }
+ if (connectTo && connectVia) {
+ return "restriction";
+ }
+ if (connectKeyFrom || connectKeyTo) {
+ if (nodeIDs.length !== 2) {
+ return "restriction";
+ }
+ var n0 = null;
+ var n1 = null;
+ for (j2 = 0; j2 < memberWays.length; j2++) {
+ way = memberWays[j2];
+ if (way.contains(nodeIDs[0])) {
+ n0 = nodeIDs[0];
+ }
+ if (way.contains(nodeIDs[1])) {
+ n1 = nodeIDs[1];
+ }
+ }
+ if (n0 && n1) {
+ var ok = false;
+ for (j2 = 0; j2 < memberWays.length; j2++) {
+ way = memberWays[j2];
+ if (way.areAdjacent(n0, n1)) {
+ ok = true;
+ break;
+ }
+ }
+ if (!ok) {
+ return "restriction";
+ }
+ }
+ }
+ for (j2 = 0; j2 < memberWays.length; j2++) {
+ way = memberWays[j2].update({});
+ 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 (way.isDegenerate()) {
+ return "restriction";
+ }
+ }
+ }
+ return false;
+ function hasDuplicates(n3, i3, arr) {
+ return arr.indexOf(n3) !== arr.lastIndexOf(n3);
+ }
+ function keyNodeFilter(froms, tos) {
+ return function(n3) {
+ return froms.indexOf(n3) === -1 && tos.indexOf(n3) === -1;
+ };
+ }
+ function collectNodes(member, collection) {
+ var entity = graph.hasEntity(member.id);
+ if (!entity)
+ return;
+ var role2 = member.role || "";
+ if (!collection[role2]) {
+ collection[role2] = [];
+ }
+ if (member.type === "node") {
+ collection[role2].push(member.id);
+ if (role2 === "via") {
+ collection.keyfrom.push(member.id);
+ collection.keyto.push(member.id);
+ }
+ } else if (member.type === "way") {
+ collection[role2].push.apply(collection[role2], entity.nodes);
+ if (role2 === "from" || role2 === "via") {
+ collection.keyfrom.push(entity.first());
+ collection.keyfrom.push(entity.last());
+ }
+ if (role2 === "to" || role2 === "via") {
+ collection.keyto.push(entity.first());
+ collection.keyto.push(entity.last());
+ }
+ }
+ }
+ };
+ return action;
+ }
+
+ // modules/actions/copy_entities.js
+ function actionCopyEntities(ids, fromGraph) {
+ var _copies = {};
+ var action = function(graph) {
+ ids.forEach(function(id3) {
+ fromGraph.entity(id3).copy(fromGraph, _copies);
+ });
+ for (var id2 in _copies) {
+ graph = graph.replace(_copies[id2]);
+ }
+ return graph;
+ };
+ action.copies = function() {
+ return _copies;
+ };
+ return action;
+ }
+
+ // modules/actions/delete_member.js
+ 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;
+ };
+ }
+
+ // modules/actions/discard_tags.js
+ function actionDiscardTags(difference, discardTags) {
+ discardTags = discardTags || {};
+ return (graph) => {
+ difference.modified().forEach(checkTags);
+ difference.created().forEach(checkTags);
+ return graph;
+ function checkTags(entity) {
+ const keys = Object.keys(entity.tags);
+ let didDiscard = false;
+ let tags = {};
+ for (let i2 = 0; i2 < keys.length; i2++) {
+ const k = keys[i2];
+ if (discardTags[k] || !entity.tags[k]) {
+ didDiscard = true;
+ } else {
+ tags[k] = entity.tags[k];
+ }
+ }
+ if (didDiscard) {
+ graph = graph.replace(entity.update({ tags }));
+ }
+ }
+ };
+ }
+
+ // modules/actions/disconnect.js
+ function actionDisconnect(nodeId, newNodeId) {
+ var wayIds;
+ var disconnectableRelationTypes = {
+ "associatedStreet": true,
+ "enforcement": true,
+ "site": true
+ };
+ var action = function(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 (connection.index === 0 && way.isArea()) {
+ graph = graph.replace(way.replaceNode(way.nodes[0], newNode.id));
+ } else if (way.isClosed() && connection.index === way.nodes.length - 1) {
+ graph = graph.replace(way.unclose().addNode(newNode.id));
+ } else {
+ graph = graph.replace(way.updateNode(newNode.id, connection.index));
+ }
+ });
+ return graph;
+ };
+ action.connections = function(graph) {
+ var candidates = [];
+ var keeping = false;
+ var parentWays = graph.parentWays(graph.entity(nodeId));
+ var way, waynode;
+ for (var i2 = 0; i2 < parentWays.length; i2++) {
+ way = parentWays[i2];
+ if (wayIds && wayIds.indexOf(way.id) === -1) {
+ keeping = true;
+ continue;
+ }
+ if (way.isArea() && way.nodes[0] === nodeId) {
+ candidates.push({ wayID: way.id, index: 0 });
+ } else {
+ for (var j2 = 0; j2 < way.nodes.length; j2++) {
+ waynode = way.nodes[j2];
+ if (waynode === nodeId) {
+ if (way.isClosed() && parentWays.length > 1 && wayIds && wayIds.indexOf(way.id) !== -1 && j2 === way.nodes.length - 1) {
+ continue;
+ }
+ candidates.push({ wayID: way.id, index: j2 });
+ }
+ }
+ }
+ }
+ return keeping ? candidates : candidates.slice(1);
+ };
+ 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.filter((relation) => !disconnectableRelationTypes[relation.tags.type]).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";
+ };
+ action.limitWays = function(val) {
+ if (!arguments.length)
+ return wayIds;
+ wayIds = val;
+ return action;
+ };
+ return action;
+ }
+
+ // modules/actions/extract.js
+ function actionExtract(entityID, projection2) {
+ var extractedNodeID;
+ var action = function(graph) {
+ var entity = graph.entity(entityID);
+ if (entity.type === "node") {
+ return extractFromNode(entity, graph);
+ }
+ return extractFromWayOrRelation(entity, graph);
+ };
+ function extractFromNode(node, graph) {
+ extractedNodeID = node.id;
+ var replacement = osmNode({ loc: node.loc });
+ graph = graph.replace(replacement);
+ graph = graph.parentWays(node).reduce(function(accGraph, parentWay) {
+ return accGraph.replace(parentWay.replaceNode(entityID, replacement.id));
+ }, graph);
+ return graph.parentRelations(node).reduce(function(accGraph, parentRel) {
+ return accGraph.replace(parentRel.replaceMember(node, replacement));
+ }, graph);
+ }
+ function extractFromWayOrRelation(entity, graph) {
+ var fromGeometry = entity.geometry(graph);
+ var keysToCopyAndRetain = ["source", "wheelchair"];
+ var keysToRetain = ["area"];
+ var buildingKeysToRetain = ["architect", "building", "height", "layer"];
+ var extractedLoc = path_default(projection2).centroid(entity.asGeoJSON(graph));
+ extractedLoc = extractedLoc && projection2.invert(extractedLoc);
+ if (!extractedLoc || !isFinite(extractedLoc[0]) || !isFinite(extractedLoc[1])) {
+ extractedLoc = entity.extent(graph).center();
+ }
+ 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);
+ var pointTags = {};
+ for (var key in entityTags) {
+ if (entity.type === "relation" && key === "type") {
+ continue;
+ }
+ if (keysToRetain.indexOf(key) !== -1) {
+ continue;
+ }
+ if (isBuilding) {
+ if (buildingKeysToRetain.indexOf(key) !== -1 || key.match(/^building:.{1,}/) || key.match(/^roof:.{1,}/))
+ continue;
+ }
+ if (isIndoorArea && key === "indoor") {
+ continue;
+ }
+ pointTags[key] = entityTags[key];
+ if (keysToCopyAndRetain.indexOf(key) !== -1 || key.match(/^addr:.{1,}/)) {
+ continue;
+ } else if (isIndoorArea && key === "level") {
+ continue;
+ }
+ delete entityTags[key];
+ }
+ if (!isBuilding && !isIndoorArea && fromGeometry === "area") {
+ entityTags.area = "yes";
+ }
+ var replacement = osmNode({ loc: extractedLoc, tags: pointTags });
+ graph = graph.replace(replacement);
+ extractedNodeID = replacement.id;
+ return graph.replace(entity.update({ tags: entityTags }));
+ }
+ action.getExtractedNodeID = function() {
+ return extractedNodeID;
+ };
+ return action;
+ }
+
+ // modules/actions/join.js
+ function actionJoin(ids) {
+ function groupEntitiesByGeometry(graph) {
+ var entities = ids.map(function(id2) {
+ return graph.entity(id2);
+ });
+ return Object.assign(
+ { line: [] },
+ utilArrayGroupBy(entities, function(entity) {
+ return entity.geometry(graph);
+ })
+ );
+ }
+ var action = function(graph) {
+ var ways = ids.map(graph.entity, graph);
+ var survivorID = utilOldestID(ways.map((way) => way.id));
+ ways.sort(function(a, b) {
+ var aSided = a.isSided();
+ var bSided = b.isSided();
+ return aSided && !bSided ? -1 : bSided && !aSided ? 1 : 0;
+ });
+ var sequences = osmJoinWays(ways, graph);
+ var joined = sequences[0];
+ graph = sequences.actions.reduce(function(g, action2) {
+ return action2(g);
+ }, graph);
+ var survivor = graph.entity(survivorID);
+ survivor = survivor.update({ nodes: joined.nodes.map(function(n2) {
+ return n2.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);
+ });
+ function checkForSimpleMultipolygon() {
+ if (!survivor.isClosed())
+ return;
+ var multipolygons = graph.parentMultipolygons(survivor).filter(function(multipolygon2) {
+ return multipolygon2.members.length === 1;
+ });
+ 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;
+ }
+ 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 (survivor.geometry(graph) !== "area") {
+ tags.area = "yes";
+ }
+ delete tags.type;
+ survivor = survivor.update({ tags });
+ graph = graph.replace(survivor);
+ }
+ checkForSimpleMultipolygon();
+ return graph;
+ };
+ action.resultingWayNodesLength = function(graph) {
+ return ids.reduce(function(count, id2) {
+ return count + graph.entity(id2).nodes.length;
+ }, 0) - ids.length - 1;
+ };
+ action.disabled = function(graph) {
+ var geometries = groupEntitiesByGeometry(graph);
+ if (ids.length < 2 || ids.length !== geometries.line.length) {
+ return "not_eligible";
+ }
+ var joined = osmJoinWays(ids.map(graph.entity, graph), graph);
+ if (joined.length > 1) {
+ return "not_adjacent";
+ }
+ var i2;
+ var sortedParentRelations = function(id2) {
+ return graph.parentRelations(graph.entity(id2)).filter((rel) => !rel.isRestriction() && !rel.isConnectivity()).sort((a, b) => a.id - b.id);
+ };
+ var relsA = sortedParentRelations(ids[0]);
+ for (i2 = 1; i2 < ids.length; i2++) {
+ var relsB = sortedParentRelations(ids[i2]);
+ if (!utilArrayIdentical(relsA, relsB)) {
+ return "conflicting_relations";
+ }
+ }
+ for (i2 = 0; i2 < ids.length - 1; i2++) {
+ for (var j2 = i2 + 1; j2 < ids.length; j2++) {
+ var path1 = graph.childNodes(graph.entity(ids[i2])).map(function(e) {
+ return e.loc;
+ });
+ var path2 = graph.childNodes(graph.entity(ids[j2])).map(function(e) {
+ return e.loc;
+ });
+ var intersections = geoPathIntersections(path1, path2);
+ var common = utilArrayIntersection(
+ joined[0].nodes.map(function(n2) {
+ return n2.loc.toString();
+ }),
+ intersections.map(function(n2) {
+ return n2.toString();
+ })
+ );
+ if (common.length !== intersections.length) {
+ return "paths_intersect";
+ }
+ }
+ }
+ var nodeIds = joined[0].nodes.map(function(n2) {
+ return n2.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.isConnectivity()) && 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;
+ }
+ }
+ });
+ if (relation) {
+ return relation.isRestriction() ? "restriction" : "connectivity";
+ }
+ if (conflicting) {
+ return "conflicting_tags";
+ }
+ };
+ return action;
+ }
+
+ // modules/actions/merge.js
+ function actionMerge(ids) {
+ function groupEntitiesByGeometry(graph) {
+ var entities = ids.map(function(id2) {
+ return graph.entity(id2);
+ });
+ return Object.assign(
+ { point: [], area: [], line: [], relation: [] },
+ utilArrayGroupBy(entities, function(entity) {
+ return entity.geometry(graph);
+ })
+ );
+ }
+ var action = function(graph) {
+ var geometries = groupEntitiesByGeometry(graph);
+ var target = geometries.area[0] || geometries.line[0];
+ var points = geometries.point;
+ points.forEach(function(point2) {
+ target = target.mergeTags(point2.tags);
+ graph = graph.replace(target);
+ graph.parentRelations(point2).forEach(function(parent) {
+ graph = graph.replace(parent.replaceMember(point2, target));
+ });
+ var nodes = utilArrayUniq(graph.childNodes(target));
+ var removeNode = point2;
+ if (!point2.isNew()) {
+ var inserted = false;
+ var canBeReplaced = function(node2) {
+ return !(graph.parentWays(node2).length > 1 || graph.parentRelations(node2).length);
+ };
+ var replaceNode = function(node2) {
+ graph = graph.replace(point2.update({ tags: node2.tags, loc: node2.loc }));
+ target = target.replaceNode(node2.id, point2.id);
+ graph = graph.replace(target);
+ removeNode = node2;
+ inserted = true;
+ };
+ var i2;
+ var node;
+ for (i2 = 0; i2 < nodes.length; i2++) {
+ node = nodes[i2];
+ if (canBeReplaced(node) && node.isNew()) {
+ replaceNode(node);
+ break;
+ }
+ }
+ if (!inserted && point2.hasInterestingTags()) {
+ for (i2 = 0; i2 < nodes.length; i2++) {
+ node = nodes[i2];
+ if (canBeReplaced(node) && !node.hasInterestingTags()) {
+ replaceNode(node);
+ break;
+ }
+ }
+ if (!inserted) {
+ for (i2 = 0; i2 < nodes.length; i2++) {
+ node = nodes[i2];
+ if (canBeReplaced(node) && utilCompareIDs(point2.id, node.id) < 0) {
+ replaceNode(node);
+ break;
+ }
+ }
+ }
+ }
+ }
+ graph = graph.remove(removeNode);
+ });
+ if (target.tags.area === "yes") {
+ var tags = Object.assign({}, target.tags);
+ delete tags.area;
+ if (osmTagSuggestingArea(tags)) {
+ target = target.update({ tags });
+ graph = graph.replace(target);
+ }
+ }
+ return graph;
+ };
+ action.disabled = function(graph) {
+ var geometries = groupEntitiesByGeometry(graph);
+ if (geometries.point.length === 0 || geometries.area.length + geometries.line.length !== 1 || geometries.relation.length !== 0) {
+ return "not_eligible";
+ }
+ };
+ return action;
+ }
+
+ // modules/actions/merge_nodes.js
+ function actionMergeNodes(nodeIDs, loc) {
+ function chooseLoc(graph) {
+ if (!nodeIDs.length)
+ return null;
+ var sum = [0, 0];
+ var interestingCount = 0;
+ var interestingLoc;
+ for (var i2 = 0; i2 < nodeIDs.length; i2++) {
+ var node = graph.entity(nodeIDs[i2]);
+ if (node.hasInterestingTags()) {
+ interestingLoc = ++interestingCount === 1 ? node.loc : null;
+ }
+ sum = geoVecAdd(sum, node.loc);
+ }
+ return interestingLoc || geoVecScale(sum, 1 / nodeIDs.length);
+ }
+ var action = function(graph) {
+ if (nodeIDs.length < 2)
+ return graph;
+ var toLoc = loc;
+ if (!toLoc) {
+ toLoc = chooseLoc(graph);
+ }
+ for (var i2 = 0; i2 < nodeIDs.length; i2++) {
+ var node = graph.entity(nodeIDs[i2]);
+ if (node.loc !== toLoc) {
+ graph = graph.replace(node.move(toLoc));
+ }
+ }
+ return actionConnect(nodeIDs)(graph);
+ };
+ action.disabled = function(graph) {
+ if (nodeIDs.length < 2)
+ return "not_eligible";
+ for (var i2 = 0; i2 < nodeIDs.length; i2++) {
+ var entity = graph.entity(nodeIDs[i2]);
+ if (entity.type !== "node")
+ return "not_eligible";
+ }
+ return actionConnect(nodeIDs).disabled(graph);
+ };
+ return action;
+ }
+
+ // modules/osm/changeset.js
+ 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() {
+ return new geoExtent();
+ },
+ geometry: function() {
+ return "changeset";
+ },
+ asJXON: function() {
+ 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(changes) {
+ var changeset_id = this.id;
+ function nest(x, order) {
+ var groups = {};
+ for (var i2 = 0; i2 < x.length; i2++) {
+ var tagName = Object.keys(x[i2])[0];
+ if (!groups[tagName])
+ groups[tagName] = [];
+ groups[tagName].push(x[i2][tagName]);
+ }
+ var ordered = {};
+ order.forEach(function(o) {
+ if (groups[o])
+ ordered[o] = groups[o];
+ });
+ return ordered;
+ }
+ function sort(changes2) {
+ function resolve(item) {
+ return relations.find(function(relation2) {
+ return item.keyAttributes.type === "relation" && item.keyAttributes.ref === relation2["@id"];
+ });
+ }
+ function isNew(item) {
+ return !sorted[item["@id"]] && !processing.find(function(proc) {
+ return proc["@id"] === item["@id"];
+ });
+ }
+ var processing = [];
+ var sorted = {};
+ var relations = changes2.relation;
+ if (!relations)
+ return changes2;
+ for (var i2 = 0; i2 < relations.length; i2++) {
+ var relation = relations[i2];
+ if (!sorted[relation["@id"]]) {
+ processing.push(relation);
+ }
+ while (processing.length > 0) {
+ var next = processing[0], deps = next.member.map(resolve).filter(Boolean).filter(isNew);
+ if (deps.length === 0) {
+ sorted[next["@id"]] = next;
+ processing.shift();
+ } else {
+ processing = deps.concat(processing);
+ }
+ }
+ }
+ changes2.relation = Object.values(sorted);
+ return changes2;
+ }
+ function rep2(entity) {
+ return entity.asJXON(changeset_id);
+ }
+ return {
+ osmChange: {
+ "@version": 0.6,
+ "@generator": "iD",
+ "create": sort(nest(changes.created.map(rep2), ["node", "way", "relation"])),
+ "modify": nest(changes.modified.map(rep2), ["node", "way", "relation"]),
+ "delete": Object.assign(nest(changes.deleted.map(rep2), ["relation", "way", "node"]), { "@if-unused": true })
+ }
+ };
+ },
+ asGeoJSON: function() {
+ return {};
+ }
+ });
+
+ // modules/osm/note.js
+ function osmNote() {
+ if (!(this instanceof osmNote)) {
+ return new osmNote().initialize(arguments);
+ } else if (arguments.length) {
+ this.initialize(arguments);
+ }
+ }
+ osmNote.id = function() {
+ return osmNote.id.next--;
+ };
+ osmNote.id.next = -1;
+ Object.assign(osmNote.prototype, {
+ type: "note",
+ initialize: function(sources) {
+ for (var i2 = 0; i2 < sources.length; ++i2) {
+ var source = sources[i2];
+ for (var prop in source) {
+ if (Object.prototype.hasOwnProperty.call(source, prop)) {
+ if (source[prop] === void 0) {
+ delete this[prop];
+ } else {
+ this[prop] = source[prop];
+ }
+ }
+ }
+ }
+ if (!this.id) {
+ this.id = osmNote.id().toString();
+ }
+ return this;
+ },
+ extent: function() {
+ return new geoExtent(this.loc);
+ },
+ update: function(attrs) {
+ return osmNote(this, attrs);
+ },
+ isNew: function() {
+ return this.id < 0;
+ },
+ move: function(loc) {
+ return this.update({ loc });
+ }
+ });
+
+ // modules/osm/relation.js
+ 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);
+ 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;
+ };
+ Object.assign(osmRelation.prototype, {
+ type: "relation",
+ members: [],
+ copy: function(resolver, copies) {
+ if (copies[this.id])
+ return copies[this.id];
+ var copy2 = 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 });
+ });
+ copy2 = copy2.update({ members });
+ copies[this.id] = copy2;
+ return copy2;
+ },
+ extent: function(resolver, memo) {
+ return resolver.transient(this, "extent", function() {
+ if (memo && memo[this.id])
+ return geoExtent();
+ memo = memo || {};
+ memo[this.id] = true;
+ var extent = geoExtent();
+ for (var i2 = 0; i2 < this.members.length; i2++) {
+ var member = resolver.hasEntity(this.members[i2].id);
+ if (member) {
+ extent._extend(member.extent(resolver, memo));
+ }
+ }
+ return extent;
+ });
+ },
+ geometry: function(graph) {
+ return graph.transient(this, "geometry", function() {
+ return this.isMultipolygon() ? "area" : "relation";
+ });
+ },
+ isDegenerate: function() {
+ return this.members.length === 0;
+ },
+ // Return an array of members, each extended with an 'index' property whose value
+ // is the member index.
+ indexedMembers: function() {
+ var result = new Array(this.members.length);
+ for (var i2 = 0; i2 < this.members.length; i2++) {
+ result[i2] = Object.assign({}, this.members[i2], { index: i2 });
+ }
+ 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(role) {
+ for (var i2 = 0; i2 < this.members.length; i2++) {
+ if (this.members[i2].role === role) {
+ return Object.assign({}, this.members[i2], { index: i2 });
+ }
+ }
+ },
+ // Same as memberByRole, but returns all members with the given role
+ membersByRole: function(role) {
+ var result = [];
+ for (var i2 = 0; i2 < this.members.length; i2++) {
+ if (this.members[i2].role === role) {
+ result.push(Object.assign({}, this.members[i2], { index: i2 }));
+ }
+ }
+ 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(id2) {
+ for (var i2 = 0; i2 < this.members.length; i2++) {
+ if (this.members[i2].id === id2) {
+ return Object.assign({}, this.members[i2], { index: i2 });
+ }
+ }
+ },
+ // 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(id2, role) {
+ for (var i2 = 0; i2 < this.members.length; i2++) {
+ if (this.members[i2].id === id2 && this.members[i2].role === role) {
+ return Object.assign({}, this.members[i2], { index: i2 });
+ }
+ }
+ },
+ addMember: function(member, index) {
+ var members = this.members.slice();
+ members.splice(index === void 0 ? members.length : index, 0, member);
+ return this.update({ members });
+ },
+ updateMember: function(member, index) {
+ var members = this.members.slice();
+ members.splice(index, 1, Object.assign({}, members[index], member));
+ return this.update({ members });
+ },
+ removeMember: function(index) {
+ var members = this.members.slice();
+ members.splice(index, 1);
+ return this.update({ members });
+ },
+ removeMembersWithID: function(id2) {
+ var members = this.members.filter(function(m) {
+ return m.id !== id2;
+ });
+ return this.update({ members });
+ },
+ moveMember: function(fromIndex, toIndex) {
+ var members = this.members.slice();
+ members.splice(toIndex, 0, members.splice(fromIndex, 1)[0]);
+ return this.update({ 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(needle, replacement, keepDuplicates) {
+ if (!this.memberById(needle.id))
+ return this;
+ var members = [];
+ for (var i2 = 0; i2 < this.members.length; i2++) {
+ var member = this.members[i2];
+ 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 });
+ }
+ }
+ return this.update({ members });
+ },
+ asJXON: function(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, v: this.tags[k] } };
+ }, this)
+ }
+ };
+ if (changeset_id) {
+ r.relation["@changeset"] = changeset_id;
+ }
+ return r;
+ },
+ asGeoJSON: function(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(resolver) {
+ return resolver.transient(this, "area", function() {
+ return area_default(this.asGeoJSON(resolver));
+ });
+ },
+ isMultipolygon: function() {
+ return this.tags.type === "multipolygon";
+ },
+ isComplete: function(resolver) {
+ for (var i2 = 0; i2 < this.members.length; i2++) {
+ if (!resolver.hasEntity(this.members[i2].id)) {
+ return false;
+ }
+ }
+ return true;
+ },
+ hasFromViaTo: function() {
+ 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() {
+ return !!(this.tags.type && this.tags.type.match(/^restriction:?/));
+ },
+ isValidRestriction: function() {
+ 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;
+ },
+ isConnectivity: function() {
+ return !!(this.tags.type && this.tags.type.match(/^connectivity:?/));
+ },
+ // 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(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 sequenceToLineString = function(sequence) {
+ if (sequence.nodes.length > 2 && sequence.nodes[0] !== sequence.nodes[sequence.nodes.length - 1]) {
+ sequence.nodes.push(sequence.nodes[0]);
+ }
+ return sequence.nodes.map(function(node) {
+ return node.loc;
+ });
+ };
+ outers = outers.map(sequenceToLineString);
+ inners = inners.map(sequenceToLineString);
+ var result = outers.map(function(o2) {
+ return [area_default({ type: "Polygon", coordinates: [o2] }) > 2 * Math.PI ? o2.reverse() : o2];
+ });
+ function findOuter(inner2) {
+ var o2, outer;
+ for (o2 = 0; o2 < outers.length; o2++) {
+ outer = outers[o2];
+ if (geoPolygonContainsPolygon(outer, inner2)) {
+ return o2;
+ }
+ }
+ for (o2 = 0; o2 < outers.length; o2++) {
+ outer = outers[o2];
+ if (geoPolygonIntersectsPolygon(outer, inner2, false)) {
+ return o2;
+ }
+ }
+ }
+ for (var i2 = 0; i2 < inners.length; i2++) {
+ var inner = inners[i2];
+ if (area_default({ type: "Polygon", coordinates: [inner] }) < 2 * Math.PI) {
+ inner = inner.reverse();
+ }
+ var o = findOuter(inners[i2]);
+ if (o !== void 0) {
+ result[o].push(inners[i2]);
+ } else {
+ result.push([inners[i2]]);
+ }
+ }
+ return result;
+ }
+ });
+
+ // modules/osm/qa_item.js
+ var QAItem = class {
+ constructor(loc, service, itemType, id2, props) {
+ this.loc = loc;
+ this.service = service.title;
+ this.itemType = itemType;
+ this.id = id2 ? id2 : `${QAItem.id()}`;
+ this.update(props);
+ if (service && typeof service.getIcon === "function") {
+ this.icon = service.getIcon(itemType);
+ }
+ }
+ update(props) {
+ const { loc, service, itemType, id: id2 } = this;
+ Object.keys(props).forEach((prop) => this[prop] = props[prop]);
+ this.loc = loc;
+ this.service = service;
+ this.itemType = itemType;
+ this.id = id2;
+ return this;
+ }
+ // Generic handling for newly created QAItems
+ static id() {
+ return this.nextId--;
+ }
+ };
+ QAItem.nextId = -1;
+
+ // modules/actions/split.js
+ function actionSplit(nodeIds, newWayIds) {
+ if (typeof nodeIds === "string")
+ nodeIds = [nodeIds];
+ var _wayIDs;
+ var _keepHistoryOn = "longest";
+ var _createdWayIDs = [];
+ function dist(graph, nA, nB) {
+ var locA = graph.entity(nA).loc;
+ var locB = graph.entity(nB).loc;
+ var epsilon3 = 1e-6;
+ return locA && locB ? geoSphericalDistance(locA, locB) : epsilon3;
+ }
+ function splitArea(nodes, idxA, graph) {
+ var lengths = new Array(nodes.length);
+ var length;
+ var i2;
+ var best = 0;
+ var idxB;
+ function wrap2(index) {
+ return utilWrap(index, nodes.length);
+ }
+ length = 0;
+ for (i2 = wrap2(idxA + 1); i2 !== idxA; i2 = wrap2(i2 + 1)) {
+ length += dist(graph, nodes[i2], nodes[wrap2(i2 - 1)]);
+ lengths[i2] = length;
+ }
+ length = 0;
+ for (i2 = wrap2(idxA - 1); i2 !== idxA; i2 = wrap2(i2 - 1)) {
+ length += dist(graph, nodes[i2], nodes[wrap2(i2 + 1)]);
+ if (length < lengths[i2]) {
+ lengths[i2] = length;
+ }
+ }
+ for (i2 = 0; i2 < nodes.length; i2++) {
+ var cost = lengths[i2] / dist(graph, nodes[idxA], nodes[i2]);
+ if (cost > best) {
+ idxB = i2;
+ best = cost;
+ }
+ }
+ return idxB;
+ }
+ function totalLengthBetweenNodes(graph, nodes) {
+ var totalLength = 0;
+ for (var i2 = 0; i2 < nodes.length - 1; i2++) {
+ totalLength += dist(graph, nodes[i2], nodes[i2 + 1]);
+ }
+ return totalLength;
+ }
+ function split(graph, nodeId, wayA, newWayId) {
+ var wayB = osmWay({ id: newWayId, tags: wayA.tags });
+ var origNodes = wayA.nodes.slice();
+ var nodesA;
+ var nodesB;
+ var isArea = wayA.isArea();
+ var isOuter = osmIsOldMultipolygonOuterMember(wayA, graph);
+ if (wayA.isClosed()) {
+ var nodes = wayA.nodes.slice(0, -1);
+ var idxA = nodes.indexOf(nodeId);
+ var idxB = splitArea(nodes, idxA, graph);
+ 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);
+ }
+ var lengthA = totalLengthBetweenNodes(graph, nodesA);
+ var lengthB = totalLengthBetweenNodes(graph, nodesB);
+ if (_keepHistoryOn === "longest" && lengthB > lengthA) {
+ 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 });
+ }
+ if (wayA.tags.step_count) {
+ var stepCount = Number(wayA.tags.step_count);
+ 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 });
+ }
+ }
+ graph = graph.replace(wayA);
+ graph = graph.replace(wayB);
+ graph.parentRelations(wayA).forEach(function(relation) {
+ var member;
+ if (relation.hasFromViaTo()) {
+ var f2 = relation.memberByRole("from");
+ var v = relation.membersByRole("via");
+ var t = relation.memberByRole("to");
+ var i2;
+ if (f2.id === wayA.id || t.id === wayA.id) {
+ var keepB = false;
+ if (v.length === 1 && v[0].type === "node") {
+ keepB = wayB.contains(v[0].id);
+ } else {
+ for (i2 = 0; i2 < v.length; i2++) {
+ if (v[i2].type === "way") {
+ var wayVia = graph.hasEntity(v[i2].id);
+ if (wayVia && utilArrayIntersection(wayB.nodes, wayVia.nodes).length) {
+ keepB = true;
+ break;
+ }
+ }
+ }
+ }
+ if (keepB) {
+ relation = relation.replaceMember(wayA, wayB);
+ graph = graph.replace(relation);
+ }
+ } else {
+ for (i2 = 0; i2 < v.length; i2++) {
+ if (v[i2].type === "way" && v[i2].id === wayA.id) {
+ member = {
+ id: wayB.id,
+ type: "way",
+ role: "via"
+ };
+ graph = actionAddMember(relation.id, member, v[i2].index + 1)(graph);
+ break;
+ }
+ }
+ }
+ } else {
+ if (relation === isOuter) {
+ graph = graph.replace(relation.mergeTags(wayA.tags));
+ graph = graph.replace(wayA.update({ tags: {} }));
+ graph = graph.replace(wayB.update({ tags: {} }));
+ }
+ 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, void 0, insertPair)(graph);
+ }
+ });
+ 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: {} }));
+ }
+ _createdWayIDs.push(wayB.id);
+ return graph;
+ }
+ var action = function(graph) {
+ _createdWayIDs = [];
+ var newWayIndex = 0;
+ for (var i2 = 0; i2 < nodeIds.length; i2++) {
+ var nodeId = nodeIds[i2];
+ var candidates = action.waysForNode(nodeId, graph);
+ for (var j2 = 0; j2 < candidates.length; j2++) {
+ graph = split(graph, nodeId, candidates[j2], newWayIds && newWayIds[newWayIndex]);
+ newWayIndex += 1;
+ }
+ }
+ return graph;
+ };
+ action.getCreatedWayIDs = function() {
+ return _createdWayIDs;
+ };
+ action.waysForNode = function(nodeId, graph) {
+ var node = graph.entity(nodeId);
+ var splittableParents = graph.parentWays(node).filter(isSplittable);
+ if (!_wayIDs) {
+ var hasLine = splittableParents.some(function(parent) {
+ return parent.geometry(graph) === "line";
+ });
+ if (hasLine) {
+ return splittableParents.filter(function(parent) {
+ return parent.geometry(graph) === "line";
+ });
+ }
+ }
+ return splittableParents;
+ function isSplittable(parent) {
+ if (_wayIDs && _wayIDs.indexOf(parent.id) === -1)
+ return false;
+ if (parent.isClosed())
+ return true;
+ for (var i2 = 1; i2 < parent.nodes.length - 1; i2++) {
+ if (parent.nodes[i2] === nodeId)
+ return true;
+ }
+ return false;
+ }
+ };
+ action.ways = function(graph) {
+ return utilArrayUniq([].concat.apply([], nodeIds.map(function(nodeId) {
+ return action.waysForNode(nodeId, graph);
+ })));
+ };
+ action.disabled = function(graph) {
+ for (var i2 = 0; i2 < nodeIds.length; i2++) {
+ var nodeId = nodeIds[i2];
+ var candidates = action.waysForNode(nodeId, graph);
+ if (candidates.length === 0 || _wayIDs && _wayIDs.length !== candidates.length) {
+ return "not_eligible";
+ }
+ }
+ };
+ 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;
+ }
+
+ // modules/core/graph.js
+ 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 = /* @__PURE__ */ Object.create({});
+ this._parentWays = /* @__PURE__ */ Object.create({});
+ this._parentRels = /* @__PURE__ */ Object.create({});
+ this.rebase(other || [], [this]);
+ }
+ this.transients = {};
+ this._childNodes = {};
+ this.frozen = !mutable;
+ }
+ coreGraph.prototype = {
+ hasEntity: function(id2) {
+ return this.entities[id2];
+ },
+ entity: function(id2) {
+ var entity = this.entities[id2];
+ if (!entity) {
+ entity = this.entities.__proto__[id2];
+ }
+ if (!entity) {
+ throw new Error("entity " + id2 + " not found");
+ }
+ return entity;
+ },
+ geometry: function(id2) {
+ return this.entity(id2).geometry(this);
+ },
+ transient: function(entity, key, fn) {
+ var id2 = entity.id;
+ var transients = this.transients[id2] || (this.transients[id2] = {});
+ if (transients[key] !== void 0) {
+ return transients[key];
+ }
+ transients[key] = fn.call(entity);
+ return transients[key];
+ },
+ parentWays: function(entity) {
+ var parents = this._parentWays[entity.id];
+ var result = [];
+ if (parents) {
+ parents.forEach(function(id2) {
+ result.push(this.entity(id2));
+ }, this);
+ }
+ return result;
+ },
+ isPoi: function(entity) {
+ var parents = this._parentWays[entity.id];
+ return !parents || parents.size === 0;
+ },
+ isShared: function(entity) {
+ var parents = this._parentWays[entity.id];
+ return parents && parents.size > 1;
+ },
+ parentRelations: function(entity) {
+ var parents = this._parentRels[entity.id];
+ var result = [];
+ if (parents) {
+ parents.forEach(function(id2) {
+ result.push(this.entity(id2));
+ }, this);
+ }
+ return result;
+ },
+ parentMultipolygons: function(entity) {
+ return this.parentRelations(entity).filter(function(relation) {
+ return relation.isMultipolygon();
+ });
+ },
+ childNodes: function(entity) {
+ if (this._childNodes[entity.id])
+ return this._childNodes[entity.id];
+ if (!entity.nodes)
+ return [];
+ var nodes = [];
+ for (var i2 = 0; i2 < entity.nodes.length; i2++) {
+ nodes[i2] = this.entity(entity.nodes[i2]);
+ }
+ if (debug)
+ Object.freeze(nodes);
+ this._childNodes[entity.id] = nodes;
+ return this._childNodes[entity.id];
+ },
+ base: function() {
+ 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(entities, stack, force) {
+ var base = this.base();
+ var i2, j2, k, id2;
+ for (i2 = 0; i2 < entities.length; i2++) {
+ var entity = entities[i2];
+ if (!entity.visible || !force && base.entities[entity.id])
+ continue;
+ base.entities[entity.id] = entity;
+ this._updateCalculated(void 0, entity, base.parentWays, base.parentRels);
+ if (entity.type === "way") {
+ for (j2 = 0; j2 < entity.nodes.length; j2++) {
+ id2 = entity.nodes[j2];
+ for (k = 1; k < stack.length; k++) {
+ var ents = stack[k].entities;
+ if (ents.hasOwnProperty(id2) && ents[id2] === void 0) {
+ delete ents[id2];
+ }
+ }
+ }
+ }
+ }
+ for (i2 = 0; i2 < stack.length; i2++) {
+ stack[i2]._updateRebased();
+ }
+ },
+ _updateRebased: function() {
+ var base = this.base();
+ Object.keys(this._parentWays).forEach(function(child) {
+ if (base.parentWays[child]) {
+ base.parentWays[child].forEach(function(id2) {
+ if (!this.entities.hasOwnProperty(id2)) {
+ this._parentWays[child].add(id2);
+ }
+ }, this);
+ }
+ }, this);
+ Object.keys(this._parentRels).forEach(function(child) {
+ if (base.parentRels[child]) {
+ base.parentRels[child].forEach(function(id2) {
+ if (!this.entities.hasOwnProperty(id2)) {
+ this._parentRels[child].add(id2);
+ }
+ }, this);
+ }
+ }, this);
+ this.transients = {};
+ },
+ // Updates calculated properties (parentWays, parentRels) for the specified change
+ _updateCalculated: function(oldentity, entity, parentWays, parentRels) {
+ parentWays = parentWays || this._parentWays;
+ parentRels = parentRels || this._parentRels;
+ var type2 = entity && entity.type || oldentity && oldentity.type;
+ var removed, added, i2;
+ if (type2 === "way") {
+ 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;
+ }
+ for (i2 = 0; i2 < removed.length; i2++) {
+ parentWays[removed[i2]] = new Set(parentWays[removed[i2]]);
+ parentWays[removed[i2]].delete(oldentity.id);
+ }
+ for (i2 = 0; i2 < added.length; i2++) {
+ parentWays[added[i2]] = new Set(parentWays[added[i2]]);
+ parentWays[added[i2]].add(entity.id);
+ }
+ } else if (type2 === "relation") {
+ var oldentityMemberIDs = oldentity ? oldentity.members.map(function(m) {
+ return m.id;
+ }) : [];
+ var entityMemberIDs = entity ? entity.members.map(function(m) {
+ return m.id;
+ }) : [];
+ if (oldentity && entity) {
+ removed = utilArrayDifference(oldentityMemberIDs, entityMemberIDs);
+ added = utilArrayDifference(entityMemberIDs, oldentityMemberIDs);
+ } else if (oldentity) {
+ removed = oldentityMemberIDs;
+ added = [];
+ } else if (entity) {
+ removed = [];
+ added = entityMemberIDs;
+ }
+ for (i2 = 0; i2 < removed.length; i2++) {
+ parentRels[removed[i2]] = new Set(parentRels[removed[i2]]);
+ parentRels[removed[i2]].delete(oldentity.id);
+ }
+ for (i2 = 0; i2 < added.length; i2++) {
+ parentRels[added[i2]] = new Set(parentRels[added[i2]]);
+ parentRels[added[i2]].add(entity.id);
+ }
+ }
+ },
+ replace: function(entity) {
+ if (this.entities[entity.id] === entity)
+ return this;
+ return this.update(function() {
+ this._updateCalculated(this.entities[entity.id], entity);
+ this.entities[entity.id] = entity;
+ });
+ },
+ remove: function(entity) {
+ return this.update(function() {
+ this._updateCalculated(entity, void 0);
+ this.entities[entity.id] = void 0;
+ });
+ },
+ revert: function(id2) {
+ var baseEntity = this.base().entities[id2];
+ var headEntity = this.entities[id2];
+ if (headEntity === baseEntity)
+ return this;
+ return this.update(function() {
+ this._updateCalculated(headEntity, baseEntity);
+ delete this.entities[id2];
+ });
+ },
+ update: function() {
+ var graph = this.frozen ? coreGraph(this, true) : this;
+ for (var i2 = 0; i2 < arguments.length; i2++) {
+ arguments[i2].call(graph, graph);
+ }
+ if (this.frozen)
+ graph.frozen = true;
+ return graph;
+ },
+ // Obliterates any existing entities
+ load: function(entities) {
+ var base = this.base();
+ this.entities = Object.create(base.entities);
+ for (var i2 in entities) {
+ this.entities[i2] = entities[i2];
+ this._updateCalculated(base.entities[i2], this.entities[i2]);
+ }
+ return this;
+ }
+ };
+
+ // modules/osm/intersection.js
+ function osmTurn(turn) {
+ if (!(this instanceof osmTurn)) {
+ return new osmTurn(turn);
+ }
+ Object.assign(this, turn);
+ }
+ function osmIntersection(graph, startVertexId, maxDistance) {
+ maxDistance = maxDistance || 30;
+ var vgraph = coreGraph();
+ var i2, j2, k;
+ function memberOfRestriction(entity) {
+ return graph.parentRelations(entity).some(function(r) {
+ return r.isRestriction();
+ });
+ }
+ function isRoad(way2) {
+ if (way2.isArea() || way2.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[way2.tags.highway];
+ }
+ 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;
+ var actions = [];
+ while (checkVertices.length) {
+ vertex = checkVertices.pop();
+ checkWays = graph.parentWays(vertex);
+ var hasWays = false;
+ for (i2 = 0; i2 < checkWays.length; i2++) {
+ way = checkWays[i2];
+ if (!isRoad(way) && !memberOfRestriction(way))
+ continue;
+ ways.push(way);
+ hasWays = true;
+ nodes = utilArrayUniq(graph.childNodes(way));
+ for (j2 = 0; j2 < nodes.length; j2++) {
+ node = nodes[j2];
+ if (node === vertex)
+ continue;
+ if (vertices.indexOf(node) !== -1)
+ continue;
+ if (geoSphericalDistance(node.loc, startNode.loc) > maxDistance)
+ continue;
+ var hasParents = false;
+ parents = graph.parentWays(node);
+ for (k = 0; k < parents.length; k++) {
+ parent = parents[k];
+ if (parent === way)
+ continue;
+ if (ways.indexOf(parent) !== -1)
+ continue;
+ if (!isRoad(parent))
+ continue;
+ hasParents = true;
+ break;
+ }
+ if (hasParents) {
+ checkVertices.push(node);
+ }
+ }
+ }
+ if (hasWays) {
+ vertices.push(vertex);
+ }
+ }
+ vertices = utilArrayUniq(vertices);
+ ways = utilArrayUniq(ways);
+ ways.forEach(function(way2) {
+ graph.childNodes(way2).forEach(function(node2) {
+ vgraph = vgraph.replace(node2);
+ });
+ vgraph = vgraph.replace(way2);
+ graph.parentRelations(way2).forEach(function(relation) {
+ if (relation.isRestriction()) {
+ if (relation.isValidRestriction(graph)) {
+ vgraph = vgraph.replace(relation);
+ } else if (relation.isComplete(graph)) {
+ actions.push(actionDeleteRelation(relation.id));
+ }
+ }
+ });
+ });
+ ways.forEach(function(w) {
+ var way2 = vgraph.entity(w.id);
+ if (way2.tags.oneway === "-1") {
+ var action = actionReverse(way2.id, { reverseOneway: true });
+ actions.push(action);
+ vgraph = action(vgraph);
+ }
+ });
+ var origCount = osmEntity.id.next.way;
+ vertices.forEach(function(v) {
+ var splitAll = actionSplit([v.id]).keepHistoryOn("first");
+ if (!splitAll.disabled(vgraph)) {
+ splitAll.ways(vgraph).forEach(function(way2) {
+ var splitOne = actionSplit([v.id]).limitWays([way2.id]).keepHistoryOn("first");
+ actions.push(splitOne);
+ vgraph = splitOne(vgraph);
+ });
+ }
+ });
+ osmEntity.id.next.way = origCount;
+ vertexIds = vertices.map(function(v) {
+ return v.id;
+ });
+ vertices = [];
+ ways = [];
+ vertexIds.forEach(function(id2) {
+ var vertex2 = vgraph.entity(id2);
+ var parents2 = vgraph.parentWays(vertex2);
+ vertices.push(vertex2);
+ ways = ways.concat(parents2);
+ });
+ vertices = utilArrayUniq(vertices);
+ ways = utilArrayUniq(ways);
+ vertexIds = vertices.map(function(v) {
+ return v.id;
+ });
+ wayIds = ways.map(function(w) {
+ return w.id;
+ });
+ function withMetadata(way2, vertexIds2) {
+ var __oneWay = way2.isOneWay();
+ var __first = vertexIds2.indexOf(way2.first()) !== -1;
+ var __last = vertexIds2.indexOf(way2.last()) !== -1;
+ var __via = __first && __last;
+ var __from = __first && !__oneWay || __last;
+ var __to = __first || __last && !__oneWay;
+ return way2.update({
+ __first,
+ __last,
+ __from,
+ __via,
+ __to,
+ __oneWay
+ });
+ }
+ ways = [];
+ wayIds.forEach(function(id2) {
+ var way2 = withMetadata(vgraph.entity(id2), vertexIds);
+ vgraph = vgraph.replace(way2);
+ ways.push(way2);
+ });
+ var keepGoing;
+ var removeWayIds = [];
+ var removeVertexIds = [];
+ do {
+ keepGoing = false;
+ checkVertices = vertexIds.slice();
+ for (i2 = 0; i2 < checkVertices.length; i2++) {
+ var vertexId = checkVertices[i2];
+ vertex = vgraph.hasEntity(vertexId);
+ if (!vertex) {
+ if (vertexIds.indexOf(vertexId) !== -1) {
+ vertexIds.splice(vertexIds.indexOf(vertexId), 1);
+ }
+ removeVertexIds.push(vertexId);
+ continue;
+ }
+ parents = vgraph.parentWays(vertex);
+ if (parents.length < 3) {
+ if (vertexIds.indexOf(vertexId) !== -1) {
+ vertexIds.splice(vertexIds.indexOf(vertexId), 1);
+ }
+ }
+ if (parents.length === 2) {
+ var a = parents[0];
+ var b = parents[1];
+ var aIsLeaf = a && !a.__via;
+ var bIsLeaf = b && !b.__via;
+ var leaf, survivor;
+ if (aIsLeaf && !bIsLeaf) {
+ leaf = a;
+ survivor = b;
+ } else if (!aIsLeaf && bIsLeaf) {
+ leaf = b;
+ survivor = a;
+ }
+ if (leaf && survivor) {
+ survivor = withMetadata(survivor, vertexIds);
+ vgraph = vgraph.replace(survivor).remove(leaf);
+ removeWayIds.push(leaf.id);
+ keepGoing = true;
+ }
+ }
+ parents = vgraph.parentWays(vertex);
+ if (parents.length < 2) {
+ if (vertexIds.indexOf(vertexId) !== -1) {
+ vertexIds.splice(vertexIds.indexOf(vertexId), 1);
+ }
+ removeVertexIds.push(vertexId);
+ keepGoing = true;
+ }
+ if (parents.length < 1) {
+ vgraph = vgraph.remove(vertex);
+ }
+ }
+ } while (keepGoing);
+ vertices = vertices.filter(function(vertex2) {
+ return removeVertexIds.indexOf(vertex2.id) === -1;
+ }).map(function(vertex2) {
+ return vgraph.entity(vertex2.id);
+ });
+ ways = ways.filter(function(way2) {
+ return removeWayIds.indexOf(way2.id) === -1;
+ }).map(function(way2) {
+ return vgraph.entity(way2.id);
+ });
+ var intersection = {
+ graph: vgraph,
+ actions,
+ vertices,
+ ways
+ };
+ intersection.turns = function(fromWayId, maxViaWay) {
+ if (!fromWayId)
+ return [];
+ if (!maxViaWay)
+ maxViaWay = 0;
+ var vgraph2 = intersection.graph;
+ var keyVertexIds = intersection.vertices.map(function(v) {
+ return v.id;
+ });
+ var start2 = vgraph2.entity(fromWayId);
+ if (!start2 || !(start2.__from || start2.__via))
+ return [];
+ var maxPathLength = maxViaWay * 2 + 3;
+ var turns = [];
+ step(start2);
+ return turns;
+ function step(entity, currPath, currRestrictions, matchedRestriction) {
+ currPath = (currPath || []).slice();
+ if (currPath.length >= maxPathLength)
+ return;
+ currPath.push(entity.id);
+ currRestrictions = (currRestrictions || []).slice();
+ if (entity.type === "node") {
+ stepNode(entity, currPath, currRestrictions);
+ } else {
+ stepWay(entity, currPath, currRestrictions, matchedRestriction);
+ }
+ }
+ function stepNode(entity, currPath, currRestrictions) {
+ var i3, j3;
+ var parents2 = vgraph2.parentWays(entity);
+ var nextWays = [];
+ for (i3 = 0; i3 < parents2.length; i3++) {
+ var way2 = parents2[i3];
+ if (way2.__oneWay && way2.nodes[0] !== entity.id)
+ continue;
+ if (currPath.indexOf(way2.id) !== -1 && currPath.length >= 3)
+ continue;
+ var restrict = null;
+ for (j3 = 0; j3 < currRestrictions.length; j3++) {
+ var restriction = currRestrictions[j3];
+ var f2 = restriction.memberByRole("from");
+ var v = restriction.membersByRole("via");
+ var t = restriction.memberByRole("to");
+ var isNo = /^no_/.test(restriction.tags.restriction);
+ var isOnly = /^only_/.test(restriction.tags.restriction);
+ if (!(isNo || isOnly)) {
+ continue;
+ }
+ var matchesFrom = f2.id === fromWayId;
+ var matchesViaTo = false;
+ var isAlongOnlyPath = false;
+ if (t.id === way2.id) {
+ if (v.length === 1 && v[0].type === "node") {
+ matchesViaTo = v[0].id === entity.id && (matchesFrom && currPath.length === 2 || !matchesFrom && currPath.length > 2);
+ } else {
+ var pathVias = [];
+ for (k = 2; k < currPath.length; k += 2) {
+ pathVias.push(currPath[k]);
+ }
+ var restrictionVias = [];
+ for (k = 0; k < v.length; k++) {
+ if (v[k].type === "way") {
+ restrictionVias.push(v[k].id);
+ }
+ }
+ var diff = utilArrayDifference(pathVias, restrictionVias);
+ matchesViaTo = !diff.length;
+ }
+ } else if (isOnly) {
+ for (k = 0; k < v.length; k++) {
+ if (v[k].type === "way" && v[k].id === way2.id) {
+ isAlongOnlyPath = true;
+ break;
+ }
+ }
+ }
+ if (matchesViaTo) {
+ if (isOnly) {
+ restrict = { id: restriction.id, direct: matchesFrom, from: f2.id, only: true, end: true };
+ } else {
+ restrict = { id: restriction.id, direct: matchesFrom, from: f2.id, no: true, end: true };
+ }
+ } else {
+ if (isAlongOnlyPath) {
+ restrict = { id: restriction.id, direct: false, from: f2.id, only: true, end: false };
+ } else if (isOnly) {
+ restrict = { id: restriction.id, direct: false, from: f2.id, no: true, end: true };
+ }
+ }
+ if (restrict && restrict.direct)
+ break;
+ }
+ nextWays.push({ way: way2, restrict });
+ }
+ nextWays.forEach(function(nextWay) {
+ step(nextWay.way, currPath, currRestrictions, nextWay.restrict);
+ });
+ }
+ function stepWay(entity, currPath, currRestrictions, matchedRestriction) {
+ var i3;
+ if (currPath.length >= 3) {
+ var turnPath = currPath.slice();
+ if (matchedRestriction && matchedRestriction.direct === false) {
+ for (i3 = 0; i3 < turnPath.length; i3++) {
+ if (turnPath[i3] === matchedRestriction.from) {
+ turnPath = turnPath.slice(i3);
+ break;
+ }
+ }
+ }
+ var turn = pathToTurn(turnPath);
+ if (turn) {
+ if (matchedRestriction) {
+ turn.restrictionID = matchedRestriction.id;
+ turn.no = matchedRestriction.no;
+ turn.only = matchedRestriction.only;
+ turn.direct = matchedRestriction.direct;
+ }
+ turns.push(osmTurn(turn));
+ }
+ if (currPath[0] === currPath[2])
+ return;
+ }
+ if (matchedRestriction && matchedRestriction.end)
+ return;
+ var n1 = vgraph2.entity(entity.first());
+ var n2 = vgraph2.entity(entity.last());
+ var dist = geoSphericalDistance(n1.loc, n2.loc);
+ var nextNodes = [];
+ if (currPath.length > 1) {
+ if (dist > maxDistance)
+ return;
+ if (!entity.__via)
+ return;
+ }
+ if (!entity.__oneWay && // bidirectional..
+ keyVertexIds.indexOf(n1.id) !== -1 && // key vertex..
+ currPath.indexOf(n1.id) === -1) {
+ nextNodes.push(n1);
+ }
+ if (keyVertexIds.indexOf(n2.id) !== -1 && // key vertex..
+ currPath.indexOf(n2.id) === -1) {
+ nextNodes.push(n2);
+ }
+ nextNodes.forEach(function(nextNode) {
+ var fromRestrictions = vgraph2.parentRelations(entity).filter(function(r) {
+ if (!r.isRestriction())
+ return false;
+ var f2 = r.memberByRole("from");
+ if (!f2 || f2.id !== entity.id)
+ return false;
+ var isOnly = /^only_/.test(r.tags.restriction);
+ if (!isOnly)
+ return true;
+ var isOnlyVia = false;
+ var v = r.membersByRole("via");
+ if (v.length === 1 && v[0].type === "node") {
+ isOnlyVia = v[0].id === nextNode.id;
+ } else {
+ for (var i4 = 0; i4 < v.length; i4++) {
+ if (v[i4].type !== "way")
+ continue;
+ var viaWay = vgraph2.entity(v[i4].id);
+ if (viaWay.first() === nextNode.id || viaWay.last() === nextNode.id) {
+ isOnlyVia = true;
+ break;
+ }
+ }
+ }
+ return isOnlyVia;
+ });
+ step(nextNode, currPath, currRestrictions.concat(fromRestrictions), false);
+ });
+ }
+ function pathToTurn(path) {
+ if (path.length < 3)
+ return;
+ var fromWayId2, fromNodeId, fromVertexId;
+ var toWayId, toNodeId, toVertexId;
+ var viaWayIds, viaNodeId, isUturn;
+ fromWayId2 = path[0];
+ toWayId = path[path.length - 1];
+ if (path.length === 3 && fromWayId2 === toWayId) {
+ var way2 = vgraph2.entity(fromWayId2);
+ if (way2.__oneWay)
+ return null;
+ isUturn = true;
+ viaNodeId = fromVertexId = toVertexId = path[1];
+ fromNodeId = toNodeId = adjacentNode(fromWayId2, viaNodeId);
+ } else {
+ isUturn = false;
+ fromVertexId = path[1];
+ fromNodeId = adjacentNode(fromWayId2, fromVertexId);
+ toVertexId = path[path.length - 2];
+ toNodeId = adjacentNode(toWayId, toVertexId);
+ if (path.length === 3) {
+ viaNodeId = path[1];
+ } else {
+ viaWayIds = path.filter(function(entityId) {
+ return entityId[0] === "w";
+ });
+ viaWayIds = viaWayIds.slice(1, viaWayIds.length - 1);
+ }
+ }
+ return {
+ key: path.join("_"),
+ path,
+ from: { node: fromNodeId, way: fromWayId2, vertex: fromVertexId },
+ via: { node: viaNodeId, ways: viaWayIds },
+ to: { node: toNodeId, way: toWayId, vertex: toVertexId },
+ u: isUturn
+ };
+ function adjacentNode(wayId, affixId) {
+ var nodes2 = vgraph2.entity(wayId).nodes;
+ return affixId === nodes2[0] ? nodes2[1] : nodes2[nodes2.length - 2];
+ }
+ }
+ };
+ return intersection;
+ }
+ function osmInferRestriction(graph, turn, projection2) {
+ 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 angle2 = (geoAngle(fromVertex, fromNode, projection2) - geoAngle(toVertex, toNode, projection2)) * 180 / Math.PI;
+ while (angle2 < 0) {
+ angle2 += 360;
+ }
+ if (fromNode === toNode) {
+ return "no_u_turn";
+ }
+ if ((angle2 < 23 || angle2 > 336) && fromOneWay && toOneWay) {
+ return "no_u_turn";
+ }
+ if ((angle2 < 40 || angle2 > 319) && fromOneWay && toOneWay && turn.from.vertex !== turn.to.vertex) {
+ return "no_u_turn";
+ }
+ if (angle2 < 158) {
+ return "no_right_turn";
+ }
+ if (angle2 > 202) {
+ return "no_left_turn";
+ }
+ return "no_straight_on";
+ }
+
+ // modules/actions/merge_polygon.js
+ function actionMergePolygon(ids, newRelationId) {
+ function groupEntities(graph) {
+ var entities = ids.map(function(id2) {
+ return graph.entity(id2);
+ });
+ 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
+ );
+ }
+ var action = function(graph) {
+ var entities = groupEntities(graph);
+ var polygons = entities.multipolygon.reduce(function(polygons2, m) {
+ return polygons2.concat(osmJoinWays(m.members, graph));
+ }, []).concat(entities.closedWay.map(function(d) {
+ var member = [{ id: d.id }];
+ member.nodes = graph.childNodes(d);
+ return member;
+ }));
+ var contained = polygons.map(function(w, i2) {
+ return polygons.map(function(d, n2) {
+ if (i2 === n2)
+ return null;
+ return geoPolygonContainsPolygon(
+ d.nodes.map(function(n3) {
+ return n3.loc;
+ }),
+ w.nodes.map(function(n3) {
+ return n3.loc;
+ })
+ );
+ });
+ });
+ var members = [];
+ var outer = true;
+ while (polygons.length) {
+ extractUncontained(polygons);
+ polygons = polygons.filter(isContained);
+ contained = contained.filter(isContained).map(filterContained);
+ }
+ function isContained(d, i2) {
+ return contained[i2].some(function(val) {
+ return val;
+ });
+ }
+ function filterContained(d) {
+ return d.filter(isContained);
+ }
+ function extractUncontained(polygons2) {
+ polygons2.forEach(function(d, i2) {
+ if (!isContained(d, i2)) {
+ d.forEach(function(member) {
+ members.push({
+ type: "way",
+ id: member.id,
+ role: outer ? "outer" : "inner"
+ });
+ });
+ }
+ });
+ outer = !outer;
+ }
+ var relation;
+ if (entities.multipolygon.length > 0) {
+ var oldestID = utilOldestID(entities.multipolygon.map((entity) => entity.id));
+ relation = entities.multipolygon.find((entity) => entity.id === oldestID);
+ } else {
+ relation = osmRelation({ id: newRelationId, tags: { type: "multipolygon" } });
+ }
+ entities.multipolygon.forEach(function(m) {
+ if (m.id !== relation.id) {
+ 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";
+ }
+ if (members.some(isThisOuter)) {
+ relation = relation.mergeTags(way.tags);
+ graph = graph.replace(way.update({ tags: {} }));
+ }
+ });
+ return graph.replace(relation.update({
+ members,
+ tags: utilObjectOmit(relation.tags, ["area"])
+ }));
+ };
+ action.disabled = function(graph) {
+ var entities = groupEntities(graph);
+ if (entities.other.length > 0 || entities.closedWay.length + entities.multipolygon.length < 2) {
+ return "not_eligible";
+ }
+ if (!entities.multipolygon.every(function(r) {
+ return r.isComplete(graph);
+ })) {
+ return "incomplete_relation";
+ }
+ if (!entities.multipolygon.length) {
+ var sharedMultipolygons = [];
+ entities.closedWay.forEach(function(way, i2) {
+ if (i2 === 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 (sharedMultipolygons.length) {
+ return "not_eligible";
+ }
+ } else if (entities.closedWay.some(function(way) {
+ return utilArrayIntersection(graph.parentMultipolygons(way), entities.multipolygon).length;
+ })) {
+ return "not_eligible";
+ }
+ };
+ return action;
+ }
+
+ // modules/actions/merge_remote_changes.js
+ var import_fast_deep_equal = __toESM(require_fast_deep_equal());
+
+ // node_modules/node-diff3/index.mjs
+ function LCS(buffer1, buffer2) {
+ let equivalenceClasses = {};
+ for (let j2 = 0; j2 < buffer2.length; j2++) {
+ const item = buffer2[j2];
+ if (equivalenceClasses[item]) {
+ equivalenceClasses[item].push(j2);
+ } else {
+ equivalenceClasses[item] = [j2];
+ }
+ }
+ const NULLRESULT = { buffer1index: -1, buffer2index: -1, chain: null };
+ let candidates = [NULLRESULT];
+ for (let i2 = 0; i2 < buffer1.length; i2++) {
+ const item = buffer1[i2];
+ const buffer2indices = equivalenceClasses[item] || [];
+ let r = 0;
+ let c = candidates[0];
+ for (let jx = 0; jx < buffer2indices.length; jx++) {
+ const j2 = buffer2indices[jx];
+ let s;
+ for (s = r; s < candidates.length; s++) {
+ if (candidates[s].buffer2index < j2 && (s === candidates.length - 1 || candidates[s + 1].buffer2index > j2)) {
+ break;
+ }
+ }
+ if (s < candidates.length) {
+ const newCandidate = { buffer1index: i2, buffer2index: j2, chain: candidates[s] };
+ if (r === candidates.length) {
+ candidates.push(c);
+ } else {
+ candidates[r] = c;
+ }
+ r = s + 1;
+ c = newCandidate;
+ if (r === candidates.length) {
+ break;
+ }
+ }
+ }
+ candidates[r] = c;
+ }
+ return candidates[candidates.length - 1];
+ }
+ function diffIndices(buffer1, buffer2) {
+ const lcs = LCS(buffer1, buffer2);
+ let result = [];
+ let tail1 = buffer1.length;
+ let tail2 = buffer2.length;
+ for (let candidate = lcs; candidate !== null; candidate = candidate.chain) {
+ const mismatchLength1 = tail1 - candidate.buffer1index - 1;
+ const 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)
+ });
+ }
+ }
+ result.reverse();
+ return result;
+ }
+ function diff3MergeRegions(a, o, b) {
+ let hunks = [];
+ function addHunk(h, ab) {
+ hunks.push({
+ 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])
+ });
+ }
+ diffIndices(o, a).forEach((item) => addHunk(item, "a"));
+ diffIndices(o, b).forEach((item) => addHunk(item, "b"));
+ hunks.sort((x, y) => x.oStart - y.oStart);
+ let results = [];
+ let 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;
+ }
+ }
+ while (hunks.length) {
+ let hunk = hunks.shift();
+ let regionStart = hunk.oStart;
+ let regionEnd = hunk.oStart + hunk.oLength;
+ let regionHunks = [hunk];
+ advanceTo(regionStart);
+ while (hunks.length) {
+ const nextHunk = hunks[0];
+ const nextHunkStart = nextHunk.oStart;
+ if (nextHunkStart > regionEnd)
+ break;
+ regionEnd = Math.max(regionEnd, nextHunkStart + nextHunk.oLength);
+ regionHunks.push(hunks.shift());
+ }
+ if (regionHunks.length === 1) {
+ if (hunk.abLength > 0) {
+ const 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 {
+ let bounds = {
+ a: [a.length, -1, o.length, -1],
+ b: [b.length, -1, o.length, -1]
+ };
+ while (regionHunks.length) {
+ hunk = regionHunks.shift();
+ const oStart = hunk.oStart;
+ const oEnd = oStart + hunk.oLength;
+ const abStart = hunk.abStart;
+ const abEnd = abStart + hunk.abLength;
+ let b2 = bounds[hunk.ab];
+ b2[0] = Math.min(abStart, b2[0]);
+ b2[1] = Math.max(abEnd, b2[1]);
+ b2[2] = Math.min(oStart, b2[2]);
+ b2[3] = Math.max(oEnd, b2[3]);
+ }
+ const aStart = bounds.a[0] + (regionStart - bounds.a[2]);
+ const aEnd = bounds.a[1] + (regionEnd - bounds.a[3]);
+ const bStart = bounds.b[0] + (regionStart - bounds.b[2]);
+ const bEnd = bounds.b[1] + (regionEnd - bounds.b[3]);
+ let result = {
+ stable: false,
+ aStart,
+ aLength: aEnd - aStart,
+ aContent: a.slice(aStart, aEnd),
+ oStart: regionStart,
+ oLength: regionEnd - regionStart,
+ oContent: o.slice(regionStart, regionEnd),
+ bStart,
+ bLength: bEnd - bStart,
+ bContent: b.slice(bStart, bEnd)
+ };
+ results.push(result);
+ }
+ currOffset = regionEnd;
+ }
+ advanceTo(o.length);
+ return results;
+ }
+ function diff3Merge(a, o, b, options2) {
+ let defaults2 = {
+ excludeFalseConflicts: true,
+ stringSeparator: /\s+/
+ };
+ options2 = Object.assign(defaults2, options2);
+ if (typeof a === "string")
+ a = a.split(options2.stringSeparator);
+ if (typeof o === "string")
+ o = o.split(options2.stringSeparator);
+ if (typeof b === "string")
+ b = b.split(options2.stringSeparator);
+ let results = [];
+ const regions = diff3MergeRegions(a, o, b);
+ let okBuffer = [];
+ function flushOk() {
+ if (okBuffer.length) {
+ results.push({ ok: okBuffer });
+ }
+ okBuffer = [];
+ }
+ function isFalseConflict(a2, b2) {
+ if (a2.length !== b2.length)
+ return false;
+ for (let i2 = 0; i2 < a2.length; i2++) {
+ if (a2[i2] !== b2[i2])
+ return false;
+ }
+ return true;
+ }
+ regions.forEach((region) => {
+ if (region.stable) {
+ okBuffer.push(...region.bufferContent);
+ } else {
+ if (options2.excludeFalseConflicts && isFalseConflict(region.aContent, region.bContent)) {
+ okBuffer.push(...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
+ }
+ });
+ }
+ }
+ });
+ flushOk();
+ return results;
+ }
+
+ // modules/actions/merge_remote_changes.js
+ var import_lodash = __toESM(require_lodash());
+ function actionMergeRemoteChanges(id2, localGraph, remoteGraph, discardTags, formatUser) {
+ discardTags = discardTags || {};
+ var _option = "safe";
+ var _conflicts = [];
+ function user(d) {
+ return typeof formatUser === "function" ? formatUser(d) : (0, import_lodash.escape)(d);
+ }
+ function mergeLocation(remote, target) {
+ function pointEqual(a, b) {
+ var epsilon3 = 1e-6;
+ return Math.abs(a[0] - b[0]) < epsilon3 && Math.abs(a[1] - b[1]) < epsilon3;
+ }
+ if (_option === "force_local" || pointEqual(target.loc, remote.loc)) {
+ return target;
+ }
+ if (_option === "force_remote") {
+ return target.update({ loc: remote.loc });
+ }
+ _conflicts.push(_t.html("merge_remote_changes.conflict.location", { user: { html: user(remote.user) } }));
+ return target;
+ }
+ function mergeNodes(base, remote, target) {
+ if (_option === "force_local" || (0, import_fast_deep_equal.default)(target.nodes, remote.nodes)) {
+ return target;
+ }
+ if (_option === "force_remote") {
+ return target.update({ nodes: remote.nodes });
+ }
+ 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 });
+ for (var i2 = 0; i2 < hunks.length; i2++) {
+ var hunk = hunks[i2];
+ if (hunk.ok) {
+ nodes.push.apply(nodes, hunk.ok);
+ } else {
+ var c = hunk.conflict;
+ if ((0, import_fast_deep_equal.default)(c.o, c.a)) {
+ nodes.push.apply(nodes, c.b);
+ } else if ((0, import_fast_deep_equal.default)(c.o, c.b)) {
+ nodes.push.apply(nodes, c.a);
+ } else {
+ _conflicts.push(_t.html("merge_remote_changes.conflict.nodelist", { user: { html: user(remote.user) } }));
+ break;
+ }
+ }
+ }
+ return _conflicts.length === ccount ? target.update({ nodes }) : target;
+ }
+ function mergeChildren(targetWay, children2, updates, graph) {
+ function isUsed(node2, targetWay2) {
+ var hasInterestingParent = graph.parentWays(node2).some(function(way) {
+ return way.id !== targetWay2.id;
+ });
+ return node2.hasInterestingTags() || hasInterestingParent || graph.parentRelations(node2).length > 0;
+ }
+ var ccount = _conflicts.length;
+ for (var i2 = 0; i2 < children2.length; i2++) {
+ var id3 = children2[i2];
+ var node = graph.hasEntity(id3);
+ if (targetWay.nodes.indexOf(id3) === -1) {
+ if (node && !isUsed(node, targetWay)) {
+ updates.removeIds.push(id3);
+ }
+ continue;
+ }
+ var local = localGraph.hasEntity(id3);
+ var remote = remoteGraph.hasEntity(id3);
+ 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 });
+ }
+ updates.replacements.push(target);
+ } else if (_option === "safe" && local && remote && local.version !== remote.version) {
+ target = osmEntity(local, { version: remote.version });
+ if (remote.visible) {
+ target = mergeLocation(remote, target);
+ } else {
+ _conflicts.push(_t.html("merge_remote_changes.conflict.deleted", { user: { html: user(remote.user) } }));
+ }
+ if (_conflicts.length !== ccount)
+ break;
+ updates.replacements.push(target);
+ }
+ }
+ return targetWay;
+ }
+ function updateChildren(updates, graph) {
+ for (var i2 = 0; i2 < updates.replacements.length; i2++) {
+ graph = graph.replace(updates.replacements[i2]);
+ }
+ if (updates.removeIds.length) {
+ graph = actionDeleteMultiple(updates.removeIds)(graph);
+ }
+ return graph;
+ }
+ function mergeMembers(remote, target) {
+ if (_option === "force_local" || (0, import_fast_deep_equal.default)(target.members, remote.members)) {
+ return target;
+ }
+ if (_option === "force_remote") {
+ return target.update({ members: remote.members });
+ }
+ _conflicts.push(_t.html("merge_remote_changes.conflict.memberlist", { user: { html: user(remote.user) } }));
+ return target;
+ }
+ function mergeTags(base, remote, target) {
+ if (_option === "force_local" || (0, import_fast_deep_equal.default)(target.tags, remote.tags)) {
+ return target;
+ }
+ if (_option === "force_remote") {
+ return target.update({ tags: remote.tags });
+ }
+ 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(k2) {
+ return !discardTags[k2];
+ });
+ var tags = Object.assign({}, a);
+ var changed = false;
+ for (var i2 = 0; i2 < keys.length; i2++) {
+ var k = keys[i2];
+ if (o[k] !== b[k] && a[k] !== b[k]) {
+ if (o[k] !== a[k]) {
+ _conflicts.push(_t.html(
+ "merge_remote_changes.conflict.tags",
+ { tag: k, local: a[k], remote: b[k], user: { html: user(remote.user) } }
+ ));
+ } else {
+ if (b.hasOwnProperty(k)) {
+ tags[k] = b[k];
+ } else {
+ delete tags[k];
+ }
+ changed = true;
+ }
+ }
+ }
+ return changed && _conflicts.length === ccount ? target.update({ tags }) : target;
+ }
+ var action = function(graph) {
+ var updates = { replacements: [], removeIds: [] };
+ var base = graph.base().entities[id2];
+ var local = localGraph.entity(id2);
+ var remote = remoteGraph.entity(id2);
+ var target = osmEntity(local, { version: remote.version });
+ if (!remote.visible) {
+ if (_option === "force_remote") {
+ return actionDeleteMultiple([id2])(graph);
+ } else if (_option === "force_local") {
+ if (target.type === "way") {
+ target = mergeChildren(target, utilArrayUniq(local.nodes), updates, graph);
+ graph = updateChildren(updates, graph);
+ }
+ return graph.replace(target);
+ } else {
+ _conflicts.push(_t.html("merge_remote_changes.conflict.deleted", { user: { html: user(remote.user) } }));
+ return graph;
+ }
+ }
+ if (target.type === "node") {
+ target = mergeLocation(remote, target);
+ } else if (target.type === "way") {
+ 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);
+ }
+ target = mergeTags(base, remote, target);
+ if (!_conflicts.length) {
+ graph = updateChildren(updates, graph).replace(target);
+ }
+ return graph;
+ };
+ action.withOption = function(opt) {
+ _option = opt;
+ return action;
+ };
+ action.conflicts = function() {
+ return _conflicts;
+ };
+ return action;
+ }
+
+ // modules/actions/move.js
+ function actionMove(moveIDs, tryDelta, projection2, cache) {
+ var _delta = tryDelta;
+ function setupCache(graph) {
+ function canMove(nodeID) {
+ if (moveIDs.indexOf(nodeID) !== -1)
+ return true;
+ var parents = graph.parentWays(graph.entity(nodeID));
+ if (parents.length < 3)
+ return true;
+ var parentsMoving = parents.every(function(way) {
+ return cache.moving[way.id];
+ });
+ if (!parentsMoving)
+ delete cache.moving[nodeID];
+ return parentsMoving;
+ }
+ function cacheEntities(ids) {
+ for (var i2 = 0; i2 < ids.length; i2++) {
+ var id2 = ids[i2];
+ if (cache.moving[id2])
+ continue;
+ cache.moving[id2] = true;
+ var entity = graph.hasEntity(id2);
+ if (!entity)
+ continue;
+ if (entity.type === "node") {
+ cache.nodes.push(id2);
+ cache.startLoc[id2] = entity.loc;
+ } else if (entity.type === "way") {
+ cache.ways.push(id2);
+ cacheEntities(entity.nodes);
+ } else {
+ cacheEntities(entity.members.map(function(member) {
+ return member.id;
+ }));
+ }
+ }
+ }
+ function cacheIntersections(ids) {
+ function isEndpoint(way2, id3) {
+ return !way2.isClosed() && !!way2.affix(id3);
+ }
+ for (var i2 = 0; i2 < ids.length; i2++) {
+ var id2 = ids[i2];
+ var childNodes = graph.childNodes(graph.entity(id2));
+ for (var j2 = 0; j2 < childNodes.length; j2++) {
+ var node = childNodes[j2];
+ var parents = graph.parentWays(node);
+ if (parents.length !== 2)
+ continue;
+ var moved = graph.entity(id2);
+ var unmoved = null;
+ for (var k = 0; k < parents.length; k++) {
+ var way = parents[k];
+ if (!cache.moving[way.id]) {
+ unmoved = way;
+ break;
+ }
+ }
+ if (!unmoved)
+ continue;
+ 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)
+ });
+ }
+ }
+ }
+ if (!cache) {
+ cache = {};
+ }
+ 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;
+ }
+ }
+ 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;
+ }
+ var prev = graph.hasEntity(way.nodes[prevIndex]);
+ var next = graph.hasEntity(way.nodes[nextIndex]);
+ if (!prev || !next)
+ return graph;
+ var key = wayId + "_" + nodeId;
+ var orig = cache.replacedVertex[key];
+ if (!orig) {
+ orig = osmNode();
+ cache.replacedVertex[key] = orig;
+ cache.startLoc[orig.id] = cache.startLoc[nodeId];
+ }
+ var start2, end;
+ if (delta) {
+ start2 = projection2(cache.startLoc[nodeId]);
+ end = projection2.invert(geoVecAdd(start2, delta));
+ } else {
+ end = cache.startLoc[nodeId];
+ }
+ orig = orig.move(end);
+ var angle2 = Math.abs(geoAngle(orig, prev, projection2) - geoAngle(orig, next, projection2)) * 180 / Math.PI;
+ if (angle2 > 175 && angle2 < 185)
+ return graph;
+ var p1 = [prev.loc, orig.loc, moved.loc, next.loc].map(projection2);
+ var p2 = [prev.loc, moved.loc, orig.loc, next.loc].map(projection2);
+ var d1 = geoPathLength(p1);
+ var d2 = geoPathLength(p2);
+ var insertAt = d1 <= d2 ? movedIndex : nextIndex;
+ if (way.isClosed() && insertAt === 0)
+ insertAt = len;
+ way = way.addNode(orig.id, insertAt);
+ return graph.replace(orig).replace(way);
+ }
+ function removeDuplicateVertices(wayId, graph) {
+ var way = graph.entity(wayId);
+ var epsilon3 = 1e-6;
+ var prev, curr;
+ function isInteresting(node, graph2) {
+ return graph2.parentWays(node).length > 1 || graph2.parentRelations(node).length || node.hasInterestingTags();
+ }
+ for (var i2 = 0; i2 < way.nodes.length; i2++) {
+ curr = graph.entity(way.nodes[i2]);
+ if (prev && curr && geoVecEqual(prev.loc, curr.loc, epsilon3)) {
+ 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;
+ }
+ return graph;
+ }
+ 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;
+ if (isEP1 && isEP2)
+ return graph;
+ var nodes1 = graph.childNodes(way1).filter(function(n2) {
+ return n2 !== vertex;
+ });
+ var nodes2 = graph.childNodes(way2).filter(function(n2) {
+ return n2 !== 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, projection2(vertex.loc), projection2);
+ var edge2 = !isEP2 && geoChooseEdge(nodes2, projection2(vertex.loc), projection2);
+ var loc;
+ if (!isEP1 && !isEP2) {
+ var epsilon3 = 1e-6, maxIter = 10;
+ for (var i2 = 0; i2 < maxIter; i2++) {
+ loc = geoVecInterp(edge1.loc, edge2.loc, 0.5);
+ edge1 = geoChooseEdge(nodes1, projection2(loc), projection2);
+ edge2 = geoChooseEdge(nodes2, projection2(loc), projection2);
+ if (Math.abs(edge1.distance - edge2.distance) < epsilon3)
+ break;
+ }
+ } else if (!isEP1) {
+ loc = edge1.loc;
+ } else {
+ loc = edge2.loc;
+ }
+ graph = graph.replace(vertex.move(loc));
+ if (!isEP1 && edge1.index !== way1.nodes.indexOf(vertex.id)) {
+ way1 = way1.removeNode(vertex.id).addNode(vertex.id, edge1.index);
+ graph = graph.replace(way1);
+ }
+ 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;
+ }
+ function cleanupIntersections(graph) {
+ for (var i2 = 0; i2 < cache.intersections.length; i2++) {
+ var obj = cache.intersections[i2];
+ 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;
+ }
+ function limitDelta(graph) {
+ function moveNode(loc) {
+ return geoVecAdd(projection2(loc), _delta);
+ }
+ for (var i2 = 0; i2 < cache.intersections.length; i2++) {
+ var obj = cache.intersections[i2];
+ if (obj.movedIsEP && obj.unmovedIsEP)
+ continue;
+ if (!obj.movedIsEP)
+ continue;
+ var node = graph.entity(obj.nodeId);
+ var start2 = projection2(node.loc);
+ var end = geoVecAdd(start2, _delta);
+ var movedNodes = graph.childNodes(graph.entity(obj.movedId));
+ var movedPath = movedNodes.map(function(n2) {
+ return moveNode(n2.loc);
+ });
+ var unmovedNodes = graph.childNodes(graph.entity(obj.unmovedId));
+ var unmovedPath = unmovedNodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var hits = geoPathIntersections(movedPath, unmovedPath);
+ for (var j2 = 0; i2 < hits.length; i2++) {
+ if (geoVecEqual(hits[j2], end))
+ continue;
+ var edge = geoChooseEdge(unmovedNodes, end, projection2);
+ _delta = geoVecSubtract(projection2(edge.loc), start2);
+ }
+ }
+ }
+ var action = function(graph) {
+ if (_delta[0] === 0 && _delta[1] === 0)
+ return graph;
+ setupCache(graph);
+ if (cache.intersections.length) {
+ limitDelta(graph);
+ }
+ for (var i2 = 0; i2 < cache.nodes.length; i2++) {
+ var node = graph.entity(cache.nodes[i2]);
+ var start2 = projection2(node.loc);
+ var end = geoVecAdd(start2, _delta);
+ graph = graph.replace(node.move(projection2.invert(end)));
+ }
+ if (cache.intersections.length) {
+ graph = cleanupIntersections(graph);
+ }
+ return graph;
+ };
+ action.delta = function() {
+ return _delta;
+ };
+ return action;
+ }
+
+ // modules/actions/move_member.js
+ function actionMoveMember(relationId, fromIndex, toIndex) {
+ return function(graph) {
+ return graph.replace(graph.entity(relationId).moveMember(fromIndex, toIndex));
+ };
+ }
+
+ // modules/actions/move_node.js
+ function actionMoveNode(nodeID, toLoc) {
+ var action = function(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;
+ }
+
+ // modules/actions/noop.js
+ function actionNoop() {
+ return function(graph) {
+ return graph;
+ };
+ }
+
+ // modules/actions/orthogonalize.js
+ function actionOrthogonalize(wayID, projection2, vertexID, degThresh, ep) {
+ var epsilon3 = ep || 1e-4;
+ var threshold = degThresh || 13;
+ var lowerThreshold = Math.cos((90 - threshold) * Math.PI / 180);
+ var upperThreshold = Math.cos(threshold * Math.PI / 180);
+ var action = function(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("");
+ if (way.tags.nonsquare) {
+ var tags = Object.assign({}, way.tags);
+ delete tags.nonsquare;
+ way = way.update({ tags });
+ }
+ graph = graph.replace(way);
+ var isClosed = way.isClosed();
+ var nodes = graph.childNodes(way).slice();
+ if (isClosed)
+ nodes.pop();
+ if (vertexID !== void 0) {
+ nodes = nodeSubset(nodes, vertexID, isClosed);
+ if (nodes.length !== 3)
+ return graph;
+ }
+ var nodeCount = {};
+ var points = [];
+ var corner = { i: 0, dotp: 1 };
+ var node, point2, loc, score, motions, i2, j2;
+ for (i2 = 0; i2 < nodes.length; i2++) {
+ node = nodes[i2];
+ nodeCount[node.id] = (nodeCount[node.id] || 0) + 1;
+ points.push({ id: node.id, coord: projection2(node.loc) });
+ }
+ if (points.length === 3) {
+ for (i2 = 0; i2 < 1e3; i2++) {
+ motions = points.map(calcMotion);
+ points[corner.i].coord = geoVecAdd(points[corner.i].coord, motions[corner.i]);
+ score = corner.dotp;
+ if (score < epsilon3) {
+ break;
+ }
+ }
+ node = graph.entity(nodes[corner.i].id);
+ loc = projection2.invert(points[corner.i].coord);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
+ } else {
+ var straights = [];
+ var simplified = [];
+ for (i2 = 0; i2 < points.length; i2++) {
+ point2 = points[i2];
+ var dotp = 0;
+ if (isClosed || i2 > 0 && i2 < points.length - 1) {
+ var a = points[(i2 - 1 + points.length) % points.length];
+ var b = points[(i2 + 1) % points.length];
+ dotp = Math.abs(geoOrthoNormalizedDotProduct(a.coord, b.coord, point2.coord));
+ }
+ if (dotp > upperThreshold) {
+ straights.push(point2);
+ } else {
+ simplified.push(point2);
+ }
+ }
+ var bestPoints = clonePoints(simplified);
+ var originalPoints = clonePoints(simplified);
+ score = Infinity;
+ for (i2 = 0; i2 < 1e3; i2++) {
+ motions = simplified.map(calcMotion);
+ for (j2 = 0; j2 < motions.length; j2++) {
+ simplified[j2].coord = geoVecAdd(simplified[j2].coord, motions[j2]);
+ }
+ var newScore = geoOrthoCalcScore(simplified, isClosed, epsilon3, threshold);
+ if (newScore < score) {
+ bestPoints = clonePoints(simplified);
+ score = newScore;
+ }
+ if (score < epsilon3) {
+ break;
+ }
+ }
+ var bestCoords = bestPoints.map(function(p) {
+ return p.coord;
+ });
+ if (isClosed)
+ bestCoords.push(bestCoords[0]);
+ for (i2 = 0; i2 < bestPoints.length; i2++) {
+ point2 = bestPoints[i2];
+ if (!geoVecEqual(originalPoints[i2].coord, point2.coord)) {
+ node = graph.entity(point2.id);
+ loc = projection2.invert(point2.coord);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
+ }
+ }
+ for (i2 = 0; i2 < straights.length; i2++) {
+ point2 = straights[i2];
+ if (nodeCount[point2.id] > 1)
+ continue;
+ node = graph.entity(point2.id);
+ if (t === 1 && graph.parentWays(node).length === 1 && graph.parentRelations(node).length === 0 && !node.hasInterestingTags()) {
+ graph = actionDeleteNode(node.id)(graph);
+ } else {
+ var choice = geoVecProject(point2.coord, bestCoords);
+ if (choice) {
+ loc = projection2.invert(choice.target);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t)));
+ }
+ }
+ }
+ }
+ return graph;
+ function clonePoints(array2) {
+ return array2.map(function(p) {
+ return { id: p.id, coord: [p.coord[0], p.coord[1]] };
+ });
+ }
+ function calcMotion(point3, i3, array2) {
+ if (!isClosed && (i3 === 0 || i3 === array2.length - 1))
+ return [0, 0];
+ if (nodeCount[array2[i3].id] > 1)
+ return [0, 0];
+ var a2 = array2[(i3 - 1 + array2.length) % array2.length].coord;
+ var origin = point3.coord;
+ var b2 = array2[(i3 + 1) % array2.length].coord;
+ var p = geoVecSubtract(a2, origin);
+ var q = geoVecSubtract(b2, origin);
+ var scale = 2 * Math.min(geoVecLength(p), geoVecLength(q));
+ p = geoVecNormalize(p);
+ q = geoVecNormalize(q);
+ var dotp2 = p[0] * q[0] + p[1] * q[1];
+ var val = Math.abs(dotp2);
+ if (val < lowerThreshold) {
+ corner.i = i3;
+ corner.dotp = val;
+ var vec = geoVecNormalize(geoVecAdd(p, q));
+ return geoVecScale(vec, 0.1 * dotp2 * scale);
+ }
+ return [0, 0];
+ }
+ };
+ function nodeSubset(nodes, vertexID2, isClosed) {
+ var first = isClosed ? 0 : 1;
+ var last = isClosed ? nodes.length : nodes.length - 1;
+ for (var i2 = first; i2 < last; i2++) {
+ if (nodes[i2].id === vertexID2) {
+ return [
+ nodes[(i2 - 1 + nodes.length) % nodes.length],
+ nodes[i2],
+ nodes[(i2 + 1) % nodes.length]
+ ];
+ }
+ }
+ return [];
+ }
+ action.disabled = function(graph) {
+ var way = graph.entity(wayID);
+ way = way.removeNode("");
+ graph = graph.replace(way);
+ var isClosed = way.isClosed();
+ var nodes = graph.childNodes(way).slice();
+ if (isClosed)
+ nodes.pop();
+ var allowStraightAngles = false;
+ if (vertexID !== void 0) {
+ allowStraightAngles = true;
+ nodes = nodeSubset(nodes, vertexID, isClosed);
+ if (nodes.length !== 3)
+ return "end_vertex";
+ }
+ var coords = nodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var score = geoOrthoCanOrthogonalize(coords, isClosed, epsilon3, threshold, allowStraightAngles);
+ if (score === null) {
+ return "not_squarish";
+ } else if (score === 0) {
+ return "square_enough";
+ } else {
+ return false;
+ }
+ };
+ action.transitionable = true;
+ return action;
+ }
+
+ // modules/actions/restrict_turn.js
+ 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(id2) {
+ return graph.entity(id2);
+ });
+ var members = [];
+ members.push({ id: fromWay.id, type: "way", role: "from" });
+ 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" });
+ });
+ }
+ members.push({ id: toWay.id, type: "way", role: "to" });
+ return graph.replace(osmRelation({
+ id: restrictionID,
+ tags: {
+ type: "restriction",
+ restriction: restrictionType
+ },
+ members
+ }));
+ };
+ }
+
+ // modules/actions/revert.js
+ function actionRevert(id2) {
+ var action = function(graph) {
+ var entity = graph.hasEntity(id2), base = graph.base().entities[id2];
+ if (entity && !base) {
+ if (entity.type === "node") {
+ graph.parentWays(entity).forEach(function(parent) {
+ parent = parent.removeNode(id2);
+ graph = graph.replace(parent);
+ if (parent.isDegenerate()) {
+ graph = actionDeleteWay(parent.id)(graph);
+ }
+ });
+ }
+ graph.parentRelations(entity).forEach(function(parent) {
+ parent = parent.removeMembersWithID(id2);
+ graph = graph.replace(parent);
+ if (parent.isDegenerate()) {
+ graph = actionDeleteRelation(parent.id)(graph);
+ }
+ });
+ }
+ return graph.revert(id2);
+ };
+ return action;
+ }
+
+ // modules/actions/rotate.js
+ function actionRotate(rotateIds, pivot, angle2, projection2) {
+ var action = function(graph) {
+ return graph.update(function(graph2) {
+ utilGetAllNodes(rotateIds, graph2).forEach(function(node) {
+ var point2 = geoRotate([projection2(node.loc)], angle2, pivot)[0];
+ graph2 = graph2.replace(node.move(projection2.invert(point2)));
+ });
+ });
+ };
+ return action;
+ }
+
+ // modules/actions/scale.js
+ function actionScale(ids, pivotLoc, scaleFactor, projection2) {
+ return function(graph) {
+ return graph.update(function(graph2) {
+ let point2, radial;
+ utilGetAllNodes(ids, graph2).forEach(function(node) {
+ point2 = projection2(node.loc);
+ radial = [
+ point2[0] - pivotLoc[0],
+ point2[1] - pivotLoc[1]
+ ];
+ point2 = [
+ pivotLoc[0] + scaleFactor * radial[0],
+ pivotLoc[1] + scaleFactor * radial[1]
+ ];
+ graph2 = graph2.replace(node.move(projection2.invert(point2)));
+ });
+ });
+ };
+ }
+
+ // modules/actions/straighten_nodes.js
+ function actionStraightenNodes(nodeIDs, projection2) {
+ function positionAlongWay(a, o, b) {
+ return geoVecDot(a, b, o) / geoVecDot(b, b, o);
+ }
+ function getEndpoints(points) {
+ var ssr = geoGetSmallestSurroundingRectangle(points);
+ 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];
+ }
+ return [p2, q2];
+ }
+ var action = function(graph, t) {
+ if (t === null || !isFinite(t))
+ t = 1;
+ t = Math.min(Math.max(+t, 0), 1);
+ var nodes = nodeIDs.map(function(id2) {
+ return graph.entity(id2);
+ });
+ var points = nodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var endpoints = getEndpoints(points);
+ var startPoint = endpoints[0];
+ var endPoint = endpoints[1];
+ for (var i2 = 0; i2 < points.length; i2++) {
+ var node = nodes[i2];
+ var point2 = points[i2];
+ var u = positionAlongWay(point2, startPoint, endPoint);
+ var point22 = geoVecInterp(startPoint, endPoint, u);
+ var loc2 = projection2.invert(point22);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t)));
+ }
+ return graph;
+ };
+ action.disabled = function(graph) {
+ var nodes = nodeIDs.map(function(id2) {
+ return graph.entity(id2);
+ });
+ var points = nodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var endpoints = getEndpoints(points);
+ var startPoint = endpoints[0];
+ var endPoint = endpoints[1];
+ var maxDistance = 0;
+ for (var i2 = 0; i2 < points.length; i2++) {
+ var point2 = points[i2];
+ var u = positionAlongWay(point2, startPoint, endPoint);
+ var p = geoVecInterp(startPoint, endPoint, u);
+ var dist = geoVecLength(p, point2);
+ if (!isNaN(dist) && dist > maxDistance) {
+ maxDistance = dist;
+ }
+ }
+ if (maxDistance < 1e-4) {
+ return "straight_enough";
+ }
+ };
+ action.transitionable = true;
+ return action;
+ }
+
+ // modules/actions/straighten_way.js
+ function actionStraightenWay(selectedIDs, projection2) {
+ function positionAlongWay(a, o, b) {
+ return geoVecDot(a, b, o) / geoVecDot(b, b, o);
+ }
+ 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(n2) {
+ return graph.entity(n2).type === "node";
+ });
+ for (var i2 = 0; i2 < selectedWays.length; i2++) {
+ var way = graph.entity(selectedWays[i2]);
+ nodes = way.nodes.slice(0);
+ remainingWays.push(nodes);
+ startNodes.push(nodes[0]);
+ endNodes.push(nodes[nodes.length - 1]);
+ }
+ startNodes = startNodes.filter(function(n2) {
+ return startNodes.indexOf(n2) === startNodes.lastIndexOf(n2);
+ });
+ endNodes = endNodes.filter(function(n2) {
+ return endNodes.indexOf(n2) === endNodes.lastIndexOf(n2);
+ });
+ var currNode = utilArrayDifference(startNodes, endNodes).concat(utilArrayDifference(endNodes, startNodes))[0];
+ var nextWay = [];
+ nodes = [];
+ var getNextWay = function(currNode2, remainingWays2) {
+ return remainingWays2.filter(function(way2) {
+ return way2[0] === currNode2 || way2[way2.length - 1] === currNode2;
+ })[0];
+ };
+ while (remainingWays.length) {
+ nextWay = getNextWay(currNode, remainingWays);
+ remainingWays = utilArrayDifference(remainingWays, [nextWay]);
+ if (nextWay[0] !== currNode) {
+ nextWay.reverse();
+ }
+ nodes = nodes.concat(nextWay);
+ currNode = nodes[nodes.length - 1];
+ }
+ 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);
+ }
+ return nodes.map(function(n2) {
+ return graph.entity(n2);
+ });
+ }
+ function shouldKeepNode(node, graph) {
+ return graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags();
+ }
+ var action = function(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(n2) {
+ return projection2(n2.loc);
+ });
+ var startPoint = points[0];
+ var endPoint = points[points.length - 1];
+ var toDelete = [];
+ var i2;
+ for (i2 = 1; i2 < points.length - 1; i2++) {
+ var node = nodes[i2];
+ var point2 = points[i2];
+ if (t < 1 || shouldKeepNode(node, graph)) {
+ var u = positionAlongWay(point2, startPoint, endPoint);
+ var p = geoVecInterp(startPoint, endPoint, u);
+ var loc2 = projection2.invert(p);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t)));
+ } else {
+ if (toDelete.indexOf(node) === -1) {
+ toDelete.push(node);
+ }
+ }
+ }
+ for (i2 = 0; i2 < toDelete.length; i2++) {
+ graph = actionDeleteNode(toDelete[i2].id)(graph);
+ }
+ return graph;
+ };
+ action.disabled = function(graph) {
+ var nodes = allNodes(graph);
+ var points = nodes.map(function(n2) {
+ return projection2(n2.loc);
+ });
+ var startPoint = points[0];
+ var endPoint = points[points.length - 1];
+ var threshold = 0.2 * geoVecLength(startPoint, endPoint);
+ var i2;
+ if (threshold === 0) {
+ return "too_bendy";
+ }
+ var maxDistance = 0;
+ for (i2 = 1; i2 < points.length - 1; i2++) {
+ var point2 = points[i2];
+ var u = positionAlongWay(point2, startPoint, endPoint);
+ var p = geoVecInterp(startPoint, endPoint, u);
+ var dist = geoVecLength(p, point2);
+ if (isNaN(dist) || dist > threshold) {
+ return "too_bendy";
+ } else if (dist > maxDistance) {
+ maxDistance = dist;
+ }
+ }
+ var keepingAllNodes = nodes.every(function(node, i3) {
+ return i3 === 0 || i3 === nodes.length - 1 || shouldKeepNode(node, graph);
+ });
+ if (maxDistance < 1e-4 && // Allow straightening even if already straight in order to remove extraneous nodes
+ keepingAllNodes) {
+ return "straight_enough";
+ }
+ };
+ action.transitionable = true;
+ return action;
+ }
+
+ // modules/actions/unrestrict_turn.js
+ function actionUnrestrictTurn(turn) {
+ return function(graph) {
+ return actionDeleteRelation(turn.restrictionID)(graph);
+ };
+ }
+
+ // modules/actions/reflect.js
+ function actionReflect(reflectIds, projection2) {
+ var _useLongAxis = true;
+ var action = function(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(n2) {
+ return projection2(n2.loc);
+ });
+ var ssr = geoGetSmallestSurroundingRectangle(points);
+ 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 (_useLongAxis && isLong || !_useLongAxis && !isLong) {
+ p = p1;
+ q = q1;
+ } else {
+ p = p2;
+ q = q2;
+ }
+ 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 (var i2 = 0; i2 < nodes.length; i2++) {
+ var node = nodes[i2];
+ var c = projection2(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 = projection2.invert(c2);
+ node = node.move(geoVecInterp(node.loc, loc2, t));
+ graph = graph.replace(node);
+ }
+ return graph;
+ };
+ action.useLongAxis = function(val) {
+ if (!arguments.length)
+ return _useLongAxis;
+ _useLongAxis = val;
+ return action;
+ };
+ action.transitionable = true;
+ return action;
+ }
+
+ // modules/actions/upgrade_tags.js
+ function actionUpgradeTags(entityId, oldTags, replaceTags) {
+ return function(graph) {
+ var entity = graph.entity(entityId);
+ var tags = Object.assign({}, entity.tags);
+ var transferValue;
+ var semiIndex;
+ for (var oldTagKey in oldTags) {
+ if (!(oldTagKey in tags))
+ continue;
+ if (oldTags[oldTagKey] === "*") {
+ transferValue = tags[oldTagKey];
+ delete tags[oldTagKey];
+ } else if (oldTags[oldTagKey] === tags[oldTagKey]) {
+ delete tags[oldTagKey];
+ } else {
+ var vals = tags[oldTagKey].split(";").filter(Boolean);
+ var oldIndex = vals.indexOf(oldTags[oldTagKey]);
+ if (vals.length === 1 || oldIndex === -1) {
+ delete tags[oldTagKey];
+ } else {
+ if (replaceTags && replaceTags[oldTagKey]) {
+ semiIndex = oldIndex;
+ }
+ vals.splice(oldIndex, 1);
+ tags[oldTagKey] = vals.join(";");
+ }
+ }
+ }
+ if (replaceTags) {
+ for (var replaceKey in replaceTags) {
+ var replaceValue = replaceTags[replaceKey];
+ if (replaceValue === "*") {
+ if (tags[replaceKey] && tags[replaceKey] !== "no") {
+ continue;
+ } else {
+ tags[replaceKey] = "yes";
+ }
+ } else if (replaceValue === "$1") {
+ tags[replaceKey] = transferValue;
+ } else {
+ if (tags[replaceKey] && oldTags[replaceKey] && semiIndex !== void 0) {
+ 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;
+ }
+ }
+ }
+ }
+ return graph.replace(entity.update({ tags }));
+ };
+ }
+
+ // modules/behavior/edit.js
+ function behaviorEdit(context) {
+ function behavior() {
+ context.map().minzoom(context.minEditableZoom());
+ }
+ behavior.off = function() {
+ context.map().minzoom(0);
+ };
+ return behavior;
+ }
+
+ // modules/behavior/hover.js
+ function behaviorHover(context) {
+ var dispatch10 = dispatch_default("hover");
+ var _selection = select_default2(null);
+ var _newNodeId = null;
+ var _initialNodeID = null;
+ var _altDisables;
+ var _ignoreVertex;
+ var _targets = [];
+ 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);
+ dispatch10.call("hover", this, null);
+ }
+ }
+ 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);
+ dispatch10.call("hover", this, _targets);
+ }
+ }
+ function behavior(selection2) {
+ _selection = selection2;
+ _targets = [];
+ if (_initialNodeID) {
+ _newNodeId = _initialNodeID;
+ _initialNodeID = null;
+ } else {
+ _newNodeId = null;
+ }
+ _selection.on(_pointerPrefix + "over.hover", pointerover).on(_pointerPrefix + "out.hover", pointerout).on(_pointerPrefix + "down.hover", pointerover);
+ select_default2(window).on(_pointerPrefix + "up.hover pointercancel.hover", pointerout, true).on("keydown.hover", keydown).on("keyup.hover", keyup);
+ function eventTarget(d3_event) {
+ var datum2 = d3_event.target && d3_event.target.__data__;
+ if (typeof datum2 !== "object")
+ return null;
+ if (!(datum2 instanceof osmEntity) && datum2.properties && datum2.properties.entity instanceof osmEntity) {
+ return datum2.properties.entity;
+ }
+ return datum2;
+ }
+ function pointerover(d3_event) {
+ 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);
+ updateHover(d3_event, _targets);
+ }
+ }
+ function pointerout(d3_event) {
+ var target = eventTarget(d3_event);
+ var index = _targets.indexOf(target);
+ if (index !== -1) {
+ _targets.splice(index);
+ updateHover(d3_event, _targets);
+ }
+ }
+ function allowsVertex(d) {
+ return d.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d, context.graph());
+ }
+ function modeAllowsHover(target) {
+ var mode = context.mode();
+ if (mode.id === "add-point") {
+ return mode.preset.matchGeometry("vertex") || target.type !== "way" && target.geometry(context.graph()) !== "vertex";
+ }
+ return true;
+ }
+ function updateHover(d3_event, targets) {
+ _selection.selectAll(".hover").classed("hover", false);
+ _selection.selectAll(".hover-suppressed").classed("hover-suppressed", false);
+ 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;
+ }
+ targets = targets.filter(function(datum3) {
+ if (datum3 instanceof osmEntity) {
+ return datum3.id !== _newNodeId && (datum3.type !== "node" || !_ignoreVertex || allowsVertex(datum3)) && modeAllowsHover(datum3);
+ }
+ return true;
+ });
+ var selector = "";
+ for (var i2 in targets) {
+ var datum2 = targets[i2];
+ if (datum2.__featurehash__) {
+ selector += ", .data" + datum2.__featurehash__;
+ } else if (datum2 instanceof QAItem) {
+ selector += ", ." + datum2.service + ".itemId-" + datum2.id;
+ } else if (datum2 instanceof osmNote) {
+ selector += ", .note-" + datum2.id;
+ } else if (datum2 instanceof osmEntity) {
+ selector += ", ." + datum2.id;
+ if (datum2.type === "relation") {
+ for (var j2 in datum2.members) {
+ selector += ", ." + datum2.members[j2].id;
+ }
+ }
+ }
+ }
+ var suppressed = _altDisables && d3_event && d3_event.altKey;
+ if (selector.trim().length) {
+ selector = selector.slice(1);
+ _selection.selectAll(selector).classed(suppressed ? "hover-suppressed" : "hover", true);
+ }
+ dispatch10.call("hover", this, !suppressed && targets);
+ }
+ }
+ behavior.off = function(selection2) {
+ selection2.selectAll(".hover").classed("hover", false);
+ selection2.selectAll(".hover-suppressed").classed("hover-suppressed", false);
+ selection2.classed("hover-disabled", false);
+ selection2.on(_pointerPrefix + "over.hover", null).on(_pointerPrefix + "out.hover", null).on(_pointerPrefix + "down.hover", null);
+ select_default2(window).on(_pointerPrefix + "up.hover pointercancel.hover", null, true).on("keydown.hover", null).on("keyup.hover", null);
+ };
+ behavior.altDisables = function(val) {
+ if (!arguments.length)
+ return _altDisables;
+ _altDisables = val;
+ return behavior;
+ };
+ behavior.ignoreVertex = function(val) {
+ if (!arguments.length)
+ return _ignoreVertex;
+ _ignoreVertex = val;
+ return behavior;
+ };
+ behavior.initialNodeID = function(nodeId) {
+ _initialNodeID = nodeId;
+ return behavior;
+ };
+ return utilRebind(behavior, dispatch10, "on");
+ }
+
+ // modules/behavior/draw.js
+ var _disableSpace = false;
+ var _lastSpace = null;
+ function behaviorDraw(context) {
+ var dispatch10 = dispatch_default(
+ "move",
+ "down",
+ "downcancel",
+ "click",
+ "clickWay",
+ "clickNode",
+ "undo",
+ "cancel",
+ "finish"
+ );
+ var keybinding = utilKeybinding("draw");
+ var _hover = behaviorHover(context).altDisables(true).ignoreVertex(true).on("hover", context.ui().sidebar.hover);
+ var _edit = behaviorEdit(context);
+ var _closeTolerance = 4;
+ var _tolerance = 12;
+ var _mouseLeave = false;
+ var _lastMouse = null;
+ var _lastPointerUpEvent;
+ var _downPointer;
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ function datum2(d3_event) {
+ var mode = context.mode();
+ var isNote = mode && mode.id.indexOf("note") !== -1;
+ if (d3_event.altKey || isNote)
+ return {};
+ var element;
+ if (d3_event.type === "keydown") {
+ element = _lastMouse && _lastMouse.target;
+ } else {
+ element = d3_event.target;
+ }
+ var d = element.__data__;
+ return d && d.properties && d.properties.target ? d : {};
+ }
+ function pointerdown(d3_event) {
+ if (_downPointer)
+ return;
+ var pointerLocGetter = utilFastMouse(this);
+ _downPointer = {
+ id: d3_event.pointerId || "mouse",
+ pointerLocGetter,
+ downTime: +new Date(),
+ downLoc: pointerLocGetter(d3_event)
+ };
+ dispatch10.call("down", this, d3_event, datum2(d3_event));
+ }
+ 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 (dist < _closeTolerance || dist < _tolerance && t2 - downPointer.downTime < 500) {
+ select_default2(window).on("click.draw-block", function() {
+ d3_event.stopPropagation();
+ }, true);
+ context.map().dblclickZoomEnable(false);
+ window.setTimeout(function() {
+ context.map().dblclickZoomEnable(true);
+ select_default2(window).on("click.draw-block", null);
+ }, 500);
+ click(d3_event, p2);
+ }
+ }
+ function pointermove(d3_event) {
+ if (_downPointer && _downPointer.id === (d3_event.pointerId || "mouse") && !_downPointer.isCancelled) {
+ var p2 = _downPointer.pointerLocGetter(d3_event);
+ var dist = geoVecLength(_downPointer.downLoc, p2);
+ if (dist >= _closeTolerance) {
+ _downPointer.isCancelled = true;
+ dispatch10.call("downcancel", this);
+ }
+ }
+ if (d3_event.pointerType && d3_event.pointerType !== "mouse" || d3_event.buttons || _downPointer)
+ return;
+ if (_lastPointerUpEvent && _lastPointerUpEvent.pointerType !== "mouse" && d3_event.timeStamp - _lastPointerUpEvent.timeStamp < 100)
+ return;
+ _lastMouse = d3_event;
+ dispatch10.call("move", this, d3_event, datum2(d3_event));
+ }
+ function pointercancel(d3_event) {
+ if (_downPointer && _downPointer.id === (d3_event.pointerId || "mouse")) {
+ if (!_downPointer.isCancelled) {
+ dispatch10.call("downcancel", this);
+ }
+ _downPointer = null;
+ }
+ }
+ function mouseenter() {
+ _mouseLeave = false;
+ }
+ function mouseleave() {
+ _mouseLeave = true;
+ }
+ function allowsVertex(d) {
+ return d.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d, context.graph());
+ }
+ function click(d3_event, loc) {
+ var d = datum2(d3_event);
+ var target = d && d.properties && d.properties.entity;
+ var mode = context.mode();
+ if (target && target.type === "node" && allowsVertex(target)) {
+ dispatch10.call("clickNode", this, target, d);
+ return;
+ } else if (target && target.type === "way" && (mode.id !== "add-point" || mode.preset.matchGeometry("vertex"))) {
+ 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]];
+ dispatch10.call("clickWay", this, choice.loc, edge, d);
+ return;
+ }
+ } else if (mode.id !== "add-point" || mode.preset.matchGeometry("point")) {
+ var locLatLng = context.projection.invert(loc);
+ dispatch10.call("click", this, locLatLng, d);
+ }
+ }
+ function space(d3_event) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var currSpace = context.map().mouse();
+ if (_disableSpace && _lastSpace) {
+ var dist = geoVecLength(_lastSpace, currSpace);
+ if (dist > _tolerance) {
+ _disableSpace = false;
+ }
+ }
+ if (_disableSpace || _mouseLeave || !_lastMouse)
+ return;
+ _lastSpace = currSpace;
+ _disableSpace = true;
+ select_default2(window).on("keyup.space-block", function() {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ _disableSpace = false;
+ select_default2(window).on("keyup.space-block", null);
+ });
+ 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();
+ dispatch10.call("undo");
+ }
+ function del(d3_event) {
+ d3_event.preventDefault();
+ dispatch10.call("cancel");
+ }
+ function ret(d3_event) {
+ d3_event.preventDefault();
+ dispatch10.call("finish");
+ }
+ function behavior(selection2) {
+ context.install(_hover);
+ context.install(_edit);
+ _downPointer = null;
+ keybinding.on("\u232B", backspace).on("\u2326", del).on("\u238B", ret).on("\u21A9", ret).on("space", space).on("\u2325space", space);
+ selection2.on("mouseenter.draw", mouseenter).on("mouseleave.draw", mouseleave).on(_pointerPrefix + "down.draw", pointerdown).on(_pointerPrefix + "move.draw", pointermove);
+ select_default2(window).on(_pointerPrefix + "up.draw", pointerup, true).on("pointercancel.draw", pointercancel, true);
+ select_default2(document).call(keybinding);
+ return behavior;
+ }
+ behavior.off = function(selection2) {
+ context.ui().sidebar.hover.cancel();
+ context.uninstall(_hover);
+ context.uninstall(_edit);
+ selection2.on("mouseenter.draw", null).on("mouseleave.draw", null).on(_pointerPrefix + "down.draw", null).on(_pointerPrefix + "move.draw", null);
+ select_default2(window).on(_pointerPrefix + "up.draw", null).on("pointercancel.draw", null);
+ select_default2(document).call(keybinding.unbind);
+ };
+ behavior.hover = function() {
+ return _hover;
+ };
+ return utilRebind(behavior, dispatch10, "on");
+ }
+
+ // modules/behavior/breathe.js
+ var import_fast_deep_equal2 = __toESM(require_fast_deep_equal());
+
+ // node_modules/d3-scale/src/init.js
+ function initRange(domain2, range3) {
+ switch (arguments.length) {
+ case 0:
+ break;
+ case 1:
+ this.range(domain2);
+ break;
+ default:
+ this.range(range3).domain(domain2);
+ break;
+ }
+ return this;
+ }
+
+ // node_modules/d3-scale/src/constant.js
+ function constants(x) {
+ return function() {
+ return x;
+ };
+ }
+
+ // node_modules/d3-scale/src/number.js
+ function number2(x) {
+ return +x;
+ }
+
+ // node_modules/d3-scale/src/continuous.js
+ var unit = [0, 1];
+ function identity3(x) {
+ return x;
+ }
+ function normalize(a, b) {
+ return (b -= a = +a) ? function(x) {
+ return (x - a) / b;
+ } : constants(isNaN(b) ? NaN : 0.5);
+ }
+ 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));
+ };
+ }
+ function bimap(domain2, range3, interpolate) {
+ var d0 = domain2[0], d1 = domain2[1], r0 = range3[0], r1 = range3[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));
+ };
+ }
+ function polymap(domain2, range3, interpolate) {
+ var j2 = Math.min(domain2.length, range3.length) - 1, d = new Array(j2), r = new Array(j2), i2 = -1;
+ if (domain2[j2] < domain2[0]) {
+ domain2 = domain2.slice().reverse();
+ range3 = range3.slice().reverse();
+ }
+ while (++i2 < j2) {
+ d[i2] = normalize(domain2[i2], domain2[i2 + 1]);
+ r[i2] = interpolate(range3[i2], range3[i2 + 1]);
+ }
+ return function(x) {
+ var i3 = bisect_default(domain2, x, 1, j2) - 1;
+ return r[i3](d[i3](x));
+ };
+ }
+ function copy(source, target) {
+ return target.domain(source.domain()).range(source.range()).interpolate(source.interpolate()).clamp(source.clamp()).unknown(source.unknown());
+ }
+ function transformer2() {
+ var domain2 = unit, range3 = unit, interpolate = value_default, transform2, untransform, unknown, clamp3 = identity3, piecewise, output, input;
+ function rescale() {
+ var n2 = Math.min(domain2.length, range3.length);
+ if (clamp3 !== identity3)
+ clamp3 = clamper(domain2[0], domain2[n2 - 1]);
+ piecewise = n2 > 2 ? polymap : bimap;
+ output = input = null;
+ return scale;
+ }
+ function scale(x) {
+ return x == null || isNaN(x = +x) ? unknown : (output || (output = piecewise(domain2.map(transform2), range3, interpolate)))(transform2(clamp3(x)));
+ }
+ scale.invert = function(y) {
+ return clamp3(untransform((input || (input = piecewise(range3, domain2.map(transform2), number_default)))(y)));
+ };
+ scale.domain = function(_) {
+ return arguments.length ? (domain2 = Array.from(_, number2), rescale()) : domain2.slice();
+ };
+ scale.range = function(_) {
+ return arguments.length ? (range3 = Array.from(_), rescale()) : range3.slice();
+ };
+ scale.rangeRound = function(_) {
+ return range3 = Array.from(_), interpolate = round_default, rescale();
+ };
+ scale.clamp = function(_) {
+ return arguments.length ? (clamp3 = _ ? true : identity3, rescale()) : clamp3 !== identity3;
+ };
+ scale.interpolate = function(_) {
+ return arguments.length ? (interpolate = _, rescale()) : interpolate;
+ };
+ scale.unknown = function(_) {
+ return arguments.length ? (unknown = _, scale) : unknown;
+ };
+ return function(t, u) {
+ transform2 = t, untransform = u;
+ return rescale();
+ };
+ }
+ function continuous() {
+ return transformer2()(identity3, identity3);
+ }
+
+ // node_modules/d3-format/src/formatDecimal.js
+ function formatDecimal_default(x) {
+ return Math.abs(x = Math.round(x)) >= 1e21 ? x.toLocaleString("en").replace(/,/g, "") : x.toString(10);
+ }
+ function formatDecimalParts(x, p) {
+ if ((i2 = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf("e")) < 0)
+ return null;
+ var i2, coefficient = x.slice(0, i2);
+ return [
+ coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,
+ +x.slice(i2 + 1)
+ ];
+ }
+
+ // node_modules/d3-format/src/exponent.js
+ function exponent_default(x) {
+ return x = formatDecimalParts(Math.abs(x)), x ? x[1] : NaN;
+ }
+
+ // node_modules/d3-format/src/formatGroup.js
+ function formatGroup_default(grouping, thousands) {
+ return function(value, width) {
+ var i2 = value.length, t = [], j2 = 0, g = grouping[0], length = 0;
+ while (i2 > 0 && g > 0) {
+ if (length + g + 1 > width)
+ g = Math.max(1, width - length);
+ t.push(value.substring(i2 -= g, i2 + g));
+ if ((length += g + 1) > width)
+ break;
+ g = grouping[j2 = (j2 + 1) % grouping.length];
+ }
+ return t.reverse().join(thousands);
+ };
+ }
+
+ // node_modules/d3-format/src/formatNumerals.js
+ function formatNumerals_default(numerals) {
+ return function(value) {
+ return value.replace(/[0-9]/g, function(i2) {
+ return numerals[+i2];
+ });
+ };
+ }
+
+ // node_modules/d3-format/src/formatSpecifier.js
+ 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;
+ function FormatSpecifier(specifier) {
+ this.fill = specifier.fill === void 0 ? " " : specifier.fill + "";
+ this.align = specifier.align === void 0 ? ">" : specifier.align + "";
+ this.sign = specifier.sign === void 0 ? "-" : specifier.sign + "";
+ this.symbol = specifier.symbol === void 0 ? "" : specifier.symbol + "";
+ this.zero = !!specifier.zero;
+ this.width = specifier.width === void 0 ? void 0 : +specifier.width;
+ this.comma = !!specifier.comma;
+ this.precision = specifier.precision === void 0 ? void 0 : +specifier.precision;
+ this.trim = !!specifier.trim;
+ this.type = specifier.type === void 0 ? "" : specifier.type + "";
+ }
+ FormatSpecifier.prototype.toString = function() {
+ return this.fill + this.align + this.sign + this.symbol + (this.zero ? "0" : "") + (this.width === void 0 ? "" : Math.max(1, this.width | 0)) + (this.comma ? "," : "") + (this.precision === void 0 ? "" : "." + Math.max(0, this.precision | 0)) + (this.trim ? "~" : "") + this.type;
+ };
+
+ // node_modules/d3-format/src/formatTrim.js
+ function formatTrim_default(s) {
+ out:
+ for (var n2 = s.length, i2 = 1, i0 = -1, i1; i2 < n2; ++i2) {
+ switch (s[i2]) {
+ case ".":
+ i0 = i1 = i2;
+ break;
+ case "0":
+ if (i0 === 0)
+ i0 = i2;
+ i1 = i2;
+ break;
+ default:
+ if (!+s[i2])
+ break out;
+ if (i0 > 0)
+ i0 = 0;
+ break;
+ }
+ }
+ return i0 > 0 ? s.slice(0, i0) + s.slice(i1 + 1) : s;
+ }
+
+ // node_modules/d3-format/src/formatPrefixAuto.js
+ var prefixExponent;
+ function formatPrefixAuto_default(x, p) {
+ var d = formatDecimalParts(x, p);
+ if (!d)
+ return x + "";
+ var coefficient = d[0], exponent = d[1], i2 = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1, n2 = coefficient.length;
+ return i2 === n2 ? coefficient : i2 > n2 ? coefficient + new Array(i2 - n2 + 1).join("0") : i2 > 0 ? coefficient.slice(0, i2) + "." + coefficient.slice(i2) : "0." + new Array(1 - i2).join("0") + formatDecimalParts(x, Math.max(0, p + i2 - 1))[0];
+ }
+
+ // node_modules/d3-format/src/formatRounded.js
+ function formatRounded_default(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");
+ }
+
+ // node_modules/d3-format/src/formatTypes.js
+ var formatTypes_default = {
+ "%": (x, p) => (x * 100).toFixed(p),
+ "b": (x) => Math.round(x).toString(2),
+ "c": (x) => x + "",
+ "d": formatDecimal_default,
+ "e": (x, p) => x.toExponential(p),
+ "f": (x, p) => x.toFixed(p),
+ "g": (x, p) => x.toPrecision(p),
+ "o": (x) => Math.round(x).toString(8),
+ "p": (x, p) => formatRounded_default(x * 100, p),
+ "r": formatRounded_default,
+ "s": formatPrefixAuto_default,
+ "X": (x) => Math.round(x).toString(16).toUpperCase(),
+ "x": (x) => Math.round(x).toString(16)
+ };
+
+ // node_modules/d3-format/src/identity.js
+ function identity_default3(x) {
+ return x;
+ }
+
+ // node_modules/d3-format/src/locale.js
+ var map = Array.prototype.map;
+ var prefixes = ["y", "z", "a", "f", "p", "n", "\xB5", "m", "", "k", "M", "G", "T", "P", "E", "Z", "Y"];
+ function locale_default(locale2) {
+ var group = locale2.grouping === void 0 || locale2.thousands === void 0 ? identity_default3 : formatGroup_default(map.call(locale2.grouping, Number), locale2.thousands + ""), currencyPrefix = locale2.currency === void 0 ? "" : locale2.currency[0] + "", currencySuffix = locale2.currency === void 0 ? "" : locale2.currency[1] + "", decimal = locale2.decimal === void 0 ? "." : locale2.decimal + "", numerals = locale2.numerals === void 0 ? identity_default3 : formatNumerals_default(map.call(locale2.numerals, String)), percent = locale2.percent === void 0 ? "%" : locale2.percent + "", minus = locale2.minus === void 0 ? "\u2212" : locale2.minus + "", nan = locale2.nan === void 0 ? "NaN" : locale2.nan + "";
+ function newFormat(specifier) {
+ specifier = formatSpecifier(specifier);
+ var fill = specifier.fill, align = specifier.align, sign2 = specifier.sign, symbol = specifier.symbol, zero3 = specifier.zero, width = specifier.width, comma = specifier.comma, precision2 = specifier.precision, trim = specifier.trim, type2 = specifier.type;
+ if (type2 === "n")
+ comma = true, type2 = "g";
+ else if (!formatTypes_default[type2])
+ precision2 === void 0 && (precision2 = 12), trim = true, type2 = "g";
+ if (zero3 || fill === "0" && align === "=")
+ zero3 = true, fill = "0", align = "=";
+ var prefix = symbol === "$" ? currencyPrefix : symbol === "#" && /[boxX]/.test(type2) ? "0" + type2.toLowerCase() : "", suffix = symbol === "$" ? currencySuffix : /[%p]/.test(type2) ? percent : "";
+ var formatType = formatTypes_default[type2], maybeSuffix = /[defgprs%]/.test(type2);
+ precision2 = precision2 === void 0 ? 6 : /[gprs]/.test(type2) ? Math.max(1, Math.min(21, precision2)) : Math.max(0, Math.min(20, precision2));
+ function format2(value) {
+ var valuePrefix = prefix, valueSuffix = suffix, i2, n2, c;
+ if (type2 === "c") {
+ valueSuffix = formatType(value) + valueSuffix;
+ value = "";
+ } else {
+ value = +value;
+ var valueNegative = value < 0 || 1 / value < 0;
+ value = isNaN(value) ? nan : formatType(Math.abs(value), precision2);
+ if (trim)
+ value = formatTrim_default(value);
+ if (valueNegative && +value === 0 && sign2 !== "+")
+ valueNegative = false;
+ valuePrefix = (valueNegative ? sign2 === "(" ? sign2 : minus : sign2 === "-" || sign2 === "(" ? "" : sign2) + valuePrefix;
+ valueSuffix = (type2 === "s" ? prefixes[8 + prefixExponent / 3] : "") + valueSuffix + (valueNegative && sign2 === "(" ? ")" : "");
+ if (maybeSuffix) {
+ i2 = -1, n2 = value.length;
+ while (++i2 < n2) {
+ if (c = value.charCodeAt(i2), 48 > c || c > 57) {
+ valueSuffix = (c === 46 ? decimal + value.slice(i2 + 1) : value.slice(i2)) + valueSuffix;
+ value = value.slice(0, i2);
+ break;
+ }
+ }
+ }
+ }
+ if (comma && !zero3)
+ value = group(value, Infinity);
+ var length = valuePrefix.length + value.length + valueSuffix.length, padding = length < width ? new Array(width - length + 1).join(fill) : "";
+ if (comma && zero3)
+ value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = "";
+ switch (align) {
+ case "<":
+ value = valuePrefix + value + valueSuffix + padding;
+ break;
+ case "=":
+ value = valuePrefix + padding + value + valueSuffix;
+ break;
+ case "^":
+ value = padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length);
+ break;
+ default:
+ value = padding + valuePrefix + value + valueSuffix;
+ break;
+ }
+ return numerals(value);
+ }
+ format2.toString = function() {
+ return specifier + "";
+ };
+ return format2;
+ }
+ function formatPrefix2(specifier, value) {
+ var f2 = newFormat((specifier = formatSpecifier(specifier), specifier.type = "f", specifier)), e = Math.max(-8, Math.min(8, Math.floor(exponent_default(value) / 3))) * 3, k = Math.pow(10, -e), prefix = prefixes[8 + e / 3];
+ return function(value2) {
+ return f2(k * value2) + prefix;
+ };
+ }
+ return {
+ format: newFormat,
+ formatPrefix: formatPrefix2
+ };
+ }
+
+ // node_modules/d3-format/src/defaultLocale.js
+ var locale;
+ var format;
+ var formatPrefix;
+ defaultLocale({
+ thousands: ",",
+ grouping: [3],
+ currency: ["$", ""]
+ });
+ function defaultLocale(definition) {
+ locale = locale_default(definition);
+ format = locale.format;
+ formatPrefix = locale.formatPrefix;
+ return locale;
+ }
+
+ // node_modules/d3-format/src/precisionFixed.js
+ function precisionFixed_default(step) {
+ return Math.max(0, -exponent_default(Math.abs(step)));
+ }
+
+ // node_modules/d3-format/src/precisionPrefix.js
+ function precisionPrefix_default(step, value) {
+ return Math.max(0, Math.max(-8, Math.min(8, Math.floor(exponent_default(value) / 3))) * 3 - exponent_default(Math.abs(step)));
+ }
+
+ // node_modules/d3-format/src/precisionRound.js
+ function precisionRound_default(step, max3) {
+ step = Math.abs(step), max3 = Math.abs(max3) - step;
+ return Math.max(0, exponent_default(max3) - exponent_default(step)) + 1;
+ }
+
+ // node_modules/d3-scale/src/tickFormat.js
+ function tickFormat(start2, stop, count, specifier) {
+ var step = tickStep(start2, stop, count), precision2;
+ specifier = formatSpecifier(specifier == null ? ",f" : specifier);
+ switch (specifier.type) {
+ case "s": {
+ var value = Math.max(Math.abs(start2), Math.abs(stop));
+ if (specifier.precision == null && !isNaN(precision2 = precisionPrefix_default(step, value)))
+ specifier.precision = precision2;
+ return formatPrefix(specifier, value);
+ }
+ case "":
+ case "e":
+ case "g":
+ case "p":
+ case "r": {
+ if (specifier.precision == null && !isNaN(precision2 = precisionRound_default(step, Math.max(Math.abs(start2), Math.abs(stop)))))
+ specifier.precision = precision2 - (specifier.type === "e");
+ break;
+ }
+ case "f":
+ case "%": {
+ if (specifier.precision == null && !isNaN(precision2 = precisionFixed_default(step)))
+ specifier.precision = precision2 - (specifier.type === "%") * 2;
+ break;
+ }
+ }
+ return format(specifier);
+ }
+
+ // node_modules/d3-scale/src/linear.js
+ function linearish(scale) {
+ var domain2 = scale.domain;
+ scale.ticks = function(count) {
+ var d = domain2();
+ return ticks(d[0], d[d.length - 1], count == null ? 10 : count);
+ };
+ scale.tickFormat = function(count, specifier) {
+ var d = domain2();
+ return tickFormat(d[0], d[d.length - 1], count == null ? 10 : count, specifier);
+ };
+ scale.nice = function(count) {
+ if (count == null)
+ count = 10;
+ var d = domain2();
+ var i0 = 0;
+ var i1 = d.length - 1;
+ var start2 = d[i0];
+ var stop = d[i1];
+ var prestep;
+ var step;
+ var maxIter = 10;
+ if (stop < start2) {
+ step = start2, start2 = stop, stop = step;
+ step = i0, i0 = i1, i1 = step;
+ }
+ while (maxIter-- > 0) {
+ step = tickIncrement(start2, stop, count);
+ if (step === prestep) {
+ d[i0] = start2;
+ d[i1] = stop;
+ return domain2(d);
+ } else if (step > 0) {
+ start2 = Math.floor(start2 / step) * step;
+ stop = Math.ceil(stop / step) * step;
+ } else if (step < 0) {
+ start2 = Math.ceil(start2 * step) / step;
+ stop = Math.floor(stop * step) / step;
+ } else {
+ break;
+ }
+ prestep = step;
+ }
+ return scale;
+ };
+ return scale;
+ }
+ function linear3() {
+ var scale = continuous();
+ scale.copy = function() {
+ return copy(scale, linear3());
+ };
+ initRange.apply(scale, arguments);
+ return linearish(scale);
+ }
+
+ // node_modules/d3-scale/src/quantize.js
+ function quantize() {
+ var x05 = 0, x12 = 1, n2 = 1, domain2 = [0.5], range3 = [0, 1], unknown;
+ function scale(x) {
+ return x != null && x <= x ? range3[bisect_default(domain2, x, 0, n2)] : unknown;
+ }
+ function rescale() {
+ var i2 = -1;
+ domain2 = new Array(n2);
+ while (++i2 < n2)
+ domain2[i2] = ((i2 + 1) * x12 - (i2 - n2) * x05) / (n2 + 1);
+ return scale;
+ }
+ scale.domain = function(_) {
+ return arguments.length ? ([x05, x12] = _, x05 = +x05, x12 = +x12, rescale()) : [x05, x12];
+ };
+ scale.range = function(_) {
+ return arguments.length ? (n2 = (range3 = Array.from(_)).length - 1, rescale()) : range3.slice();
+ };
+ scale.invertExtent = function(y) {
+ var i2 = range3.indexOf(y);
+ return i2 < 0 ? [NaN, NaN] : i2 < 1 ? [x05, domain2[0]] : i2 >= n2 ? [domain2[n2 - 1], x12] : [domain2[i2 - 1], domain2[i2]];
+ };
+ scale.unknown = function(_) {
+ return arguments.length ? (unknown = _, scale) : scale;
+ };
+ scale.thresholds = function() {
+ return domain2.slice();
+ };
+ scale.copy = function() {
+ return quantize().domain([x05, x12]).range(range3).unknown(unknown);
+ };
+ return initRange.apply(linearish(scale), arguments);
+ }
+
+ // modules/behavior/breathe.js
+ function behaviorBreathe() {
+ var duration = 800;
+ var steps = 4;
+ var selector = ".selected.shadow, .selected .shadow";
+ var _selected = select_default2(null);
+ var _classed = "";
+ var _params = {};
+ var _done = false;
+ var _timer;
+ function ratchetyInterpolator(a, b, steps2, units) {
+ a = Number(a);
+ b = Number(b);
+ var sample = quantize().domain([0, 1]).range(quantize_default(number_default(a, b), steps2));
+ return function(t) {
+ return String(sample(t)) + (units || "");
+ };
+ }
+ function reset(selection2) {
+ selection2.style("stroke-opacity", null).style("stroke-width", null).style("fill-opacity", null).style("r", null);
+ }
+ function setAnimationParams(transition2, fromTo) {
+ var toFrom = fromTo === "from" ? "to" : "from";
+ transition2.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"
+ );
+ });
+ }
+ function calcAnimationParams(selection2) {
+ selection2.call(reset).each(function(d) {
+ var s = select_default2(this);
+ var tag = s.node().tagName;
+ var p = { "from": {}, "to": {} };
+ var opacity;
+ var width;
+ if (tag === "circle") {
+ opacity = Number(s.style("fill-opacity") || 0.5);
+ width = Number(s.style("r") || 15.5);
+ } else {
+ opacity = Number(s.style("stroke-opacity") || 0.7);
+ width = Number(s.style("stroke-width") || 10);
+ }
+ 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;
+ });
+ }
+ 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);
+ _selected = select_default2(null);
+ return;
+ }
+ if (!(0, import_fast_deep_equal2.default)(currSelected.data(), _selected.data()) || currClassed !== _classed) {
+ _selected.call(reset);
+ _classed = currClassed;
+ _selected = currSelected.call(calcAnimationParams);
+ }
+ var didCallNextRun = false;
+ _selected.transition().duration(duration).call(setAnimationParams, fromTo).on("end", function() {
+ if (!didCallNextRun) {
+ surface.call(run, toFrom);
+ didCallNextRun = true;
+ }
+ if (!select_default2(this).classed("selected")) {
+ reset(select_default2(this));
+ }
+ });
+ }
+ function behavior(surface) {
+ _done = false;
+ _timer = timer(function() {
+ if (surface.selectAll(selector).empty()) {
+ return false;
+ }
+ surface.call(run, "from");
+ _timer.stop();
+ return true;
+ }, 20);
+ }
+ behavior.restartIfNeeded = function(surface) {
+ if (_selected.empty()) {
+ surface.call(run, "from");
+ if (_timer) {
+ _timer.stop();
+ }
+ }
+ };
+ behavior.off = function() {
+ _done = true;
+ if (_timer) {
+ _timer.stop();
+ }
+ _selected.interrupt().call(reset);
+ };
+ return behavior;
+ }
+
+ // modules/behavior/operation.js
+ function behaviorOperation(context) {
+ var _operation;
+ function keypress(d3_event) {
+ if (!context.map().withinEditableZoom())
+ return;
+ if (_operation.availableForKeypress && !_operation.availableForKeypress())
+ return;
+ d3_event.preventDefault();
+ var disabled = _operation.disabled();
+ if (disabled) {
+ context.ui().flash.duration(4e3).iconName("#iD-operation-" + _operation.id).iconClass("operation disabled").label(_operation.tooltip())();
+ } else {
+ context.ui().flash.duration(2e3).iconName("#iD-operation-" + _operation.id).iconClass("operation").label(_operation.annotation() || _operation.title)();
+ if (_operation.point)
+ _operation.point(null);
+ _operation();
+ }
+ }
+ function behavior() {
+ if (_operation && _operation.available()) {
+ context.keybinding().on(_operation.keys, keypress);
+ }
+ return behavior;
+ }
+ behavior.off = function() {
+ context.keybinding().off(_operation.keys);
+ };
+ behavior.which = function(_) {
+ if (!arguments.length)
+ return _operation;
+ _operation = _;
+ return behavior;
+ };
+ return behavior;
+ }
+
+ // modules/operations/circularize.js
+ function operationCircularize(context, selectedIDs) {
+ var _extent;
+ var _actions = selectedIDs.map(getAction).filter(Boolean);
+ var _amount = _actions.length === 1 ? "single" : "multiple";
+ var _coords = utilGetAllNodes(selectedIDs, context.graph()).map(function(n2) {
+ return n2.loc;
+ });
+ function getAction(entityID) {
+ var entity = context.entity(entityID);
+ if (entity.type !== "way" || new Set(entity.nodes).size <= 1)
+ return null;
+ if (!_extent) {
+ _extent = entity.extent(context.graph());
+ } else {
+ _extent = _extent.extend(entity.extent(context.graph()));
+ }
+ return actionCircularize(entityID, context.projection);
+ }
+ var operation = function() {
+ if (!_actions.length)
+ return;
+ var combinedAction = function(graph, t) {
+ _actions.forEach(function(action) {
+ if (!action.disabled(graph)) {
+ graph = action(graph, t);
+ }
+ });
+ return graph;
+ };
+ combinedAction.transitionable = true;
+ context.perform(combinedAction, operation.annotation());
+ window.setTimeout(function() {
+ context.validator().validate();
+ }, 300);
+ };
+ operation.available = function() {
+ return _actions.length && selectedIDs.length === _actions.length;
+ };
+ operation.disabled = function() {
+ if (!_actions.length)
+ return "";
+ var actionDisableds = _actions.map(function(action) {
+ return action.disabled(context.graph());
+ }).filter(Boolean);
+ if (actionDisableds.length === _actions.length) {
+ if (new Set(actionDisableds).size > 1) {
+ return "multiple_blockers";
+ }
+ 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 false;
+ function someMissing() {
+ if (context.inIntro())
+ return false;
+ var osm = context.connection();
+ if (osm) {
+ var missing = _coords.filter(function(loc) {
+ return !osm.isDataLoaded(loc);
+ });
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
+ }
+ }
+ return false;
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.circularize." + disable + "." + _amount) : _t.append("operations.circularize.description." + _amount);
+ };
+ operation.annotation = function() {
+ return _t("operations.circularize.annotation.feature", { n: _actions.length });
+ };
+ operation.id = "circularize";
+ operation.keys = [_t("operations.circularize.key")];
+ operation.title = _t.append("operations.circularize.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/ui/cmd.js
+ var uiCmd = function(code) {
+ var detected = utilDetect();
+ if (detected.os === "mac") {
+ return code;
+ }
+ if (detected.os === "win") {
+ if (code === "\u2318\u21E7Z")
+ return "Ctrl+Y";
+ }
+ var result = "", replacements = {
+ "\u2318": "Ctrl",
+ "\u21E7": "Shift",
+ "\u2325": "Alt",
+ "\u232B": "Backspace",
+ "\u2326": "Delete"
+ };
+ for (var i2 = 0; i2 < code.length; i2++) {
+ if (code[i2] in replacements) {
+ result += replacements[code[i2]] + (i2 < code.length - 1 ? "+" : "");
+ } else {
+ result += code[i2];
+ }
+ }
+ return result;
+ };
+ uiCmd.display = function(code) {
+ if (code.length !== 1)
+ return code;
+ var detected = utilDetect();
+ var mac = detected.os === "mac";
+ var replacements = {
+ "\u2318": mac ? "\u2318 " + _t("shortcuts.key.cmd") : _t("shortcuts.key.ctrl"),
+ "\u21E7": mac ? "\u21E7 " + _t("shortcuts.key.shift") : _t("shortcuts.key.shift"),
+ "\u2325": mac ? "\u2325 " + _t("shortcuts.key.option") : _t("shortcuts.key.alt"),
+ "\u2303": mac ? "\u2303 " + _t("shortcuts.key.ctrl") : _t("shortcuts.key.ctrl"),
+ "\u232B": mac ? "\u232B " + _t("shortcuts.key.delete") : _t("shortcuts.key.backspace"),
+ "\u2326": mac ? "\u2326 " + _t("shortcuts.key.del") : _t("shortcuts.key.del"),
+ "\u2196": mac ? "\u2196 " + _t("shortcuts.key.pgup") : _t("shortcuts.key.pgup"),
+ "\u2198": mac ? "\u2198 " + _t("shortcuts.key.pgdn") : _t("shortcuts.key.pgdn"),
+ "\u21DE": mac ? "\u21DE " + _t("shortcuts.key.home") : _t("shortcuts.key.home"),
+ "\u21DF": mac ? "\u21DF " + _t("shortcuts.key.end") : _t("shortcuts.key.end"),
+ "\u21B5": mac ? "\u23CE " + _t("shortcuts.key.return") : _t("shortcuts.key.enter"),
+ "\u238B": mac ? "\u238B " + _t("shortcuts.key.esc") : _t("shortcuts.key.esc"),
+ "\u2630": mac ? "\u2630 " + _t("shortcuts.key.menu") : _t("shortcuts.key.menu")
+ };
+ return replacements[code] || code;
+ };
+
+ // modules/operations/delete.js
+ 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(n2) {
+ return n2.loc;
+ });
+ var extent = utilTotalExtent(selectedIDs, context.graph());
+ var operation = function() {
+ var nextSelectedID;
+ var nextSelectedLoc;
+ if (selectedIDs.length === 1) {
+ var id2 = selectedIDs[0];
+ var entity = context.entity(id2);
+ var geometry = entity.geometry(context.graph());
+ var parents = context.graph().parentWays(entity);
+ var parent = parents[0];
+ if (geometry === "vertex") {
+ var nodes2 = parent.nodes;
+ var i2 = nodes2.indexOf(id2);
+ if (i2 === 0) {
+ i2++;
+ } else if (i2 === nodes2.length - 1) {
+ i2--;
+ } else {
+ var a = geoSphericalDistance(entity.loc, context.entity(nodes2[i2 - 1]).loc);
+ var b = geoSphericalDistance(entity.loc, context.entity(nodes2[i2 + 1]).loc);
+ i2 = a < b ? i2 - 1 : i2 + 1;
+ }
+ nextSelectedID = nodes2[i2];
+ nextSelectedLoc = context.entity(nextSelectedID).loc;
+ }
+ }
+ context.perform(action, operation.annotation());
+ context.validator().validate();
+ if (nextSelectedID && nextSelectedLoc) {
+ if (context.hasEntity(nextSelectedID)) {
+ context.enter(modeSelect(context, [nextSelectedID]).follow(true));
+ } else {
+ context.map().centerEase(nextSelectedLoc);
+ context.enter(modeBrowse(context));
+ }
+ } else {
+ context.enter(modeBrowse(context));
+ }
+ };
+ operation.available = function() {
+ return true;
+ };
+ operation.disabled = function() {
+ if (extent.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";
+ }
+ return false;
+ function someMissing() {
+ if (context.inIntro())
+ return false;
+ var osm = context.connection();
+ if (osm) {
+ var missing = coords.filter(function(loc) {
+ return !osm.isDataLoaded(loc);
+ });
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
+ }
+ }
+ return false;
+ }
+ function hasWikidataTag(id2) {
+ var entity = context.entity(id2);
+ return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
+ }
+ function incompleteRelation(id2) {
+ var entity = context.entity(id2);
+ return entity.type === "relation" && !entity.isComplete(context.graph());
+ }
+ function protectedMember(id2) {
+ var entity = context.entity(id2);
+ if (entity.type !== "way")
+ return false;
+ var parents = context.graph().parentRelations(entity);
+ for (var i2 = 0; i2 < parents.length; i2++) {
+ var parent = parents[i2];
+ var type2 = parent.tags.type;
+ var role = parent.memberById(id2).role || "outer";
+ if (type2 === "route" || type2 === "boundary" || type2 === "multipolygon" && role === "outer") {
+ return true;
+ }
+ }
+ return false;
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.delete." + disable + "." + multi) : _t.append("operations.delete.description." + multi);
+ };
+ operation.annotation = function() {
+ return selectedIDs.length === 1 ? _t("operations.delete.annotation." + context.graph().geometry(selectedIDs[0])) : _t("operations.delete.annotation.feature", { n: selectedIDs.length });
+ };
+ operation.id = "delete";
+ operation.keys = [uiCmd("\u2318\u232B"), uiCmd("\u2318\u2326"), uiCmd("\u2326")];
+ operation.title = _t.append("operations.delete.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/orthogonalize.js
+ function operationOrthogonalize(context, selectedIDs) {
+ var _extent;
+ var _type;
+ var _actions = selectedIDs.map(chooseAction).filter(Boolean);
+ var _amount = _actions.length === 1 ? "single" : "multiple";
+ var _coords = utilGetAllNodes(selectedIDs, context.graph()).map(function(n2) {
+ return n2.loc;
+ });
+ function chooseAction(entityID) {
+ var entity = context.entity(entityID);
+ var geometry = entity.geometry(context.graph());
+ if (!_extent) {
+ _extent = entity.extent(context.graph());
+ } else {
+ _extent = _extent.extend(entity.extent(context.graph()));
+ }
+ if (entity.type === "way" && new Set(entity.nodes).size > 2) {
+ if (_type && _type !== "feature")
+ return null;
+ _type = "feature";
+ return actionOrthogonalize(entityID, context.projection);
+ } else if (geometry === "vertex") {
+ if (_type && _type !== "corner")
+ return null;
+ _type = "corner";
+ var graph = context.graph();
+ var parents = graph.parentWays(entity);
+ if (parents.length === 1) {
+ var way = parents[0];
+ if (way.nodes.indexOf(entityID) !== -1) {
+ return actionOrthogonalize(way.id, context.projection, entityID);
+ }
+ }
+ }
+ return null;
+ }
+ var operation = function() {
+ if (!_actions.length)
+ return;
+ var combinedAction = function(graph, t) {
+ _actions.forEach(function(action) {
+ if (!action.disabled(graph)) {
+ graph = action(graph, t);
+ }
+ });
+ return graph;
+ };
+ combinedAction.transitionable = true;
+ context.perform(combinedAction, operation.annotation());
+ window.setTimeout(function() {
+ context.validator().validate();
+ }, 300);
+ };
+ operation.available = function() {
+ return _actions.length && selectedIDs.length === _actions.length;
+ };
+ operation.disabled = function() {
+ if (!_actions.length)
+ return "";
+ var actionDisableds = _actions.map(function(action) {
+ return action.disabled(context.graph());
+ }).filter(Boolean);
+ if (actionDisableds.length === _actions.length) {
+ if (new Set(actionDisableds).size > 1) {
+ return "multiple_blockers";
+ }
+ 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 (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;
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.orthogonalize." + disable + "." + _amount) : _t.append("operations.orthogonalize.description." + _type + "." + _amount);
+ };
+ operation.annotation = function() {
+ return _t("operations.orthogonalize.annotation." + _type, { n: _actions.length });
+ };
+ operation.id = "orthogonalize";
+ operation.keys = [_t("operations.orthogonalize.key")];
+ operation.title = _t.append("operations.orthogonalize.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/reflect.js
+ 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(n2) {
+ return n2.loc;
+ });
+ var extent = utilTotalExtent(selectedIDs, context.graph());
+ var operation = function() {
+ var action = actionReflect(selectedIDs, context.projection).useLongAxis(Boolean(axis === "long"));
+ context.perform(action, operation.annotation());
+ window.setTimeout(function() {
+ context.validator().validate();
+ }, 300);
+ };
+ operation.available = function() {
+ return nodes.length >= 3;
+ };
+ 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 someMissing() {
+ if (context.inIntro())
+ return false;
+ var osm = context.connection();
+ if (osm) {
+ var missing = coords.filter(function(loc) {
+ return !osm.isDataLoaded(loc);
+ });
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
+ }
+ }
+ return false;
+ }
+ function incompleteRelation(id2) {
+ var entity = context.entity(id2);
+ return entity.type === "relation" && !entity.isComplete(context.graph());
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.reflect." + disable + "." + multi) : _t.append("operations.reflect.description." + axis + "." + multi);
+ };
+ operation.annotation = function() {
+ return _t("operations.reflect.annotation." + axis + ".feature", { n: selectedIDs.length });
+ };
+ operation.id = "reflect-" + axis;
+ operation.keys = [_t("operations.reflect.key." + axis)];
+ operation.title = _t.append("operations.reflect.title." + axis);
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/move.js
+ function operationMove(context, selectedIDs) {
+ var multi = selectedIDs.length === 1 ? "single" : "multiple";
+ var nodes = utilGetAllNodes(selectedIDs, context.graph());
+ var coords = nodes.map(function(n2) {
+ return n2.loc;
+ });
+ var extent = utilTotalExtent(selectedIDs, context.graph());
+ var operation = function() {
+ context.enter(modeMove(context, selectedIDs));
+ };
+ operation.available = function() {
+ return selectedIDs.length > 0;
+ };
+ 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 someMissing() {
+ if (context.inIntro())
+ return false;
+ var osm = context.connection();
+ if (osm) {
+ var missing = coords.filter(function(loc) {
+ return !osm.isDataLoaded(loc);
+ });
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
+ }
+ }
+ return false;
+ }
+ function incompleteRelation(id2) {
+ var entity = context.entity(id2);
+ return entity.type === "relation" && !entity.isComplete(context.graph());
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.move." + disable + "." + multi) : _t.append("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 });
+ };
+ operation.id = "move";
+ operation.keys = [_t("operations.move.key")];
+ operation.title = _t.append("operations.move.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ operation.mouseOnly = true;
+ return operation;
+ }
+
+ // modules/modes/rotate.js
+ function modeRotate(context, entityIDs) {
+ var _tolerancePx = 4;
+ 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 });
+ var _prevGraph;
+ var _prevAngle;
+ var _prevTransform;
+ var _pivot;
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ function doRotate(d3_event) {
+ var fn;
+ if (context.graph() !== _prevGraph) {
+ fn = context.perform;
+ } else {
+ fn = context.replace;
+ }
+ var projection2 = context.projection;
+ var currTransform = projection2.transform();
+ 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(n2) {
+ return projection2(n2.loc);
+ });
+ _pivot = getPivot(points);
+ _prevAngle = void 0;
+ }
+ 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, projection2));
+ _prevTransform = currTransform;
+ _prevAngle = currAngle;
+ _prevGraph = context.graph();
+ }
+ function getPivot(points) {
+ var _pivot2;
+ if (points.length === 1) {
+ _pivot2 = points[0];
+ } else if (points.length === 2) {
+ _pivot2 = geoVecInterp(points[0], points[1], 0.5);
+ } else {
+ var polygonHull = hull_default(points);
+ if (polygonHull.length === 2) {
+ _pivot2 = geoVecInterp(points[0], points[1], 0.5);
+ } else {
+ _pivot2 = centroid_default2(hull_default(points));
+ }
+ }
+ return _pivot2;
+ }
+ function finish(d3_event) {
+ d3_event.stopPropagation();
+ context.replace(actionNoop(), annotation);
+ context.enter(modeSelect(context, entityIDs));
+ }
+ function cancel() {
+ if (_prevGraph)
+ context.pop();
+ context.enter(modeSelect(context, entityIDs));
+ }
+ function undone() {
+ context.enter(modeBrowse(context));
+ }
+ 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_default2(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("\u238B", cancel).on("\u21A9", finish);
+ select_default2(document).call(keybinding);
+ };
+ mode.exit = function() {
+ behaviors.forEach(context.uninstall);
+ context.surface().on(_pointerPrefix + "down.modeRotate", null);
+ select_default2(window).on(_pointerPrefix + "move.modeRotate", null, true).on(_pointerPrefix + "up.modeRotate", null, true);
+ context.history().on("undone.modeRotate", null);
+ select_default2(document).call(keybinding.unbind);
+ context.features().forceVisible([]);
+ };
+ mode.selectedIDs = function() {
+ if (!arguments.length)
+ return entityIDs;
+ return mode;
+ };
+ return mode;
+ }
+
+ // modules/operations/rotate.js
+ function operationRotate(context, selectedIDs) {
+ var multi = selectedIDs.length === 1 ? "single" : "multiple";
+ var nodes = utilGetAllNodes(selectedIDs, context.graph());
+ var coords = nodes.map(function(n2) {
+ return n2.loc;
+ });
+ var extent = utilTotalExtent(selectedIDs, context.graph());
+ var operation = function() {
+ context.enter(modeRotate(context, selectedIDs));
+ };
+ operation.available = function() {
+ return nodes.length >= 2;
+ };
+ 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 someMissing() {
+ if (context.inIntro())
+ return false;
+ var osm = context.connection();
+ if (osm) {
+ var missing = coords.filter(function(loc) {
+ return !osm.isDataLoaded(loc);
+ });
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
+ }
+ }
+ return false;
+ }
+ function incompleteRelation(id2) {
+ var entity = context.entity(id2);
+ return entity.type === "relation" && !entity.isComplete(context.graph());
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.rotate." + disable + "." + multi) : _t.append("operations.rotate.description." + multi);
+ };
+ 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.append("operations.rotate.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ operation.mouseOnly = true;
+ return operation;
+ }
+
+ // modules/modes/move.js
+ function modeMove(context, entityIDs, baseGraph) {
+ var _tolerancePx = 4;
+ 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 _prevGraph;
+ var _cache4;
+ var _origin;
+ var _nudgeInterval;
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ function doMove(nudge) {
+ nudge = nudge || [0, 0];
+ var fn;
+ if (_prevGraph !== context.graph()) {
+ _cache4 = {};
+ _origin = context.map().mouseCoordinates();
+ fn = context.perform;
+ } else {
+ fn = context.overwrite;
+ }
+ var currMouse = context.map().mouse();
+ var origMouse = context.projection(_origin);
+ var delta = geoVecSubtract(geoVecSubtract(currMouse, origMouse), nudge);
+ fn(actionMove(entityIDs, delta, context.projection, _cache4));
+ _prevGraph = context.graph();
+ }
+ function startNudge(nudge) {
+ if (_nudgeInterval)
+ window.clearInterval(_nudgeInterval);
+ _nudgeInterval = window.setInterval(function() {
+ context.map().pan(nudge);
+ doMove(nudge);
+ }, 50);
+ }
+ function stopNudge() {
+ if (_nudgeInterval) {
+ window.clearInterval(_nudgeInterval);
+ _nudgeInterval = null;
+ }
+ }
+ function move() {
+ doMove();
+ var nudge = geoViewportEdge(context.map().mouse(), context.map().dimensions());
+ if (nudge) {
+ startNudge(nudge);
+ } else {
+ stopNudge();
+ }
+ }
+ function finish(d3_event) {
+ d3_event.stopPropagation();
+ context.replace(actionNoop(), annotation);
+ context.enter(modeSelect(context, entityIDs));
+ stopNudge();
+ }
+ function cancel() {
+ if (baseGraph) {
+ while (context.graph() !== baseGraph)
+ context.pop();
+ context.enter(modeBrowse(context));
+ } else {
+ if (_prevGraph)
+ context.pop();
+ context.enter(modeSelect(context, entityIDs));
+ }
+ stopNudge();
+ }
+ function undone() {
+ context.enter(modeBrowse(context));
+ }
+ mode.enter = function() {
+ _origin = context.map().mouseCoordinates();
+ _prevGraph = null;
+ _cache4 = {};
+ context.features().forceVisible(entityIDs);
+ behaviors.forEach(context.install);
+ var downEvent;
+ context.surface().on(_pointerPrefix + "down.modeMove", function(d3_event) {
+ downEvent = d3_event;
+ });
+ select_default2(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("\u238B", cancel).on("\u21A9", finish);
+ select_default2(document).call(keybinding);
+ };
+ mode.exit = function() {
+ stopNudge();
+ behaviors.forEach(function(behavior) {
+ context.uninstall(behavior);
+ });
+ context.surface().on(_pointerPrefix + "down.modeMove", null);
+ select_default2(window).on(_pointerPrefix + "move.modeMove", null, true).on(_pointerPrefix + "up.modeMove", null, true);
+ context.history().on("undone.modeMove", null);
+ select_default2(document).call(keybinding.unbind);
+ context.features().forceVisible([]);
+ };
+ mode.selectedIDs = function() {
+ if (!arguments.length)
+ return entityIDs;
+ return mode;
+ };
+ return mode;
+ }
+
+ // modules/behavior/paste.js
+ function behaviorPaste(context) {
+ function doPaste(d3_event) {
+ if (!context.map().withinEditableZoom())
+ return;
+ d3_event.preventDefault();
+ var baseGraph = context.graph();
+ var mouse = context.map().mouse();
+ var projection2 = context.projection;
+ var viewport = geoExtent(projection2.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 = /* @__PURE__ */ new Set();
+ Object.values(copies).forEach(function(entity) {
+ originals.add(entity.id);
+ });
+ for (var id2 in copies) {
+ var oldEntity = oldGraph.entity(id2);
+ var newEntity = copies[id2];
+ extent._extend(oldEntity.extent(oldGraph));
+ var parents = context.graph().parentWays(newEntity);
+ var parentCopied = parents.some(function(parent) {
+ return originals.has(parent.id);
+ });
+ if (!parentCopied) {
+ newIDs.push(newEntity.id);
+ }
+ }
+ var copyPoint = context.copyLonLat() && projection2(context.copyLonLat()) || projection2(extent.center());
+ var delta = geoVecSubtract(mouse, copyPoint);
+ context.perform(actionMove(newIDs, delta, projection2));
+ context.enter(modeMove(context, newIDs, baseGraph));
+ }
+ function behavior() {
+ context.keybinding().on(uiCmd("\u2318V"), doPaste);
+ return behavior;
+ }
+ behavior.off = function() {
+ context.keybinding().off(uiCmd("\u2318V"));
+ };
+ return behavior;
+ }
+
+ // modules/behavior/drag.js
+ function behaviorDrag() {
+ var dispatch10 = dispatch_default("start", "move", "end");
+ var _tolerancePx = 1;
+ var _penTolerancePx = 4;
+ var _origin = null;
+ var _selector = "";
+ var _targetNode;
+ var _targetEntity;
+ var _surface;
+ var _pointerId;
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ var d3_event_userSelectProperty = utilPrefixCSSProperty("UserSelect");
+ var d3_event_userSelectSuppress = function() {
+ var selection2 = selection_default();
+ var select = selection2.style(d3_event_userSelectProperty);
+ selection2.style(d3_event_userSelectProperty, "none");
+ return function() {
+ selection2.style(d3_event_userSelectProperty, select);
+ };
+ };
+ function pointerdown(d3_event) {
+ if (_pointerId)
+ return;
+ _pointerId = d3_event.pointerId || "mouse";
+ _targetNode = this;
+ var pointerLocGetter = utilFastMouse(_surface || _targetNode.parentNode);
+ var offset;
+ var startOrigin = pointerLocGetter(d3_event);
+ var started = false;
+ var selectEnable = d3_event_userSelectSuppress();
+ select_default2(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];
+ }
+ d3_event.stopPropagation();
+ function pointermove(d3_event2) {
+ if (_pointerId !== (d3_event2.pointerId || "mouse"))
+ return;
+ var p = pointerLocGetter(d3_event2);
+ if (!started) {
+ var dist = geoVecLength(startOrigin, p);
+ var tolerance = d3_event2.pointerType === "pen" ? _penTolerancePx : _tolerancePx;
+ if (dist < tolerance)
+ return;
+ started = true;
+ dispatch10.call("start", this, d3_event2, _targetEntity);
+ } else {
+ startOrigin = p;
+ d3_event2.stopPropagation();
+ d3_event2.preventDefault();
+ var dx = p[0] - startOrigin[0];
+ var dy = p[1] - startOrigin[1];
+ dispatch10.call("move", this, d3_event2, _targetEntity, [p[0] + offset[0], p[1] + offset[1]], [dx, dy]);
+ }
+ }
+ function pointerup(d3_event2) {
+ if (_pointerId !== (d3_event2.pointerId || "mouse"))
+ return;
+ _pointerId = null;
+ if (started) {
+ dispatch10.call("end", this, d3_event2, _targetEntity);
+ d3_event2.preventDefault();
+ }
+ select_default2(window).on(_pointerPrefix + "move.drag", null).on(_pointerPrefix + "up.drag pointercancel.drag", null);
+ selectEnable();
+ }
+ }
+ function behavior(selection2) {
+ var matchesSelector = utilPrefixDOMProperty("matchesSelector");
+ var delegate = pointerdown;
+ if (_selector) {
+ delegate = function(d3_event) {
+ var root3 = this;
+ var target = d3_event.target;
+ for (; target && target !== root3; target = target.parentNode) {
+ var datum2 = target.__data__;
+ _targetEntity = datum2 instanceof osmNote ? datum2 : datum2 && datum2.properties && datum2.properties.entity;
+ if (_targetEntity && target[matchesSelector](_selector)) {
+ return pointerdown.call(target, d3_event);
+ }
+ }
+ };
+ }
+ selection2.on(_pointerPrefix + "down.drag" + _selector, delegate);
+ }
+ behavior.off = function(selection2) {
+ selection2.on(_pointerPrefix + "down.drag" + _selector, null);
+ };
+ behavior.selector = function(_) {
+ if (!arguments.length)
+ return _selector;
+ _selector = _;
+ return behavior;
+ };
+ behavior.origin = function(_) {
+ if (!arguments.length)
+ return _origin;
+ _origin = _;
+ return behavior;
+ };
+ behavior.cancel = function() {
+ select_default2(window).on(_pointerPrefix + "move.drag", null).on(_pointerPrefix + "up.drag pointercancel.drag", null);
+ return behavior;
+ };
+ behavior.targetNode = function(_) {
+ if (!arguments.length)
+ return _targetNode;
+ _targetNode = _;
+ return behavior;
+ };
+ behavior.targetEntity = function(_) {
+ if (!arguments.length)
+ return _targetEntity;
+ _targetEntity = _;
+ return behavior;
+ };
+ behavior.surface = function(_) {
+ if (!arguments.length)
+ return _surface;
+ _surface = _;
+ return behavior;
+ };
+ return utilRebind(behavior, dispatch10, "on");
+ }
+
+ // modules/modes/drag_node.js
+ function modeDragNode(context) {
+ var mode = {
+ id: "drag-node",
+ button: "browse"
+ };
+ var hover = behaviorHover(context).altDisables(true).on("hover", context.ui().sidebar.hover);
+ var edit2 = behaviorEdit(context);
+ var _nudgeInterval;
+ var _restoreSelectedIDs = [];
+ var _wasMidpoint = false;
+ var _isCancelled = false;
+ var _activeEntity;
+ var _startLoc;
+ var _lastLoc;
+ 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);
+ }
+ function stopNudge() {
+ if (_nudgeInterval) {
+ window.clearInterval(_nudgeInterval);
+ _nudgeInterval = null;
+ }
+ }
+ 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());
+ if (nodeGeometry === "vertex" && targetGeometry === "vertex") {
+ var nodeParentWayIDs = context.graph().parentWays(nodeEntity);
+ var targetParentWayIDs = context.graph().parentWays(targetEntity);
+ var sharedParentWays = utilArrayIntersection(nodeParentWayIDs, targetParentWayIDs);
+ if (sharedParentWays.length !== 0) {
+ if (sharedParentWays[0].areAdjacent(nodeEntity.id, targetEntity.id)) {
+ return _t("operations.connect.annotation.from_vertex.to_adjacent_vertex");
+ }
+ return _t("operations.connect.annotation.from_vertex.to_sibling_vertex");
+ }
+ }
+ return _t("operations.connect.annotation.from_" + nodeGeometry + ".to_" + targetGeometry);
+ }
+ function shouldSnapToNode(target) {
+ if (!_activeEntity)
+ return false;
+ return _activeEntity.geometry(context.graph()) !== "vertex" || (target.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(target, context.graph()));
+ }
+ function origin(entity) {
+ return context.projection(entity.loc);
+ }
+ function keydown(d3_event) {
+ if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+ if (context.surface().classed("nope")) {
+ context.surface().classed("nope-suppressed", true);
+ }
+ 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);
+ }
+ context.surface().classed("nope-suppressed", false).classed("nope-disabled", false);
+ }
+ }
+ function start2(d3_event, entity) {
+ _wasMidpoint = entity.type === "midpoint";
+ var hasHidden = context.features().hasHiddenConnections(entity, context.graph());
+ _isCancelled = !context.editable() || d3_event.shiftKey || hasHidden;
+ if (_isCancelled) {
+ if (hasHidden) {
+ context.ui().flash.duration(4e3).iconName("#iD-icon-no").label(_t.append("modes.drag_node.connected_to_hidden"))();
+ }
+ return drag.cancel();
+ }
+ if (_wasMidpoint) {
+ var midpoint = entity;
+ entity = osmNode();
+ context.perform(actionAddMidpoint(midpoint, entity));
+ entity = context.entity(entity.id);
+ var vertex = context.surface().selectAll("." + entity.id);
+ drag.targetNode(vertex.node()).targetEntity(entity);
+ } else {
+ context.perform(actionNoop());
+ }
+ _activeEntity = entity;
+ _startLoc = entity.loc;
+ hover.ignoreVertex(entity.geometry(context.graph()) === "vertex");
+ context.surface().selectAll("." + _activeEntity.id).classed("active", true);
+ context.enter(mode);
+ }
+ function datum2(d3_event) {
+ if (!d3_event || d3_event.altKey) {
+ return {};
+ } else {
+ var d = d3_event.target.__data__;
+ return d && d.properties && d.properties.target ? d : {};
+ }
+ }
+ 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 (!_nudgeInterval) {
+ var d = datum2(d3_event);
+ target = d && d.properties && d.properties.entity;
+ var targetLoc = target && target.loc;
+ var targetNodes = d && d.properties && d.properties.nodes;
+ if (targetLoc) {
+ if (shouldSnapToNode(target)) {
+ loc = targetLoc;
+ }
+ } else if (targetNodes) {
+ edge = geoChooseEdge(targetNodes, context.map().mouse(), context.projection, end.id);
+ if (edge) {
+ loc = edge.loc;
+ }
+ }
+ }
+ context.replace(
+ actionMoveNode(entity.id, loc)
+ );
+ var isInvalid = false;
+ if (target) {
+ isInvalid = hasRelationConflict(entity, target, edge, context.graph());
+ }
+ if (!isInvalid) {
+ isInvalid = hasInvalidGeometry(entity, context.graph());
+ }
+ var nope = context.surface().classed("nope");
+ if (isInvalid === "relation" || isInvalid === "restriction") {
+ if (!nope) {
+ context.ui().flash.duration(4e3).iconName("#iD-icon-no").label(_t.append(
+ "operations.connect." + isInvalid,
+ { relation: _mainPresetIndex.item("type/restriction").name() }
+ ))();
+ }
+ } else if (isInvalid) {
+ var errorID = isInvalid === "line" ? "lines" : "areas";
+ context.ui().flash.duration(3e3).iconName("#iD-icon-no").label(_t.append("self_intersection.error." + errorID))();
+ } else {
+ if (nope) {
+ context.ui().flash.duration(1).label("")();
+ }
+ }
+ 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);
+ }
+ _lastLoc = loc;
+ }
+ function hasRelationConflict(entity, target, edge, graph) {
+ var testGraph = graph.update();
+ 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;
+ }
+ var ids = [entity.id, target.id];
+ return actionConnect(ids).disabled(testGraph);
+ }
+ function hasInvalidGeometry(entity, graph) {
+ var parents = graph.parentWays(entity);
+ var i2, j2, k;
+ for (i2 = 0; i2 < parents.length; i2++) {
+ var parent = parents[i2];
+ var nodes = [];
+ var activeIndex = null;
+ var relations = graph.parentRelations(parent);
+ for (j2 = 0; j2 < relations.length; j2++) {
+ if (!relations[j2].isMultipolygon())
+ continue;
+ var rings = osmJoinWays(relations[j2].members, graph);
+ for (k = 0; k < rings.length; k++) {
+ nodes = rings[k].nodes;
+ if (nodes.find(function(n2) {
+ return n2.id === entity.id;
+ })) {
+ activeIndex = k;
+ if (geoHasSelfIntersections(nodes, entity.id)) {
+ return "multipolygonMember";
+ }
+ }
+ rings[k].coords = nodes.map(function(n2) {
+ return n2.loc;
+ });
+ }
+ for (k = 0; k < rings.length; k++) {
+ if (k === activeIndex)
+ continue;
+ if (geoHasLineIntersections(rings[activeIndex].nodes, rings[k].nodes, entity.id)) {
+ return "multipolygonRing";
+ }
+ }
+ }
+ if (activeIndex === null) {
+ nodes = parent.nodes.map(function(nodeID) {
+ return graph.entity(nodeID);
+ });
+ if (nodes.length && geoHasSelfIntersections(nodes, entity.id)) {
+ return parent.geometry(graph);
+ }
+ }
+ }
+ return false;
+ }
+ function move(d3_event, entity, point2) {
+ if (_isCancelled)
+ return;
+ d3_event.stopPropagation();
+ context.surface().classed("nope-disabled", d3_event.altKey);
+ _lastLoc = context.projection.invert(point2);
+ doMove(d3_event, entity);
+ var nudge = geoViewportEdge(point2, context.map().dimensions());
+ if (nudge) {
+ startNudge(d3_event, entity, nudge);
+ } else {
+ stopNudge();
+ }
+ }
+ function end(d3_event, entity) {
+ if (_isCancelled)
+ return;
+ var wasPoint = entity.geometry(context.graph()) === "point";
+ var d = datum2(d3_event);
+ var nope = d && d.properties && d.properties.nope || context.surface().classed("nope");
+ var target = d && d.properties && d.properties.entity;
+ if (nope) {
+ 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)
+ );
+ }
+ if (wasPoint) {
+ context.enter(modeSelect(context, [entity.id]));
+ } else {
+ var reselection = _restoreSelectedIDs.filter(function(id2) {
+ return context.graph().hasEntity(id2);
+ });
+ if (reselection.length) {
+ context.enter(modeSelect(context, reselection));
+ } else {
+ context.enter(modeBrowse(context));
+ }
+ }
+ }
+ function _actionBounceBack(nodeID, toLoc) {
+ var moveNode = actionMoveNode(nodeID, toLoc);
+ var action = function(graph, t) {
+ if (t === 1)
+ context.pop();
+ return moveNode(graph, t);
+ };
+ action.transitionable = true;
+ return action;
+ }
+ function cancel() {
+ drag.cancel();
+ context.enter(modeBrowse(context));
+ }
+ var drag = behaviorDrag().selector(".layer-touch.points .target").surface(context.container().select(".main-map").node()).origin(origin).on("start", start2).on("move", move).on("end", end);
+ mode.enter = function() {
+ context.install(hover);
+ context.install(edit2);
+ select_default2(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(edit2);
+ select_default2(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();
+ };
+ mode.selectedIDs = function() {
+ if (!arguments.length)
+ return _activeEntity ? [_activeEntity.id] : [];
+ return mode;
+ };
+ mode.activeID = function() {
+ if (!arguments.length)
+ return _activeEntity && _activeEntity.id;
+ return mode;
+ };
+ mode.restoreSelectedIDs = function(_) {
+ if (!arguments.length)
+ return _restoreSelectedIDs;
+ _restoreSelectedIDs = _;
+ return mode;
+ };
+ mode.behavior = drag;
+ return mode;
+ }
+
+ // modules/services/keepRight.js
+ var import_rbush = __toESM(require_rbush_min());
+
+ // node_modules/d3-fetch/src/text.js
+ function responseText(response) {
+ if (!response.ok)
+ throw new Error(response.status + " " + response.statusText);
+ return response.text();
+ }
+ function text_default3(input, init2) {
+ return fetch(input, init2).then(responseText);
+ }
+
+ // node_modules/d3-fetch/src/json.js
+ 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 json_default(input, init2) {
+ return fetch(input, init2).then(responseJson);
+ }
+
+ // node_modules/d3-fetch/src/xml.js
+ function parser(type2) {
+ return (input, init2) => text_default3(input, init2).then((text2) => new DOMParser().parseFromString(text2, type2));
+ }
+ var xml_default = parser("application/xml");
+ var html = parser("text/html");
+ var svg = parser("image/svg+xml");
+
+ // modules/services/keepRight.js
+ var tiler = utilTiler();
+ var dispatch2 = dispatch_default("loaded");
+ var _tileZoom = 14;
+ var _krUrlRoot = "https://www.keepright.at";
+ var _krData = { errorTypes: {}, localizeStrings: {} };
+ var _cache;
+ 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
+ ];
+ function abortRequest(controller) {
+ if (controller) {
+ controller.abort();
+ }
+ }
+ function abortUnwantedRequests(cache, tiles) {
+ Object.keys(cache.inflightTile).forEach((k) => {
+ const wanted = tiles.find((tile) => k === tile.id);
+ if (!wanted) {
+ abortRequest(cache.inflightTile[k]);
+ delete cache.inflightTile[k];
+ }
+ });
+ }
+ function encodeIssueRtree(d) {
+ return { minX: d.loc[0], minY: d.loc[1], maxX: d.loc[0], maxY: d.loc[1], data: d };
+ }
+ function updateRtree(item, replace) {
+ _cache.rtree.remove(item, (a, b) => a.data.id === b.data.id);
+ if (replace) {
+ _cache.rtree.insert(item);
+ }
+ }
+ function tokenReplacements(d) {
+ if (!(d instanceof QAItem))
+ return;
+ const replacements = {};
+ const issueTemplate = _krData.errorTypes[d.whichType];
+ if (!issueTemplate) {
+ console.log("No Template: ", d.whichType);
+ console.log(" ", d.description);
+ return;
+ }
+ if (!issueTemplate.regex)
+ return;
+ const errorRegex = new RegExp(issueTemplate.regex, "i");
+ const errorMatch = errorRegex.exec(d.description);
+ if (!errorMatch) {
+ console.log("Unmatched: ", d.whichType);
+ console.log(" ", d.description);
+ console.log(" ", errorRegex);
+ return;
+ }
+ for (let i2 = 1; i2 < errorMatch.length; i2++) {
+ let capture = errorMatch[i2];
+ let idType;
+ idType = "IDs" in issueTemplate ? issueTemplate.IDs[i2 - 1] : "";
+ if (idType && capture) {
+ capture = parseError(capture, idType);
+ } else {
+ const compare = capture.toLowerCase();
+ if (_krData.localizeStrings[compare]) {
+ capture = _t("QA.keepRight.error_parts." + _krData.localizeStrings[compare]);
+ } else {
+ capture = unescape_default(capture);
+ }
+ }
+ replacements["var" + i2] = capture;
+ }
+ return replacements;
+ }
+ function parseError(capture, idType) {
+ const compare = capture.toLowerCase();
+ if (_krData.localizeStrings[compare]) {
+ capture = _t("QA.keepRight.error_parts." + _krData.localizeStrings[compare]);
+ }
+ switch (idType) {
+ case "this":
+ capture = linkErrorObject2(capture);
+ break;
+ case "url":
+ capture = linkURL(capture);
+ break;
+ case "n":
+ case "w":
+ case "r":
+ capture = linkEntity2(idType + capture);
+ break;
+ case "20":
+ capture = parse20(capture);
+ break;
+ case "211":
+ capture = parse211(capture);
+ break;
+ case "231":
+ capture = parse231(capture);
+ break;
+ case "294":
+ capture = parse294(capture);
+ break;
+ case "370":
+ capture = parse370(capture);
+ break;
+ }
+ return capture;
+ function linkErrorObject2(d) {
+ return { html: `<a class="error_object_link">${d}</a>` };
+ }
+ function linkEntity2(d) {
+ return { html: `<a class="error_entity_link">${d}</a>` };
+ }
+ function linkURL(d) {
+ return { html: `<a class="kr_external_link" target="_blank" href="${d}">${d}</a>` };
+ }
+ function parse211(capture2) {
+ let newList = [];
+ const items = capture2.split(", ");
+ items.forEach((item) => {
+ let id2 = linkEntity2("n" + item.slice(1));
+ newList.push(id2);
+ });
+ return newList.join(", ");
+ }
+ function parse231(capture2) {
+ let newList = [];
+ const items = capture2.split("),");
+ items.forEach((item) => {
+ const match = item.match(/\#(\d+)\((.+)\)?/);
+ if (match !== null && match.length > 2) {
+ newList.push(
+ linkEntity2("w" + match[1]) + " " + _t("QA.keepRight.errorTypes.231.layer", { layer: match[2] })
+ );
+ }
+ });
+ return newList.join(", ");
+ }
+ function parse294(capture2) {
+ let newList = [];
+ const items = capture2.split(",");
+ items.forEach((item) => {
+ item = item.split(" ");
+ const role = `"${item[0]}"`;
+ const idType2 = item[1].slice(0, 1);
+ let id2 = item[2].slice(1);
+ id2 = linkEntity2(idType2 + id2);
+ newList.push(`${role} ${item[1]} ${id2}`);
+ });
+ return newList.join(", ");
+ }
+ function parse370(capture2) {
+ if (!capture2)
+ return "";
+ const match = capture2.match(/\(including the name (\'.+\')\)/);
+ if (match && match.length) {
+ return _t("QA.keepRight.errorTypes.370.including_the_name", { name: match[1] });
+ }
+ return "";
+ }
+ function parse20(capture2) {
+ let newList = [];
+ const items = capture2.split(",");
+ items.forEach((item) => {
+ const id2 = linkEntity2("n" + item.slice(1));
+ newList.push(id2);
+ });
+ return newList.join(", ");
+ }
+ }
+ var keepRight_default = {
+ title: "keepRight",
+ init() {
+ _mainFileFetcher.get("keepRight").then((d) => _krData = d);
+ if (!_cache) {
+ this.reset();
+ }
+ this.event = utilRebind(this, dispatch2, "on");
+ },
+ reset() {
+ if (_cache) {
+ Object.values(_cache.inflightTile).forEach(abortRequest);
+ }
+ _cache = {
+ data: {},
+ loadedTile: {},
+ inflightTile: {},
+ inflightPost: {},
+ closed: {},
+ rtree: new import_rbush.default()
+ };
+ },
+ // KeepRight API: http://osm.mueschelsoft.de/keepright/interfacing.php
+ loadIssues(projection2) {
+ const options2 = {
+ format: "geojson",
+ ch: _krRuleset
+ };
+ const tiles = tiler.zoomExtent([_tileZoom, _tileZoom]).getTiles(projection2);
+ abortUnwantedRequests(_cache, tiles);
+ tiles.forEach((tile) => {
+ if (_cache.loadedTile[tile.id] || _cache.inflightTile[tile.id])
+ return;
+ const [left, top, right, bottom] = tile.extent.rectangle();
+ const params = Object.assign({}, options2, { left, bottom, right, top });
+ const url = `${_krUrlRoot}/export.php?` + utilQsString(params);
+ const controller = new AbortController();
+ _cache.inflightTile[tile.id] = controller;
+ json_default(url, { signal: controller.signal }).then((data) => {
+ delete _cache.inflightTile[tile.id];
+ _cache.loadedTile[tile.id] = true;
+ if (!data || !data.features || !data.features.length) {
+ throw new Error("No Data");
+ }
+ data.features.forEach((feature3) => {
+ const {
+ properties: {
+ error_type: itemType,
+ error_id: id2,
+ comment = null,
+ object_id: objectId,
+ object_type: objectType,
+ schema,
+ title
+ }
+ } = feature3;
+ let {
+ geometry: { coordinates: loc },
+ properties: { description = "" }
+ } = feature3;
+ const issueTemplate = _krData.errorTypes[itemType];
+ const parentIssueType = (Math.floor(itemType / 10) * 10).toString();
+ const whichType = issueTemplate ? itemType : parentIssueType;
+ const whichTemplate = _krData.errorTypes[whichType];
+ switch (whichType) {
+ case "170":
+ description = `This feature has a FIXME tag: ${description}`;
+ break;
+ case "292":
+ case "293":
+ description = description.replace("A turn-", "This turn-");
+ break;
+ case "294":
+ case "295":
+ case "296":
+ case "297":
+ case "298":
+ description = `This turn-restriction~${description}`;
+ break;
+ case "300":
+ description = "This highway is missing a maxspeed tag";
+ break;
+ case "411":
+ case "412":
+ case "413":
+ description = `This feature~${description}`;
+ break;
+ }
+ let coincident = false;
+ do {
+ let delta = coincident ? [1e-5, 0] : [0, 1e-5];
+ loc = geoVecAdd(loc, delta);
+ let bbox2 = geoExtent(loc).bbox();
+ coincident = _cache.rtree.search(bbox2).length;
+ } while (coincident);
+ let d = new QAItem(loc, this, itemType, id2, {
+ comment,
+ description,
+ whichType,
+ parentIssueType,
+ severity: whichTemplate.severity || "error",
+ objectId,
+ objectType,
+ schema,
+ title
+ });
+ d.replacements = tokenReplacements(d);
+ _cache.data[id2] = d;
+ _cache.rtree.insert(encodeIssueRtree(d));
+ });
+ dispatch2.call("loaded");
+ }).catch(() => {
+ delete _cache.inflightTile[tile.id];
+ _cache.loadedTile[tile.id] = true;
+ });
+ });
+ },
+ postUpdate(d, callback) {
+ if (_cache.inflightPost[d.id]) {
+ return callback({ message: "Error update already inflight", status: -2 }, d);
+ }
+ const params = { schema: d.schema, id: d.id };
+ if (d.newStatus) {
+ params.st = d.newStatus;
+ }
+ if (d.newComment !== void 0) {
+ params.co = d.newComment;
+ }
+ const url = `${_krUrlRoot}/comment.php?` + utilQsString(params);
+ const controller = new AbortController();
+ _cache.inflightPost[d.id] = controller;
+ json_default(url, { signal: controller.signal }).finally(() => {
+ delete _cache.inflightPost[d.id];
+ if (d.newStatus === "ignore") {
+ this.removeItem(d);
+ } else if (d.newStatus === "ignore_t") {
+ this.removeItem(d);
+ _cache.closed[`${d.schema}:${d.id}`] = true;
+ } else {
+ d = this.replaceItem(d.update({
+ comment: d.newComment,
+ newComment: void 0,
+ newState: void 0
+ }));
+ }
+ if (callback)
+ callback(null, d);
+ });
+ },
+ // Get all cached QAItems covering the viewport
+ getItems(projection2) {
+ const viewport = projection2.clipExtent();
+ const min3 = [viewport[0][0], viewport[1][1]];
+ const max3 = [viewport[1][0], viewport[0][1]];
+ const bbox2 = geoExtent(projection2.invert(min3), projection2.invert(max3)).bbox();
+ return _cache.rtree.search(bbox2).map((d) => d.data);
+ },
+ // Get a QAItem from cache
+ // NOTE: Don't change method name until UI v3 is merged
+ getError(id2) {
+ return _cache.data[id2];
+ },
+ // Replace a single QAItem in the cache
+ replaceItem(item) {
+ if (!(item instanceof QAItem) || !item.id)
+ return;
+ _cache.data[item.id] = item;
+ updateRtree(encodeIssueRtree(item), true);
+ return item;
+ },
+ // Remove a single QAItem from the cache
+ removeItem(item) {
+ if (!(item instanceof QAItem) || !item.id)
+ return;
+ delete _cache.data[item.id];
+ updateRtree(encodeIssueRtree(item), false);
+ },
+ issueURL(item) {
+ return `${_krUrlRoot}/report_map.php?schema=${item.schema}&error=${item.id}`;
+ },
+ // Get an array of issues closed during this session.
+ // Used to populate `closed:keepright` changeset tag
+ getClosedIDs() {
+ return Object.keys(_cache.closed).sort();
+ }
+ };
+
+ // modules/services/improveOSM.js
+ var import_rbush2 = __toESM(require_rbush_min());
+ var tiler2 = utilTiler();
+ var dispatch3 = dispatch_default("loaded");
+ var _tileZoom2 = 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: {} };
+ var _cache2;
+ function abortRequest2(i2) {
+ Object.values(i2).forEach((controller) => {
+ if (controller) {
+ controller.abort();
+ }
+ });
+ }
+ function abortUnwantedRequests2(cache, tiles) {
+ Object.keys(cache.inflightTile).forEach((k) => {
+ const wanted = tiles.find((tile) => k === tile.id);
+ if (!wanted) {
+ abortRequest2(cache.inflightTile[k]);
+ delete cache.inflightTile[k];
+ }
+ });
+ }
+ function encodeIssueRtree2(d) {
+ return { minX: d.loc[0], minY: d.loc[1], maxX: d.loc[0], maxY: d.loc[1], data: d };
+ }
+ function updateRtree2(item, replace) {
+ _cache2.rtree.remove(item, (a, b) => a.data.id === b.data.id);
+ if (replace) {
+ _cache2.rtree.insert(item);
+ }
+ }
+ function linkErrorObject(d) {
+ return { html: `<a class="error_object_link">${d}</a>` };
+ }
+ function linkEntity(d) {
+ return { html: `<a class="error_entity_link">${d}</a>` };
+ }
+ function pointAverage(points) {
+ if (points.length) {
+ const sum = points.reduce(
+ (acc, point2) => geoVecAdd(acc, [point2.lon, point2.lat]),
+ [0, 0]
+ );
+ return geoVecScale(sum, 1 / points.length);
+ } else {
+ return [0, 0];
+ }
+ }
+ function relativeBearing(p1, p2) {
+ let angle2 = Math.atan2(p2.lon - p1.lon, p2.lat - p1.lat);
+ if (angle2 < 0) {
+ angle2 += 2 * Math.PI;
+ }
+ return angle2 * 180 / Math.PI;
+ }
+ function cardinalDirection(bearing) {
+ const dir = 45 * Math.round(bearing / 45);
+ const compass = {
+ 0: "north",
+ 45: "northeast",
+ 90: "east",
+ 135: "southeast",
+ 180: "south",
+ 225: "southwest",
+ 270: "west",
+ 315: "northwest",
+ 360: "north"
+ };
+ return _t(`QA.improveOSM.directions.${compass[dir]}`);
+ }
+ function preventCoincident(loc, bumpUp) {
+ let coincident = false;
+ do {
+ let delta = coincident ? [1e-5, 0] : bumpUp ? [0, 1e-5] : [0, 0];
+ loc = geoVecAdd(loc, delta);
+ let bbox2 = geoExtent(loc).bbox();
+ coincident = _cache2.rtree.search(bbox2).length;
+ } while (coincident);
+ return loc;
+ }
+ var improveOSM_default = {
+ title: "improveOSM",
+ init() {
+ _mainFileFetcher.get("qa_data").then((d) => _impOsmData = d.improveOSM);
+ if (!_cache2) {
+ this.reset();
+ }
+ this.event = utilRebind(this, dispatch3, "on");
+ },
+ reset() {
+ if (_cache2) {
+ Object.values(_cache2.inflightTile).forEach(abortRequest2);
+ }
+ _cache2 = {
+ data: {},
+ loadedTile: {},
+ inflightTile: {},
+ inflightPost: {},
+ closed: {},
+ rtree: new import_rbush2.default()
+ };
+ },
+ loadIssues(projection2) {
+ const options2 = {
+ client: "iD",
+ status: "OPEN",
+ zoom: "19"
+ // Use a high zoom so that clusters aren't returned
+ };
+ const tiles = tiler2.zoomExtent([_tileZoom2, _tileZoom2]).getTiles(projection2);
+ abortUnwantedRequests2(_cache2, tiles);
+ tiles.forEach((tile) => {
+ if (_cache2.loadedTile[tile.id] || _cache2.inflightTile[tile.id])
+ return;
+ const [east, north, west, south] = tile.extent.rectangle();
+ const params = Object.assign({}, options2, { east, south, west, north });
+ const requests = {};
+ Object.keys(_impOsmUrls).forEach((k) => {
+ const kParams = Object.assign(
+ {},
+ params,
+ k === "mr" ? { type: "PARKING,ROAD,BOTH,PATH" } : { confidenceLevel: "C1" }
+ );
+ const url = `${_impOsmUrls[k]}/search?` + utilQsString(kParams);
+ const controller = new AbortController();
+ requests[k] = controller;
+ json_default(url, { signal: controller.signal }).then((data) => {
+ delete _cache2.inflightTile[tile.id][k];
+ if (!Object.keys(_cache2.inflightTile[tile.id]).length) {
+ delete _cache2.inflightTile[tile.id];
+ _cache2.loadedTile[tile.id] = true;
+ }
+ if (data.roadSegments) {
+ data.roadSegments.forEach((feature3) => {
+ const { points, wayId, fromNodeId, toNodeId } = feature3;
+ const itemId = `${wayId}${fromNodeId}${toNodeId}`;
+ let mid = points.length / 2;
+ let loc;
+ if (mid % 1 === 0) {
+ loc = pointAverage([points[mid - 1], points[mid]]);
+ } else {
+ mid = points[Math.floor(mid)];
+ loc = [mid.lon, mid.lat];
+ }
+ loc = preventCoincident(loc, false);
+ let d = new QAItem(loc, this, k, itemId, {
+ issueKey: k,
+ // used as a category
+ identifier: {
+ // used to post changes
+ wayId,
+ fromNodeId,
+ toNodeId
+ },
+ objectId: wayId,
+ objectType: "way"
+ });
+ d.replacements = {
+ percentage: feature3.percentOfTrips,
+ num_trips: feature3.numberOfTrips,
+ highway: linkErrorObject(_t("QA.keepRight.error_parts.highway")),
+ from_node: linkEntity("n" + feature3.fromNodeId),
+ to_node: linkEntity("n" + feature3.toNodeId)
+ };
+ _cache2.data[d.id] = d;
+ _cache2.rtree.insert(encodeIssueRtree2(d));
+ });
+ }
+ if (data.tiles) {
+ data.tiles.forEach((feature3) => {
+ const { type: type2, x, y, numberOfTrips } = feature3;
+ const geoType = type2.toLowerCase();
+ const itemId = `${geoType}${x}${y}${numberOfTrips}`;
+ let loc = pointAverage(feature3.points);
+ loc = preventCoincident(loc, false);
+ let d = new QAItem(loc, this, `${k}-${geoType}`, itemId, {
+ issueKey: k,
+ identifier: { x, y }
+ });
+ d.replacements = {
+ num_trips: numberOfTrips,
+ geometry_type: _t(`QA.improveOSM.geometry_types.${geoType}`)
+ };
+ if (numberOfTrips === -1) {
+ d.desc = _t("QA.improveOSM.error_types.mr.description_alt", d.replacements);
+ }
+ _cache2.data[d.id] = d;
+ _cache2.rtree.insert(encodeIssueRtree2(d));
+ });
+ }
+ if (data.entities) {
+ data.entities.forEach((feature3) => {
+ const { point: point2, id: id2, segments, numberOfPasses, turnType } = feature3;
+ const itemId = `${id2.replace(/[,:+#]/g, "_")}`;
+ const loc = preventCoincident([point2.lon, point2.lat], true);
+ const ids = id2.split(",");
+ const from_way = ids[0];
+ const via_node = ids[3];
+ const to_way = ids[2].split(":")[1];
+ let d = new QAItem(loc, this, k, itemId, {
+ issueKey: k,
+ identifier: id2,
+ objectId: via_node,
+ objectType: "node"
+ });
+ const [p1, p2] = segments[0].points;
+ const dir_of_travel = cardinalDirection(relativeBearing(p1, p2));
+ 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"))
+ };
+ _cache2.data[d.id] = d;
+ _cache2.rtree.insert(encodeIssueRtree2(d));
+ dispatch3.call("loaded");
+ });
+ }
+ }).catch(() => {
+ delete _cache2.inflightTile[tile.id][k];
+ if (!Object.keys(_cache2.inflightTile[tile.id]).length) {
+ delete _cache2.inflightTile[tile.id];
+ _cache2.loadedTile[tile.id] = true;
+ }
+ });
+ });
+ _cache2.inflightTile[tile.id] = requests;
+ });
+ },
+ getComments(item) {
+ if (item.comments) {
+ return Promise.resolve(item);
+ }
+ const key = item.issueKey;
+ let qParams = {};
+ 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;
+ }
+ const url = `${_impOsmUrls[key]}/retrieveComments?` + utilQsString(qParams);
+ const cacheComments = (data) => {
+ item.comments = data.comments ? data.comments.reverse() : [];
+ this.replaceItem(item);
+ };
+ return json_default(url).then(cacheComments).then(() => item);
+ },
+ postUpdate(d, callback) {
+ if (!osm_default.authenticated()) {
+ return callback({ message: "Not Authenticated", status: -3 }, d);
+ }
+ if (_cache2.inflightPost[d.id]) {
+ return callback({ message: "Error update already inflight", status: -2 }, d);
+ }
+ osm_default.userDetails(sendPayload.bind(this));
+ function sendPayload(err, user) {
+ if (err) {
+ return callback(err, d);
+ }
+ const key = d.issueKey;
+ const url = `${_impOsmUrls[key]}/comment`;
+ const payload = {
+ username: user.display_name,
+ targetIds: [d.identifier]
+ };
+ if (d.newStatus) {
+ payload.status = d.newStatus;
+ payload.text = "status changed";
+ }
+ if (d.newComment) {
+ payload.text = d.newComment;
+ }
+ const controller = new AbortController();
+ _cache2.inflightPost[d.id] = controller;
+ const options2 = {
+ method: "POST",
+ signal: controller.signal,
+ body: JSON.stringify(payload)
+ };
+ json_default(url, options2).then(() => {
+ delete _cache2.inflightPost[d.id];
+ if (!d.newStatus) {
+ const now3 = new Date();
+ let comments = d.comments ? d.comments : [];
+ comments.push({
+ username: payload.username,
+ text: payload.text,
+ timestamp: now3.getTime() / 1e3
+ });
+ this.replaceItem(d.update({
+ comments,
+ newComment: void 0
+ }));
+ } else {
+ this.removeItem(d);
+ if (d.newStatus === "SOLVED") {
+ if (!(d.issueKey in _cache2.closed)) {
+ _cache2.closed[d.issueKey] = 0;
+ }
+ _cache2.closed[d.issueKey] += 1;
+ }
+ }
+ if (callback)
+ callback(null, d);
+ }).catch((err2) => {
+ delete _cache2.inflightPost[d.id];
+ if (callback)
+ callback(err2.message);
+ });
+ }
+ },
+ // Get all cached QAItems covering the viewport
+ getItems(projection2) {
+ const viewport = projection2.clipExtent();
+ const min3 = [viewport[0][0], viewport[1][1]];
+ const max3 = [viewport[1][0], viewport[0][1]];
+ const bbox2 = geoExtent(projection2.invert(min3), projection2.invert(max3)).bbox();
+ return _cache2.rtree.search(bbox2).map((d) => d.data);
+ },
+ // Get a QAItem from cache
+ // NOTE: Don't change method name until UI v3 is merged
+ getError(id2) {
+ return _cache2.data[id2];
+ },
+ // get the name of the icon to display for this item
+ getIcon(itemType) {
+ return _impOsmData.icons[itemType];
+ },
+ // Replace a single QAItem in the cache
+ replaceItem(issue) {
+ if (!(issue instanceof QAItem) || !issue.id)
+ return;
+ _cache2.data[issue.id] = issue;
+ updateRtree2(encodeIssueRtree2(issue), true);
+ return issue;
+ },
+ // Remove a single QAItem from the cache
+ removeItem(issue) {
+ if (!(issue instanceof QAItem) || !issue.id)
+ return;
+ delete _cache2.data[issue.id];
+ updateRtree2(encodeIssueRtree2(issue), false);
+ },
+ // Used to populate `closed:improveosm:*` changeset tags
+ getClosedCounts() {
+ return _cache2.closed;
+ }
+ };
+
+ // modules/services/osmose.js
+ var import_rbush3 = __toESM(require_rbush_min());
+
+ // node_modules/marked/lib/marked.esm.js
+ function getDefaults() {
+ return {
+ async: false,
+ baseUrl: null,
+ breaks: false,
+ extensions: null,
+ gfm: true,
+ headerIds: true,
+ headerPrefix: "",
+ highlight: null,
+ langPrefix: "language-",
+ mangle: true,
+ pedantic: false,
+ renderer: null,
+ sanitize: false,
+ sanitizer: null,
+ silent: false,
+ smartypants: false,
+ tokenizer: null,
+ walkTokens: null,
+ xhtml: false
+ };
+ }
+ var defaults = getDefaults();
+ function changeDefaults(newDefaults) {
+ defaults = newDefaults;
+ }
+ var escapeTest = /[&<>"']/;
+ var escapeReplace = /[&<>"']/g;
+ var escapeTestNoEncode = /[<>"']|&(?!#?\w+;)/;
+ var escapeReplaceNoEncode = /[<>"']|&(?!#?\w+;)/g;
+ var escapeReplacements = {
+ "&": "&",
+ "<": "<",
+ ">": ">",
+ '"': """,
+ "'": "'"
+ };
+ var getEscapeReplacement = (ch) => escapeReplacements[ch];
+ function escape4(html2, encode) {
+ if (encode) {
+ if (escapeTest.test(html2)) {
+ return html2.replace(escapeReplace, getEscapeReplacement);
+ }
+ } else {
+ if (escapeTestNoEncode.test(html2)) {
+ return html2.replace(escapeReplaceNoEncode, getEscapeReplacement);
+ }
+ }
+ return html2;
+ }
+ var unescapeTest = /&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/ig;
+ function unescape3(html2) {
+ return html2.replace(unescapeTest, (_, n2) => {
+ n2 = n2.toLowerCase();
+ if (n2 === "colon")
+ return ":";
+ if (n2.charAt(0) === "#") {
+ return n2.charAt(1) === "x" ? String.fromCharCode(parseInt(n2.substring(2), 16)) : String.fromCharCode(+n2.substring(1));
+ }
+ return "";
+ });
+ }
+ var caret = /(^|[^\[])\^/g;
+ function edit(regex, opt) {
+ regex = typeof regex === "string" ? regex : regex.source;
+ opt = opt || "";
+ const obj = {
+ replace: (name, val) => {
+ val = val.source || val;
+ val = val.replace(caret, "$1");
+ regex = regex.replace(name, val);
+ return obj;
+ },
+ getRegex: () => {
+ return new RegExp(regex, opt);
+ }
+ };
+ return obj;
+ }
+ var nonWordAndColonTest = /[^\w:]/g;
+ var originIndependentUrl = /^$|^[a-z][a-z0-9+.-]*:|^[?#]/i;
+ function cleanUrl(sanitize, base, href) {
+ if (sanitize) {
+ let prot;
+ try {
+ prot = decodeURIComponent(unescape3(href)).replace(nonWordAndColonTest, "").toLowerCase();
+ } catch (e) {
+ return null;
+ }
+ if (prot.indexOf("javascript:") === 0 || prot.indexOf("vbscript:") === 0 || prot.indexOf("data:") === 0) {
+ return null;
+ }
+ }
+ if (base && !originIndependentUrl.test(href)) {
+ href = resolveUrl(base, href);
+ }
+ try {
+ href = encodeURI(href).replace(/%25/g, "%");
+ } catch (e) {
+ return null;
+ }
+ return href;
+ }
+ var baseUrls = {};
+ var justDomain = /^[^:]+:\/*[^/]*$/;
+ var protocol = /^([^:]+:)[\s\S]*$/;
+ var domain = /^([^:]+:\/*[^/]*)[\s\S]*$/;
+ function resolveUrl(base, href) {
+ if (!baseUrls[" " + base]) {
+ if (justDomain.test(base)) {
+ baseUrls[" " + base] = base + "/";
+ } else {
+ baseUrls[" " + base] = rtrim(base, "/", true);
+ }
+ }
+ base = baseUrls[" " + base];
+ const relativeBase = base.indexOf(":") === -1;
+ if (href.substring(0, 2) === "//") {
+ if (relativeBase) {
+ return href;
+ }
+ return base.replace(protocol, "$1") + href;
+ } else if (href.charAt(0) === "/") {
+ if (relativeBase) {
+ return href;
+ }
+ return base.replace(domain, "$1") + href;
+ } else {
+ return base + href;
+ }
+ }
+ var noopTest = { exec: function noopTest2() {
+ } };
+ function merge2(obj) {
+ let i2 = 1, target, key;
+ for (; i2 < arguments.length; i2++) {
+ target = arguments[i2];
+ for (key in target) {
+ if (Object.prototype.hasOwnProperty.call(target, key)) {
+ obj[key] = target[key];
+ }
+ }
+ }
+ return obj;
+ }
+ function splitCells(tableRow, count) {
+ const row = tableRow.replace(/\|/g, (match, offset, str2) => {
+ let escaped = false, curr = offset;
+ while (--curr >= 0 && str2[curr] === "\\")
+ escaped = !escaped;
+ if (escaped) {
+ return "|";
+ } else {
+ return " |";
+ }
+ }), cells = row.split(/ \|/);
+ let i2 = 0;
+ if (!cells[0].trim()) {
+ cells.shift();
+ }
+ if (cells.length > 0 && !cells[cells.length - 1].trim()) {
+ cells.pop();
+ }
+ if (cells.length > count) {
+ cells.splice(count);
+ } else {
+ while (cells.length < count)
+ cells.push("");
+ }
+ for (; i2 < cells.length; i2++) {
+ cells[i2] = cells[i2].trim().replace(/\\\|/g, "|");
+ }
+ return cells;
+ }
+ function rtrim(str2, c, invert) {
+ const l = str2.length;
+ if (l === 0) {
+ return "";
+ }
+ let suffLen = 0;
+ while (suffLen < l) {
+ const currChar = str2.charAt(l - suffLen - 1);
+ if (currChar === c && !invert) {
+ suffLen++;
+ } else if (currChar !== c && invert) {
+ suffLen++;
+ } else {
+ break;
+ }
+ }
+ return str2.slice(0, l - suffLen);
+ }
+ function findClosingBracket(str2, b) {
+ if (str2.indexOf(b[1]) === -1) {
+ return -1;
+ }
+ const l = str2.length;
+ let level = 0, i2 = 0;
+ for (; i2 < l; i2++) {
+ if (str2[i2] === "\\") {
+ i2++;
+ } else if (str2[i2] === b[0]) {
+ level++;
+ } else if (str2[i2] === b[1]) {
+ level--;
+ if (level < 0) {
+ return i2;
+ }
+ }
+ }
+ return -1;
+ }
+ 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 repeatString(pattern, count) {
+ if (count < 1) {
+ return "";
+ }
+ let result = "";
+ while (count > 1) {
+ if (count & 1) {
+ result += pattern;
+ }
+ count >>= 1;
+ pattern += pattern;
+ }
+ return result + pattern;
+ }
+ function outputLink(cap, link2, raw, lexer2) {
+ const href = link2.href;
+ const title = link2.title ? escape4(link2.title) : null;
+ const text2 = cap[1].replace(/\\([\[\]])/g, "$1");
+ if (cap[0].charAt(0) !== "!") {
+ lexer2.state.inLink = true;
+ const token = {
+ type: "link",
+ raw,
+ href,
+ title,
+ text: text2,
+ tokens: lexer2.inlineTokens(text2)
+ };
+ lexer2.state.inLink = false;
+ return token;
+ }
+ return {
+ type: "image",
+ raw,
+ href,
+ title,
+ text: escape4(text2)
+ };
+ }
+ function indentCodeCompensation(raw, text2) {
+ const matchIndentToCode = raw.match(/^(\s+)(?:```)/);
+ if (matchIndentToCode === null) {
+ return text2;
+ }
+ const indentToCode = matchIndentToCode[1];
+ return text2.split("\n").map((node) => {
+ const matchIndentInNode = node.match(/^\s+/);
+ if (matchIndentInNode === null) {
+ return node;
+ }
+ const [indentInNode] = matchIndentInNode;
+ if (indentInNode.length >= indentToCode.length) {
+ return node.slice(indentToCode.length);
+ }
+ return node;
+ }).join("\n");
+ }
+ var Tokenizer = class {
+ constructor(options2) {
+ this.options = options2 || defaults;
+ }
+ space(src) {
+ const cap = this.rules.block.newline.exec(src);
+ if (cap && cap[0].length > 0) {
+ return {
+ type: "space",
+ raw: cap[0]
+ };
+ }
+ }
+ code(src) {
+ const cap = this.rules.block.code.exec(src);
+ if (cap) {
+ const text2 = cap[0].replace(/^ {1,4}/gm, "");
+ return {
+ type: "code",
+ raw: cap[0],
+ codeBlockStyle: "indented",
+ text: !this.options.pedantic ? rtrim(text2, "\n") : text2
+ };
+ }
+ }
+ fences(src) {
+ const cap = this.rules.block.fences.exec(src);
+ if (cap) {
+ const raw = cap[0];
+ const text2 = indentCodeCompensation(raw, cap[3] || "");
+ return {
+ type: "code",
+ raw,
+ lang: cap[2] ? cap[2].trim().replace(this.rules.inline._escapes, "$1") : cap[2],
+ text: text2
+ };
+ }
+ }
+ heading(src) {
+ const cap = this.rules.block.heading.exec(src);
+ if (cap) {
+ let text2 = cap[2].trim();
+ if (/#$/.test(text2)) {
+ const trimmed = rtrim(text2, "#");
+ if (this.options.pedantic) {
+ text2 = trimmed.trim();
+ } else if (!trimmed || / $/.test(trimmed)) {
+ text2 = trimmed.trim();
+ }
+ }
+ return {
+ type: "heading",
+ raw: cap[0],
+ depth: cap[1].length,
+ text: text2,
+ tokens: this.lexer.inline(text2)
+ };
+ }
+ }
+ hr(src) {
+ const cap = this.rules.block.hr.exec(src);
+ if (cap) {
+ return {
+ type: "hr",
+ raw: cap[0]
+ };
+ }
+ }
+ blockquote(src) {
+ const cap = this.rules.block.blockquote.exec(src);
+ if (cap) {
+ const text2 = cap[0].replace(/^ *>[ \t]?/gm, "");
+ return {
+ type: "blockquote",
+ raw: cap[0],
+ tokens: this.lexer.blockTokens(text2, []),
+ text: text2
+ };
+ }
+ }
+ list(src) {
+ let cap = this.rules.block.list.exec(src);
+ if (cap) {
+ let raw, istask, ischecked, indent2, i2, blankLine, endsWithBlankLine, line, nextLine, rawLine, itemContents, endEarly;
+ let bull = cap[1].trim();
+ const isordered = bull.length > 1;
+ const list = {
+ type: "list",
+ raw: "",
+ ordered: isordered,
+ start: isordered ? +bull.slice(0, -1) : "",
+ loose: false,
+ items: []
+ };
+ bull = isordered ? `\\d{1,9}\\${bull.slice(-1)}` : `\\${bull}`;
+ if (this.options.pedantic) {
+ bull = isordered ? bull : "[*+-]";
+ }
+ const itemRegex = new RegExp(`^( {0,3}${bull})((?:[ ][^\\n]*)?(?:\\n|$))`);
+ while (src) {
+ endEarly = false;
+ if (!(cap = itemRegex.exec(src))) {
+ break;
+ }
+ if (this.rules.block.hr.test(src)) {
+ break;
+ }
+ raw = cap[0];
+ src = src.substring(raw.length);
+ line = cap[2].split("\n", 1)[0];
+ nextLine = src.split("\n", 1)[0];
+ if (this.options.pedantic) {
+ indent2 = 2;
+ itemContents = line.trimLeft();
+ } else {
+ indent2 = cap[2].search(/[^ ]/);
+ indent2 = indent2 > 4 ? 1 : indent2;
+ itemContents = line.slice(indent2);
+ indent2 += cap[1].length;
+ }
+ blankLine = false;
+ if (!line && /^ *$/.test(nextLine)) {
+ raw += nextLine + "\n";
+ src = src.substring(nextLine.length + 1);
+ endEarly = true;
+ }
+ if (!endEarly) {
+ const nextBulletRegex = new RegExp(`^ {0,${Math.min(3, indent2 - 1)}}(?:[*+-]|\\d{1,9}[.)])((?: [^\\n]*)?(?:\\n|$))`);
+ const hrRegex = new RegExp(`^ {0,${Math.min(3, indent2 - 1)}}((?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$)`);
+ const fencesBeginRegex = new RegExp(`^ {0,${Math.min(3, indent2 - 1)}}(?:\`\`\`|~~~)`);
+ const headingBeginRegex = new RegExp(`^ {0,${Math.min(3, indent2 - 1)}}#`);
+ while (src) {
+ rawLine = src.split("\n", 1)[0];
+ line = rawLine;
+ if (this.options.pedantic) {
+ line = line.replace(/^ {1,4}(?=( {4})*[^ ])/g, " ");
+ }
+ if (fencesBeginRegex.test(line)) {
+ break;
+ }
+ if (headingBeginRegex.test(line)) {
+ break;
+ }
+ if (nextBulletRegex.test(line)) {
+ break;
+ }
+ if (hrRegex.test(src)) {
+ break;
+ }
+ if (line.search(/[^ ]/) >= indent2 || !line.trim()) {
+ itemContents += "\n" + line.slice(indent2);
+ } else if (!blankLine) {
+ itemContents += "\n" + line;
+ } else {
+ break;
+ }
+ if (!blankLine && !line.trim()) {
+ blankLine = true;
+ }
+ raw += rawLine + "\n";
+ src = src.substring(rawLine.length + 1);
+ }
+ }
+ if (!list.loose) {
+ if (endsWithBlankLine) {
+ list.loose = true;
+ } else if (/\n *\n *$/.test(raw)) {
+ endsWithBlankLine = true;
+ }
+ }
+ if (this.options.gfm) {
+ istask = /^\[[ xX]\] /.exec(itemContents);
+ if (istask) {
+ ischecked = istask[0] !== "[ ] ";
+ itemContents = itemContents.replace(/^\[[ xX]\] +/, "");
+ }
+ }
+ list.items.push({
+ type: "list_item",
+ raw,
+ task: !!istask,
+ checked: ischecked,
+ loose: false,
+ text: itemContents
+ });
+ list.raw += raw;
+ }
+ list.items[list.items.length - 1].raw = raw.trimRight();
+ list.items[list.items.length - 1].text = itemContents.trimRight();
+ list.raw = list.raw.trimRight();
+ const l = list.items.length;
+ for (i2 = 0; i2 < l; i2++) {
+ this.lexer.state.top = false;
+ list.items[i2].tokens = this.lexer.blockTokens(list.items[i2].text, []);
+ const spacers = list.items[i2].tokens.filter((t) => t.type === "space");
+ const hasMultipleLineBreaks = spacers.every((t) => {
+ const chars = t.raw.split("");
+ let lineBreaks = 0;
+ for (const char of chars) {
+ if (char === "\n") {
+ lineBreaks += 1;
+ }
+ if (lineBreaks > 1) {
+ return true;
+ }
+ }
+ return false;
+ });
+ if (!list.loose && spacers.length && hasMultipleLineBreaks) {
+ list.loose = true;
+ list.items[i2].loose = true;
+ }
+ }
+ return list;
+ }
+ }
+ html(src) {
+ const cap = this.rules.block.html.exec(src);
+ if (cap) {
+ const token = {
+ type: "html",
+ raw: cap[0],
+ pre: !this.options.sanitizer && (cap[1] === "pre" || cap[1] === "script" || cap[1] === "style"),
+ text: cap[0]
+ };
+ if (this.options.sanitize) {
+ const text2 = this.options.sanitizer ? this.options.sanitizer(cap[0]) : escape4(cap[0]);
+ token.type = "paragraph";
+ token.text = text2;
+ token.tokens = this.lexer.inline(text2);
+ }
+ return token;
+ }
+ }
+ def(src) {
+ const cap = this.rules.block.def.exec(src);
+ if (cap) {
+ if (cap[3])
+ cap[3] = cap[3].substring(1, cap[3].length - 1);
+ const tag = cap[1].toLowerCase().replace(/\s+/g, " ");
+ return {
+ type: "def",
+ tag,
+ raw: cap[0],
+ href: cap[2] ? cap[2].replace(this.rules.inline._escapes, "$1") : cap[2],
+ title: cap[3] ? cap[3].replace(this.rules.inline._escapes, "$1") : cap[3]
+ };
+ }
+ }
+ table(src) {
+ const cap = this.rules.block.table.exec(src);
+ if (cap) {
+ const item = {
+ type: "table",
+ header: splitCells(cap[1]).map((c) => {
+ return { text: c };
+ }),
+ align: cap[2].replace(/^ *|\| *$/g, "").split(/ *\| */),
+ rows: cap[3] && cap[3].trim() ? cap[3].replace(/\n[ \t]*$/, "").split("\n") : []
+ };
+ if (item.header.length === item.align.length) {
+ item.raw = cap[0];
+ let l = item.align.length;
+ let i2, j2, k, row;
+ for (i2 = 0; i2 < l; i2++) {
+ if (/^ *-+: *$/.test(item.align[i2])) {
+ item.align[i2] = "right";
+ } else if (/^ *:-+: *$/.test(item.align[i2])) {
+ item.align[i2] = "center";
+ } else if (/^ *:-+ *$/.test(item.align[i2])) {
+ item.align[i2] = "left";
+ } else {
+ item.align[i2] = null;
+ }
+ }
+ l = item.rows.length;
+ for (i2 = 0; i2 < l; i2++) {
+ item.rows[i2] = splitCells(item.rows[i2], item.header.length).map((c) => {
+ return { text: c };
+ });
+ }
+ l = item.header.length;
+ for (j2 = 0; j2 < l; j2++) {
+ item.header[j2].tokens = this.lexer.inline(item.header[j2].text);
+ }
+ l = item.rows.length;
+ for (j2 = 0; j2 < l; j2++) {
+ row = item.rows[j2];
+ for (k = 0; k < row.length; k++) {
+ row[k].tokens = this.lexer.inline(row[k].text);
+ }
+ }
+ return item;
+ }
+ }
+ }
+ lheading(src) {
+ const cap = this.rules.block.lheading.exec(src);
+ if (cap) {
+ return {
+ type: "heading",
+ raw: cap[0],
+ depth: cap[2].charAt(0) === "=" ? 1 : 2,
+ text: cap[1],
+ tokens: this.lexer.inline(cap[1])
+ };
+ }
+ }
+ paragraph(src) {
+ const cap = this.rules.block.paragraph.exec(src);
+ if (cap) {
+ const text2 = cap[1].charAt(cap[1].length - 1) === "\n" ? cap[1].slice(0, -1) : cap[1];
+ return {
+ type: "paragraph",
+ raw: cap[0],
+ text: text2,
+ tokens: this.lexer.inline(text2)
+ };
+ }
+ }
+ text(src) {
+ const cap = this.rules.block.text.exec(src);
+ if (cap) {
+ return {
+ type: "text",
+ raw: cap[0],
+ text: cap[0],
+ tokens: this.lexer.inline(cap[0])
+ };
+ }
+ }
+ escape(src) {
+ const cap = this.rules.inline.escape.exec(src);
+ if (cap) {
+ return {
+ type: "escape",
+ raw: cap[0],
+ text: escape4(cap[1])
+ };
+ }
+ }
+ tag(src) {
+ const cap = this.rules.inline.tag.exec(src);
+ if (cap) {
+ if (!this.lexer.state.inLink && /^<a /i.test(cap[0])) {
+ this.lexer.state.inLink = true;
+ } else if (this.lexer.state.inLink && /^<\/a>/i.test(cap[0])) {
+ this.lexer.state.inLink = false;
+ }
+ if (!this.lexer.state.inRawBlock && /^<(pre|code|kbd|script)(\s|>)/i.test(cap[0])) {
+ this.lexer.state.inRawBlock = true;
+ } else if (this.lexer.state.inRawBlock && /^<\/(pre|code|kbd|script)(\s|>)/i.test(cap[0])) {
+ this.lexer.state.inRawBlock = false;
+ }
+ return {
+ type: this.options.sanitize ? "text" : "html",
+ raw: cap[0],
+ inLink: this.lexer.state.inLink,
+ inRawBlock: this.lexer.state.inRawBlock,
+ text: this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : escape4(cap[0]) : cap[0]
+ };
+ }
+ }
+ link(src) {
+ const cap = this.rules.inline.link.exec(src);
+ if (cap) {
+ const trimmedUrl = cap[2].trim();
+ if (!this.options.pedantic && /^</.test(trimmedUrl)) {
+ if (!/>$/.test(trimmedUrl)) {
+ return;
+ }
+ const rtrimSlash = rtrim(trimmedUrl.slice(0, -1), "\\");
+ if ((trimmedUrl.length - rtrimSlash.length) % 2 === 0) {
+ return;
+ }
+ } else {
+ const lastParenIndex = findClosingBracket(cap[2], "()");
+ if (lastParenIndex > -1) {
+ const start2 = cap[0].indexOf("!") === 0 ? 5 : 4;
+ const linkLen = start2 + cap[1].length + lastParenIndex;
+ cap[2] = cap[2].substring(0, lastParenIndex);
+ cap[0] = cap[0].substring(0, linkLen).trim();
+ cap[3] = "";
+ }
+ }
+ let href = cap[2];
+ let title = "";
+ if (this.options.pedantic) {
+ const link2 = /^([^'"]*[^\s])\s+(['"])(.*)\2/.exec(href);
+ if (link2) {
+ href = link2[1];
+ title = link2[3];
+ }
+ } else {
+ title = cap[3] ? cap[3].slice(1, -1) : "";
+ }
+ href = href.trim();
+ if (/^</.test(href)) {
+ if (this.options.pedantic && !/>$/.test(trimmedUrl)) {
+ href = href.slice(1);
+ } else {
+ href = href.slice(1, -1);
+ }
+ }
+ 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], this.lexer);
+ }
+ }
+ reflink(src, links) {
+ let cap;
+ if ((cap = this.rules.inline.reflink.exec(src)) || (cap = this.rules.inline.nolink.exec(src))) {
+ let link2 = (cap[2] || cap[1]).replace(/\s+/g, " ");
+ link2 = links[link2.toLowerCase()];
+ if (!link2 || !link2.href) {
+ const text2 = cap[0].charAt(0);
+ return {
+ type: "text",
+ raw: text2,
+ text: text2
+ };
+ }
+ return outputLink(cap, link2, cap[0], this.lexer);
+ }
+ }
+ emStrong(src, maskedSrc, prevChar = "") {
+ let match = this.rules.inline.emStrong.lDelim.exec(src);
+ if (!match)
+ return;
+ if (match[3] && prevChar.match(/[\p{L}\p{N}]/u))
+ return;
+ const nextChar = match[1] || match[2] || "";
+ if (!nextChar || nextChar && (prevChar === "" || this.rules.inline.punctuation.exec(prevChar))) {
+ const lLength = match[0].length - 1;
+ let rDelim, rLength, delimTotal = lLength, midDelimTotal = 0;
+ const endReg = match[0][0] === "*" ? this.rules.inline.emStrong.rDelimAst : this.rules.inline.emStrong.rDelimUnd;
+ endReg.lastIndex = 0;
+ maskedSrc = maskedSrc.slice(-1 * src.length + lLength);
+ while ((match = endReg.exec(maskedSrc)) != null) {
+ rDelim = match[1] || match[2] || match[3] || match[4] || match[5] || match[6];
+ if (!rDelim)
+ continue;
+ rLength = rDelim.length;
+ if (match[3] || match[4]) {
+ delimTotal += rLength;
+ continue;
+ } else if (match[5] || match[6]) {
+ if (lLength % 3 && !((lLength + rLength) % 3)) {
+ midDelimTotal += rLength;
+ continue;
+ }
+ }
+ delimTotal -= rLength;
+ if (delimTotal > 0)
+ continue;
+ rLength = Math.min(rLength, rLength + delimTotal + midDelimTotal);
+ const raw = src.slice(0, lLength + match.index + (match[0].length - rDelim.length) + rLength);
+ if (Math.min(lLength, rLength) % 2) {
+ const text3 = raw.slice(1, -1);
+ return {
+ type: "em",
+ raw,
+ text: text3,
+ tokens: this.lexer.inlineTokens(text3)
+ };
+ }
+ const text2 = raw.slice(2, -2);
+ return {
+ type: "strong",
+ raw,
+ text: text2,
+ tokens: this.lexer.inlineTokens(text2)
+ };
+ }
+ }
+ }
+ codespan(src) {
+ const cap = this.rules.inline.code.exec(src);
+ if (cap) {
+ let text2 = cap[2].replace(/\n/g, " ");
+ const hasNonSpaceChars = /[^ ]/.test(text2);
+ const hasSpaceCharsOnBothEnds = /^ /.test(text2) && / $/.test(text2);
+ if (hasNonSpaceChars && hasSpaceCharsOnBothEnds) {
+ text2 = text2.substring(1, text2.length - 1);
+ }
+ text2 = escape4(text2, true);
+ return {
+ type: "codespan",
+ raw: cap[0],
+ text: text2
+ };
+ }
+ }
+ br(src) {
+ const cap = this.rules.inline.br.exec(src);
+ if (cap) {
+ return {
+ type: "br",
+ raw: cap[0]
+ };
+ }
+ }
+ del(src) {
+ const cap = this.rules.inline.del.exec(src);
+ if (cap) {
+ return {
+ type: "del",
+ raw: cap[0],
+ text: cap[2],
+ tokens: this.lexer.inlineTokens(cap[2])
+ };
+ }
+ }
+ autolink(src, mangle2) {
+ const cap = this.rules.inline.autolink.exec(src);
+ if (cap) {
+ let text2, href;
+ if (cap[2] === "@") {
+ text2 = escape4(this.options.mangle ? mangle2(cap[1]) : cap[1]);
+ href = "mailto:" + text2;
+ } else {
+ text2 = escape4(cap[1]);
+ href = text2;
+ }
+ return {
+ type: "link",
+ raw: cap[0],
+ text: text2,
+ href,
+ tokens: [
+ {
+ type: "text",
+ raw: text2,
+ text: text2
+ }
+ ]
+ };
+ }
+ }
+ url(src, mangle2) {
+ let cap;
+ if (cap = this.rules.inline.url.exec(src)) {
+ let text2, href;
+ if (cap[2] === "@") {
+ text2 = escape4(this.options.mangle ? mangle2(cap[0]) : cap[0]);
+ href = "mailto:" + text2;
+ } else {
+ let prevCapZero;
+ do {
+ prevCapZero = cap[0];
+ cap[0] = this.rules.inline._backpedal.exec(cap[0])[0];
+ } while (prevCapZero !== cap[0]);
+ text2 = escape4(cap[0]);
+ if (cap[1] === "www.") {
+ href = "http://" + text2;
+ } else {
+ href = text2;
+ }
+ }
+ return {
+ type: "link",
+ raw: cap[0],
+ text: text2,
+ href,
+ tokens: [
+ {
+ type: "text",
+ raw: text2,
+ text: text2
+ }
+ ]
+ };
+ }
+ }
+ inlineText(src, smartypants2) {
+ const cap = this.rules.inline.text.exec(src);
+ if (cap) {
+ let text2;
+ if (this.lexer.state.inRawBlock) {
+ text2 = this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : escape4(cap[0]) : cap[0];
+ } else {
+ text2 = escape4(this.options.smartypants ? smartypants2(cap[0]) : cap[0]);
+ }
+ return {
+ type: "text",
+ raw: cap[0],
+ text: text2
+ };
+ }
+ }
+ };
+ 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}((?:-[\t ]*){3,}|(?:_[ \t]*){3,}|(?:\*[ \t]*){3,})(?:\n+|$)/,
+ heading: /^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/,
+ blockquote: /^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/,
+ list: /^( {0,3}bull)([ \t][^\n]+?)?(?:\n|$)/,
+ html: "^ {0,3}(?:<(script|pre|style|textarea)[\\s>][\\s\\S]*?(?:</\\1>[^\\n]*\\n+|$)|comment[^\\n]*(\\n+|$)|<\\?[\\s\\S]*?(?:\\?>\\n*|$)|<![A-Z][\\s\\S]*?(?:>\\n*|$)|<!\\[CDATA\\[[\\s\\S]*?(?:\\]\\]>\\n*|$)|</?(tag)(?: +|\\n|/?>)[\\s\\S]*?(?:(?:\\n *)+\\n|$)|<(?!script|pre|style|textarea)([a-z][\\w-]*)(?:attribute)*? */?>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n *)+\\n|$)|</(?!script|pre|style|textarea)[a-z][\\w-]*\\s*>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n *)+\\n|$))",
+ def: /^ {0,3}\[(label)\]: *(?:\n *)?<?([^\s>]+)>?(?:(?: +(?:\n *)?| *\n *)(title))? *(?:\n+|$)/,
+ 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|table| +\n)[^\n]+)*)/,
+ text: /^[^\n]+/
+ };
+ block._label = /(?!\s*\])(?:\\.|[^\[\]\\])+/;
+ block._title = /(?:"(?:\\"?|[^"\\])*"|'[^'\n]*(?:\n[^'\n]+)*\n?'|\([^()]*\))/;
+ block.def = edit(block.def).replace("label", block._label).replace("title", block._title).getRegex();
+ block.bullet = /(?:[*+-]|\d{1,9}[.)])/;
+ block.listItemStart = edit(/^( *)(bull) */).replace("bull", block.bullet).getRegex();
+ block.list = edit(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(block.html, "i").replace("comment", block._comment).replace("tag", block._tag).replace("attribute", / +[a-zA-Z:_][\w.:-]*(?: *= *"[^"\n]*"| *= *'[^'\n]*'| *= *[^\s"'=<>`]+)?/).getRegex();
+ block.paragraph = edit(block._paragraph).replace("hr", block.hr).replace("heading", " {0,3}#{1,6} ").replace("|lheading", "").replace("|table", "").replace("blockquote", " {0,3}>").replace("fences", " {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list", " {0,3}(?:[*+-]|1[.)]) ").replace("html", "</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag", block._tag).getRegex();
+ block.blockquote = edit(block.blockquote).replace("paragraph", block.paragraph).getRegex();
+ block.normal = merge2({}, block);
+ block.gfm = merge2({}, block.normal, {
+ table: "^ *([^\\n ].*\\|.*)\\n {0,3}(?:\\| *)?(:?-+:? *(?:\\| *:?-+:? *)*)(?:\\| *)?(?:\\n((?:(?! *\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)"
+ // Cells
+ });
+ block.gfm.table = edit(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[.)]) ").replace("html", "</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag", block._tag).getRegex();
+ block.gfm.paragraph = edit(block._paragraph).replace("hr", block.hr).replace("heading", " {0,3}#{1,6} ").replace("|lheading", "").replace("table", block.gfm.table).replace("blockquote", " {0,3}>").replace("fences", " {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list", " {0,3}(?:[*+-]|1[.)]) ").replace("html", "</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag", block._tag).getRegex();
+ block.pedantic = merge2({}, block.normal, {
+ html: edit(
+ `^ *(?:comment *(?:\\n|\\s*$)|<(tag)[\\s\\S]+?</\\1> *(?:\\n{2,}|\\s*$)|<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,
+ // fences not supported
+ paragraph: edit(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()
+ });
+ var inline = {
+ escape: /^\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/,
+ autolink: /^<(scheme:[^\s\x00-\x1f<>]*|email)>/,
+ url: noopTest,
+ tag: "^comment|^</[a-zA-Z][\\w:-]*\\s*>|^<[a-zA-Z][\\w-]*(?:attribute)*?\\s*/?>|^<\\?[\\s\\S]*?\\?>|^<![a-zA-Z]+\\s[\\s\\S]*?>|^<!\\[CDATA\\[[\\s\\S]*?\\]\\]>",
+ // CDATA section
+ link: /^!?\[(label)\]\(\s*(href)(?:\s+(title))?\s*\)/,
+ reflink: /^!?\[(label)\]\[(ref)\]/,
+ nolink: /^!?\[(ref)\](?:\[\])?/,
+ 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 orphan inside strong () Consume to delim (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 _
+ },
+ code: /^(`+)([^`]|[^`][\s\S]*?[^`])\1(?!`)/,
+ br: /^( {2,}|\\)\n(?!\s*$)/,
+ del: noopTest,
+ text: /^(`+|[^`])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\<!\[`*_]|\b_|$)|[^ ](?= {2,}\n)))/,
+ punctuation: /^([\spunctuation])/
+ };
+ inline._punctuation = "!\"#$%&'()+\\-.,/:;<=>?@\\[\\]`^{|}~";
+ inline.punctuation = edit(inline.punctuation).replace(/punctuation/g, inline._punctuation).getRegex();
+ inline.blockSkip = /\[[^\]]*?\]\([^\)]*?\)|`[^`]*?`|<[^>]*?>/g;
+ inline.escapedEmSt = /(?:^|[^\\])(?:\\\\)*\\[*_]/g;
+ inline._comment = edit(block._comment).replace("(?:-->|$)", "-->").getRegex();
+ inline.emStrong.lDelim = edit(inline.emStrong.lDelim).replace(/punct/g, inline._punctuation).getRegex();
+ inline.emStrong.rDelimAst = edit(inline.emStrong.rDelimAst, "g").replace(/punct/g, inline._punctuation).getRegex();
+ inline.emStrong.rDelimUnd = edit(inline.emStrong.rDelimUnd, "g").replace(/punct/g, inline._punctuation).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(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(inline.tag).replace("comment", inline._comment).replace("attribute", inline._attribute).getRegex();
+ inline._label = /(?:\[(?:\\.|[^\[\]\\])*\]|\\.|`[^`]*`|[^\[\]\\`])*?/;
+ inline._href = /<(?:\\.|[^\n<>\\])+>|[^\s\x00-\x1f]*/;
+ inline._title = /"(?:\\"?|[^"\\])*"|'(?:\\'?|[^'\\])*'|\((?:\\\)?|[^)\\])*\)/;
+ inline.link = edit(inline.link).replace("label", inline._label).replace("href", inline._href).replace("title", inline._title).getRegex();
+ inline.reflink = edit(inline.reflink).replace("label", inline._label).replace("ref", block._label).getRegex();
+ inline.nolink = edit(inline.nolink).replace("ref", block._label).getRegex();
+ inline.reflinkSearch = edit(inline.reflinkSearch, "g").replace("reflink", inline.reflink).replace("nolink", inline.nolink).getRegex();
+ inline.normal = merge2({}, inline);
+ inline.pedantic = merge2({}, 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(/^!?\[(label)\]\((.*?)\)/).replace("label", inline._label).getRegex(),
+ reflink: edit(/^!?\[(label)\]\s*\[([^\]]*)\]/).replace("label", inline._label).getRegex()
+ });
+ inline.gfm = merge2({}, inline.normal, {
+ escape: edit(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)|(?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)|[\s\S]*?(?:(?=[\\<!\[`*~_]|\b_|https?:\/\/|ftp:\/\/|www\.|$)|[^ ](?= {2,}\n)|[^a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-](?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)))/
+ });
+ inline.gfm.url = edit(inline.gfm.url, "i").replace("email", inline.gfm._extended_email).getRegex();
+ inline.breaks = merge2({}, inline.gfm, {
+ br: edit(inline.br).replace("{2,}", "*").getRegex(),
+ text: edit(inline.gfm.text).replace("\\b_", "\\b_| {2,}\\n").replace(/\{2,\}/g, "*").getRegex()
+ });
+ function smartypants(text2) {
+ return text2.replace(/---/g, "\u2014").replace(/--/g, "\u2013").replace(/(^|[-\u2014/(\[{"\s])'/g, "$1\u2018").replace(/'/g, "\u2019").replace(/(^|[-\u2014/(\[{\u2018\s])"/g, "$1\u201C").replace(/"/g, "\u201D").replace(/\.{3}/g, "\u2026");
+ }
+ function mangle(text2) {
+ let out = "", i2, ch;
+ const l = text2.length;
+ for (i2 = 0; i2 < l; i2++) {
+ ch = text2.charCodeAt(i2);
+ if (Math.random() > 0.5) {
+ ch = "x" + ch.toString(16);
+ }
+ out += "&#" + ch + ";";
+ }
+ return out;
+ }
+ var Lexer = class {
+ constructor(options2) {
+ this.tokens = [];
+ this.tokens.links = /* @__PURE__ */ Object.create(null);
+ this.options = options2 || defaults;
+ this.options.tokenizer = this.options.tokenizer || new Tokenizer();
+ this.tokenizer = this.options.tokenizer;
+ this.tokenizer.options = this.options;
+ this.tokenizer.lexer = this;
+ this.inlineQueue = [];
+ this.state = {
+ inLink: false,
+ inRawBlock: false,
+ top: true
+ };
+ const rules = {
+ block: block.normal,
+ inline: inline.normal
+ };
+ if (this.options.pedantic) {
+ rules.block = block.pedantic;
+ rules.inline = inline.pedantic;
+ } else if (this.options.gfm) {
+ rules.block = block.gfm;
+ if (this.options.breaks) {
+ rules.inline = inline.breaks;
+ } else {
+ rules.inline = inline.gfm;
+ }
+ }
+ this.tokenizer.rules = rules;
+ }
+ /**
+ * Expose Rules
+ */
+ static get rules() {
+ return {
+ block,
+ inline
+ };
+ }
+ /**
+ * Static Lex Method
+ */
+ static lex(src, options2) {
+ const lexer2 = new Lexer(options2);
+ return lexer2.lex(src);
+ }
+ /**
+ * Static Lex Inline Method
+ */
+ static lexInline(src, options2) {
+ const lexer2 = new Lexer(options2);
+ return lexer2.inlineTokens(src);
+ }
+ /**
+ * Preprocessing
+ */
+ lex(src) {
+ src = src.replace(/\r\n|\r/g, "\n");
+ this.blockTokens(src, this.tokens);
+ let next;
+ while (next = this.inlineQueue.shift()) {
+ this.inlineTokens(next.src, next.tokens);
+ }
+ return this.tokens;
+ }
+ /**
+ * Lexing
+ */
+ blockTokens(src, tokens = []) {
+ if (this.options.pedantic) {
+ src = src.replace(/\t/g, " ").replace(/^ +$/gm, "");
+ } else {
+ src = src.replace(/^( *)(\t+)/gm, (_, leading, tabs) => {
+ return leading + " ".repeat(tabs.length);
+ });
+ }
+ let token, lastToken, cutSrc, lastParagraphClipped;
+ while (src) {
+ if (this.options.extensions && this.options.extensions.block && this.options.extensions.block.some((extTokenizer) => {
+ if (token = extTokenizer.call({ lexer: this }, src, tokens)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ return true;
+ }
+ return false;
+ })) {
+ continue;
+ }
+ if (token = this.tokenizer.space(src)) {
+ src = src.substring(token.raw.length);
+ if (token.raw.length === 1 && tokens.length > 0) {
+ tokens[tokens.length - 1].raw += "\n";
+ } else {
+ tokens.push(token);
+ }
+ continue;
+ }
+ if (token = this.tokenizer.code(src)) {
+ src = src.substring(token.raw.length);
+ lastToken = tokens[tokens.length - 1];
+ if (lastToken && (lastToken.type === "paragraph" || lastToken.type === "text")) {
+ lastToken.raw += "\n" + token.raw;
+ lastToken.text += "\n" + token.text;
+ this.inlineQueue[this.inlineQueue.length - 1].src = lastToken.text;
+ } else {
+ tokens.push(token);
+ }
+ continue;
+ }
+ if (token = this.tokenizer.fences(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.heading(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.hr(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.blockquote(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.list(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.html(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.def(src)) {
+ src = src.substring(token.raw.length);
+ lastToken = tokens[tokens.length - 1];
+ if (lastToken && (lastToken.type === "paragraph" || lastToken.type === "text")) {
+ lastToken.raw += "\n" + token.raw;
+ lastToken.text += "\n" + token.raw;
+ this.inlineQueue[this.inlineQueue.length - 1].src = lastToken.text;
+ } else if (!this.tokens.links[token.tag]) {
+ this.tokens.links[token.tag] = {
+ href: token.href,
+ title: token.title
+ };
+ }
+ continue;
+ }
+ if (token = this.tokenizer.table(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.lheading(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ cutSrc = src;
+ if (this.options.extensions && this.options.extensions.startBlock) {
+ let startIndex = Infinity;
+ const tempSrc = src.slice(1);
+ let tempStart;
+ this.options.extensions.startBlock.forEach(function(getStartIndex) {
+ tempStart = getStartIndex.call({ lexer: this }, tempSrc);
+ if (typeof tempStart === "number" && tempStart >= 0) {
+ startIndex = Math.min(startIndex, tempStart);
+ }
+ });
+ if (startIndex < Infinity && startIndex >= 0) {
+ cutSrc = src.substring(0, startIndex + 1);
+ }
+ }
+ if (this.state.top && (token = this.tokenizer.paragraph(cutSrc))) {
+ lastToken = tokens[tokens.length - 1];
+ if (lastParagraphClipped && lastToken.type === "paragraph") {
+ lastToken.raw += "\n" + token.raw;
+ lastToken.text += "\n" + token.text;
+ this.inlineQueue.pop();
+ this.inlineQueue[this.inlineQueue.length - 1].src = lastToken.text;
+ } else {
+ tokens.push(token);
+ }
+ lastParagraphClipped = cutSrc.length !== src.length;
+ src = src.substring(token.raw.length);
+ continue;
+ }
+ if (token = this.tokenizer.text(src)) {
+ src = src.substring(token.raw.length);
+ lastToken = tokens[tokens.length - 1];
+ if (lastToken && lastToken.type === "text") {
+ lastToken.raw += "\n" + token.raw;
+ lastToken.text += "\n" + token.text;
+ this.inlineQueue.pop();
+ this.inlineQueue[this.inlineQueue.length - 1].src = lastToken.text;
+ } else {
+ tokens.push(token);
+ }
+ continue;
+ }
+ if (src) {
+ const errMsg = "Infinite loop on byte: " + src.charCodeAt(0);
+ if (this.options.silent) {
+ console.error(errMsg);
+ break;
+ } else {
+ throw new Error(errMsg);
+ }
+ }
+ }
+ this.state.top = true;
+ return tokens;
+ }
+ inline(src, tokens = []) {
+ this.inlineQueue.push({ src, tokens });
+ return tokens;
+ }
+ /**
+ * Lexing/Compiling
+ */
+ inlineTokens(src, tokens = []) {
+ let token, lastToken, cutSrc;
+ let maskedSrc = src;
+ let match;
+ let keepPrevChar, prevChar;
+ if (this.tokens.links) {
+ const links = Object.keys(this.tokens.links);
+ 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);
+ }
+ }
+ }
+ }
+ 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);
+ }
+ while ((match = this.tokenizer.rules.inline.escapedEmSt.exec(maskedSrc)) != null) {
+ maskedSrc = maskedSrc.slice(0, match.index + match[0].length - 2) + "++" + maskedSrc.slice(this.tokenizer.rules.inline.escapedEmSt.lastIndex);
+ this.tokenizer.rules.inline.escapedEmSt.lastIndex--;
+ }
+ while (src) {
+ if (!keepPrevChar) {
+ prevChar = "";
+ }
+ keepPrevChar = false;
+ if (this.options.extensions && this.options.extensions.inline && this.options.extensions.inline.some((extTokenizer) => {
+ if (token = extTokenizer.call({ lexer: this }, src, tokens)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ return true;
+ }
+ return false;
+ })) {
+ continue;
+ }
+ if (token = this.tokenizer.escape(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.tag(src)) {
+ src = src.substring(token.raw.length);
+ lastToken = tokens[tokens.length - 1];
+ if (lastToken && token.type === "text" && lastToken.type === "text") {
+ lastToken.raw += token.raw;
+ lastToken.text += token.text;
+ } else {
+ tokens.push(token);
+ }
+ continue;
+ }
+ if (token = this.tokenizer.link(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.reflink(src, this.tokens.links)) {
+ src = src.substring(token.raw.length);
+ lastToken = tokens[tokens.length - 1];
+ if (lastToken && token.type === "text" && lastToken.type === "text") {
+ lastToken.raw += token.raw;
+ lastToken.text += token.text;
+ } else {
+ tokens.push(token);
+ }
+ continue;
+ }
+ if (token = this.tokenizer.emStrong(src, maskedSrc, prevChar)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.codespan(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.br(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.del(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (token = this.tokenizer.autolink(src, mangle)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (!this.state.inLink && (token = this.tokenizer.url(src, mangle))) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ cutSrc = src;
+ if (this.options.extensions && this.options.extensions.startInline) {
+ let startIndex = Infinity;
+ const tempSrc = src.slice(1);
+ let tempStart;
+ this.options.extensions.startInline.forEach(function(getStartIndex) {
+ tempStart = getStartIndex.call({ lexer: this }, tempSrc);
+ if (typeof tempStart === "number" && tempStart >= 0) {
+ startIndex = Math.min(startIndex, tempStart);
+ }
+ });
+ if (startIndex < Infinity && startIndex >= 0) {
+ cutSrc = src.substring(0, startIndex + 1);
+ }
+ }
+ if (token = this.tokenizer.inlineText(cutSrc, smartypants)) {
+ src = src.substring(token.raw.length);
+ if (token.raw.slice(-1) !== "_") {
+ prevChar = token.raw.slice(-1);
+ }
+ keepPrevChar = true;
+ lastToken = tokens[tokens.length - 1];
+ if (lastToken && lastToken.type === "text") {
+ lastToken.raw += token.raw;
+ lastToken.text += token.text;
+ } else {
+ tokens.push(token);
+ }
+ continue;
+ }
+ if (src) {
+ const errMsg = "Infinite loop on byte: " + src.charCodeAt(0);
+ if (this.options.silent) {
+ console.error(errMsg);
+ break;
+ } else {
+ throw new Error(errMsg);
+ }
+ }
+ }
+ return tokens;
+ }
+ };
+ var Renderer = class {
+ constructor(options2) {
+ this.options = options2 || defaults;
+ }
+ code(code, infostring, escaped) {
+ const lang = (infostring || "").match(/\S*/)[0];
+ if (this.options.highlight) {
+ const out = this.options.highlight(code, lang);
+ if (out != null && out !== code) {
+ escaped = true;
+ code = out;
+ }
+ }
+ code = code.replace(/\n$/, "") + "\n";
+ if (!lang) {
+ return "<pre><code>" + (escaped ? code : escape4(code, true)) + "</code></pre>\n";
+ }
+ return '<pre><code class="' + this.options.langPrefix + escape4(lang, true) + '">' + (escaped ? code : escape4(code, true)) + "</code></pre>\n";
+ }
+ /**
+ * @param {string} quote
+ */
+ blockquote(quote2) {
+ return `<blockquote>
+${quote2}</blockquote>
+`;
+ }
+ html(html2) {
+ return html2;
+ }
+ /**
+ * @param {string} text
+ * @param {string} level
+ * @param {string} raw
+ * @param {any} slugger
+ */
+ heading(text2, level, raw, slugger) {
+ if (this.options.headerIds) {
+ const id2 = this.options.headerPrefix + slugger.slug(raw);
+ return `<h${level} id="${id2}">${text2}</h${level}>
+`;
+ }
+ return `<h${level}>${text2}</h${level}>
+`;
+ }
+ hr() {
+ return this.options.xhtml ? "<hr/>\n" : "<hr>\n";
+ }
+ list(body, ordered, start2) {
+ const type2 = ordered ? "ol" : "ul", startatt = ordered && start2 !== 1 ? ' start="' + start2 + '"' : "";
+ return "<" + type2 + startatt + ">\n" + body + "</" + type2 + ">\n";
+ }
+ /**
+ * @param {string} text
+ */
+ listitem(text2) {
+ return `<li>${text2}</li>
+`;
+ }
+ checkbox(checked) {
+ return "<input " + (checked ? 'checked="" ' : "") + 'disabled="" type="checkbox"' + (this.options.xhtml ? " /" : "") + "> ";
+ }
+ /**
+ * @param {string} text
+ */
+ paragraph(text2) {
+ return `<p>${text2}</p>
+`;
+ }
+ /**
+ * @param {string} header
+ * @param {string} body
+ */
+ table(header, body) {
+ if (body)
+ body = `<tbody>${body}</tbody>`;
+ return "<table>\n<thead>\n" + header + "</thead>\n" + body + "</table>\n";
+ }
+ /**
+ * @param {string} content
+ */
+ tablerow(content) {
+ return `<tr>
+${content}</tr>
+`;
+ }
+ tablecell(content, flags) {
+ const type2 = flags.header ? "th" : "td";
+ const tag = flags.align ? `<${type2} align="${flags.align}">` : `<${type2}>`;
+ return tag + content + `</${type2}>
+`;
+ }
+ /**
+ * span level renderer
+ * @param {string} text
+ */
+ strong(text2) {
+ return `<strong>${text2}</strong>`;
+ }
+ /**
+ * @param {string} text
+ */
+ em(text2) {
+ return `<em>${text2}</em>`;
+ }
+ /**
+ * @param {string} text
+ */
+ codespan(text2) {
+ return `<code>${text2}</code>`;
+ }
+ br() {
+ return this.options.xhtml ? "<br/>" : "<br>";
+ }
+ /**
+ * @param {string} text
+ */
+ del(text2) {
+ return `<del>${text2}</del>`;
+ }
+ /**
+ * @param {string} href
+ * @param {string} title
+ * @param {string} text
+ */
+ link(href, title, text2) {
+ href = cleanUrl(this.options.sanitize, this.options.baseUrl, href);
+ if (href === null) {
+ return text2;
+ }
+ let out = '<a href="' + escape4(href) + '"';
+ if (title) {
+ out += ' title="' + title + '"';
+ }
+ out += ">" + text2 + "</a>";
+ return out;
+ }
+ /**
+ * @param {string} href
+ * @param {string} title
+ * @param {string} text
+ */
+ image(href, title, text2) {
+ href = cleanUrl(this.options.sanitize, this.options.baseUrl, href);
+ if (href === null) {
+ return text2;
+ }
+ let out = `<img src="${href}" alt="${text2}"`;
+ if (title) {
+ out += ` title="${title}"`;
+ }
+ out += this.options.xhtml ? "/>" : ">";
+ return out;
+ }
+ text(text2) {
+ return text2;
+ }
+ };
+ var TextRenderer = class {
+ // no need for block level renderers
+ strong(text2) {
+ return text2;
+ }
+ em(text2) {
+ return text2;
+ }
+ codespan(text2) {
+ return text2;
+ }
+ del(text2) {
+ return text2;
+ }
+ html(text2) {
+ return text2;
+ }
+ text(text2) {
+ return text2;
+ }
+ link(href, title, text2) {
+ return "" + text2;
+ }
+ image(href, title, text2) {
+ return "" + text2;
+ }
+ br() {
+ return "";
+ }
+ };
+ var Slugger = class {
+ constructor() {
+ this.seen = {};
+ }
+ /**
+ * @param {string} value
+ */
+ serialize(value) {
+ return value.toLowerCase().trim().replace(/<[!\/a-z].*?>/ig, "").replace(/[\u2000-\u206F\u2E00-\u2E7F\\'!"#$%&()*+,./:;<=>?@[\]^`{|}~]/g, "").replace(/\s/g, "-");
+ }
+ /**
+ * Finds the next safe (unique) slug to use
+ * @param {string} originalSlug
+ * @param {boolean} isDryRun
+ */
+ getNextSafeSlug(originalSlug, isDryRun) {
+ let slug = originalSlug;
+ let occurenceAccumulator = 0;
+ if (this.seen.hasOwnProperty(slug)) {
+ occurenceAccumulator = this.seen[originalSlug];
+ do {
+ occurenceAccumulator++;
+ slug = originalSlug + "-" + occurenceAccumulator;
+ } while (this.seen.hasOwnProperty(slug));
+ }
+ 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.
+ */
+ slug(value, options2 = {}) {
+ const slug = this.serialize(value);
+ return this.getNextSafeSlug(slug, options2.dryrun);
+ }
+ };
+ var Parser = class {
+ constructor(options2) {
+ this.options = options2 || defaults;
+ this.options.renderer = this.options.renderer || new Renderer();
+ this.renderer = this.options.renderer;
+ this.renderer.options = this.options;
+ this.textRenderer = new TextRenderer();
+ this.slugger = new Slugger();
+ }
+ /**
+ * Static Parse Method
+ */
+ static parse(tokens, options2) {
+ const parser3 = new Parser(options2);
+ return parser3.parse(tokens);
+ }
+ /**
+ * Static Parse Inline Method
+ */
+ static parseInline(tokens, options2) {
+ const parser3 = new Parser(options2);
+ return parser3.parseInline(tokens);
+ }
+ /**
+ * Parse Loop
+ */
+ parse(tokens, top = true) {
+ let out = "", i2, j2, k, l2, l3, row, cell, header, body, token, ordered, start2, loose, itemBody, item, checked, task, checkbox, ret;
+ const l = tokens.length;
+ for (i2 = 0; i2 < l; i2++) {
+ token = tokens[i2];
+ if (this.options.extensions && this.options.extensions.renderers && this.options.extensions.renderers[token.type]) {
+ ret = this.options.extensions.renderers[token.type].call({ parser: this }, token);
+ if (ret !== false || !["space", "hr", "heading", "code", "table", "blockquote", "list", "html", "paragraph", "text"].includes(token.type)) {
+ out += ret || "";
+ continue;
+ }
+ }
+ switch (token.type) {
+ case "space": {
+ continue;
+ }
+ case "hr": {
+ out += this.renderer.hr();
+ continue;
+ }
+ case "heading": {
+ out += this.renderer.heading(
+ this.parseInline(token.tokens),
+ token.depth,
+ unescape3(this.parseInline(token.tokens, this.textRenderer)),
+ this.slugger
+ );
+ continue;
+ }
+ case "code": {
+ out += this.renderer.code(
+ token.text,
+ token.lang,
+ token.escaped
+ );
+ continue;
+ }
+ case "table": {
+ header = "";
+ cell = "";
+ l2 = token.header.length;
+ for (j2 = 0; j2 < l2; j2++) {
+ cell += this.renderer.tablecell(
+ this.parseInline(token.header[j2].tokens),
+ { header: true, align: token.align[j2] }
+ );
+ }
+ header += this.renderer.tablerow(cell);
+ body = "";
+ l2 = token.rows.length;
+ for (j2 = 0; j2 < l2; j2++) {
+ row = token.rows[j2];
+ cell = "";
+ l3 = row.length;
+ for (k = 0; k < l3; k++) {
+ cell += this.renderer.tablecell(
+ this.parseInline(row[k].tokens),
+ { header: false, align: token.align[k] }
+ );
+ }
+ body += this.renderer.tablerow(cell);
+ }
+ out += this.renderer.table(header, body);
+ continue;
+ }
+ case "blockquote": {
+ body = this.parse(token.tokens);
+ out += this.renderer.blockquote(body);
+ continue;
+ }
+ case "list": {
+ ordered = token.ordered;
+ start2 = token.start;
+ loose = token.loose;
+ l2 = token.items.length;
+ body = "";
+ for (j2 = 0; j2 < l2; j2++) {
+ item = token.items[j2];
+ checked = item.checked;
+ task = item.task;
+ itemBody = "";
+ if (item.task) {
+ checkbox = this.renderer.checkbox(checked);
+ if (loose) {
+ if (item.tokens.length > 0 && item.tokens[0].type === "paragraph") {
+ item.tokens[0].text = checkbox + " " + item.tokens[0].text;
+ 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;
+ }
+ }
+ itemBody += this.parse(item.tokens, loose);
+ body += this.renderer.listitem(itemBody, task, checked);
+ }
+ out += this.renderer.list(body, ordered, start2);
+ continue;
+ }
+ case "html": {
+ out += this.renderer.html(token.text);
+ continue;
+ }
+ case "paragraph": {
+ out += this.renderer.paragraph(this.parseInline(token.tokens));
+ continue;
+ }
+ case "text": {
+ body = token.tokens ? this.parseInline(token.tokens) : token.text;
+ while (i2 + 1 < l && tokens[i2 + 1].type === "text") {
+ token = tokens[++i2];
+ body += "\n" + (token.tokens ? this.parseInline(token.tokens) : token.text);
+ }
+ out += top ? this.renderer.paragraph(body) : body;
+ continue;
+ }
+ default: {
+ const errMsg = 'Token with "' + token.type + '" type was not found.';
+ if (this.options.silent) {
+ console.error(errMsg);
+ return;
+ } else {
+ throw new Error(errMsg);
+ }
+ }
+ }
+ }
+ return out;
+ }
+ /**
+ * Parse Inline Tokens
+ */
+ parseInline(tokens, renderer) {
+ renderer = renderer || this.renderer;
+ let out = "", i2, token, ret;
+ const l = tokens.length;
+ for (i2 = 0; i2 < l; i2++) {
+ token = tokens[i2];
+ if (this.options.extensions && this.options.extensions.renderers && this.options.extensions.renderers[token.type]) {
+ ret = this.options.extensions.renderers[token.type].call({ parser: this }, token);
+ if (ret !== false || !["escape", "html", "link", "image", "strong", "em", "codespan", "br", "del", "text"].includes(token.type)) {
+ out += ret || "";
+ continue;
+ }
+ }
+ switch (token.type) {
+ case "escape": {
+ out += renderer.text(token.text);
+ break;
+ }
+ case "html": {
+ out += renderer.html(token.text);
+ break;
+ }
+ case "link": {
+ out += renderer.link(token.href, token.title, this.parseInline(token.tokens, renderer));
+ break;
+ }
+ case "image": {
+ out += renderer.image(token.href, token.title, token.text);
+ break;
+ }
+ case "strong": {
+ out += renderer.strong(this.parseInline(token.tokens, renderer));
+ break;
+ }
+ case "em": {
+ out += renderer.em(this.parseInline(token.tokens, renderer));
+ break;
+ }
+ case "codespan": {
+ out += renderer.codespan(token.text);
+ break;
+ }
+ case "br": {
+ out += renderer.br();
+ break;
+ }
+ case "del": {
+ out += renderer.del(this.parseInline(token.tokens, renderer));
+ break;
+ }
+ case "text": {
+ out += renderer.text(token.text);
+ break;
+ }
+ default: {
+ const errMsg = 'Token with "' + token.type + '" type was not found.';
+ if (this.options.silent) {
+ console.error(errMsg);
+ return;
+ } else {
+ throw new Error(errMsg);
+ }
+ }
+ }
+ }
+ return out;
+ }
+ };
+ function marked(src, opt, callback) {
+ if (typeof src === "undefined" || src === null) {
+ throw new Error("marked(): input parameter is undefined or null");
+ }
+ if (typeof src !== "string") {
+ throw new Error("marked(): input parameter is of type " + Object.prototype.toString.call(src) + ", string expected");
+ }
+ if (typeof opt === "function") {
+ callback = opt;
+ opt = null;
+ }
+ opt = merge2({}, marked.defaults, opt || {});
+ checkSanitizeDeprecation(opt);
+ if (callback) {
+ const highlight = opt.highlight;
+ let tokens;
+ try {
+ tokens = Lexer.lex(src, opt);
+ } catch (e) {
+ return callback(e);
+ }
+ const done = function(err) {
+ let out;
+ if (!err) {
+ try {
+ if (opt.walkTokens) {
+ marked.walkTokens(tokens, opt.walkTokens);
+ }
+ out = Parser.parse(tokens, opt);
+ } catch (e) {
+ err = e;
+ }
+ }
+ opt.highlight = highlight;
+ return err ? callback(err) : callback(null, out);
+ };
+ if (!highlight || highlight.length < 3) {
+ return done();
+ }
+ delete opt.highlight;
+ if (!tokens.length)
+ return done();
+ let pending = 0;
+ marked.walkTokens(tokens, function(token) {
+ if (token.type === "code") {
+ pending++;
+ setTimeout(() => {
+ highlight(token.text, token.lang, function(err, code) {
+ if (err) {
+ return done(err);
+ }
+ if (code != null && code !== token.text) {
+ token.text = code;
+ token.escaped = true;
+ }
+ pending--;
+ if (pending === 0) {
+ done();
+ }
+ });
+ }, 0);
+ }
+ });
+ if (pending === 0) {
+ done();
+ }
+ return;
+ }
+ function onError(e) {
+ e.message += "\nPlease report this to https://github.com/markedjs/marked.";
+ if (opt.silent) {
+ return "<p>An error occurred:</p><pre>" + escape4(e.message + "", true) + "</pre>";
+ }
+ throw e;
+ }
+ try {
+ const tokens = Lexer.lex(src, opt);
+ if (opt.walkTokens) {
+ if (opt.async) {
+ return Promise.all(marked.walkTokens(tokens, opt.walkTokens)).then(() => {
+ return Parser.parse(tokens, opt);
+ }).catch(onError);
+ }
+ marked.walkTokens(tokens, opt.walkTokens);
+ }
+ return Parser.parse(tokens, opt);
+ } catch (e) {
+ onError(e);
+ }
+ }
+ marked.options = marked.setOptions = function(opt) {
+ merge2(marked.defaults, opt);
+ changeDefaults(marked.defaults);
+ return marked;
+ };
+ marked.getDefaults = getDefaults;
+ marked.defaults = defaults;
+ marked.use = function(...args) {
+ const opts = merge2({}, ...args);
+ const extensions = marked.defaults.extensions || { renderers: {}, childTokens: {} };
+ let hasExtensions;
+ args.forEach((pack) => {
+ if (pack.extensions) {
+ hasExtensions = true;
+ pack.extensions.forEach((ext) => {
+ if (!ext.name) {
+ throw new Error("extension name required");
+ }
+ if (ext.renderer) {
+ const prevRenderer = extensions.renderers ? extensions.renderers[ext.name] : null;
+ if (prevRenderer) {
+ extensions.renderers[ext.name] = function(...args2) {
+ let ret = ext.renderer.apply(this, args2);
+ if (ret === false) {
+ ret = prevRenderer.apply(this, args2);
+ }
+ return ret;
+ };
+ } else {
+ extensions.renderers[ext.name] = ext.renderer;
+ }
+ }
+ if (ext.tokenizer) {
+ if (!ext.level || ext.level !== "block" && ext.level !== "inline") {
+ throw new Error("extension level must be 'block' or 'inline'");
+ }
+ if (extensions[ext.level]) {
+ extensions[ext.level].unshift(ext.tokenizer);
+ } else {
+ extensions[ext.level] = [ext.tokenizer];
+ }
+ if (ext.start) {
+ if (ext.level === "block") {
+ if (extensions.startBlock) {
+ extensions.startBlock.push(ext.start);
+ } else {
+ extensions.startBlock = [ext.start];
+ }
+ } else if (ext.level === "inline") {
+ if (extensions.startInline) {
+ extensions.startInline.push(ext.start);
+ } else {
+ extensions.startInline = [ext.start];
+ }
+ }
+ }
+ }
+ if (ext.childTokens) {
+ extensions.childTokens[ext.name] = ext.childTokens;
+ }
+ });
+ }
+ if (pack.renderer) {
+ const renderer = marked.defaults.renderer || new Renderer();
+ for (const prop in pack.renderer) {
+ const prevRenderer = renderer[prop];
+ renderer[prop] = (...args2) => {
+ let ret = pack.renderer[prop].apply(renderer, args2);
+ if (ret === false) {
+ ret = prevRenderer.apply(renderer, args2);
+ }
+ return ret;
+ };
+ }
+ opts.renderer = renderer;
+ }
+ if (pack.tokenizer) {
+ const tokenizer = marked.defaults.tokenizer || new Tokenizer();
+ for (const prop in pack.tokenizer) {
+ const prevTokenizer = tokenizer[prop];
+ tokenizer[prop] = (...args2) => {
+ let ret = pack.tokenizer[prop].apply(tokenizer, args2);
+ if (ret === false) {
+ ret = prevTokenizer.apply(tokenizer, args2);
+ }
+ return ret;
+ };
+ }
+ opts.tokenizer = tokenizer;
+ }
+ if (pack.walkTokens) {
+ const walkTokens2 = marked.defaults.walkTokens;
+ opts.walkTokens = function(token) {
+ let values = [];
+ values.push(pack.walkTokens.call(this, token));
+ if (walkTokens2) {
+ values = values.concat(walkTokens2.call(this, token));
+ }
+ return values;
+ };
+ }
+ if (hasExtensions) {
+ opts.extensions = extensions;
+ }
+ marked.setOptions(opts);
+ });
+ };
+ marked.walkTokens = function(tokens, callback) {
+ let values = [];
+ for (const token of tokens) {
+ values = values.concat(callback.call(marked, token));
+ switch (token.type) {
+ case "table": {
+ for (const cell of token.header) {
+ values = values.concat(marked.walkTokens(cell.tokens, callback));
+ }
+ for (const row of token.rows) {
+ for (const cell of row) {
+ values = values.concat(marked.walkTokens(cell.tokens, callback));
+ }
+ }
+ break;
+ }
+ case "list": {
+ values = values.concat(marked.walkTokens(token.items, callback));
+ break;
+ }
+ default: {
+ if (marked.defaults.extensions && marked.defaults.extensions.childTokens && marked.defaults.extensions.childTokens[token.type]) {
+ marked.defaults.extensions.childTokens[token.type].forEach(function(childTokens) {
+ values = values.concat(marked.walkTokens(token[childTokens], callback));
+ });
+ } else if (token.tokens) {
+ values = values.concat(marked.walkTokens(token.tokens, callback));
+ }
+ }
+ }
+ }
+ return values;
+ };
+ marked.parseInline = function(src, opt) {
+ 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 = merge2({}, marked.defaults, opt || {});
+ checkSanitizeDeprecation(opt);
+ try {
+ const tokens = Lexer.lexInline(src, opt);
+ if (opt.walkTokens) {
+ marked.walkTokens(tokens, opt.walkTokens);
+ }
+ return Parser.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>" + escape4(e.message + "", true) + "</pre>";
+ }
+ throw e;
+ }
+ };
+ marked.Parser = Parser;
+ marked.parser = Parser.parse;
+ marked.Renderer = Renderer;
+ marked.TextRenderer = TextRenderer;
+ marked.Lexer = Lexer;
+ marked.lexer = Lexer.lex;
+ marked.Tokenizer = Tokenizer;
+ marked.Slugger = Slugger;
+ marked.parse = marked;
+ var options = marked.options;
+ var setOptions = marked.setOptions;
+ var use = marked.use;
+ var walkTokens = marked.walkTokens;
+ var parseInline = marked.parseInline;
+ var parser2 = Parser.parse;
+ var lexer = Lexer.lex;
+
+ // modules/services/osmose.js
+ var tiler3 = utilTiler();
+ var dispatch4 = dispatch_default("loaded");
+ var _tileZoom3 = 14;
+ var _osmoseUrlRoot = "https://osmose.openstreetmap.fr/api/0.3";
+ var _osmoseData = { icons: {}, items: [] };
+ var _cache3;
+ function abortRequest3(controller) {
+ if (controller) {
+ controller.abort();
+ }
+ }
+ function abortUnwantedRequests3(cache, tiles) {
+ Object.keys(cache.inflightTile).forEach((k) => {
+ let wanted = tiles.find((tile) => k === tile.id);
+ if (!wanted) {
+ abortRequest3(cache.inflightTile[k]);
+ delete cache.inflightTile[k];
+ }
+ });
+ }
+ function encodeIssueRtree3(d) {
+ return { minX: d.loc[0], minY: d.loc[1], maxX: d.loc[0], maxY: d.loc[1], data: d };
+ }
+ function updateRtree3(item, replace) {
+ _cache3.rtree.remove(item, (a, b) => a.data.id === b.data.id);
+ if (replace) {
+ _cache3.rtree.insert(item);
+ }
+ }
+ function preventCoincident2(loc) {
+ let coincident = false;
+ do {
+ let delta = coincident ? [1e-5, 0] : [0, 1e-5];
+ loc = geoVecAdd(loc, delta);
+ let bbox2 = geoExtent(loc).bbox();
+ coincident = _cache3.rtree.search(bbox2).length;
+ } while (coincident);
+ return loc;
+ }
+ var osmose_default = {
+ title: "osmose",
+ init() {
+ _mainFileFetcher.get("qa_data").then((d) => {
+ _osmoseData = d.osmose;
+ _osmoseData.items = Object.keys(d.osmose.icons).map((s) => s.split("-")[0]).reduce((unique, item) => unique.indexOf(item) !== -1 ? unique : [...unique, item], []);
+ });
+ if (!_cache3) {
+ this.reset();
+ }
+ this.event = utilRebind(this, dispatch4, "on");
+ },
+ reset() {
+ let _strings = {};
+ let _colors = {};
+ if (_cache3) {
+ Object.values(_cache3.inflightTile).forEach(abortRequest3);
+ _strings = _cache3.strings;
+ _colors = _cache3.colors;
+ }
+ _cache3 = {
+ data: {},
+ loadedTile: {},
+ inflightTile: {},
+ inflightPost: {},
+ closed: {},
+ rtree: new import_rbush3.default(),
+ strings: _strings,
+ colors: _colors
+ };
+ },
+ loadIssues(projection2) {
+ let params = {
+ // Tiles return a maximum # of issues
+ // So we want to filter our request for only types iD supports
+ item: _osmoseData.items
+ };
+ let tiles = tiler3.zoomExtent([_tileZoom3, _tileZoom3]).getTiles(projection2);
+ abortUnwantedRequests3(_cache3, tiles);
+ tiles.forEach((tile) => {
+ if (_cache3.loadedTile[tile.id] || _cache3.inflightTile[tile.id])
+ return;
+ let [x, y, z] = tile.xyz;
+ let url = `${_osmoseUrlRoot}/issues/${z}/${x}/${y}.geojson?` + utilQsString(params);
+ let controller = new AbortController();
+ _cache3.inflightTile[tile.id] = controller;
+ json_default(url, { signal: controller.signal }).then((data) => {
+ delete _cache3.inflightTile[tile.id];
+ _cache3.loadedTile[tile.id] = true;
+ if (data.features) {
+ data.features.forEach((issue) => {
+ const { item, class: cl, uuid: id2 } = issue.properties;
+ const itemType = `${item}-${cl}`;
+ if (itemType in _osmoseData.icons) {
+ let loc = issue.geometry.coordinates;
+ loc = preventCoincident2(loc);
+ let d = new QAItem(loc, this, itemType, id2, { item });
+ if (item === 8300 || item === 8360) {
+ d.elems = [];
+ }
+ _cache3.data[d.id] = d;
+ _cache3.rtree.insert(encodeIssueRtree3(d));
+ }
+ });
+ }
+ dispatch4.call("loaded");
+ }).catch(() => {
+ delete _cache3.inflightTile[tile.id];
+ _cache3.loadedTile[tile.id] = true;
+ });
+ });
+ },
+ loadIssueDetail(issue) {
+ if (issue.elems !== void 0) {
+ return Promise.resolve(issue);
+ }
+ const url = `${_osmoseUrlRoot}/issue/${issue.id}?langs=${_mainLocalizer.localeCode()}`;
+ const cacheDetails = (data) => {
+ issue.elems = data.elems.map((e) => e.type.substring(0, 1) + e.id);
+ issue.detail = data.subtitle ? marked(data.subtitle.auto) : "";
+ this.replaceItem(issue);
+ };
+ return json_default(url).then(cacheDetails).then(() => issue);
+ },
+ loadStrings(locale2 = _mainLocalizer.localeCode()) {
+ const items = Object.keys(_osmoseData.icons);
+ if (locale2 in _cache3.strings && Object.keys(_cache3.strings[locale2]).length === items.length) {
+ return Promise.resolve(_cache3.strings[locale2]);
+ }
+ if (!(locale2 in _cache3.strings)) {
+ _cache3.strings[locale2] = {};
+ }
+ const allRequests = items.map((itemType) => {
+ if (itemType in _cache3.strings[locale2])
+ return null;
+ const cacheData = (data) => {
+ const [cat = { items: [] }] = data.categories;
+ const [item2 = { class: [] }] = cat.items;
+ const [cl2 = null] = item2.class;
+ if (!cl2) {
+ console.log(`Osmose strings request (${itemType}) had unexpected data`);
+ return;
+ }
+ const { item: itemInt, color: color2 } = item2;
+ if (/^#[A-Fa-f0-9]{6}|[A-Fa-f0-9]{3}/.test(color2)) {
+ _cache3.colors[itemInt] = color2;
+ }
+ const { title, detail, fix, trap } = cl2;
+ let issueStrings = {};
+ if (title)
+ issueStrings.title = title.auto;
+ if (detail)
+ issueStrings.detail = marked(detail.auto);
+ if (trap)
+ issueStrings.trap = marked(trap.auto);
+ if (fix)
+ issueStrings.fix = marked(fix.auto);
+ _cache3.strings[locale2][itemType] = issueStrings;
+ };
+ const [item, cl] = itemType.split("-");
+ const url = `${_osmoseUrlRoot}/items/${item}/class/${cl}?langs=${locale2}`;
+ return json_default(url).then(cacheData);
+ }).filter(Boolean);
+ return Promise.all(allRequests).then(() => _cache3.strings[locale2]);
+ },
+ getStrings(itemType, locale2 = _mainLocalizer.localeCode()) {
+ return locale2 in _cache3.strings ? _cache3.strings[locale2][itemType] : {};
+ },
+ getColor(itemType) {
+ return itemType in _cache3.colors ? _cache3.colors[itemType] : "#FFFFFF";
+ },
+ postUpdate(issue, callback) {
+ if (_cache3.inflightPost[issue.id]) {
+ return callback({ message: "Issue update already inflight", status: -2 }, issue);
+ }
+ const url = `${_osmoseUrlRoot}/issue/${issue.id}/${issue.newStatus}`;
+ const controller = new AbortController();
+ const after = () => {
+ delete _cache3.inflightPost[issue.id];
+ this.removeItem(issue);
+ if (issue.newStatus === "done") {
+ if (!(issue.item in _cache3.closed)) {
+ _cache3.closed[issue.item] = 0;
+ }
+ _cache3.closed[issue.item] += 1;
+ }
+ if (callback)
+ callback(null, issue);
+ };
+ _cache3.inflightPost[issue.id] = controller;
+ fetch(url, { signal: controller.signal }).then(after).catch((err) => {
+ delete _cache3.inflightPost[issue.id];
+ if (callback)
+ callback(err.message);
+ });
+ },
+ // Get all cached QAItems covering the viewport
+ getItems(projection2) {
+ const viewport = projection2.clipExtent();
+ const min3 = [viewport[0][0], viewport[1][1]];
+ const max3 = [viewport[1][0], viewport[0][1]];
+ const bbox2 = geoExtent(projection2.invert(min3), projection2.invert(max3)).bbox();
+ return _cache3.rtree.search(bbox2).map((d) => d.data);
+ },
+ // Get a QAItem from cache
+ // NOTE: Don't change method name until UI v3 is merged
+ getError(id2) {
+ return _cache3.data[id2];
+ },
+ // get the name of the icon to display for this item
+ getIcon(itemType) {
+ return _osmoseData.icons[itemType];
+ },
+ // Replace a single QAItem in the cache
+ replaceItem(item) {
+ if (!(item instanceof QAItem) || !item.id)
+ return;
+ _cache3.data[item.id] = item;
+ updateRtree3(encodeIssueRtree3(item), true);
+ return item;
+ },
+ // Remove a single QAItem from the cache
+ removeItem(item) {
+ if (!(item instanceof QAItem) || !item.id)
+ return;
+ delete _cache3.data[item.id];
+ updateRtree3(encodeIssueRtree3(item), false);
+ },
+ // Used to populate `closed:osmose:*` changeset tags
+ getClosedCounts() {
+ return _cache3.closed;
+ },
+ itemURL(item) {
+ return `https://osmose.openstreetmap.fr/en/error/${item.id}`;
+ }
+ };
+
+ // modules/services/mapillary.js
+ var import_pbf = __toESM(require_pbf());
+ var import_rbush4 = __toESM(require_rbush_min());
+ var import_vector_tile = __toESM(require_vector_tile());
+ var accessToken = "MLY|4100327730013843|5bb78b81720791946a9a7b956c57b7cf";
+ var apiUrl = "https://graph.mapillary.com/";
+ var baseTileUrl = "https://tiles.mapillary.com/maps/vtp";
+ var mapFeatureTileUrl = `${baseTileUrl}/mly_map_feature_point/2/{z}/{x}/{y}?access_token=${accessToken}`;
+ var tileUrl = `${baseTileUrl}/mly1_public/2/{z}/{x}/{y}?access_token=${accessToken}`;
+ var trafficSignTileUrl = `${baseTileUrl}/mly_map_feature_traffic_sign/2/{z}/{x}/{y}?access_token=${accessToken}`;
+ var viewercss = "mapillary-js/mapillary.css";
+ var viewerjs = "mapillary-js/mapillary.js";
+ var minZoom = 14;
+ var dispatch5 = dispatch_default("change", "loadedImages", "loadedSigns", "loadedMapFeatures", "bearingChanged", "imageChanged");
+ var _loadViewerPromise;
+ var _mlyActiveImage;
+ var _mlyCache;
+ var _mlyFallback = false;
+ var _mlyHighlightedDetection;
+ var _mlyShowFeatureDetections = false;
+ var _mlyShowSignDetections = false;
+ var _mlyViewer;
+ var _mlyViewerFilter = ["all"];
+ function loadTiles(which, url, maxZoom2, projection2) {
+ const tiler8 = utilTiler().zoomExtent([minZoom, maxZoom2]).skipNullIsland(true);
+ const tiles = tiler8.getTiles(projection2);
+ tiles.forEach(function(tile) {
+ loadTile(which, url, tile);
+ });
+ }
+ function loadTile(which, url, tile) {
+ const cache = _mlyCache.requests;
+ const tileId = `${tile.id}-${which}`;
+ if (cache.loaded[tileId] || cache.inflight[tileId])
+ return;
+ const controller = new AbortController();
+ cache.inflight[tileId] = controller;
+ const 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);
+ }
+ cache.loaded[tileId] = true;
+ delete cache.inflight[tileId];
+ return response.arrayBuffer();
+ }).then(function(data) {
+ if (!data) {
+ throw new Error("No Data");
+ }
+ loadTileDataToCache(data, tile, which);
+ if (which === "images") {
+ dispatch5.call("loadedImages");
+ } else if (which === "signs") {
+ dispatch5.call("loadedSigns");
+ } else if (which === "points") {
+ dispatch5.call("loadedMapFeatures");
+ }
+ }).catch(function() {
+ cache.loaded[tileId] = true;
+ delete cache.inflight[tileId];
+ });
+ }
+ function loadTileDataToCache(data, tile, which) {
+ const vectorTile = new import_vector_tile.VectorTile(new import_pbf.default(data));
+ let features, cache, layer, i2, feature3, loc, d;
+ if (vectorTile.layers.hasOwnProperty("image")) {
+ features = [];
+ cache = _mlyCache.images;
+ layer = vectorTile.layers.image;
+ for (i2 = 0; i2 < layer.length; i2++) {
+ feature3 = layer.feature(i2).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+ loc = feature3.geometry.coordinates;
+ d = {
+ loc,
+ captured_at: feature3.properties.captured_at,
+ ca: feature3.properties.compass_angle,
+ id: feature3.properties.id,
+ is_pano: feature3.properties.is_pano,
+ sequence_id: feature3.properties.sequence_id
+ };
+ cache.forImageId[d.id] = d;
+ features.push({
+ minX: loc[0],
+ minY: loc[1],
+ maxX: loc[0],
+ maxY: loc[1],
+ data: d
+ });
+ }
+ if (cache.rtree) {
+ cache.rtree.load(features);
+ }
+ }
+ if (vectorTile.layers.hasOwnProperty("sequence")) {
+ features = [];
+ cache = _mlyCache.sequences;
+ layer = vectorTile.layers.sequence;
+ for (i2 = 0; i2 < layer.length; i2++) {
+ feature3 = layer.feature(i2).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+ if (cache.lineString[feature3.properties.id]) {
+ cache.lineString[feature3.properties.id].push(feature3);
+ } else {
+ cache.lineString[feature3.properties.id] = [feature3];
+ }
+ }
+ }
+ if (vectorTile.layers.hasOwnProperty("point")) {
+ features = [];
+ cache = _mlyCache[which];
+ layer = vectorTile.layers.point;
+ for (i2 = 0; i2 < layer.length; i2++) {
+ feature3 = layer.feature(i2).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+ loc = feature3.geometry.coordinates;
+ d = {
+ loc,
+ id: feature3.properties.id,
+ first_seen_at: feature3.properties.first_seen_at,
+ last_seen_at: feature3.properties.last_seen_at,
+ value: feature3.properties.value
+ };
+ features.push({
+ minX: loc[0],
+ minY: loc[1],
+ maxX: loc[0],
+ maxY: loc[1],
+ data: d
+ });
+ }
+ if (cache.rtree) {
+ cache.rtree.load(features);
+ }
+ }
+ if (vectorTile.layers.hasOwnProperty("traffic_sign")) {
+ features = [];
+ cache = _mlyCache[which];
+ layer = vectorTile.layers.traffic_sign;
+ for (i2 = 0; i2 < layer.length; i2++) {
+ feature3 = layer.feature(i2).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+ loc = feature3.geometry.coordinates;
+ d = {
+ loc,
+ id: feature3.properties.id,
+ first_seen_at: feature3.properties.first_seen_at,
+ last_seen_at: feature3.properties.last_seen_at,
+ value: feature3.properties.value
+ };
+ features.push({
+ minX: loc[0],
+ minY: loc[1],
+ maxX: loc[0],
+ maxY: loc[1],
+ data: d
+ });
+ }
+ if (cache.rtree) {
+ cache.rtree.load(features);
+ }
+ }
+ }
+ function loadData(url) {
+ return fetch(url).then(function(response) {
+ if (!response.ok) {
+ throw new Error(response.status + " " + response.statusText);
+ }
+ return response.json();
+ }).then(function(result) {
+ if (!result) {
+ return [];
+ }
+ return result.data || [];
+ });
+ }
+ function partitionViewport(projection2) {
+ const z = geoScaleToZoom(projection2.scale());
+ const z2 = Math.ceil(z * 2) / 2 + 2.5;
+ const tiler8 = utilTiler().zoomExtent([z2, z2]);
+ return tiler8.getTiles(projection2).map(function(tile) {
+ return tile.extent;
+ });
+ }
+ function searchLimited(limit, projection2, rtree) {
+ limit = limit || 5;
+ return partitionViewport(projection2).reduce(function(result, extent) {
+ const found = rtree.search(extent.bbox()).slice(0, limit).map(function(d) {
+ return d.data;
+ });
+ return found.length ? result.concat(found) : result;
+ }, []);
+ }
+ var mapillary_default = {
+ // Initialize Mapillary
+ init: function() {
+ if (!_mlyCache) {
+ this.reset();
+ }
+ this.event = utilRebind(this, dispatch5, "on");
+ },
+ // Reset cache and state
+ reset: function() {
+ if (_mlyCache) {
+ Object.values(_mlyCache.requests.inflight).forEach(function(request3) {
+ request3.abort();
+ });
+ }
+ _mlyCache = {
+ images: { rtree: new import_rbush4.default(), forImageId: {} },
+ image_detections: { forImageId: {} },
+ signs: { rtree: new import_rbush4.default() },
+ points: { rtree: new import_rbush4.default() },
+ sequences: { rtree: new import_rbush4.default(), lineString: {} },
+ requests: { loaded: {}, inflight: {} }
+ };
+ _mlyActiveImage = null;
+ },
+ // Get visible images
+ images: function(projection2) {
+ const limit = 5;
+ return searchLimited(limit, projection2, _mlyCache.images.rtree);
+ },
+ // Get visible traffic signs
+ signs: function(projection2) {
+ const limit = 5;
+ return searchLimited(limit, projection2, _mlyCache.signs.rtree);
+ },
+ // Get visible map (point) features
+ mapFeatures: function(projection2) {
+ const limit = 5;
+ return searchLimited(limit, projection2, _mlyCache.points.rtree);
+ },
+ // Get cached image by id
+ cachedImage: function(imageId) {
+ return _mlyCache.images.forImageId[imageId];
+ },
+ // Get visible sequences
+ sequences: function(projection2) {
+ const viewport = projection2.clipExtent();
+ const min3 = [viewport[0][0], viewport[1][1]];
+ const max3 = [viewport[1][0], viewport[0][1]];
+ const bbox2 = geoExtent(projection2.invert(min3), projection2.invert(max3)).bbox();
+ const sequenceIds = {};
+ let lineStrings = [];
+ _mlyCache.images.rtree.search(bbox2).forEach(function(d) {
+ if (d.data.sequence_id) {
+ sequenceIds[d.data.sequence_id] = true;
+ }
+ });
+ 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(projection2) {
+ loadTiles("images", tileUrl, 14, projection2);
+ },
+ // Load traffic signs in the visible area
+ loadSigns: function(projection2) {
+ loadTiles("signs", trafficSignTileUrl, 14, projection2);
+ },
+ // Load map (point) features in the visible area
+ loadMapFeatures: function(projection2) {
+ loadTiles("points", mapFeatureTileUrl, 14, projection2);
+ },
+ // Return a promise that resolves when the image viewer (Mapillary JS) library has finished loading
+ ensureViewerLoaded: function(context) {
+ if (_loadViewerPromise)
+ return _loadViewerPromise;
+ const wrap2 = context.container().select(".photoviewer").selectAll(".mly-wrapper").data([0]);
+ wrap2.enter().append("div").attr("id", "ideditor-mly").attr("class", "photo-wrapper mly-wrapper").classed("hide", true);
+ const that = this;
+ _loadViewerPromise = new Promise((resolve, reject) => {
+ let loadedCount = 0;
+ function loaded() {
+ loadedCount += 1;
+ if (loadedCount === 2)
+ resolve();
+ }
+ const head = select_default2("head");
+ 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();
+ });
+ 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;
+ },
+ // Load traffic sign image sprites
+ loadSignResources: function(context) {
+ context.ui().svgDefs.addSprites(
+ ["mapillary-sprite"],
+ false
+ /* don't override colors */
+ );
+ return this;
+ },
+ // Load map (point) feature image sprites
+ loadObjectResources: function(context) {
+ context.ui().svgDefs.addSprites(
+ ["mapillary-object-sprite"],
+ false
+ /* don't override colors */
+ );
+ return this;
+ },
+ // Remove previous detections in image viewer
+ resetTags: function() {
+ if (_mlyViewer && !_mlyFallback) {
+ _mlyViewer.getComponent("tag").removeAll();
+ }
+ },
+ // Show map feature detections in image viewer
+ showFeatureDetections: function(value) {
+ _mlyShowFeatureDetections = value;
+ if (!_mlyShowFeatureDetections && !_mlyShowSignDetections) {
+ this.resetTags();
+ }
+ },
+ // Show traffic sign detections in image viewer
+ showSignDetections: function(value) {
+ _mlyShowSignDetections = value;
+ if (!_mlyShowFeatureDetections && !_mlyShowSignDetections) {
+ this.resetTags();
+ }
+ },
+ // Apply filter to image viewer
+ filterViewer: function(context) {
+ const showsPano = context.photos().showsPanoramic();
+ const showsFlat = context.photos().showsFlat();
+ const fromDate = context.photos().fromDate();
+ const toDate = context.photos().toDate();
+ const filter2 = ["all"];
+ if (!showsPano)
+ filter2.push(["!=", "cameraType", "spherical"]);
+ if (!showsFlat && showsPano)
+ filter2.push(["==", "pano", true]);
+ if (fromDate) {
+ filter2.push([">=", "capturedAt", new Date(fromDate).getTime()]);
+ }
+ if (toDate) {
+ filter2.push([">=", "capturedAt", new Date(toDate).getTime()]);
+ }
+ if (_mlyViewer) {
+ _mlyViewer.setFilter(filter2);
+ }
+ _mlyViewerFilter = filter2;
+ return filter2;
+ },
+ // Make the image viewer visible
+ showViewer: function(context) {
+ const wrap2 = context.container().select(".photoviewer").classed("hide", false);
+ const isHidden = wrap2.selectAll(".photo-wrapper.mly-wrapper.hide").size();
+ if (isHidden && _mlyViewer) {
+ wrap2.selectAll(".photo-wrapper:not(.mly-wrapper)").classed("hide", true);
+ wrap2.selectAll(".photo-wrapper.mly-wrapper").classed("hide", false);
+ _mlyViewer.resize();
+ }
+ return this;
+ },
+ // Hide the image viewer and resets map markers
+ hideViewer: function(context) {
+ _mlyActiveImage = null;
+ if (!_mlyFallback && _mlyViewer) {
+ _mlyViewer.getComponent("sequence").stop();
+ }
+ const viewer = context.container().select(".photoviewer");
+ if (!viewer.empty())
+ viewer.datum(null);
+ viewer.classed("hide", true).selectAll(".photo-wrapper").classed("hide", true);
+ this.updateUrlImage(null);
+ dispatch5.call("imageChanged");
+ dispatch5.call("loadedMapFeatures");
+ dispatch5.call("loadedSigns");
+ return this.setStyles(context, null);
+ },
+ // Update the URL with current image id
+ updateUrlImage: function(imageId) {
+ if (!window.mocha) {
+ const hash = utilStringQs(window.location.hash);
+ if (imageId) {
+ hash.photo = "mapillary/" + imageId;
+ } else {
+ delete hash.photo;
+ }
+ window.location.replace("#" + utilQsString(hash, true));
+ }
+ },
+ // Highlight the detection in the viewer that is related to the clicked map feature
+ highlightDetection: function(detection) {
+ if (detection) {
+ _mlyHighlightedDetection = detection.id;
+ }
+ return this;
+ },
+ // Initialize image viewer (Mapillar JS)
+ initViewer: function(context) {
+ const that = this;
+ if (!window.mapillary)
+ return;
+ const opts = {
+ accessToken,
+ component: {
+ cover: false,
+ keyboard: false,
+ tag: true
+ },
+ container: "ideditor-mly"
+ };
+ 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
+ };
+ }
+ _mlyViewer = new mapillary.Viewer(opts);
+ _mlyViewer.on("image", imageChanged);
+ _mlyViewer.on("bearing", bearingChanged);
+ if (_mlyViewerFilter) {
+ _mlyViewer.setFilter(_mlyViewerFilter);
+ }
+ context.ui().photoviewer.on("resize.mapillary", function() {
+ if (_mlyViewer)
+ _mlyViewer.resize();
+ });
+ function imageChanged(node) {
+ that.resetTags();
+ const image = node.image;
+ that.setActiveImage(image);
+ that.setStyles(context, null);
+ const loc = [image.originalLngLat.lng, image.originalLngLat.lat];
+ context.map().centerEase(loc);
+ that.updateUrlImage(image.id);
+ if (_mlyShowFeatureDetections || _mlyShowSignDetections) {
+ that.updateDetections(image.id, `${apiUrl}/${image.id}/detections?access_token=${accessToken}&fields=id,image,geometry,value`);
+ }
+ dispatch5.call("imageChanged");
+ }
+ function bearingChanged(e) {
+ dispatch5.call("bearingChanged", void 0, e);
+ }
+ },
+ // Move to an image
+ selectImage: function(context, imageId) {
+ if (_mlyViewer && imageId) {
+ _mlyViewer.moveTo(imageId).catch(function(e) {
+ console.error("mly3", e);
+ });
+ }
+ return this;
+ },
+ // Return the currently displayed image
+ getActiveImage: function() {
+ return _mlyActiveImage;
+ },
+ // Return a list of detection objects for the given id
+ getDetections: function(id2) {
+ return loadData(`${apiUrl}/${id2}/detections?access_token=${accessToken}&fields=id,value,image`);
+ },
+ // Set the currently visible image
+ setActiveImage: function(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;
+ }
+ },
+ // Update the currently highlighted sequence and selected bubble.
+ setStyles: function(context, hovered) {
+ const hoveredImageId = hovered && hovered.id;
+ const hoveredSequenceId = hovered && hovered.sequence_id;
+ const 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(imageId, url) {
+ if (!_mlyViewer || _mlyFallback)
+ return;
+ if (!imageId)
+ return;
+ const cache = _mlyCache.image_detections;
+ if (cache.forImageId[imageId]) {
+ showDetections(_mlyCache.image_detections.forImageId[imageId]);
+ } else {
+ loadData(url).then((detections) => {
+ detections.forEach(function(detection) {
+ if (!cache.forImageId[imageId]) {
+ cache.forImageId[imageId] = [];
+ }
+ cache.forImageId[imageId].push({
+ geometry: detection.geometry,
+ id: detection.id,
+ image_id: imageId,
+ value: detection.value
+ });
+ });
+ showDetections(_mlyCache.image_detections.forImageId[imageId] || []);
+ });
+ }
+ function showDetections(detections) {
+ const tagComponent = _mlyViewer.getComponent("tag");
+ detections.forEach(function(data) {
+ const tag = makeTag(data);
+ if (tag) {
+ tagComponent.add([tag]);
+ }
+ });
+ }
+ function makeTag(data) {
+ const valueParts = data.value.split("--");
+ if (!valueParts.length)
+ return;
+ let tag;
+ let text2;
+ let color2 = 16777215;
+ if (_mlyHighlightedDetection === data.id) {
+ color2 = 16776960;
+ text2 = valueParts[1];
+ if (text2 === "flat" || text2 === "discrete" || text2 === "sign") {
+ text2 = valueParts[2];
+ }
+ text2 = text2.replace(/-/g, " ");
+ text2 = text2.charAt(0).toUpperCase() + text2.slice(1);
+ _mlyHighlightedDetection = null;
+ }
+ var decodedGeometry = window.atob(data.geometry);
+ var uintArray = new Uint8Array(decodedGeometry.length);
+ for (var i2 = 0; i2 < decodedGeometry.length; i2++) {
+ uintArray[i2] = decodedGeometry.charCodeAt(i2);
+ }
+ const tile = new import_vector_tile.VectorTile(new import_pbf.default(uintArray.buffer));
+ const layer = tile.layers["mpy-or"];
+ const geometries = layer.feature(0).loadGeometry();
+ const polygon2 = geometries.map((ring) => ring.map((point2) => [point2.x / layer.extent, point2.y / layer.extent]));
+ tag = new mapillary.OutlineTag(
+ data.id,
+ new mapillary.PolygonGeometry(polygon2[0]),
+ {
+ text: text2,
+ textColor: color2,
+ lineColor: color2,
+ lineWidth: 2,
+ fillColor: color2,
+ fillOpacity: 0.3
+ }
+ );
+ return tag;
+ }
+ },
+ // Return the current cache
+ cache: function() {
+ return _mlyCache;
+ }
+ };
+
+ // modules/core/validation/models.js
+ function validationIssue(attrs) {
+ this.type = attrs.type;
+ this.subtype = attrs.subtype;
+ this.severity = attrs.severity;
+ this.message = attrs.message;
+ this.reference = attrs.reference;
+ this.entityIds = attrs.entityIds;
+ this.loc = attrs.loc;
+ this.data = attrs.data;
+ this.dynamicFixes = attrs.dynamicFixes;
+ this.hash = attrs.hash;
+ this.id = generateID.apply(this);
+ this.key = generateKey.apply(this);
+ this.autoFix = null;
+ function generateID() {
+ var parts = [this.type];
+ if (this.hash) {
+ parts.push(this.hash);
+ }
+ if (this.subtype) {
+ parts.push(this.subtype);
+ }
+ if (this.entityIds) {
+ var entityKeys = this.entityIds.slice().sort();
+ parts.push.apply(parts, entityKeys);
+ }
+ return parts.join(":");
+ }
+ function generateKey() {
+ return this.id + ":" + Date.now().toString();
+ }
+ this.extent = function(resolver) {
+ if (this.loc) {
+ return geoExtent(this.loc);
+ }
+ if (this.entityIds && this.entityIds.length) {
+ return this.entityIds.reduce(function(extent, entityId) {
+ return extent.extend(resolver.entity(entityId).extent(resolver));
+ }, geoExtent());
+ }
+ return null;
+ };
+ this.fixes = function(context) {
+ var fixes = this.dynamicFixes ? this.dynamicFixes(context) : [];
+ var issue = this;
+ if (issue.severity === "warning") {
+ fixes.push(new validationIssueFix({
+ title: _t.append("issues.fix.ignore_issue.title"),
+ icon: "iD-icon-close",
+ onClick: function() {
+ context.validator().ignoreIssue(this.issue.id);
+ }
+ }));
+ }
+ fixes.forEach(function(fix) {
+ fix.id = fix.title.stringId;
+ fix.issue = issue;
+ if (fix.autoArgs) {
+ issue.autoFix = fix;
+ }
+ });
+ return fixes;
+ };
+ }
+ function validationIssueFix(attrs) {
+ this.title = attrs.title;
+ this.onClick = attrs.onClick;
+ this.disabledReason = attrs.disabledReason;
+ this.icon = attrs.icon;
+ this.entityIds = attrs.entityIds || [];
+ this.autoArgs = attrs.autoArgs;
+ this.issue = null;
+ }
+
+ // modules/services/maprules.js
+ var buildRuleChecks = function() {
+ return {
+ equals: function(equals) {
+ return function(tags) {
+ return Object.keys(equals).every(function(k) {
+ return equals[k] === tags[k];
+ });
+ };
+ },
+ notEquals: function(notEquals) {
+ return function(tags) {
+ return Object.keys(notEquals).some(function(k) {
+ return notEquals[k] !== tags[k];
+ });
+ };
+ },
+ absence: function(absence) {
+ return function(tags) {
+ return Object.keys(tags).indexOf(absence) === -1;
+ };
+ },
+ presence: function(presence) {
+ return function(tags) {
+ return Object.keys(tags).indexOf(presence) > -1;
+ };
+ },
+ greaterThan: function(greaterThan) {
+ var key = Object.keys(greaterThan)[0];
+ var value = greaterThan[key];
+ return function(tags) {
+ return tags[key] > value;
+ };
+ },
+ greaterThanEqual: function(greaterThanEqual) {
+ var key = Object.keys(greaterThanEqual)[0];
+ var value = greaterThanEqual[key];
+ return function(tags) {
+ return tags[key] >= value;
+ };
+ },
+ lessThan: function(lessThan) {
+ var key = Object.keys(lessThan)[0];
+ var value = lessThan[key];
+ return function(tags) {
+ return tags[key] < value;
+ };
+ },
+ lessThanEqual: function(lessThanEqual) {
+ var key = Object.keys(lessThanEqual)[0];
+ var value = lessThanEqual[key];
+ return function(tags) {
+ return tags[key] <= value;
+ };
+ },
+ positiveRegex: function(positiveRegex) {
+ var tagKey = Object.keys(positiveRegex)[0];
+ var expression = positiveRegex[tagKey].join("|");
+ var regex = new RegExp(expression);
+ return function(tags) {
+ return regex.test(tags[tagKey]);
+ };
+ },
+ negativeRegex: function(negativeRegex) {
+ var tagKey = Object.keys(negativeRegex)[0];
+ var expression = negativeRegex[tagKey].join("|");
+ var regex = new RegExp(expression);
+ return function(tags) {
+ return !regex.test(tags[tagKey]);
+ };
+ }
+ };
+ };
+ var buildLineKeys = function() {
+ return {
+ highway: {
+ rest_area: true,
+ services: true
+ },
+ railway: {
+ roundhouse: true,
+ station: true,
+ traverser: true,
+ turntable: true,
+ wash: true
+ }
+ };
+ };
+ var maprules_default = {
+ init: function() {
+ this._ruleChecks = buildRuleChecks();
+ this._validationRules = [];
+ this._areaKeys = osmAreaKeys;
+ this._lineKeys = buildLineKeys();
+ },
+ // list of rules only relevant to tag checks...
+ filterRuleChecks: function(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 rules;
+ }, []);
+ },
+ // builds tagMap from mapcss-parse selector object...
+ buildTagMap: function(selector) {
+ var getRegexValues = function(regexes) {
+ return regexes.map(function(regex) {
+ return regex.replace(/\$|\^/g, "");
+ });
+ };
+ var tagMap = Object.keys(selector).reduce(function(expectedTags, key) {
+ var values;
+ var isRegex = /regex/gi.test(key);
+ var isEqual = /equals/gi.test(key);
+ if (isRegex || isEqual) {
+ Object.keys(selector[key]).forEach(function(selectorKey) {
+ values = isEqual ? [selector[key][selectorKey]] : getRegexValues(selector[key][selectorKey]);
+ if (expectedTags.hasOwnProperty(selectorKey)) {
+ values = values.concat(expectedTags[selectorKey]);
+ }
+ 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 (expectedTags.hasOwnProperty(tagKey)) {
+ values = values.concat(expectedTags[tagKey]);
+ }
+ expectedTags[tagKey] = values;
+ }
+ return expectedTags;
+ }, {});
+ return tagMap;
+ },
+ // inspired by osmWay#isArea()
+ inferGeometry: function(tagMap) {
+ var _lineKeys = this._lineKeys;
+ var _areaKeys = this._areaKeys;
+ var keyValueDoesNotImplyArea = function(key2) {
+ return utilArrayIntersection(tagMap[key2], Object.keys(_areaKeys[key2])).length > 0;
+ };
+ var keyValueImpliesLine = function(key2) {
+ return utilArrayIntersection(tagMap[key2], Object.keys(_lineKeys[key2])).length > 0;
+ };
+ if (tagMap.hasOwnProperty("area")) {
+ if (tagMap.area.indexOf("yes") > -1) {
+ return "area";
+ }
+ if (tagMap.area.indexOf("no") > -1) {
+ return "line";
+ }
+ }
+ for (var key in tagMap) {
+ if (key in _areaKeys && !keyValueDoesNotImplyArea(key)) {
+ return "area";
+ }
+ if (key in _lineKeys && keyValueImpliesLine(key)) {
+ return "area";
+ }
+ }
+ return "line";
+ },
+ // adds from mapcss-parse selector check...
+ addRule: function(selector) {
+ var rule = {
+ // checks relevant to mapcss-selector
+ checks: this.filterRuleChecks(selector),
+ // true if all conditions for a tag error are true..
+ matches: function(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(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(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,
+ message: function() {
+ return message;
+ },
+ entityIds: [entity.id]
+ }));
+ }
+ }
+ };
+ this._validationRules.push(rule);
+ },
+ clearRules: function() {
+ this._validationRules = [];
+ },
+ // returns validationRules...
+ validationRules: function() {
+ return this._validationRules;
+ },
+ // returns ruleChecks
+ ruleChecks: function() {
+ return this._ruleChecks;
+ }
+ };
+
+ // modules/services/nominatim.js
+ var import_rbush5 = __toESM(require_rbush_min());
+ var apibase = nominatimApiUrl;
+ var _inflight = {};
+ var _nominatimCache;
+ var nominatim_default = {
+ init: function() {
+ _inflight = {};
+ _nominatimCache = new import_rbush5.default();
+ },
+ reset: function() {
+ Object.values(_inflight).forEach(function(controller) {
+ controller.abort();
+ });
+ _inflight = {};
+ _nominatimCache = new import_rbush5.default();
+ },
+ countryCode: function(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(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;
+ }
+ var params = { zoom: 13, format: "json", addressdetails: 1, lat: loc[1], lon: loc[0] };
+ var url = apibase + "reverse?" + utilQsString(params);
+ if (_inflight[url])
+ return;
+ var controller = new AbortController();
+ _inflight[url] = controller;
+ json_default(url, { signal: controller.signal }).then(function(result) {
+ delete _inflight[url];
+ if (result && result.error) {
+ throw new Error(result.error);
+ }
+ var extent = geoExtent(loc).padByMeters(200);
+ _nominatimCache.insert(Object.assign(extent.bbox(), { data: result }));
+ if (callback)
+ callback(null, result);
+ }).catch(function(err) {
+ delete _inflight[url];
+ if (err.name === "AbortError")
+ return;
+ if (callback)
+ callback(err.message);
+ });
+ },
+ search: function(val, callback) {
+ var searchVal = encodeURIComponent(val);
+ var url = apibase + "search/" + searchVal + "?limit=10&format=json";
+ if (_inflight[url])
+ return;
+ var controller = new AbortController();
+ _inflight[url] = controller;
+ json_default(url, { signal: controller.signal }).then(function(result) {
+ delete _inflight[url];
+ if (result && result.error) {
+ throw new Error(result.error);
+ }
+ if (callback)
+ callback(null, result);
+ }).catch(function(err) {
+ delete _inflight[url];
+ if (err.name === "AbortError")
+ return;
+ if (callback)
+ callback(err.message);
+ });
+ }
+ };
+
+ // node_modules/name-suggestion-index/lib/matcher.js
+ var import_which_polygon3 = __toESM(require_which_polygon(), 1);
+
+ // node_modules/name-suggestion-index/lib/simplify.js
+ var import_diacritics2 = __toESM(require_diacritics(), 1);
+ function simplify(str2) {
+ if (typeof str2 !== "string")
+ return "";
+ return import_diacritics2.default.remove(
+ str2.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\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()
+ );
+ }
+
+ // node_modules/name-suggestion-index/config/matchGroups.json
+ var matchGroups_default = {
+ 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/beer",
+ "shop/beverages",
+ "shop/kiosk",
+ "shop/wine"
+ ],
+ camping: [
+ "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/laboratory",
+ "healthcare/physiotherapist",
+ "healthcare/sample_collection",
+ "healthcare/dialysis"
+ ],
+ convenience: [
+ "shop/beauty",
+ "shop/chemist",
+ "shop/convenience",
+ "shop/cosmetics",
+ "shop/grocery",
+ "shop/kiosk",
+ "shop/newsagent",
+ "shop/perfumery"
+ ],
+ 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/kiosk",
+ "shop/mobile",
+ "shop/mobile_phone",
+ "shop/telecommunication"
+ ],
+ fabric: [
+ "shop/fabric",
+ "shop/haberdashery",
+ "shop/sewing"
+ ],
+ fashion: [
+ "shop/accessories",
+ "shop/bag",
+ "shop/boutique",
+ "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/bar",
+ "amenity/cafe",
+ "amenity/fast_food",
+ "amenity/ice_cream",
+ "amenity/pub",
+ "amenity/restaurant",
+ "shop/bakery",
+ "shop/candy",
+ "shop/chocolate",
+ "shop/coffee",
+ "shop/confectionary",
+ "shop/confectionery",
+ "shop/food",
+ "shop/kiosk",
+ "shop/ice_cream",
+ "shop/pastry",
+ "shop/tea"
+ ],
+ fuel: [
+ "amenity/fuel",
+ "shop/gas",
+ "shop/convenience;gas",
+ "shop/gas;convenience"
+ ],
+ gift: [
+ "shop/gift",
+ "shop/card",
+ "shop/cards",
+ "shop/kiosk",
+ "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/clothes",
+ "shop/outdoor",
+ "shop/sports"
+ ],
+ parcel_locker: [
+ "amenity/parcel_locker",
+ "amenity/vending_machine"
+ ],
+ 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/kiosk",
+ "shop/rental"
+ ],
+ school: [
+ "amenity/childcare",
+ "amenity/college",
+ "amenity/kindergarten",
+ "amenity/language_school",
+ "amenity/prep_school",
+ "amenity/school",
+ "amenity/university"
+ ],
+ storage: [
+ "shop/storage_units",
+ "shop/storage_rental"
+ ],
+ substation: [
+ "power/station",
+ "power/substation",
+ "power/sub_station"
+ ],
+ 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/kiosk",
+ "shop/vending_machine"
+ ],
+ weight_loss: [
+ "amenity/clinic",
+ "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"
+ ]
+ }
+ };
+
+ // node_modules/name-suggestion-index/config/genericWords.json
+ var genericWords_default = {
+ 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 \xE1rbol|fountain|generic|golf|government|graveyard)$",
+ "^(fixme|n\\s?\\/?\\s?a|name|no\\s?name|none|null|temporary|test|unknown)$",
+ "^(hofladen|librairie|magazine?|maison)$",
+ "^(mobile home|skate)?\\s?park$",
+ "^(obuwie|pond|pool|sale|shops?|sklep|stores?)$",
+ "^\\?+$",
+ "^private$",
+ "^tattoo( studio)?$",
+ "^windmill$",
+ "^\u0446\u0435\u0440\u043A\u043E\u0432\u043D\u0430\u044F( \u043B\u0430\u0432\u043A\u0430)?$"
+ ]
+ };
+
+ // node_modules/name-suggestion-index/config/trees.json
+ var trees_default = {
+ trees: {
+ brands: {
+ emoji: "\u{1F354}",
+ 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: "\u{1F6A9}",
+ mainTag: "flag:wikidata",
+ nameTags: {
+ primary: "^(flag:name|flag:name:\\w+)$",
+ alternate: "^(country|country:\\w+|flag|flag:\\w+|subject|subject:\\w+)$"
+ }
+ },
+ operators: {
+ emoji: "\u{1F4BC}",
+ mainTag: "operator:wikidata",
+ sourceTags: ["operator"],
+ nameTags: {
+ primary: "^(name|name:\\w+|operator|operator:\\w+)$",
+ alternate: "^(brand|brand:\\w+|\\w+_name|\\w+_name:\\w+)$"
+ }
+ },
+ transit: {
+ emoji: "\u{1F687}",
+ mainTag: "network:wikidata",
+ sourceTags: ["network"],
+ nameTags: {
+ primary: "^network$",
+ alternate: "^(operator|operator:\\w+|network:\\w+|\\w+_name|\\w+_name:\\w+)$"
+ }
+ }
+ }
+ };
+
+ // node_modules/name-suggestion-index/lib/matcher.js
+ var matchGroups = matchGroups_default.matchGroups;
+ var trees = trees_default.trees;
+ var Matcher = class {
+ //
+ // `constructor`
+ // initialize the genericWords regexes
+ constructor() {
+ this.matchIndex = void 0;
+ this.genericWords = /* @__PURE__ */ new Map();
+ (genericWords_default.genericWords || []).forEach((s) => this.genericWords.set(s, new RegExp(s, "i")));
+ this.itemLocation = void 0;
+ this.locationSets = void 0;
+ this.locationIndex = void 0;
+ 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: [ {}, {}, … ] },
+ // …
+ // }
+ //
+ buildMatchIndex(data) {
+ const that = this;
+ if (that.matchIndex)
+ return;
+ that.matchIndex = /* @__PURE__ */ new Map();
+ const seenTree = /* @__PURE__ */ new Map();
+ Object.keys(data).forEach((tkv) => {
+ const category = data[tkv];
+ const parts = tkv.split("/", 3);
+ const t = parts[0];
+ const k = parts[1];
+ const v = parts[2];
+ const thiskv = `${k}/${v}`;
+ const tree = trees[t];
+ let branch = that.matchIndex.get(thiskv);
+ if (!branch) {
+ branch = {
+ primary: /* @__PURE__ */ new Map(),
+ alternate: /* @__PURE__ */ new Map(),
+ excludeGeneric: /* @__PURE__ */ new Map(),
+ excludeNamed: /* @__PURE__ */ new Map()
+ };
+ that.matchIndex.set(thiskv, branch);
+ }
+ const properties = category.properties || {};
+ const exclude = properties.exclude || {};
+ (exclude.generic || []).forEach((s) => branch.excludeGeneric.set(s, new RegExp(s, "i")));
+ (exclude.named || []).forEach((s) => branch.excludeNamed.set(s, new RegExp(s, "i")));
+ const excludeRegexes = [...branch.excludeGeneric.values(), ...branch.excludeNamed.values()];
+ let items = category.items;
+ if (!Array.isArray(items) || !items.length)
+ return;
+ const primaryName = new RegExp(tree.nameTags.primary, "i");
+ const alternateName = new RegExp(tree.nameTags.alternate, "i");
+ const notName = /:(colou?r|type|forward|backward|left|right|etymology|pronunciation|wikipedia)$/i;
+ const skipGenericKV = skipGenericKVMatches(t, k, v);
+ const genericKV = /* @__PURE__ */ new Set([`${k}/yes`, `building/yes`]);
+ const matchGroupKV = /* @__PURE__ */ new Set();
+ Object.values(matchGroups).forEach((matchGroup) => {
+ const inGroup = matchGroup.some((otherkv) => otherkv === thiskv);
+ if (!inGroup)
+ return;
+ matchGroup.forEach((otherkv) => {
+ if (otherkv === thiskv)
+ return;
+ matchGroupKV.add(otherkv);
+ const otherk = otherkv.split("/", 2)[0];
+ genericKV.add(`${otherk}/yes`);
+ });
+ });
+ items.forEach((item) => {
+ if (!item.id)
+ return;
+ if (Array.isArray(item.matchTags) && item.matchTags.length) {
+ item.matchTags = item.matchTags.filter((matchTag) => !matchGroupKV.has(matchTag) && !genericKV.has(matchTag));
+ if (!item.matchTags.length)
+ delete item.matchTags;
+ }
+ let kvTags = [`${thiskv}`].concat(item.matchTags || []);
+ if (!skipGenericKV) {
+ kvTags = kvTags.concat(Array.from(genericKV));
+ }
+ Object.keys(item.tags).forEach((osmkey) => {
+ if (notName.test(osmkey))
+ return;
+ const osmvalue = item.tags[osmkey];
+ if (!osmvalue || excludeRegexes.some((regex) => regex.test(osmvalue)))
+ return;
+ if (primaryName.test(osmkey)) {
+ kvTags.forEach((kv) => insertName("primary", t, kv, simplify(osmvalue), item.id));
+ } else if (alternateName.test(osmkey)) {
+ kvTags.forEach((kv) => insertName("alternate", t, kv, simplify(osmvalue), item.id));
+ }
+ });
+ let keepMatchNames = /* @__PURE__ */ new Set();
+ (item.matchNames || []).forEach((matchName) => {
+ const nsimple = simplify(matchName);
+ kvTags.forEach((kv) => {
+ const branch2 = that.matchIndex.get(kv);
+ const primaryLeaf = branch2 && branch2.primary.get(nsimple);
+ const alternateLeaf = branch2 && branch2.alternate.get(nsimple);
+ const inPrimary = primaryLeaf && primaryLeaf.has(item.id);
+ const inAlternate = alternateLeaf && alternateLeaf.has(item.id);
+ if (!inPrimary && !inAlternate) {
+ insertName("alternate", t, kv, nsimple, item.id);
+ keepMatchNames.add(matchName);
+ }
+ });
+ });
+ if (keepMatchNames.size) {
+ item.matchNames = Array.from(keepMatchNames);
+ } else {
+ delete item.matchNames;
+ }
+ });
+ });
+ function insertName(which, t, kv, nsimple, itemID) {
+ if (!nsimple) {
+ that.warnings.push(`Warning: skipping empty ${which} name for item ${t}/${kv}: ${itemID}`);
+ return;
+ }
+ let branch = that.matchIndex.get(kv);
+ if (!branch) {
+ branch = {
+ primary: /* @__PURE__ */ new Map(),
+ alternate: /* @__PURE__ */ new Map(),
+ excludeGeneric: /* @__PURE__ */ new Map(),
+ excludeNamed: /* @__PURE__ */ new Map()
+ };
+ that.matchIndex.set(kv, branch);
+ }
+ let leaf = branch[which].get(nsimple);
+ if (!leaf) {
+ leaf = /* @__PURE__ */ new Set();
+ branch[which].set(nsimple, leaf);
+ }
+ leaf.add(itemID);
+ if (!/yes$/.test(kv)) {
+ const kvnsimple = `${kv}/${nsimple}`;
+ const existing = seenTree.get(kvnsimple);
+ if (existing && existing !== t) {
+ const items = Array.from(leaf);
+ that.warnings.push(`Duplicate cache key "${kvnsimple}" in trees "${t}" and "${existing}", check items: ${items}`);
+ return;
+ }
+ seenTree.set(kvnsimple, t);
+ }
+ }
+ 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: [ {}, {}, … ] },
+ // …
+ // }
+ //
+ buildLocationIndex(data, loco) {
+ const that = this;
+ if (that.locationIndex)
+ return;
+ that.itemLocation = /* @__PURE__ */ new Map();
+ that.locationSets = /* @__PURE__ */ new Map();
+ Object.keys(data).forEach((tkv) => {
+ const items = data[tkv].items;
+ if (!Array.isArray(items) || !items.length)
+ return;
+ items.forEach((item) => {
+ if (that.itemLocation.has(item.id))
+ return;
+ let resolved;
+ try {
+ resolved = loco.resolveLocationSet(item.locationSet);
+ } catch (err) {
+ console.warn(`buildLocationIndex: ${err.message}`);
+ }
+ if (!resolved || !resolved.id)
+ return;
+ that.itemLocation.set(item.id, resolved.id);
+ if (that.locationSets.has(resolved.id))
+ return;
+ let feature3 = _cloneDeep2(resolved.feature);
+ feature3.id = resolved.id;
+ feature3.properties.id = resolved.id;
+ if (!feature3.geometry.coordinates.length || !feature3.properties.area) {
+ console.warn(`buildLocationIndex: locationSet ${resolved.id} for ${item.id} resolves to an empty feature:`);
+ console.warn(JSON.stringify(feature3));
+ return;
+ }
+ that.locationSets.set(resolved.id, feature3);
+ });
+ });
+ that.locationIndex = (0, import_which_polygon3.default)({ type: "FeatureCollection", features: [...that.locationSets.values()] });
+ function _cloneDeep2(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`
+ //
+ //
+ match(k, v, n2, loc) {
+ const that = this;
+ if (!that.matchIndex) {
+ throw new Error("match: matchIndex not built.");
+ }
+ let matchLocations;
+ if (Array.isArray(loc) && that.locationIndex) {
+ matchLocations = that.locationIndex([loc[0], loc[1], loc[0], loc[1]], true);
+ }
+ const nsimple = simplify(n2);
+ let seen = /* @__PURE__ */ new Set();
+ let results = [];
+ gatherResults("primary");
+ gatherResults("alternate");
+ if (results.length)
+ return results;
+ gatherResults("exclude");
+ return results.length ? results : null;
+ function gatherResults(which) {
+ const kv = `${k}/${v}`;
+ let didMatch = tryMatch(which, kv);
+ if (didMatch)
+ return;
+ for (let mg in matchGroups) {
+ const matchGroup = matchGroups[mg];
+ const inGroup = matchGroup.some((otherkv) => otherkv === kv);
+ if (!inGroup)
+ continue;
+ for (let i2 = 0; i2 < matchGroup.length; i2++) {
+ const otherkv = matchGroup[i2];
+ if (otherkv === kv)
+ continue;
+ didMatch = tryMatch(which, otherkv);
+ if (didMatch)
+ return;
+ }
+ }
+ if (which === "exclude") {
+ const regex = [...that.genericWords.values()].find((regex2) => regex2.test(n2));
+ if (regex) {
+ results.push({ match: "excludeGeneric", pattern: String(regex) });
+ return;
+ }
+ }
+ }
+ function tryMatch(which, kv) {
+ const branch = that.matchIndex.get(kv);
+ if (!branch)
+ return;
+ if (which === "exclude") {
+ let regex = [...branch.excludeNamed.values()].find((regex2) => regex2.test(n2));
+ if (regex) {
+ results.push({ match: "excludeNamed", pattern: String(regex), kv });
+ return;
+ }
+ regex = [...branch.excludeGeneric.values()].find((regex2) => regex2.test(n2));
+ if (regex) {
+ results.push({ match: "excludeGeneric", pattern: String(regex), kv });
+ return;
+ }
+ return;
+ }
+ const leaf = branch[which].get(nsimple);
+ if (!leaf || !leaf.size)
+ return;
+ let hits = Array.from(leaf).map((itemID) => {
+ let area = Infinity;
+ if (that.itemLocation && that.locationSets) {
+ const location = that.locationSets.get(that.itemLocation.get(itemID));
+ area = location && location.properties.area || Infinity;
+ }
+ return { match: which, itemID, area, kv, nsimple };
+ });
+ let sortFn = byAreaDescending;
+ if (matchLocations) {
+ hits = hits.filter(isValidLocation);
+ sortFn = byAreaAscending;
+ }
+ if (!hits.length)
+ return;
+ hits.sort(sortFn).forEach((hit) => {
+ if (seen.has(hit.itemID))
+ return;
+ seen.add(hit.itemID);
+ results.push(hit);
+ });
+ return true;
+ function isValidLocation(hit) {
+ if (!that.itemLocation)
+ return true;
+ return matchLocations.find((props) => props.id === that.itemLocation.get(hit.itemID));
+ }
+ function byAreaAscending(hitA, hitB) {
+ return hitA.area - hitB.area;
+ }
+ function byAreaDescending(hitA, hitB) {
+ return hitB.area - hitA.area;
+ }
+ }
+ }
+ //
+ // `getWarnings()`
+ // Return any warnings discovered when buiding the index.
+ // (currently this does nothing)
+ //
+ getWarnings() {
+ return this.warnings;
+ }
+ };
+
+ // modules/services/nsi.js
+ var import_vparse2 = __toESM(require_vparse());
+
+ // modules/core/difference.js
+ var import_fast_deep_equal3 = __toESM(require_fast_deep_equal());
+ function coreDifference(base, head) {
+ var _changes = {};
+ var _didChange = {};
+ var _diff = {};
+ function checkEntityID(id2) {
+ var h = head.entities[id2];
+ var b = base.entities[id2];
+ if (h === b)
+ return;
+ if (_changes[id2])
+ return;
+ if (!h && b) {
+ _changes[id2] = { base: b, head: h };
+ _didChange.deletion = true;
+ return;
+ }
+ if (h && !b) {
+ _changes[id2] = { base: b, head: h };
+ _didChange.addition = true;
+ return;
+ }
+ if (h && b) {
+ if (h.members && b.members && !(0, import_fast_deep_equal3.default)(h.members, b.members)) {
+ _changes[id2] = { base: b, head: h };
+ _didChange.geometry = true;
+ _didChange.properties = true;
+ return;
+ }
+ if (h.loc && b.loc && !geoVecEqual(h.loc, b.loc)) {
+ _changes[id2] = { base: b, head: h };
+ _didChange.geometry = true;
+ }
+ if (h.nodes && b.nodes && !(0, import_fast_deep_equal3.default)(h.nodes, b.nodes)) {
+ _changes[id2] = { base: b, head: h };
+ _didChange.geometry = true;
+ }
+ if (h.tags && b.tags && !(0, import_fast_deep_equal3.default)(h.tags, b.tags)) {
+ _changes[id2] = { base: b, head: h };
+ _didChange.properties = true;
+ }
+ }
+ }
+ function load() {
+ var ids = utilArrayUniq(Object.keys(head.entities).concat(Object.keys(base.entities)));
+ for (var i2 = 0; i2 < ids.length; i2++) {
+ checkEntityID(ids[i2]);
+ }
+ }
+ load();
+ _diff.length = function length() {
+ return Object.keys(_changes).length;
+ };
+ _diff.changes = function changes() {
+ return _changes;
+ };
+ _diff.didChange = _didChange;
+ _diff.extantIDs = function extantIDs(includeRelMembers) {
+ var result = /* @__PURE__ */ new Set();
+ Object.keys(_changes).forEach(function(id2) {
+ if (_changes[id2].head) {
+ result.add(id2);
+ }
+ var h = _changes[id2].head;
+ var b = _changes[id2].base;
+ var entity = h || b;
+ 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);
+ };
+ _diff.modified = function modified() {
+ var result = [];
+ Object.values(_changes).forEach(function(change) {
+ if (change.base && change.head) {
+ result.push(change.head);
+ }
+ });
+ return result;
+ };
+ _diff.created = function created() {
+ var result = [];
+ Object.values(_changes).forEach(function(change) {
+ if (!change.base && change.head) {
+ result.push(change.head);
+ }
+ });
+ return result;
+ };
+ _diff.deleted = function deleted() {
+ var result = [];
+ Object.values(_changes).forEach(function(change) {
+ if (change.base && !change.head) {
+ result.push(change.base);
+ }
+ });
+ return result;
+ };
+ _diff.summary = function summary() {
+ var relevant = {};
+ var keys = Object.keys(_changes);
+ for (var i2 = 0; i2 < keys.length; i2++) {
+ var change = _changes[keys[i2]];
+ 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) {
+ var moved = !(0, import_fast_deep_equal3.default)(change.base.loc, change.head.loc);
+ var retagged = !(0, import_fast_deep_equal3.default)(change.base.tags, change.head.tags);
+ if (moved) {
+ addParents(change.head);
+ }
+ if (retagged || moved && change.head.hasInterestingTags()) {
+ addEntity(change.head, head, "modified");
+ }
+ } else if (change.head && change.head.hasInterestingTags()) {
+ addEntity(change.head, head, "created");
+ } else if (change.base && change.base.hasInterestingTags()) {
+ addEntity(change.base, base, "deleted");
+ }
+ }
+ return Object.values(relevant);
+ function addEntity(entity, graph, changeType) {
+ relevant[entity.id] = {
+ entity,
+ graph,
+ changeType
+ };
+ }
+ function addParents(entity) {
+ var parents = head.parentWays(entity);
+ for (var j2 = parents.length - 1; j2 >= 0; j2--) {
+ var parent = parents[j2];
+ if (!(parent.id in relevant)) {
+ addEntity(parent, head, "modified");
+ }
+ }
+ }
+ };
+ _diff.complete = function complete(extent) {
+ var result = {};
+ var id2, change;
+ for (id2 in _changes) {
+ change = _changes[id2];
+ var h = change.head;
+ var b = change.base;
+ var entity = h || b;
+ var i2;
+ if (extent && (!h || !h.intersects(extent, head)) && (!b || !b.intersects(extent, base))) {
+ continue;
+ }
+ result[id2] = h;
+ if (entity.type === "way") {
+ var nh = h ? h.nodes : [];
+ var nb = b ? b.nodes : [];
+ var diff;
+ diff = utilArrayDifference(nh, nb);
+ for (i2 = 0; i2 < diff.length; i2++) {
+ result[diff[i2]] = head.hasEntity(diff[i2]);
+ }
+ diff = utilArrayDifference(nb, nh);
+ for (i2 = 0; i2 < diff.length; i2++) {
+ result[diff[i2]] = head.hasEntity(diff[i2]);
+ }
+ }
+ 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 (i2 = 0; i2 < ids.length; i2++) {
+ var member = head.hasEntity(ids[i2]);
+ if (!member)
+ continue;
+ if (extent && !member.intersects(extent, head))
+ continue;
+ result[ids[i2]] = member;
+ }
+ }
+ addParents(head.parentWays(entity), result);
+ addParents(head.parentRelations(entity), result);
+ }
+ return result;
+ function addParents(parents, result2) {
+ for (var i3 = 0; i3 < parents.length; i3++) {
+ var parent = parents[i3];
+ if (parent.id in result2)
+ continue;
+ result2[parent.id] = parent;
+ addParents(head.parentRelations(parent), result2);
+ }
+ }
+ };
+ return _diff;
+ }
+
+ // modules/core/tree.js
+ var import_rbush6 = __toESM(require_rbush_min());
+ function coreTree(head) {
+ var _rtree = new import_rbush6.default();
+ var _bboxes = {};
+ var _segmentsRTree = new import_rbush6.default();
+ var _segmentsBBoxes = {};
+ var _segmentsByWayId = {};
+ var tree = {};
+ function entityBBox(entity) {
+ var bbox2 = entity.extent(head).bbox();
+ bbox2.id = entity.id;
+ _bboxes[entity.id] = bbox2;
+ return bbox2;
+ }
+ function segmentBBox(segment) {
+ var extent = segment.extent(head);
+ if (!extent)
+ return null;
+ var bbox2 = extent.bbox();
+ bbox2.segment = segment;
+ _segmentsBBoxes[segment.id] = bbox2;
+ return bbox2;
+ }
+ function removeEntity(entity) {
+ _rtree.remove(_bboxes[entity.id]);
+ delete _bboxes[entity.id];
+ if (_segmentsByWayId[entity.id]) {
+ _segmentsByWayId[entity.id].forEach(function(segment) {
+ _segmentsRTree.remove(_segmentsBBoxes[segment.id]);
+ delete _segmentsBBoxes[segment.id];
+ });
+ delete _segmentsByWayId[entity.id];
+ }
+ }
+ function loadEntities(entities) {
+ _rtree.load(entities.map(entityBBox));
+ var segments = [];
+ entities.forEach(function(entity) {
+ if (entity.segments) {
+ var entitySegments = entity.segments(head);
+ _segmentsByWayId[entity.id] = entitySegments;
+ segments = segments.concat(entitySegments);
+ }
+ });
+ if (segments.length)
+ _segmentsRTree.load(segments.map(segmentBBox).filter(Boolean));
+ }
+ 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 i2 = 0; i2 < entities.length; i2++) {
+ var entity = entities[i2];
+ 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));
+ }
+ tree.intersects = function(extent, graph) {
+ updateToGraph(graph);
+ return _rtree.search(extent.bbox()).map(function(bbox2) {
+ return graph.entity(bbox2.id);
+ });
+ };
+ tree.waySegments = function(extent, graph) {
+ updateToGraph(graph);
+ return _segmentsRTree.search(extent.bbox()).map(function(bbox2) {
+ return bbox2.segment;
+ });
+ };
+ return tree;
+ }
+
+ // modules/svg/icon.js
+ function svgIcon(name, svgklass, useklass) {
+ return function drawIcon(selection2) {
+ selection2.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);
+ };
+ }
+
+ // modules/ui/modal.js
+ function uiModal(selection2, blocking) {
+ let keybinding = utilKeybinding("modal");
+ let previous = selection2.select("div.modal");
+ let animate = previous.empty();
+ previous.transition().duration(200).style("opacity", 0).remove();
+ let shaded = selection2.append("div").attr("class", "shaded").style("opacity", 0);
+ shaded.close = () => {
+ shaded.transition().duration(200).style("opacity", 0).remove();
+ modal.transition().duration(200).style("top", "0px");
+ select_default2(document).call(keybinding.unbind);
+ };
+ let 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", (d3_event) => {
+ if (d3_event.target === this) {
+ shaded.close();
+ }
+ });
+ modal.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", shaded.close).call(svgIcon("#iD-icon-close"));
+ keybinding.on("\u232B", shaded.close).on("\u238B", shaded.close);
+ select_default2(document).call(keybinding);
+ }
+ 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 shaded;
+ function moveFocusToFirst() {
+ let node = modal.select("a, button, input:not(.keytrap), select, textarea").node();
+ if (node) {
+ node.focus();
+ } else {
+ select_default2(this).node().blur();
+ }
+ }
+ function moveFocusToLast() {
+ let nodes = modal.selectAll("a, button, input:not(.keytrap), select, textarea").nodes();
+ if (nodes.length) {
+ nodes[nodes.length - 1].focus();
+ } else {
+ select_default2(this).node().blur();
+ }
+ }
+ }
+
+ // modules/ui/loading.js
+ function uiLoading(context) {
+ let _modalSelection = select_default2(null);
+ let _message = "";
+ let _blocking = false;
+ let loading = (selection2) => {
+ _modalSelection = uiModal(selection2, _blocking);
+ let loadertext = _modalSelection.select(".content").classed("loading-modal", true).append("div").attr("class", "modal-section fillL");
+ 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 = () => {
+ _modalSelection.remove();
+ };
+ loading.isShown = () => {
+ return _modalSelection && !_modalSelection.empty() && _modalSelection.node().parentNode;
+ };
+ return loading;
+ }
+
+ // modules/core/history.js
+ function coreHistory(context) {
+ var dispatch10 = dispatch_default("reset", "change", "merge", "restore", "undone", "redone", "storage_error");
+ var lock = utilSessionMutex("lock");
+ var _hasUnresolvedRestorableChanges = lock.lock() && !!corePreferences(getKey("saved_history"));
+ var duration = 150;
+ var _imageryUsed = [];
+ var _photoOverlaysUsed = [];
+ var _checkpoints = {};
+ var _pausedGraph;
+ var _stack;
+ var _index;
+ var _tree;
+ function _act(actions, t) {
+ actions = Array.prototype.slice.call(actions);
+ var annotation;
+ if (typeof actions[actions.length - 1] !== "function") {
+ annotation = actions.pop();
+ }
+ var graph = _stack[_index].graph;
+ for (var i2 = 0; i2 < actions.length; i2++) {
+ graph = actions[i2](graph, t);
+ }
+ return {
+ graph,
+ annotation,
+ imageryUsed: _imageryUsed,
+ photoOverlaysUsed: _photoOverlaysUsed,
+ transform: context.projection.transform(),
+ selectedIDs: context.selectedIDs()
+ };
+ }
+ function _perform(args, t) {
+ var previous = _stack[_index].graph;
+ _stack = _stack.slice(0, _index + 1);
+ var actionResult = _act(args, t);
+ _stack.push(actionResult);
+ _index++;
+ return change(previous);
+ }
+ function _replace(args, t) {
+ var previous = _stack[_index].graph;
+ var actionResult = _act(args, t);
+ _stack[_index] = actionResult;
+ return change(previous);
+ }
+ 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);
+ }
+ function change(previous) {
+ var difference = coreDifference(previous, history.graph());
+ if (!_pausedGraph) {
+ dispatch10.call("change", this, difference);
+ }
+ return difference;
+ }
+ function getKey(n2) {
+ return "iD_" + window.location.origin + "_" + n2;
+ }
+ var history = {
+ graph: function() {
+ return _stack[_index].graph;
+ },
+ tree: function() {
+ return _tree;
+ },
+ base: function() {
+ return _stack[0].graph;
+ },
+ merge: function(entities) {
+ var stack = _stack.map(function(state) {
+ return state.graph;
+ });
+ _stack[0].graph.rebase(entities, stack, false);
+ _tree.rebase(entities, false);
+ dispatch10.call("merge", this, entities);
+ },
+ perform: function() {
+ select_default2(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;
+ }
+ if (transitionable) {
+ var origArguments = arguments;
+ select_default2(document).transition("history.perform").duration(duration).ease(linear2).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() {
+ select_default2(document).interrupt("history.perform");
+ return _replace(arguments, 1);
+ },
+ // Same as calling pop and then perform
+ overwrite: function() {
+ select_default2(document).interrupt("history.perform");
+ return _overwrite(arguments, 1);
+ },
+ pop: function(n2) {
+ select_default2(document).interrupt("history.perform");
+ var previous = _stack[_index].graph;
+ if (isNaN(+n2) || +n2 < 0) {
+ n2 = 1;
+ }
+ while (n2-- > 0 && _index > 0) {
+ _index--;
+ _stack.pop();
+ }
+ return change(previous);
+ },
+ // Back to the previous annotated state or _index = 0.
+ undo: function() {
+ select_default2(document).interrupt("history.perform");
+ var previousStack = _stack[_index];
+ var previous = previousStack.graph;
+ while (_index > 0) {
+ _index--;
+ if (_stack[_index].annotation)
+ break;
+ }
+ dispatch10.call("undone", this, _stack[_index], previousStack);
+ return change(previous);
+ },
+ // Forward to the next annotated state.
+ redo: function() {
+ select_default2(document).interrupt("history.perform");
+ var previousStack = _stack[_index];
+ var previous = previousStack.graph;
+ var tryIndex = _index;
+ while (tryIndex < _stack.length - 1) {
+ tryIndex++;
+ if (_stack[tryIndex].annotation) {
+ _index = tryIndex;
+ dispatch10.call("redone", this, _stack[_index], previousStack);
+ break;
+ }
+ }
+ return change(previous);
+ },
+ pauseChangeDispatch: function() {
+ if (!_pausedGraph) {
+ _pausedGraph = _stack[_index].graph;
+ }
+ },
+ resumeChangeDispatch: function() {
+ if (_pausedGraph) {
+ var previous = _pausedGraph;
+ _pausedGraph = null;
+ return change(previous);
+ }
+ },
+ undoAnnotation: function() {
+ var i2 = _index;
+ while (i2 >= 0) {
+ if (_stack[i2].annotation)
+ return _stack[i2].annotation;
+ i2--;
+ }
+ },
+ redoAnnotation: function() {
+ var i2 = _index + 1;
+ while (i2 <= _stack.length - 1) {
+ if (_stack[i2].annotation)
+ return _stack[i2].annotation;
+ i2++;
+ }
+ },
+ // Returns the entities from the active graph with bounding boxes
+ // overlapping the given `extent`.
+ intersects: function(extent) {
+ return _tree.intersects(extent, _stack[_index].graph);
+ },
+ difference: function() {
+ var base = _stack[0].graph;
+ var head = _stack[_index].graph;
+ return coreDifference(base, head);
+ },
+ changes: function(action) {
+ var base = _stack[0].graph;
+ var head = _stack[_index].graph;
+ if (action) {
+ head = action(head);
+ }
+ var difference = coreDifference(base, head);
+ return {
+ modified: difference.modified(),
+ created: difference.created(),
+ deleted: difference.deleted()
+ };
+ },
+ hasChanges: function() {
+ return this.difference().length() > 0;
+ },
+ imageryUsed: function(sources) {
+ if (sources) {
+ _imageryUsed = sources;
+ return history;
+ } else {
+ var s = /* @__PURE__ */ 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(sources) {
+ if (sources) {
+ _photoOverlaysUsed = sources;
+ return history;
+ } else {
+ var s = /* @__PURE__ */ new Set();
+ _stack.slice(1, _index + 1).forEach(function(state) {
+ if (state.photoOverlaysUsed && Array.isArray(state.photoOverlaysUsed)) {
+ state.photoOverlaysUsed.forEach(function(photoOverlay) {
+ s.add(photoOverlay);
+ });
+ }
+ });
+ return Array.from(s);
+ }
+ },
+ // save the current history state
+ checkpoint: function(key) {
+ _checkpoints[key] = {
+ stack: _stack,
+ index: _index
+ };
+ return history;
+ },
+ // restore history state to a given checkpoint or reset completely
+ reset: function(key) {
+ if (key !== void 0 && _checkpoints.hasOwnProperty(key)) {
+ _stack = _checkpoints[key].stack;
+ _index = _checkpoints[key].index;
+ } else {
+ _stack = [{ graph: coreGraph() }];
+ _index = 0;
+ _tree = coreTree(_stack[0].graph);
+ _checkpoints = {};
+ }
+ dispatch10.call("reset");
+ dispatch10.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() {
+ var nextID = { n: 0, r: 0, w: 0 };
+ var permIDs = {};
+ var graph = this.graph();
+ var baseEntities = {};
+ Object.values(graph.base().entities).forEach(function(entity) {
+ var copy2 = copyIntroEntity(entity);
+ baseEntities[copy2.id] = copy2;
+ });
+ Object.keys(graph.entities).forEach(function(id2) {
+ var entity = graph.entities[id2];
+ if (entity) {
+ var copy2 = copyIntroEntity(entity);
+ baseEntities[copy2.id] = copy2;
+ } else {
+ delete baseEntities[id2];
+ }
+ });
+ Object.values(baseEntities).forEach(function(entity) {
+ if (Array.isArray(entity.nodes)) {
+ entity.nodes = entity.nodes.map(function(node) {
+ return permIDs[node] || node;
+ });
+ }
+ 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 });
+ function copyIntroEntity(source) {
+ var copy2 = utilObjectOmit(source, ["type", "user", "v", "version", "visible"]);
+ if (copy2.tags && !Object.keys(copy2.tags)) {
+ delete copy2.tags;
+ }
+ if (Array.isArray(copy2.loc)) {
+ copy2.loc[0] = +copy2.loc[0].toFixed(6);
+ copy2.loc[1] = +copy2.loc[1].toFixed(6);
+ }
+ var match = source.id.match(/([nrw])-\d*/);
+ if (match !== null) {
+ var nrw = match[1];
+ var permID;
+ do {
+ permID = nrw + ++nextID[nrw];
+ } while (baseEntities.hasOwnProperty(permID));
+ copy2.id = permIDs[source.id] = permID;
+ }
+ return copy2;
+ }
+ },
+ toJSON: function() {
+ if (!this.hasChanges())
+ return;
+ var allEntities = {};
+ var baseEntities = {};
+ var base = _stack[0];
+ var s = _stack.map(function(i2) {
+ var modified = [];
+ var deleted = [];
+ Object.keys(i2.graph.entities).forEach(function(id2) {
+ var entity = i2.graph.entities[id2];
+ if (entity) {
+ var key = osmEntity.key(entity);
+ allEntities[key] = entity;
+ modified.push(key);
+ } else {
+ deleted.push(id2);
+ }
+ if (id2 in base.graph.entities) {
+ baseEntities[id2] = base.graph.entities[id2];
+ }
+ if (entity && entity.nodes) {
+ entity.nodes.forEach(function(nodeID) {
+ if (nodeID in base.graph.entities) {
+ baseEntities[nodeID] = base.graph.entities[nodeID];
+ }
+ });
+ }
+ var baseParents = base.graph._parentWays[id2];
+ 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 (i2.imageryUsed)
+ x.imageryUsed = i2.imageryUsed;
+ if (i2.photoOverlaysUsed)
+ x.photoOverlaysUsed = i2.photoOverlaysUsed;
+ if (i2.annotation)
+ x.annotation = i2.annotation;
+ if (i2.transform)
+ x.transform = i2.transform;
+ if (i2.selectedIDs)
+ x.selectedIDs = i2.selectedIDs;
+ return x;
+ });
+ 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(json, loadChildNodes) {
+ var h = JSON.parse(json);
+ var loadComplete = true;
+ osmEntity.id.next = h.nextIDs;
+ _index = h.index;
+ if (h.version === 2 || h.version === 3) {
+ var allEntities = {};
+ h.entities.forEach(function(entity) {
+ allEntities[osmEntity.key(entity)] = osmEntity(entity);
+ });
+ if (h.version === 3) {
+ var baseEntities = h.baseEntities.map(function(d) {
+ return osmEntity(d);
+ });
+ var stack = _stack.map(function(state) {
+ return state.graph;
+ });
+ _stack[0].graph.rebase(baseEntities, stack, true);
+ _tree.rebase(baseEntities, true);
+ 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(n2) {
+ return !_stack[0].graph.hasEntity(n2);
+ });
+ if (missing.length && osm) {
+ loadComplete = false;
+ context.map().redrawEnable(false);
+ var loading = uiLoading(context).blocking(true);
+ context.container().call(loading);
+ var childNodesLoaded = function(err, result) {
+ if (!err) {
+ var visibleGroups = utilArrayGroupBy(result.data, "visible");
+ var visibles = visibleGroups.true || [];
+ var invisibles = visibleGroups.false || [];
+ if (visibles.length) {
+ var visibleIDs = visibles.map(function(entity) {
+ return entity.id;
+ });
+ var stack2 = _stack.map(function(state) {
+ return state.graph;
+ });
+ missing = utilArrayDifference(missing, visibleIDs);
+ _stack[0].graph.rebase(visibles, stack2, true);
+ _tree.rebase(visibles, true);
+ }
+ invisibles.forEach(function(entity) {
+ osm.loadEntityVersion(entity.id, +entity.version - 1, childNodesLoaded);
+ });
+ }
+ if (err || !missing.length) {
+ loading.close();
+ context.map().redrawEnable(true);
+ dispatch10.call("change");
+ dispatch10.call("restore", this);
+ }
+ };
+ osm.loadMultiple(missing, childNodesLoaded);
+ }
+ }
+ }
+ _stack = h.stack.map(function(d) {
+ var entities = {}, entity;
+ if (d.modified) {
+ d.modified.forEach(function(key) {
+ entity = allEntities[key];
+ entities[entity.id] = entity;
+ });
+ }
+ if (d.deleted) {
+ d.deleted.forEach(function(id2) {
+ entities[id2] = void 0;
+ });
+ }
+ return {
+ graph: coreGraph(_stack[0].graph).load(entities),
+ annotation: d.annotation,
+ imageryUsed: d.imageryUsed,
+ photoOverlaysUsed: d.photoOverlaysUsed,
+ transform: d.transform,
+ selectedIDs: d.selectedIDs
+ };
+ });
+ } else {
+ _stack = h.stack.map(function(d) {
+ var entities = {};
+ for (var i2 in d.entities) {
+ var entity = d.entities[i2];
+ entities[i2] = entity === "undefined" ? void 0 : osmEntity(entity);
+ }
+ d.graph = coreGraph(_stack[0].graph).load(entities);
+ return d;
+ });
+ }
+ var transform2 = _stack[_index].transform;
+ if (transform2) {
+ context.map().transformEase(transform2, 0);
+ }
+ if (loadComplete) {
+ dispatch10.call("change");
+ dispatch10.call("restore", this);
+ }
+ return history;
+ },
+ lock: function() {
+ return lock.lock();
+ },
+ unlock: function() {
+ lock.unlock();
+ },
+ save: function() {
+ if (lock.locked() && // don't overwrite existing, unresolved changes
+ !_hasUnresolvedRestorableChanges) {
+ const success = corePreferences(getKey("saved_history"), history.toJSON() || null);
+ if (!success)
+ dispatch10.call("storage_error");
+ }
+ return history;
+ },
+ // delete the history version saved in localStorage
+ clearSaved: function() {
+ context.debouncedSave.cancel();
+ if (lock.locked()) {
+ _hasUnresolvedRestorableChanges = false;
+ corePreferences(getKey("saved_history"), null);
+ corePreferences("comment", null);
+ corePreferences("hashtags", null);
+ corePreferences("source", null);
+ }
+ return history;
+ },
+ savedHistoryJSON: function() {
+ return corePreferences(getKey("saved_history"));
+ },
+ hasRestorableChanges: function() {
+ return _hasUnresolvedRestorableChanges;
+ },
+ // load history from a version stored in localStorage
+ restore: function() {
+ if (lock.locked()) {
+ _hasUnresolvedRestorableChanges = false;
+ var json = this.savedHistoryJSON();
+ if (json)
+ history.fromJSON(json, true);
+ }
+ },
+ _getKey: getKey
+ };
+ history.reset();
+ return utilRebind(history, dispatch10, "on");
+ }
+
+ // modules/validations/index.js
+ var validations_exports = {};
+ __export(validations_exports, {
+ 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
+ });
+
+ // modules/validations/almost_junction.js
+ function validationAlmostJunction(context) {
+ const type2 = "almost_junction";
+ const EXTEND_TH_METERS = 5;
+ const WELD_TH_METERS = 0.75;
+ const CLOSE_NODE_TH = EXTEND_TH_METERS - WELD_TH_METERS;
+ const SIG_ANGLE_TH = Math.atan(WELD_TH_METERS / EXTEND_TH_METERS);
+ function isHighway(entity) {
+ return entity.type === "way" && osmRoutableHighwayTagValues[entity.tags.highway];
+ }
+ function isTaggedAsNotContinuing(node) {
+ return node.tags.noexit === "yes" || node.tags.amenity === "parking_entrance" || node.tags.entrance && node.tags.entrance !== "no";
+ }
+ const validation = function checkAlmostJunction(entity, graph) {
+ if (!isHighway(entity))
+ return [];
+ if (entity.isDegenerate())
+ return [];
+ const tree = context.history().tree();
+ const extendableNodeInfos = findConnectableEndNodesByExtension(entity);
+ let issues = [];
+ extendableNodeInfos.forEach((extendableNodeInfo) => {
+ issues.push(new validationIssue({
+ type: type2,
+ subtype: "highway-highway",
+ severity: "warning",
+ message: function(context2) {
+ const entity1 = context2.hasEntity(this.entityIds[0]);
+ if (this.entityIds[0] === this.entityIds[2]) {
+ return entity1 ? _t.append("issues.almost_junction.self.message", {
+ feature: utilDisplayLabel(entity1, context2.graph())
+ }) : "";
+ } else {
+ const entity2 = context2.hasEntity(this.entityIds[2]);
+ return entity1 && entity2 ? _t.append("issues.almost_junction.message", {
+ feature: utilDisplayLabel(entity1, context2.graph()),
+ feature2: utilDisplayLabel(entity2, context2.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;
+ function makeFixes(context2) {
+ let fixes = [new validationIssueFix({
+ icon: "iD-icon-abutment",
+ title: _t.append("issues.fix.connect_features.title"),
+ onClick: function(context3) {
+ const annotation = _t("issues.fix.connect_almost_junction.annotation");
+ const [, endNodeId, crossWayId] = this.issue.entityIds;
+ const midNode = context3.entity(this.issue.data.midId);
+ const endNode = context3.entity(endNodeId);
+ const crossWay = context3.entity(crossWayId);
+ const nearEndNodes = findNearbyEndNodes(endNode, crossWay);
+ if (nearEndNodes.length > 0) {
+ const collinear = findSmallJoinAngle(midNode, endNode, nearEndNodes);
+ if (collinear) {
+ context3.perform(
+ actionMergeNodes([collinear.id, endNode.id], collinear.loc),
+ annotation
+ );
+ return;
+ }
+ }
+ const targetEdge = this.issue.data.edge;
+ const crossLoc = this.issue.data.cross_loc;
+ const edgeNodes = [context3.entity(targetEdge[0]), context3.entity(targetEdge[1])];
+ const closestNodeInfo = geoSphericalClosestNode(edgeNodes, crossLoc);
+ if (closestNodeInfo.distance < WELD_TH_METERS) {
+ context3.perform(
+ actionMergeNodes([closestNodeInfo.node.id, endNode.id], closestNodeInfo.node.loc),
+ annotation
+ );
+ } else {
+ context3.perform(
+ actionAddMidpoint({ loc: crossLoc, edge: targetEdge }, endNode),
+ annotation
+ );
+ }
+ }
+ })];
+ const node = context2.hasEntity(this.entityIds[1]);
+ if (node && !node.hasInterestingTags()) {
+ fixes.push(new validationIssueFix({
+ icon: "maki-barrier",
+ title: _t.append("issues.fix.tag_as_disconnected.title"),
+ onClick: function(context3) {
+ const nodeID = this.issue.entityIds[1];
+ const tags = Object.assign({}, context3.entity(nodeID).tags);
+ tags.noexit = "yes";
+ context3.perform(
+ actionChangeTags(nodeID, tags),
+ _t("issues.fix.tag_as_disconnected.annotation")
+ );
+ }
+ }));
+ }
+ return fixes;
+ }
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.almost_junction.highway-highway.reference"));
+ }
+ function isExtendableCandidate(node, way) {
+ const osm = services.osm;
+ if (osm && !osm.isDataLoaded(node.loc)) {
+ return false;
+ }
+ if (isTaggedAsNotContinuing(node) || graph.parentWays(node).length !== 1) {
+ return false;
+ }
+ let occurrences = 0;
+ for (const index in way.nodes) {
+ if (way.nodes[index] === node.id) {
+ occurrences += 1;
+ if (occurrences > 1) {
+ return false;
+ }
+ }
+ }
+ return true;
+ }
+ function findConnectableEndNodesByExtension(way) {
+ let results = [];
+ if (way.isClosed())
+ return results;
+ let testNodes;
+ const indices = [0, way.nodes.length - 1];
+ indices.forEach((nodeIndex) => {
+ const nodeID = way.nodes[nodeIndex];
+ const node = graph.entity(nodeID);
+ if (!isExtendableCandidate(node, way))
+ return;
+ const connectionInfo = canConnectByExtend(way, nodeIndex);
+ if (!connectionInfo)
+ return;
+ testNodes = graph.childNodes(way).slice();
+ testNodes[nodeIndex] = testNodes[nodeIndex].move(connectionInfo.cross_loc);
+ if (geoHasSelfIntersections(testNodes, nodeID))
+ return;
+ results.push(connectionInfo);
+ });
+ return results;
+ }
+ function findNearbyEndNodes(node, way) {
+ return [
+ way.nodes[0],
+ way.nodes[way.nodes.length - 1]
+ ].map((d) => graph.entity(d)).filter((d) => {
+ return d.id !== node.id && geoSphericalDistance(node.loc, d.loc) <= CLOSE_NODE_TH;
+ });
+ }
+ function findSmallJoinAngle(midNode, tipNode, endNodes) {
+ let joinTo;
+ let minAngle = Infinity;
+ endNodes.forEach((endNode) => {
+ const a1 = geoAngle(midNode, tipNode, context.projection) + Math.PI;
+ const a2 = geoAngle(midNode, endNode, context.projection) + Math.PI;
+ const diff = Math.max(a1, a2) - Math.min(a1, a2);
+ if (diff < minAngle) {
+ joinTo = endNode;
+ minAngle = diff;
+ }
+ });
+ if (minAngle <= SIG_ANGLE_TH)
+ return joinTo;
+ return null;
+ }
+ function hasTag(tags, key) {
+ return tags[key] !== void 0 && tags[key] !== "no";
+ }
+ function canConnectWays(way, way2) {
+ if (way.id === way2.id)
+ return true;
+ 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;
+ const layer1 = way.tags.layer || "0", layer2 = way2.tags.layer || "0";
+ if (layer1 !== layer2)
+ return false;
+ const level1 = way.tags.level || "0", level2 = way2.tags.level || "0";
+ if (level1 !== level2)
+ return false;
+ return true;
+ }
+ function canConnectByExtend(way, endNodeIdx) {
+ const tipNid = way.nodes[endNodeIdx];
+ const midNid = endNodeIdx === 0 ? way.nodes[1] : way.nodes[way.nodes.length - 2];
+ const tipNode = graph.entity(tipNid);
+ const midNode = graph.entity(midNid);
+ const lon = tipNode.loc[0];
+ const lat = tipNode.loc[1];
+ const lon_range = geoMetersToLon(EXTEND_TH_METERS, lat) / 2;
+ const lat_range = geoMetersToLat(EXTEND_TH_METERS) / 2;
+ const queryExtent = geoExtent([
+ [lon - lon_range, lat - lat_range],
+ [lon + lon_range, lat + lat_range]
+ ]);
+ const edgeLen = geoSphericalDistance(midNode.loc, tipNode.loc);
+ const t = EXTEND_TH_METERS / edgeLen + 1;
+ const extTipLoc = geoVecInterp(midNode.loc, tipNode.loc, t);
+ const segmentInfos = tree.waySegments(queryExtent, graph);
+ for (let i2 = 0; i2 < segmentInfos.length; i2++) {
+ let segmentInfo = segmentInfos[i2];
+ let way2 = graph.entity(segmentInfo.wayId);
+ if (!isHighway(way2))
+ continue;
+ if (!canConnectWays(way, way2))
+ continue;
+ let nAid = segmentInfo.nodes[0], nBid = segmentInfo.nodes[1];
+ if (nAid === tipNid || nBid === tipNid)
+ continue;
+ let nA = graph.entity(nAid), nB = graph.entity(nBid);
+ let 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;
+ }
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/close_nodes.js
+ function validationCloseNodes(context) {
+ var type2 = "close_nodes";
+ var pointThresholdMeters = 0.2;
+ var validation = function(entity, graph) {
+ if (entity.type === "node") {
+ return getIssuesForNode(entity);
+ } else if (entity.type === "way") {
+ return getIssuesForWay(entity);
+ }
+ return [];
+ function getIssuesForNode(node) {
+ var parentWays = graph.parentWays(node);
+ if (parentWays.length) {
+ return getIssuesForVertex(node, parentWays);
+ } else {
+ return getIssuesForDetachedPoint(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);
+ for (var i2 in parentRelations) {
+ var relation = parentRelations[i2];
+ if (relation.tags.type === "boundary")
+ return "boundary";
+ 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";
+ }
+ }
+ return "other";
+ }
+ function shouldCheckWay(way) {
+ if (way.nodes.length <= 2 || way.isClosed() && way.nodes.length <= 4)
+ return false;
+ var bbox2 = way.extent(graph).bbox();
+ var hypotenuseMeters = geoSphericalDistance([bbox2.minX, bbox2.minY], [bbox2.maxX, bbox2.maxY]);
+ if (hypotenuseMeters < 1.5)
+ return false;
+ return true;
+ }
+ function getIssuesForWay(way) {
+ if (!shouldCheckWay(way))
+ return [];
+ var issues = [], nodes = graph.childNodes(way);
+ for (var i2 = 0; i2 < nodes.length - 1; i2++) {
+ var node1 = nodes[i2];
+ var node2 = nodes[i2 + 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);
+ }
+ for (var i2 = 0; i2 < parentWays.length; i2++) {
+ var parentWay = parentWays[i2];
+ if (!shouldCheckWay(parentWay))
+ continue;
+ var lastIndex = parentWay.nodes.length - 1;
+ for (var j2 = 0; j2 < parentWay.nodes.length; j2++) {
+ if (j2 !== 0) {
+ if (parentWay.nodes[j2 - 1] === node.id) {
+ checkForCloseness(node, graph.entity(parentWay.nodes[j2]), parentWay);
+ }
+ }
+ if (j2 !== lastIndex) {
+ if (parentWay.nodes[j2 + 1] === node.id) {
+ checkForCloseness(graph.entity(parentWay.nodes[j2]), node, parentWay);
+ }
+ }
+ }
+ }
+ return issues;
+ }
+ function thresholdMetersForWay(way) {
+ if (!shouldCheckWay(way))
+ return 0;
+ var wayType = wayTypeFor(way);
+ if (wayType === "boundary")
+ return 0;
+ if (wayType === "indoor")
+ return 0.01;
+ if (wayType === "building")
+ return 0.05;
+ if (wayType === "path")
+ return 0.1;
+ return 0.2;
+ }
+ 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 j2 = 0; j2 < intersected.length; j2++) {
+ var nearby = intersected[j2];
+ if (nearby.id === node.id)
+ continue;
+ if (nearby.type !== "node" || nearby.geometry(graph) !== "point")
+ continue;
+ if (nearby.loc === node.loc || geoSphericalDistance(node.loc, nearby.loc) < pointThresholdMeters) {
+ if ("memorial:type" in node.tags && "memorial:type" in nearby.tags && node.tags["memorial:type"] === "stolperstein" && nearby.tags["memorial:type"] === "stolperstein")
+ continue;
+ if ("memorial" in node.tags && "memorial" in nearby.tags && node.tags.memorial === "stolperstein" && nearby.tags.memorial === "stolperstein")
+ continue;
+ 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: type2,
+ subtype: "detached",
+ severity: "warning",
+ message: function(context2) {
+ var entity2 = context2.hasEntity(this.entityIds[0]), entity22 = context2.hasEntity(this.entityIds[1]);
+ return entity2 && entity22 ? _t.append("issues.close_nodes.detached.message", {
+ feature: utilDisplayLabel(entity2, context2.graph()),
+ feature2: utilDisplayLabel(entity22, context2.graph())
+ }) : "";
+ },
+ reference: showReference,
+ entityIds: [node.id, nearby.id],
+ dynamicFixes: function() {
+ return [
+ new validationIssueFix({
+ icon: "iD-operation-disconnect",
+ title: _t.append("issues.fix.move_points_apart.title")
+ }),
+ new validationIssueFix({
+ icon: "iD-icon-layers",
+ title: _t.append("issues.fix.use_different_layers_or_levels.title")
+ })
+ ];
+ }
+ }));
+ }
+ }
+ return issues;
+ function showReference(selection2) {
+ var referenceText = _t("issues.close_nodes.detached.reference");
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").html(referenceText);
+ }
+ }
+ function getWayIssueIfAny(node1, node2, way) {
+ if (node1.id === node2.id || node1.hasInterestingTags() && node2.hasInterestingTags()) {
+ return null;
+ }
+ 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(...thresholds);
+ var distance = geoSphericalDistance(node1.loc, node2.loc);
+ if (distance > threshold)
+ return null;
+ }
+ return new validationIssue({
+ type: type2,
+ subtype: "vertices",
+ severity: "warning",
+ message: function(context2) {
+ var entity2 = context2.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.close_nodes.message", { way: utilDisplayLabel(entity2, context2.graph()) }) : "";
+ },
+ reference: showReference,
+ entityIds: [way.id, node1.id, node2.id],
+ loc: node1.loc,
+ dynamicFixes: function() {
+ return [
+ new validationIssueFix({
+ icon: "iD-icon-plus",
+ title: _t.append("issues.fix.merge_points.title"),
+ onClick: function(context2) {
+ var entityIds = this.issue.entityIds;
+ var action = actionMergeNodes([entityIds[1], entityIds[2]]);
+ context2.perform(action, _t("issues.fix.merge_close_vertices.annotation"));
+ }
+ }),
+ new validationIssueFix({
+ icon: "iD-operation-disconnect",
+ title: _t.append("issues.fix.move_points_apart.title")
+ })
+ ];
+ }
+ });
+ function showReference(selection2) {
+ var referenceText = _t("issues.close_nodes.reference");
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").html(referenceText);
+ }
+ }
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/crossing_ways.js
+ function validationCrossingWays(context) {
+ var type2 = "crossing_ways";
+ function getFeatureWithFeatureTypeTagsForWay(way, graph) {
+ if (getFeatureType(way, graph) === null) {
+ var parentRels = graph.parentRelations(way);
+ for (var i2 = 0; i2 < parentRels.length; i2++) {
+ var rel = parentRels[i2];
+ if (getFeatureType(rel, graph) !== null) {
+ return rel;
+ }
+ }
+ }
+ return way;
+ }
+ function hasTag(tags, key) {
+ return tags[key] !== void 0 && tags[key] !== "no";
+ }
+ function taggedAsIndoor(tags) {
+ return hasTag(tags, "indoor") || hasTag(tags, "level") || tags.highway === "corridor";
+ }
+ function allowsBridge(featureType) {
+ return featureType === "highway" || featureType === "railway" || featureType === "waterway";
+ }
+ function allowsTunnel(featureType) {
+ return featureType === "highway" || featureType === "railway" || featureType === "waterway";
+ }
+ 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";
+ 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;
+ }
+ function isLegitCrossing(tags1, featureType1, tags2, featureType2) {
+ var level1 = tags1.level || "0";
+ var level2 = tags2.level || "0";
+ if (taggedAsIndoor(tags1) && taggedAsIndoor(tags2) && level1 !== level2) {
+ return true;
+ }
+ 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;
+ 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;
+ 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;
+ 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" || taggedAsIndoor(tags1) || taggedAsIndoor(tags2)) {
+ if (layer1 !== layer2)
+ return true;
+ }
+ return false;
+ }
+ 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 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 (featureType1 === featureType2) {
+ if (featureType1 === "highway") {
+ var entity1IsPath = osmPathHighwayTagValues[entity1.tags.highway];
+ var entity2IsPath = osmPathHighwayTagValues[entity2.tags.highway];
+ if ((entity1IsPath || entity2IsPath) && entity1IsPath !== entity2IsPath) {
+ var roadFeature = entity1IsPath ? entity2 : entity1;
+ if (nonCrossingHighways[roadFeature.tags.highway]) {
+ return {};
+ }
+ var pathFeature = entity1IsPath ? entity1 : entity2;
+ if (["marked", "unmarked", "traffic_signals", "uncontrolled"].indexOf(pathFeature.tags.crossing) !== -1) {
+ return bothLines ? { highway: "crossing", crossing: pathFeature.tags.crossing } : {};
+ }
+ return bothLines ? { highway: "crossing" } : {};
+ }
+ return {};
+ }
+ 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]) {
+ if (isTram)
+ return { railway: "tram_crossing" };
+ return { railway: "crossing" };
+ } else {
+ if (isTram)
+ return { railway: "tram_level_crossing" };
+ return { railway: "level_crossing" };
+ }
+ }
+ if (featureTypes.indexOf("waterway") !== -1) {
+ if (hasTag(entity1.tags, "tunnel") && hasTag(entity2.tags, "tunnel"))
+ return null;
+ if (hasTag(entity1.tags, "bridge") && hasTag(entity2.tags, "bridge"))
+ return null;
+ if (highwaysDisallowingFords[entity1.tags.highway] || highwaysDisallowingFords[entity2.tags.highway]) {
+ return null;
+ }
+ return bothLines ? { ford: "yes" } : {};
+ }
+ }
+ }
+ return null;
+ }
+ 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 = {};
+ var i2, j2;
+ 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 = {};
+ for (i2 = 0; i2 < way1Nodes.length - 1; i2++) {
+ n1 = way1Nodes[i2];
+ n2 = way1Nodes[i2 + 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])
+ ]
+ ]);
+ segmentInfos = tree.waySegments(extent, graph);
+ for (j2 = 0; j2 < segmentInfos.length; j2++) {
+ segment2Info = segmentInfos[j2];
+ if (segment2Info.wayId === way1.id)
+ continue;
+ if (checkedSingleCrossingWays[segment2Info.wayId])
+ continue;
+ comparedWays[segment2Info.wayId] = true;
+ way2 = graph.hasEntity(segment2Info.wayId);
+ if (!way2)
+ continue;
+ taggedFeature2 = getFeatureWithFeatureTypeTagsForWay(way2, graph);
+ way2FeatureType = getFeatureType(taggedFeature2, graph);
+ if (way2FeatureType === null || isLegitCrossing(taggedFeature1.tags, way1FeatureType, taggedFeature2.tags, way2FeatureType)) {
+ continue;
+ }
+ oneOnly = way1FeatureType === "building" || way2FeatureType === "building";
+ nAId = segment2Info.nodes[0];
+ nBId = segment2Info.nodes[1];
+ if (nAId === n1.id || nAId === n2.id || nBId === n1.id || nBId === n2.id) {
+ continue;
+ }
+ 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 point2 = geoLineIntersection(segment1, segment2);
+ if (point2) {
+ edgeCrossInfos.push({
+ wayInfos: [
+ {
+ way: way1,
+ featureType: way1FeatureType,
+ edge: [n1.id, n2.id]
+ },
+ {
+ way: way2,
+ featureType: way2FeatureType,
+ edge: [nA.id, nB.id]
+ }
+ ],
+ crossPoint: point2
+ });
+ if (oneOnly) {
+ checkedSingleCrossingWays[way2.id] = true;
+ break;
+ }
+ }
+ }
+ }
+ return edgeCrossInfos;
+ }
+ function waysToCheck(entity, graph) {
+ var featureType = getFeatureType(entity, graph);
+ if (!featureType)
+ return [];
+ if (entity.type === "way") {
+ return [entity];
+ } else if (entity.type === "relation") {
+ return entity.members.reduce(function(array2, member) {
+ if (member.type === "way" && // only look at geometry ways
+ (!member.role || member.role === "outer" || member.role === "inner")) {
+ var entity2 = graph.hasEntity(member.id);
+ if (entity2 && array2.indexOf(entity2) === -1) {
+ array2.push(entity2);
+ }
+ }
+ return array2;
+ }, []);
+ }
+ return [];
+ }
+ var validation = function checkCrossingWays(entity, graph) {
+ var tree = context.history().tree();
+ var ways = waysToCheck(entity, graph);
+ var issues = [];
+ var wayIndex, crossingIndex, crossings;
+ for (wayIndex in ways) {
+ crossings = findCrossingsByWay(ways[wayIndex], graph, tree);
+ for (crossingIndex in crossings) {
+ issues.push(createIssue(crossings[crossingIndex], graph));
+ }
+ }
+ return issues;
+ };
+ function createIssue(crossing, graph) {
+ crossing.wayInfos.sort(function(way1Info, way2Info) {
+ var type1 = way1Info.featureType;
+ var type22 = way2Info.featureType;
+ if (type1 === type22) {
+ return utilDisplayLabel(way1Info.way, graph) > utilDisplayLabel(way2Info.way, graph);
+ } else if (type1 === "waterway") {
+ return true;
+ } else if (type22 === "waterway") {
+ return false;
+ }
+ return type1 < type22;
+ });
+ 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";
+ }
+ if (connectionTags && (isCrossingIndoors || isCrossingTunnels || isCrossingBridges)) {
+ crossingTypeID += "_connectable";
+ }
+ var uniqueID = crossing.crossPoint[0].toFixed(4) + "," + crossing.crossPoint[1].toFixed(4);
+ return new validationIssue({
+ type: type2,
+ subtype,
+ severity: "warning",
+ message: function(context2) {
+ var graph2 = context2.graph();
+ var entity1 = graph2.hasEntity(this.entityIds[0]), entity2 = graph2.hasEntity(this.entityIds[1]);
+ return entity1 && entity2 ? _t.append("issues.crossing_ways.message", {
+ feature: utilDisplayLabel(entity1, graph2),
+ feature2: utilDisplayLabel(entity2, graph2)
+ }) : "";
+ },
+ reference: showReference,
+ entityIds: entities.map(function(entity) {
+ return entity.id;
+ }),
+ data: {
+ edges,
+ featureTypes,
+ connectionTags
+ },
+ hash: uniqueID,
+ loc: crossing.crossPoint,
+ dynamicFixes: function(context2) {
+ var mode = context2.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 = [];
+ if (connectionTags) {
+ fixes.push(makeConnectWaysFix(this.data.connectionTags));
+ }
+ if (isCrossingIndoors) {
+ fixes.push(new validationIssueFix({
+ icon: "iD-icon-layers",
+ title: _t.append("issues.fix.use_different_levels.title")
+ }));
+ } else if (isCrossingTunnels || isCrossingBridges || featureType1 === "building" || featureType2 === "building") {
+ fixes.push(makeChangeLayerFix("higher"));
+ fixes.push(makeChangeLayerFix("lower"));
+ } else if (context2.graph().geometry(this.entityIds[0]) === "line" && context2.graph().geometry(this.entityIds[1]) === "line") {
+ if (allowsBridge(selectedFeatureType) && selectedFeatureType !== "waterway") {
+ fixes.push(makeAddBridgeOrTunnelFix("add_a_bridge", "temaki-bridge", "bridge"));
+ }
+ var skipTunnelFix = otherFeatureType === "waterway" && selectedFeatureType !== "waterway";
+ if (allowsTunnel(selectedFeatureType) && !skipTunnelFix) {
+ fixes.push(makeAddBridgeOrTunnelFix("add_a_tunnel", "temaki-tunnel", "tunnel"));
+ }
+ }
+ fixes.push(new validationIssueFix({
+ icon: "iD-operation-move",
+ title: _t.append("issues.fix.reposition_features.title")
+ }));
+ return fixes;
+ }
+ });
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.crossing_ways." + crossingTypeID + ".reference"));
+ }
+ }
+ function makeAddBridgeOrTunnelFix(fixTitleID, iconName, bridgeOrTunnel) {
+ return new validationIssueFix({
+ icon: iconName,
+ title: _t.append("issues.fix." + fixTitleID + ".title"),
+ onClick: function(context2) {
+ var mode = context2.mode();
+ if (!mode || mode.id !== "select")
+ return;
+ var selectedIDs = mode.selectedIDs();
+ if (selectedIDs.length !== 1)
+ return;
+ var selectedWayID = selectedIDs[0];
+ if (!context2.hasEntity(selectedWayID))
+ return;
+ var resultWayIDs = [selectedWayID];
+ var edge, crossedEdge, crossedWayID;
+ 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 projection2 = context2.projection;
+ var action = function actionAddStructure(graph) {
+ var edgeNodes = [graph.entity(edge[0]), graph.entity(edge[1])];
+ var crossedWay = graph.hasEntity(crossedWayID);
+ var structLengthMeters = crossedWay && isFinite(crossedWay.tags.width) && Number(crossedWay.tags.width);
+ if (!structLengthMeters) {
+ structLengthMeters = crossedWay && crossedWay.impliedLineWidthMeters();
+ }
+ if (structLengthMeters) {
+ if (getFeatureType(crossedWay, graph) === "railway") {
+ structLengthMeters *= 2;
+ }
+ } else {
+ structLengthMeters = 8;
+ }
+ var a1 = geoAngle(edgeNodes[0], edgeNodes[1], projection2) + Math.PI;
+ var a2 = geoAngle(graph.entity(crossedEdge[0]), graph.entity(crossedEdge[1]), projection2) + Math.PI;
+ var crossingAngle = Math.max(a1, a2) - Math.min(a1, a2);
+ if (crossingAngle > Math.PI)
+ crossingAngle -= Math.PI;
+ structLengthMeters = structLengthMeters / 2 / Math.sin(crossingAngle) * 2;
+ structLengthMeters += 4;
+ structLengthMeters = Math.min(Math.max(structLengthMeters, 4), 50);
+ function geomToProj(geoPoint) {
+ return [
+ geoLonToMeters(geoPoint[0], geoPoint[1]),
+ geoLatToMeters(geoPoint[1])
+ ];
+ }
+ function projToGeom(projPoint) {
+ var lat = geoMetersToLat(projPoint[1]);
+ return [
+ geoMetersToLon(projPoint[0], lat),
+ lat
+ ];
+ }
+ 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(angle2, distanceMeters) {
+ var lengthSphericalMeters = distanceMeters * linearToSphericalMetersRatio;
+ return projToGeom([
+ projectedCrossingLoc[0] + Math.cos(angle2) * lengthSphericalMeters,
+ projectedCrossingLoc[1] + Math.sin(angle2) * lengthSphericalMeters
+ ]);
+ }
+ var endpointLocGetter1 = function(lengthMeters) {
+ return locSphericalDistanceFromCrossingLoc(projectedAngle, lengthMeters);
+ };
+ var endpointLocGetter2 = function(lengthMeters) {
+ return locSphericalDistanceFromCrossingLoc(projectedAngle + Math.PI, lengthMeters);
+ };
+ var minEdgeLengthMeters = 0.55;
+ function determineEndpoint(edge2, endNode, locGetter) {
+ var newNode;
+ var idealLengthMeters = structLengthMeters / 2;
+ var crossingToEdgeEndDistance = geoSphericalDistance(crossingLoc, endNode.loc);
+ if (crossingToEdgeEndDistance - idealLengthMeters > minEdgeLengthMeters) {
+ var idealNodeLoc = locGetter(idealLengthMeters);
+ newNode = osmNode();
+ graph = actionAddMidpoint({ loc: idealNodeLoc, edge: edge2 }, 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;
+ }
+ }
+ });
+ });
+ if (edgeCount >= 3) {
+ var insetLength = crossingToEdgeEndDistance - minEdgeLengthMeters;
+ if (insetLength > minEdgeLengthMeters) {
+ var insetNodeLoc = locGetter(insetLength);
+ newNode = osmNode();
+ graph = actionAddMidpoint({ loc: insetNodeLoc, edge: edge2 }, newNode)(graph);
+ }
+ }
+ }
+ if (!newNode)
+ newNode = endNode;
+ var splitAction = actionSplit([newNode.id]).limitWays(resultWayIDs);
+ graph = splitAction(graph);
+ if (splitAction.getCreatedWayIDs().length) {
+ resultWayIDs.push(splitAction.getCreatedWayIDs()[0]);
+ }
+ return newNode;
+ }
+ var structEndNode1 = determineEndpoint(edge, edgeNodes[1], endpointLocGetter1);
+ var structEndNode2 = determineEndpoint([edgeNodes[0].id, structEndNode1.id], edgeNodes[0], endpointLocGetter2);
+ var structureWay = resultWayIDs.map(function(id2) {
+ return graph.entity(id2);
+ }).find(function(way) {
+ return way.nodes.indexOf(structEndNode1.id) !== -1 && way.nodes.indexOf(structEndNode2.id) !== -1;
+ });
+ var tags = Object.assign({}, structureWay.tags);
+ if (bridgeOrTunnel === "bridge") {
+ tags.bridge = "yes";
+ tags.layer = "1";
+ } else {
+ var tunnelValue = "yes";
+ if (getFeatureType(structureWay, graph) === "waterway") {
+ tunnelValue = "culvert";
+ }
+ tags.tunnel = tunnelValue;
+ tags.layer = "-1";
+ }
+ graph = actionChangeTags(structureWay.id, tags)(graph);
+ return graph;
+ };
+ context2.perform(action, _t("issues.fix." + fixTitleID + ".annotation"));
+ context2.enter(modeSelect(context2, resultWayIDs));
+ }
+ });
+ }
+ function makeConnectWaysFix(connectionTags) {
+ var fixTitleID = "connect_features";
+ if (connectionTags.ford) {
+ fixTitleID = "connect_using_ford";
+ }
+ return new validationIssueFix({
+ icon: "iD-icon-crossing",
+ title: _t.append("issues.fix." + fixTitleID + ".title"),
+ onClick: function(context2) {
+ var loc = this.issue.loc;
+ var connectionTags2 = this.issue.data.connectionTags;
+ var edges = this.issue.data.edges;
+ context2.perform(
+ function actionConnectCrossingWays(graph) {
+ var node = osmNode({ loc, tags: connectionTags2 });
+ 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 ((!nearby.node.hasInterestingTags() || nearby.node.isCrossing()) && nearby.distance < mergeThresholdInMeters) {
+ nodesToMerge.push(nearby.node.id);
+ } else {
+ graph = actionAddMidpoint({ loc, edge }, node)(graph);
+ }
+ });
+ if (nodesToMerge.length > 1) {
+ graph = actionMergeNodes(nodesToMerge, loc)(graph);
+ }
+ return graph;
+ },
+ _t("issues.fix.connect_crossing_features.annotation")
+ );
+ }
+ });
+ }
+ function makeChangeLayerFix(higherOrLower) {
+ return new validationIssueFix({
+ icon: "iD-icon-" + (higherOrLower === "higher" ? "up" : "down"),
+ title: _t.append("issues.fix.tag_this_as_" + higherOrLower + ".title"),
+ onClick: function(context2) {
+ var mode = context2.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 = context2.hasEntity(selectedID);
+ if (!entity)
+ return;
+ var tags = Object.assign({}, entity.tags);
+ var layer = tags.layer && Number(tags.layer);
+ if (layer && !isNaN(layer)) {
+ if (higherOrLower === "higher") {
+ layer += 1;
+ } else {
+ layer -= 1;
+ }
+ } else {
+ if (higherOrLower === "higher") {
+ layer = 1;
+ } else {
+ layer = -1;
+ }
+ }
+ tags.layer = layer.toString();
+ context2.perform(
+ actionChangeTags(entity.id, tags),
+ _t("operations.change_tags.annotation")
+ );
+ }
+ });
+ }
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/behavior/draw_way.js
+ function behaviorDrawWay(context, wayID, mode, startGraph) {
+ const keybinding = utilKeybinding("drawWay");
+ var dispatch10 = dispatch_default("rejectedSelfIntersection");
+ var behavior = behaviorDraw(context);
+ var _nodeIndex;
+ var _origWay;
+ var _wayGeometry;
+ var _headNodeID;
+ var _annotation;
+ var _pointerHasMoved = false;
+ var _drawNode;
+ var _didResolveTempEdit = false;
+ function createDrawNode(loc) {
+ _drawNode = osmNode({ loc });
+ context.pauseChangeDispatch();
+ context.replace(function actionAddDrawNode(graph) {
+ var way = graph.entity(wayID);
+ return graph.replace(_drawNode).replace(way.addNode(_drawNode.id, _nodeIndex));
+ }, _annotation);
+ context.resumeChangeDispatch();
+ setActiveElements();
+ }
+ 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 = void 0;
+ context.resumeChangeDispatch();
+ }
+ function keydown(d3_event) {
+ if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+ if (context.surface().classed("nope")) {
+ context.surface().classed("nope-suppressed", true);
+ }
+ 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);
+ }
+ context.surface().classed("nope-suppressed", false).classed("nope-disabled", false);
+ }
+ }
+ function allowsVertex(d) {
+ return d.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d, context.graph());
+ }
+ function move(d3_event, datum2) {
+ var loc = context.map().mouseCoordinates();
+ if (!_drawNode)
+ createDrawNode(loc);
+ context.surface().classed("nope-disabled", d3_event.altKey);
+ var targetLoc = datum2 && datum2.properties && datum2.properties.entity && allowsVertex(datum2.properties.entity) && datum2.properties.entity.loc;
+ var targetNodes = datum2 && datum2.properties && datum2.properties.nodes;
+ if (targetLoc) {
+ loc = targetLoc;
+ } else if (targetNodes) {
+ var choice = geoChooseEdge(targetNodes, context.map().mouse(), context.projection, _drawNode.id);
+ if (choice) {
+ loc = choice.loc;
+ }
+ }
+ context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
+ _drawNode = context.entity(_drawNode.id);
+ checkGeometry(
+ true
+ /* includeDrawNode */
+ );
+ }
+ function checkGeometry(includeDrawNode) {
+ var nopeDisabled = context.surface().classed("nope-disabled");
+ var isInvalid = isInvalidGeometry(includeDrawNode);
+ if (nopeDisabled) {
+ context.surface().classed("nope", false).classed("nope-suppressed", isInvalid);
+ } else {
+ context.surface().classed("nope", isInvalid).classed("nope-suppressed", false);
+ }
+ }
+ function isInvalidGeometry(includeDrawNode) {
+ var testNode = _drawNode;
+ var parentWay = context.graph().entity(wayID);
+ var nodes = context.graph().childNodes(parentWay).slice();
+ if (includeDrawNode) {
+ if (parentWay.isClosed()) {
+ nodes.pop();
+ }
+ } else {
+ if (parentWay.isClosed()) {
+ if (nodes.length < 3)
+ return false;
+ if (_drawNode)
+ nodes.splice(-2, 1);
+ testNode = nodes[nodes.length - 2];
+ } else {
+ return false;
+ }
+ }
+ return testNode && geoHasSelfIntersections(nodes, testNode.id);
+ }
+ function undone() {
+ _didResolveTempEdit = true;
+ context.pauseChangeDispatch();
+ var nextMode;
+ if (context.graph() === startGraph) {
+ nextMode = modeSelect(context, [wayID]);
+ } else {
+ context.pop(1);
+ nextMode = mode;
+ }
+ context.perform(actionNoop());
+ context.pop(1);
+ context.resumeChangeDispatch();
+ context.enter(nextMode);
+ }
+ function setActiveElements() {
+ if (!_drawNode)
+ return;
+ context.surface().selectAll("." + _drawNode.id).classed("active", true);
+ }
+ function resetToStartGraph() {
+ while (context.graph() !== startGraph) {
+ context.pop();
+ }
+ }
+ var drawWay = function(surface) {
+ _drawNode = void 0;
+ _didResolveTempEdit = false;
+ _origWay = context.entity(wayID);
+ 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];
+ }
+ _wayGeometry = _origWay.geometry(context.graph());
+ _annotation = _t(
+ (_origWay.nodes.length === (_origWay.isClosed() ? 2 : 1) ? "operations.start.annotation." : "operations.continue.annotation.") + _wayGeometry
+ );
+ _pointerHasMoved = false;
+ 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_default2(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) {
+ context.pauseChangeDispatch();
+ resetToStartGraph();
+ context.resumeChangeDispatch();
+ }
+ _drawNode = void 0;
+ _nodeIndex = void 0;
+ 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_default2(window).on("keydown.drawWay", null).on("keyup.drawWay", null);
+ context.history().on("undone.draw", null);
+ };
+ function attemptAdd(d, loc, doAdd) {
+ if (_drawNode) {
+ context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
+ _drawNode = context.entity(_drawNode.id);
+ } else {
+ createDrawNode(loc);
+ }
+ checkGeometry(
+ true
+ /* includeDrawNode */
+ );
+ if (d && d.properties && d.properties.nope || context.surface().classed("nope")) {
+ if (!_pointerHasMoved) {
+ removeDrawNode();
+ }
+ dispatch10.call("rejectedSelfIntersection", this);
+ return;
+ }
+ context.pauseChangeDispatch();
+ doAdd();
+ _didResolveTempEdit = true;
+ context.resumeChangeDispatch();
+ context.enter(mode);
+ }
+ drawWay.add = function(loc, d) {
+ attemptAdd(d, loc, function() {
+ });
+ };
+ drawWay.addWay = function(loc, edge, d) {
+ attemptAdd(d, loc, function() {
+ context.replace(
+ actionAddMidpoint({ loc, edge }, _drawNode),
+ _annotation
+ );
+ });
+ };
+ drawWay.addNode = function(node, d) {
+ 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) {
+ graph = graph.replace(graph.entity(wayID).removeNode(_drawNode.id)).remove(_drawNode);
+ return graph.replace(graph.entity(wayID).addNode(node.id, _nodeIndex));
+ },
+ _annotation
+ );
+ });
+ };
+ function getFeatureType(ways) {
+ if (ways.every((way) => way.isClosed()))
+ return "area";
+ if (ways.every((way) => !way.isClosed()))
+ return "line";
+ return "generic";
+ }
+ function followMode() {
+ if (_didResolveTempEdit)
+ return;
+ try {
+ const isDrawingArea = _origWay.nodes[0] === _origWay.nodes.slice(-1)[0];
+ const [secondLastNodeId, lastNodeId] = _origWay.nodes.slice(isDrawingArea ? -3 : -2);
+ const historyGraph = context.history().graph();
+ if (!lastNodeId || !secondLastNodeId || !historyGraph.hasEntity(lastNodeId) || !historyGraph.hasEntity(secondLastNodeId)) {
+ context.ui().flash.duration(4e3).iconName("#iD-icon-no").label(_t.append("operations.follow.error.needs_more_initial_nodes"))();
+ return;
+ }
+ const lastNodesParents = historyGraph.parentWays(historyGraph.entity(lastNodeId)).filter((w) => w.id !== wayID);
+ const secondLastNodesParents = historyGraph.parentWays(historyGraph.entity(secondLastNodeId)).filter((w) => w.id !== wayID);
+ const featureType = getFeatureType(lastNodesParents);
+ if (lastNodesParents.length !== 1 || secondLastNodesParents.length === 0) {
+ context.ui().flash.duration(4e3).iconName("#iD-icon-no").label(_t.append(`operations.follow.error.intersection_of_multiple_ways.${featureType}`))();
+ return;
+ }
+ if (!secondLastNodesParents.some((n2) => n2.id === lastNodesParents[0].id)) {
+ context.ui().flash.duration(4e3).iconName("#iD-icon-no").label(_t.append(`operations.follow.error.intersection_of_different_ways.${featureType}`))();
+ return;
+ }
+ const way = lastNodesParents[0];
+ const indexOfLast = way.nodes.indexOf(lastNodeId);
+ const indexOfSecondLast = way.nodes.indexOf(secondLastNodeId);
+ const isDescendingPastZero = indexOfLast === way.nodes.length - 2 && indexOfSecondLast === 0;
+ let nextNodeIndex = indexOfLast + (indexOfLast > indexOfSecondLast && !isDescendingPastZero ? 1 : -1);
+ if (nextNodeIndex === -1)
+ nextNodeIndex = indexOfSecondLast === 1 ? way.nodes.length - 2 : 1;
+ const nextNode = historyGraph.entity(way.nodes[nextNodeIndex]);
+ drawWay.addNode(nextNode, {
+ geometry: { type: "Point", coordinates: nextNode.loc },
+ id: nextNode.id,
+ properties: { target: true, entity: nextNode }
+ });
+ } catch (ex) {
+ context.ui().flash.duration(4e3).iconName("#iD-icon-no").label(_t.append("operations.follow.error.unknown"))();
+ }
+ }
+ keybinding.on(_t("operations.follow.key"), followMode);
+ select_default2(document).call(keybinding);
+ drawWay.finish = function() {
+ checkGeometry(
+ false
+ /* includeDrawNode */
+ );
+ if (context.surface().classed("nope")) {
+ dispatch10.call("rejectedSelfIntersection", this);
+ return;
+ }
+ context.pauseChangeDispatch();
+ context.pop(1);
+ _didResolveTempEdit = true;
+ context.resumeChangeDispatch();
+ var way = context.hasEntity(wayID);
+ if (!way || way.isDegenerate()) {
+ drawWay.cancel();
+ return;
+ }
+ window.setTimeout(function() {
+ context.map().dblclickZoomEnable(true);
+ }, 1e3);
+ var isNewFeature = !mode.isContinuing;
+ context.enter(modeSelect(context, [wayID]).newFeature(isNewFeature));
+ };
+ drawWay.cancel = function() {
+ context.pauseChangeDispatch();
+ resetToStartGraph();
+ context.resumeChangeDispatch();
+ window.setTimeout(function() {
+ context.map().dblclickZoomEnable(true);
+ }, 1e3);
+ 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;
+ };
+ drawWay.activeID = function() {
+ if (!arguments.length)
+ return _drawNode && _drawNode.id;
+ return drawWay;
+ };
+ return utilRebind(drawWay, dispatch10, "on");
+ }
+
+ // modules/modes/draw_line.js
+ function modeDrawLine(context, wayID, startGraph, button, affix, continuing) {
+ var mode = {
+ button,
+ id: "draw-line"
+ };
+ var behavior = behaviorDrawWay(context, wayID, mode, startGraph).on("rejectedSelfIntersection.modeDrawLine", function() {
+ context.ui().flash.iconName("#iD-icon-no").label(_t.append("self_intersection.error.lines"))();
+ });
+ mode.wayID = wayID;
+ mode.isContinuing = continuing;
+ mode.enter = function() {
+ behavior.nodeIndex(affix === "prefix" ? 0 : void 0);
+ context.install(behavior);
+ };
+ mode.exit = function() {
+ context.uninstall(behavior);
+ };
+ mode.selectedIDs = function() {
+ return [wayID];
+ };
+ mode.activeID = function() {
+ return behavior && behavior.activeID() || [];
+ };
+ return mode;
+ }
+
+ // modules/validations/disconnected_way.js
+ function validationDisconnectedWay() {
+ var type2 = "disconnected_way";
+ function isTaggedAsHighway(entity) {
+ return osmRoutableHighwayTagValues[entity.tags.highway];
+ }
+ var validation = function checkDisconnectedWay(entity, graph) {
+ var routingIslandWays = routingIslandForEntity(entity);
+ if (!routingIslandWays)
+ return [];
+ return [new validationIssue({
+ type: type2,
+ subtype: "highway",
+ severity: "warning",
+ message: function(context) {
+ var entity2 = this.entityIds.length && context.hasEntity(this.entityIds[0]);
+ var label = entity2 && utilDisplayLabel(entity2, context.graph());
+ return _t.append("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
+ })];
+ function makeFixes(context) {
+ var fixes = [];
+ var singleEntity = this.entityIds.length === 1 && context.hasEntity(this.entityIds[0]);
+ 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 (!fixes.length) {
+ fixes.push(new validationIssueFix({
+ title: _t.append("issues.fix.connect_feature.title")
+ }));
+ }
+ fixes.push(new validationIssueFix({
+ icon: "iD-operation-delete",
+ title: _t.append("issues.fix.delete_feature.title"),
+ entityIds: [singleEntity.id],
+ onClick: function(context2) {
+ var id2 = this.issue.entityIds[0];
+ var operation = operationDelete(context2, [id2]);
+ if (!operation.disabled()) {
+ operation();
+ }
+ }
+ }));
+ } else {
+ fixes.push(new validationIssueFix({
+ title: _t.append("issues.fix.connect_features.title")
+ }));
+ }
+ return fixes;
+ }
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.disconnected_way.routable.reference"));
+ }
+ function routingIslandForEntity(entity2) {
+ var routingIsland = /* @__PURE__ */ new Set();
+ var waysToCheck = [];
+ function queueParentWays(node) {
+ graph.parentWays(node).forEach(function(parentWay) {
+ if (!routingIsland.has(parentWay) && // only check each feature once
+ isRoutableWay(parentWay, false)) {
+ routingIsland.add(parentWay);
+ waysToCheck.push(parentWay);
+ }
+ });
+ }
+ if (entity2.type === "way" && isRoutableWay(entity2, true)) {
+ routingIsland.add(entity2);
+ waysToCheck.push(entity2);
+ } else if (entity2.type === "node" && isRoutableNode(entity2)) {
+ routingIsland.add(entity2);
+ queueParentWays(entity2);
+ } else {
+ return null;
+ }
+ while (waysToCheck.length) {
+ var wayToCheck = waysToCheck.pop();
+ var childNodes = graph.childNodes(wayToCheck);
+ for (var i2 in childNodes) {
+ var vertex = childNodes[i2];
+ if (isConnectedVertex(vertex)) {
+ return null;
+ }
+ if (isRoutableNode(vertex)) {
+ routingIsland.add(vertex);
+ }
+ queueParentWays(vertex);
+ }
+ }
+ return routingIsland;
+ }
+ function isConnectedVertex(vertex) {
+ var osm = services.osm;
+ if (osm && !osm.isDataLoaded(vertex.loc))
+ return true;
+ if (vertex.tags.entrance && vertex.tags.entrance !== "no")
+ return true;
+ if (vertex.tags.amenity === "parking_entrance")
+ return true;
+ return false;
+ }
+ function isRoutableNode(node) {
+ if (node.tags.highway === "elevator")
+ return true;
+ return false;
+ }
+ 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;
+ });
+ }
+ 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.append("issues.fix.continue_from_" + whichEnd + ".title"),
+ entityIds: [vertexID],
+ onClick: function(context) {
+ var wayId = this.issue.entityIds[0];
+ var way = context.hasEntity(wayId);
+ var vertexId = this.entityIds[0];
+ var vertex2 = context.hasEntity(vertexId);
+ if (!way || !vertex2)
+ return;
+ var map2 = context.map();
+ if (!context.editable() || !map2.trimmedExtent().contains(vertex2.loc)) {
+ map2.zoomToEase(vertex2);
+ }
+ context.enter(
+ modeDrawLine(context, wayId, context.graph(), "line", way.affix(vertexId), true)
+ );
+ }
+ });
+ }
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/invalid_format.js
+ function validationFormatting() {
+ var type2 = "invalid_format";
+ var validation = function(entity) {
+ var issues = [];
+ function isValidEmail(email) {
+ var valid_email = /^[^\(\)\\,":;<>@\[\]]+@[^\(\)\\,":;<>@\[\]\.]+(?:\.[a-z0-9-]+)*$/i;
+ return !email || valid_email.test(email);
+ }
+ function showReferenceEmail(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.invalid_format.email.reference"));
+ }
+ if (entity.tags.email) {
+ var emails = entity.tags.email.split(";").map(function(s) {
+ return s.trim();
+ }).filter(function(x) {
+ return !isValidEmail(x);
+ });
+ if (emails.length) {
+ issues.push(new validationIssue({
+ type: type2,
+ subtype: "email",
+ severity: "warning",
+ message: function(context) {
+ var entity2 = context.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append(
+ "issues.invalid_format.email.message" + this.data,
+ { feature: utilDisplayLabel(entity2, context.graph()), email: emails.join(", ") }
+ ) : "";
+ },
+ reference: showReferenceEmail,
+ entityIds: [entity.id],
+ hash: emails.join(),
+ data: emails.length > 1 ? "_multi" : ""
+ }));
+ }
+ }
+ return issues;
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/help_request.js
+ function validationHelpRequest(context) {
+ var type2 = "help_request";
+ var validation = function checkFixmeTag(entity) {
+ if (!entity.tags.fixme)
+ return [];
+ if (entity.version === void 0)
+ return [];
+ if (entity.v !== void 0) {
+ var baseEntity = context.history().base().hasEntity(entity.id);
+ if (!baseEntity || !baseEntity.tags.fixme)
+ return [];
+ }
+ return [new validationIssue({
+ type: type2,
+ subtype: "fixme_tag",
+ severity: "warning",
+ message: function(context2) {
+ var entity2 = context2.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.fixme_tag.message", {
+ feature: utilDisplayLabel(
+ entity2,
+ context2.graph(),
+ true
+ /* verbose */
+ )
+ }) : "";
+ },
+ dynamicFixes: function() {
+ return [
+ new validationIssueFix({
+ title: _t.append("issues.fix.address_the_concern.title")
+ })
+ ];
+ },
+ reference: showReference,
+ entityIds: [entity.id]
+ })];
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.fixme_tag.reference"));
+ }
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/impossible_oneway.js
+ function validationImpossibleOneway() {
+ var type2 = "impossible_oneway";
+ 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);
+ 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;
+ }
+ function isOneway(way) {
+ if (way.tags.oneway === "yes")
+ return true;
+ if (way.tags.oneway)
+ return false;
+ for (var key in way.tags) {
+ if (osmOneWayTags[key] && osmOneWayTags[key][way.tags[key]]) {
+ return true;
+ }
+ }
+ return false;
+ }
+ function nodeOccursMoreThanOnce(way, nodeID) {
+ var occurrences = 0;
+ for (var index in way.nodes) {
+ if (way.nodes[index] === nodeID) {
+ occurrences += 1;
+ if (occurrences > 1)
+ return true;
+ }
+ }
+ return false;
+ }
+ function isConnectedViaOtherTypes(way, node) {
+ var wayType = typeForWay(way);
+ if (wayType === "highway") {
+ 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()) {
+ if (node.tags.natural === "spring")
+ return true;
+ } else {
+ if (node.tags.manhole === "drain")
+ return true;
+ }
+ }
+ return graph.parentWays(node).some(function(parentWay) {
+ if (parentWay.id === way.id)
+ return false;
+ if (wayType === "highway") {
+ if (parentWay.geometry(graph) === "area" && osmRoutableHighwayTagValues[parentWay.tags.highway])
+ return 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;
+ return parentRelation.isMultipolygon() && osmRoutableHighwayTagValues[parentRelation.tags.highway];
+ });
+ } else if (wayType === "waterway") {
+ if (parentWay.tags.natural === "water" || parentWay.tags.natural === "coastline")
+ return true;
+ }
+ return false;
+ });
+ }
+ function issuesForNode(way, nodeID) {
+ var isFirst = nodeID === way.first();
+ var wayType = typeForWay(way);
+ if (nodeOccursMoreThanOnce(way, nodeID))
+ return [];
+ var osm = services.osm;
+ if (!osm)
+ return [];
+ var node = graph.hasEntity(nodeID);
+ 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;
+ });
+ if (wayType === "waterway" && attachedWaysOfSameType.length === 0)
+ return [];
+ var attachedOneways = attachedWaysOfSameType.filter(function(attachedWay) {
+ return isOneway(attachedWay);
+ });
+ if (attachedOneways.length < attachedWaysOfSameType.length)
+ return [];
+ 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 + ".";
+ if (wayType === "waterway") {
+ messageID += "connected." + placement;
+ referenceID += "connected";
+ } else {
+ messageID += placement;
+ referenceID += placement;
+ }
+ return [new validationIssue({
+ type: type2,
+ subtype: wayType,
+ severity: "warning",
+ message: function(context) {
+ var entity2 = context.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.impossible_oneway." + messageID + ".message", {
+ feature: utilDisplayLabel(entity2, context.graph())
+ }) : "";
+ },
+ reference: getReference(referenceID),
+ entityIds: [way.id, node.id],
+ dynamicFixes: function() {
+ var fixes = [];
+ if (attachedOneways.length) {
+ fixes.push(new validationIssueFix({
+ icon: "iD-operation-reverse",
+ title: _t.append("issues.fix.reverse_feature.title"),
+ entityIds: [way.id],
+ onClick: function(context) {
+ var id2 = this.issue.entityIds[0];
+ context.perform(actionReverse(id2), _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.append("issues.fix.continue_from_" + (isFirst ? "start" : "end") + ".title"),
+ onClick: function(context) {
+ var entityID = this.issue.entityIds[0];
+ var vertexID = this.issue.entityIds[1];
+ var way2 = context.entity(entityID);
+ var vertex = context.entity(vertexID);
+ continueDrawing(way2, vertex, context);
+ }
+ }));
+ }
+ return fixes;
+ },
+ loc: node.loc
+ })];
+ function getReference(referenceID2) {
+ return function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.impossible_oneway." + referenceID2 + ".reference"));
+ };
+ }
+ }
+ };
+ function continueDrawing(way, vertex, context) {
+ var map2 = context.map();
+ if (!context.editable() || !map2.trimmedExtent().contains(vertex.loc)) {
+ map2.zoomToEase(vertex);
+ }
+ context.enter(
+ modeDrawLine(context, way.id, context.graph(), "line", way.affix(vertex.id), true)
+ );
+ }
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/incompatible_source.js
+ function validationIncompatibleSource() {
+ const type2 = "incompatible_source";
+ const incompatibleRules = [
+ {
+ id: "amap",
+ regex: /(^amap$|^amap\.com|autonavi|mapabc|高德)/i
+ },
+ {
+ id: "baidu",
+ regex: /(baidu|mapbar|百度)/i
+ },
+ {
+ id: "google",
+ regex: /google/i,
+ exceptRegex: /((books|drive)\.google|google\s?(books|drive|plus))|(esri\/Google_Africa_Buildings)/i
+ }
+ ];
+ const validation = function checkIncompatibleSource(entity) {
+ const entitySources = entity.tags && entity.tags.source && entity.tags.source.split(";");
+ if (!entitySources)
+ return [];
+ const entityID = entity.id;
+ return entitySources.map((source) => {
+ const matchRule = incompatibleRules.find((rule) => {
+ if (!rule.regex.test(source))
+ return false;
+ if (rule.exceptRegex && rule.exceptRegex.test(source))
+ return false;
+ return true;
+ });
+ if (!matchRule)
+ return null;
+ return new validationIssue({
+ type: type2,
+ severity: "warning",
+ message: (context) => {
+ const entity2 = context.hasEntity(entityID);
+ return entity2 ? _t.append("issues.incompatible_source.feature.message", {
+ feature: utilDisplayLabel(
+ entity2,
+ context.graph(),
+ true
+ /* verbose */
+ ),
+ value: source
+ }) : "";
+ },
+ reference: getReference(matchRule.id),
+ entityIds: [entityID],
+ hash: source,
+ dynamicFixes: () => {
+ return [
+ new validationIssueFix({ title: _t.append("issues.fix.remove_proprietary_data.title") })
+ ];
+ }
+ });
+ }).filter(Boolean);
+ function getReference(id2) {
+ return function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append(`issues.incompatible_source.reference.${id2}`));
+ };
+ }
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/maprules.js
+ function validationMaprules() {
+ var type2 = "maprules";
+ var validation = function checkMaprules(entity, graph) {
+ if (!services.maprules)
+ return [];
+ var rules = services.maprules.validationRules();
+ var issues = [];
+ for (var i2 = 0; i2 < rules.length; i2++) {
+ var rule = rules[i2];
+ rule.findIssues(entity, graph, issues);
+ }
+ return issues;
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/mismatched_geometry.js
+ var import_fast_deep_equal4 = __toESM(require_fast_deep_equal());
+ function validationMismatchedGeometry() {
+ var type2 = "mismatched_geometry";
+ function tagSuggestingLineIsArea(entity) {
+ if (entity.type !== "way" || entity.isClosed())
+ return null;
+ var tagSuggestingArea = entity.tagSuggestingArea();
+ if (!tagSuggestingArea) {
+ return null;
+ }
+ var asLine = _mainPresetIndex.matchTags(tagSuggestingArea, "line");
+ var asArea = _mainPresetIndex.matchTags(tagSuggestingArea, "area");
+ if (asLine && asArea && asLine === asArea) {
+ return null;
+ }
+ return tagSuggestingArea;
+ }
+ function makeConnectEndpointsFixOnClick(way, graph) {
+ 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 (firstToLastDistanceMeters < 0.75) {
+ testNodes = nodes.slice();
+ testNodes.pop();
+ testNodes.push(testNodes[0]);
+ if (!geoHasSelfIntersections(testNodes, testNodes[0].id)) {
+ return function(context) {
+ var way2 = context.entity(this.issue.entityIds[0]);
+ context.perform(
+ actionMergeNodes([way2.nodes[0], way2.nodes[way2.nodes.length - 1]], nodes[0].loc),
+ _t("issues.fix.connect_endpoints.annotation")
+ );
+ };
+ }
+ }
+ testNodes = nodes.slice();
+ testNodes.push(testNodes[0]);
+ if (!geoHasSelfIntersections(testNodes, testNodes[0].id)) {
+ return function(context) {
+ var wayId = this.issue.entityIds[0];
+ var way2 = context.entity(wayId);
+ var nodeId = way2.nodes[0];
+ var index = way2.nodes.length;
+ context.perform(
+ actionAddVertex(wayId, nodeId, index),
+ _t("issues.fix.connect_endpoints.annotation")
+ );
+ };
+ }
+ }
+ function lineTaggedAsAreaIssue(entity) {
+ var tagSuggestingArea = tagSuggestingLineIsArea(entity);
+ if (!tagSuggestingArea)
+ return null;
+ var validAsLine = false;
+ var presetAsLine = _mainPresetIndex.matchTags(entity.tags, "line");
+ if (presetAsLine) {
+ validAsLine = true;
+ var key = Object.keys(tagSuggestingArea)[0];
+ if (presetAsLine.tags[key] && presetAsLine.tags[key] === "*") {
+ validAsLine = false;
+ }
+ if (Object.keys(presetAsLine.tags).length === 0) {
+ validAsLine = false;
+ }
+ }
+ return new validationIssue({
+ type: type2,
+ subtype: "area_as_line",
+ severity: "warning",
+ message: function(context) {
+ var entity2 = context.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.tag_suggests_area.message", {
+ feature: utilDisplayLabel(
+ entity2,
+ "area",
+ true
+ /* verbose */
+ ),
+ tag: utilTagText({ tags: tagSuggestingArea })
+ }) : "";
+ },
+ reference: showReference,
+ entityIds: [entity.id],
+ hash: JSON.stringify(tagSuggestingArea),
+ dynamicFixes: function(context) {
+ var fixes = [];
+ var entity2 = context.entity(this.entityIds[0]);
+ var connectEndsOnClick = makeConnectEndpointsFixOnClick(entity2, context.graph());
+ if (!validAsLine) {
+ fixes.push(new validationIssueFix({
+ title: _t.append("issues.fix.connect_endpoints.title"),
+ onClick: connectEndsOnClick
+ }));
+ }
+ fixes.push(new validationIssueFix({
+ icon: "iD-operation-delete",
+ title: _t.append("issues.fix.remove_tag.title"),
+ onClick: function(context2) {
+ var entityId = this.issue.entityIds[0];
+ var entity3 = context2.entity(entityId);
+ var tags = Object.assign({}, entity3.tags);
+ for (var key2 in tagSuggestingArea) {
+ delete tags[key2];
+ }
+ context2.perform(
+ actionChangeTags(entityId, tags),
+ _t("issues.fix.remove_tag.annotation")
+ );
+ }
+ }));
+ return fixes;
+ }
+ });
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.tag_suggests_area.reference"));
+ }
+ }
+ function vertexPointIssue(entity, graph) {
+ if (entity.type !== "node")
+ return null;
+ if (Object.keys(entity.tags).length === 0)
+ return null;
+ if (entity.isOnAddressLine(graph))
+ return null;
+ var geometry = entity.geometry(graph);
+ var allowedGeometries = osmNodeGeometriesForTags(entity.tags);
+ if (geometry === "point" && !allowedGeometries.point && allowedGeometries.vertex) {
+ return new validationIssue({
+ type: type2,
+ subtype: "vertex_as_point",
+ severity: "warning",
+ message: function(context) {
+ var entity2 = context.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.vertex_as_point.message", {
+ feature: utilDisplayLabel(
+ entity2,
+ "vertex",
+ true
+ /* verbose */
+ )
+ }) : "";
+ },
+ reference: function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.vertex_as_point.reference"));
+ },
+ entityIds: [entity.id]
+ });
+ } else if (geometry === "vertex" && !allowedGeometries.vertex && allowedGeometries.point) {
+ return new validationIssue({
+ type: type2,
+ subtype: "point_as_vertex",
+ severity: "warning",
+ message: function(context) {
+ var entity2 = context.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.point_as_vertex.message", {
+ feature: utilDisplayLabel(
+ entity2,
+ "point",
+ true
+ /* verbose */
+ )
+ }) : "";
+ },
+ reference: function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.point_as_vertex.reference"));
+ },
+ entityIds: [entity.id],
+ dynamicFixes: extractPointDynamicFixes
+ });
+ }
+ return null;
+ }
+ function otherMismatchIssue(entity, graph) {
+ if (!entity.hasInterestingTags())
+ return null;
+ if (entity.type !== "node" && entity.type !== "way")
+ return null;
+ 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 asSource = _mainPresetIndex.match(entity, graph);
+ var targetGeom = targetGeoms.find((nodeGeom) => {
+ var asTarget = _mainPresetIndex.matchTags(entity.tags, nodeGeom);
+ if (!asSource || !asTarget || asSource === asTarget || // sometimes there are two presets with the same tags for different geometries
+ (0, import_fast_deep_equal4.default)(asSource.tags, asTarget.tags))
+ return false;
+ if (asTarget.isFallback())
+ return false;
+ var primaryKey = Object.keys(asTarget.tags)[0];
+ 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;
+ }
+ return new validationIssue({
+ type: type2,
+ subtype,
+ severity: "warning",
+ message: function(context) {
+ var entity2 = context.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues." + referenceId + ".message", {
+ feature: utilDisplayLabel(
+ entity2,
+ targetGeom,
+ true
+ /* verbose */
+ )
+ }) : "";
+ },
+ reference: function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.mismatched_geometry.reference"));
+ },
+ entityIds: [entity.id],
+ dynamicFixes
+ });
+ }
+ function lineToAreaDynamicFixes(context) {
+ var convertOnClick;
+ var entityId = this.entityIds[0];
+ var entity = context.entity(entityId);
+ var tags = Object.assign({}, entity.tags);
+ delete tags.area;
+ if (!osmTagSuggestingArea(tags)) {
+ convertOnClick = function(context2) {
+ var entityId2 = this.issue.entityIds[0];
+ var entity2 = context2.entity(entityId2);
+ var tags2 = Object.assign({}, entity2.tags);
+ if (tags2.area) {
+ delete tags2.area;
+ }
+ context2.perform(
+ actionChangeTags(entityId2, tags2),
+ _t("issues.fix.convert_to_line.annotation")
+ );
+ };
+ }
+ return [
+ new validationIssueFix({
+ icon: "iD-icon-line",
+ title: _t.append("issues.fix.convert_to_line.title"),
+ onClick: convertOnClick
+ })
+ ];
+ }
+ function extractPointDynamicFixes(context) {
+ var entityId = this.entityIds[0];
+ var extractOnClick = null;
+ if (!context.hasHiddenConnections(entityId)) {
+ extractOnClick = function(context2) {
+ var entityId2 = this.issue.entityIds[0];
+ var action = actionExtract(entityId2, context2.projection);
+ context2.perform(
+ action,
+ _t("operations.extract.annotation", { n: 1 })
+ );
+ context2.enter(modeSelect(context2, [action.getExtractedNodeID()]));
+ };
+ }
+ return [
+ new validationIssueFix({
+ icon: "iD-operation-extract",
+ title: _t.append("issues.fix.extract_point.title"),
+ onClick: extractOnClick
+ })
+ ];
+ }
+ 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 i2 in sequences) {
+ var sequence = sequences[i2];
+ if (!sequence.nodes)
+ continue;
+ var firstNode = sequence.nodes[0];
+ var lastNode = sequence.nodes[sequence.nodes.length - 1];
+ if (firstNode === lastNode)
+ continue;
+ var issue = new validationIssue({
+ type: type2,
+ subtype: "unclosed_multipolygon_part",
+ severity: "warning",
+ message: function(context) {
+ var entity2 = context.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.unclosed_multipolygon_part.message", {
+ feature: utilDisplayLabel(
+ entity2,
+ context.graph(),
+ true
+ /* verbose */
+ )
+ }) : "";
+ },
+ reference: showReference,
+ loc: sequence.nodes[0].loc,
+ entityIds: [entity.id],
+ hash: sequence.map(function(way) {
+ return way.id;
+ }).join()
+ });
+ issues.push(issue);
+ }
+ return issues;
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.unclosed_multipolygon_part.reference"));
+ }
+ }
+ 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);
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/missing_role.js
+ function validationMissingRole() {
+ var type2 = "missing_role";
+ var validation = function checkMissingRole(entity, graph) {
+ var issues = [];
+ if (entity.type === "way") {
+ graph.parentRelations(entity).forEach(function(relation) {
+ if (!relation.isMultipolygon())
+ return;
+ var member = relation.memberById(entity.id);
+ 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 (way && isMissingRole(member)) {
+ issues.push(makeIssue(way, entity, member));
+ }
+ });
+ }
+ return issues;
+ };
+ function isMissingRole(member) {
+ return !member.role || !member.role.trim().length;
+ }
+ function makeIssue(way, relation, member) {
+ return new validationIssue({
+ type: type2,
+ severity: "warning",
+ message: function(context) {
+ var member2 = context.hasEntity(this.entityIds[1]), relation2 = context.hasEntity(this.entityIds[0]);
+ return member2 && relation2 ? _t.append("issues.missing_role.message", {
+ member: utilDisplayLabel(member2, context.graph()),
+ relation: utilDisplayLabel(relation2, context.graph())
+ }) : "";
+ },
+ reference: showReference,
+ entityIds: [relation.id, way.id],
+ data: {
+ member
+ },
+ hash: member.index.toString(),
+ dynamicFixes: function() {
+ return [
+ makeAddRoleFix("inner"),
+ makeAddRoleFix("outer"),
+ new validationIssueFix({
+ icon: "iD-operation-delete",
+ title: _t.append("issues.fix.remove_from_relation.title"),
+ onClick: function(context) {
+ context.perform(
+ actionDeleteMember(this.issue.entityIds[0], this.issue.data.member.index),
+ _t("operations.delete_member.annotation", {
+ n: 1
+ })
+ );
+ }
+ })
+ ];
+ }
+ });
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.missing_role.multipolygon.reference"));
+ }
+ }
+ function makeAddRoleFix(role) {
+ return new validationIssueFix({
+ title: _t.append("issues.fix.set_as_" + role + ".title"),
+ onClick: function(context) {
+ var oldMember = this.issue.data.member;
+ var member = { id: this.issue.entityIds[1], type: oldMember.type, role };
+ context.perform(
+ actionChangeMember(this.issue.entityIds[0], member, oldMember.index),
+ _t("operations.change_role.annotation", {
+ n: 1
+ })
+ );
+ }
+ });
+ }
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/missing_tag.js
+ function validationMissingTag(context) {
+ var type2 = "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") {
+ return osmOldMultipolygonOuterMemberOfRelation(entity, graph);
+ }
+ return entityDescriptiveKeys.length > 0;
+ }
+ function isUnknownRoad(entity) {
+ return entity.type === "way" && entity.tags.highway === "road";
+ }
+ function isUntypedRelation(entity) {
+ return entity.type === "relation" && !entity.tags.type;
+ }
+ var validation = function checkMissingTag(entity, graph) {
+ var subtype;
+ var osm = context.connection();
+ var isUnloadedNode = entity.type === "node" && osm && !osm.isDataLoaded(entity.loc);
+ 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";
+ }
+ }
+ if (!subtype && isUnknownRoad(entity)) {
+ subtype = "highway_classification";
+ }
+ if (!subtype)
+ return [];
+ var messageID = subtype === "highway_classification" ? "unknown_road" : "missing_tag." + subtype;
+ var referenceID = subtype === "highway_classification" ? "unknown_road" : "missing_tag";
+ var canDelete = entity.version === void 0 || entity.v !== void 0;
+ var severity = canDelete && subtype !== "highway_classification" ? "error" : "warning";
+ return [new validationIssue({
+ type: type2,
+ subtype,
+ severity,
+ message: function(context2) {
+ var entity2 = context2.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues." + messageID + ".message", {
+ feature: utilDisplayLabel(entity2, context2.graph())
+ }) : "";
+ },
+ reference: showReference,
+ entityIds: [entity.id],
+ dynamicFixes: function(context2) {
+ var fixes = [];
+ var selectFixType = subtype === "highway_classification" ? "select_road_type" : "select_preset";
+ fixes.push(new validationIssueFix({
+ icon: "iD-icon-search",
+ title: _t.append("issues.fix." + selectFixType + ".title"),
+ onClick: function(context3) {
+ context3.ui().sidebar.showPresetList();
+ }
+ }));
+ var deleteOnClick;
+ var id2 = this.entityIds[0];
+ var operation = operationDelete(context2, [id2]);
+ var disabledReasonID = operation.disabled();
+ if (!disabledReasonID) {
+ deleteOnClick = function(context3) {
+ var id3 = this.issue.entityIds[0];
+ var operation2 = operationDelete(context3, [id3]);
+ if (!operation2.disabled()) {
+ operation2();
+ }
+ };
+ }
+ fixes.push(
+ new validationIssueFix({
+ icon: "iD-operation-delete",
+ title: _t.append("issues.fix.delete_feature.title"),
+ disabledReason: disabledReasonID ? _t("operations.delete." + disabledReasonID + ".single") : void 0,
+ onClick: deleteOnClick
+ })
+ );
+ return fixes;
+ }
+ })];
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues." + referenceID + ".reference"));
+ }
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/outdated_tags.js
+ function validationOutdatedTags() {
+ const type2 = "outdated_tags";
+ let _waitingForDeprecated = true;
+ let _dataDeprecated;
+ _mainFileFetcher.get("deprecated").then((d) => _dataDeprecated = d).catch(() => {
+ }).finally(() => _waitingForDeprecated = false);
+ function oldTagIssues(entity, graph) {
+ if (!entity.hasInterestingTags())
+ return [];
+ let preset = _mainPresetIndex.match(entity, graph);
+ if (!preset)
+ return [];
+ const oldTags = Object.assign({}, entity.tags);
+ let subtype = "deprecated_tags";
+ if (preset.replacement) {
+ const newPreset = _mainPresetIndex.item(preset.replacement);
+ graph = actionChangePreset(
+ entity.id,
+ preset,
+ newPreset,
+ true
+ /* skip field defaults */
+ )(graph);
+ entity = graph.entity(entity.id);
+ preset = newPreset;
+ }
+ if (_dataDeprecated) {
+ const deprecatedTags = entity.deprecatedTags(_dataDeprecated);
+ if (deprecatedTags.length) {
+ deprecatedTags.forEach((tag) => {
+ graph = actionUpgradeTags(entity.id, tag.old, tag.replace)(graph);
+ });
+ entity = graph.entity(entity.id);
+ }
+ }
+ let newTags = Object.assign({}, entity.tags);
+ if (preset.tags !== preset.addTags) {
+ Object.keys(preset.addTags).forEach((k) => {
+ if (!newTags[k]) {
+ if (preset.addTags[k] === "*") {
+ newTags[k] = "yes";
+ } else {
+ newTags[k] = preset.addTags[k];
+ }
+ }
+ });
+ }
+ const nsi = services.nsi;
+ let waitingForNsi = false;
+ let nsiResult;
+ if (nsi) {
+ waitingForNsi = nsi.status() === "loading";
+ if (!waitingForNsi) {
+ const loc = entity.extent(graph).center();
+ nsiResult = nsi.upgradeTags(newTags, loc);
+ if (nsiResult) {
+ newTags = nsiResult.newTags;
+ subtype = "noncanonical_brand";
+ }
+ }
+ }
+ let issues = [];
+ issues.provisional = _waitingForDeprecated || waitingForNsi;
+ const tagDiff = utilTagDiff(oldTags, newTags);
+ if (!tagDiff.length)
+ return issues;
+ const isOnlyAddingTags = tagDiff.every((d) => d.type === "+");
+ let prefix = "";
+ if (nsiResult) {
+ prefix = "noncanonical_brand.";
+ } else if (subtype === "deprecated_tags" && isOnlyAddingTags) {
+ subtype = "incomplete_tags";
+ prefix = "incomplete.";
+ }
+ let autoArgs = subtype !== "noncanonical_brand" ? [doUpgrade, _t("issues.fix.upgrade_tags.annotation")] : null;
+ issues.push(new validationIssue({
+ type: type2,
+ subtype,
+ severity: "warning",
+ message: showMessage,
+ reference: showReference,
+ entityIds: [entity.id],
+ hash: utilHashcode(JSON.stringify(tagDiff)),
+ dynamicFixes: () => {
+ let fixes = [
+ new validationIssueFix({
+ autoArgs,
+ title: _t.append("issues.fix.upgrade_tags.title"),
+ onClick: (context) => {
+ context.perform(doUpgrade, _t("issues.fix.upgrade_tags.annotation"));
+ }
+ })
+ ];
+ const item = nsiResult && nsiResult.matched;
+ if (item) {
+ fixes.push(
+ new validationIssueFix({
+ title: _t.append("issues.fix.tag_as_not.title", { name: item.displayName }),
+ onClick: (context) => {
+ context.perform(addNotTag, _t("issues.fix.tag_as_not.annotation"));
+ }
+ })
+ );
+ }
+ return fixes;
+ }
+ }));
+ return issues;
+ function doUpgrade(graph2) {
+ const currEntity = graph2.hasEntity(entity.id);
+ if (!currEntity)
+ return graph2;
+ let newTags2 = Object.assign({}, currEntity.tags);
+ tagDiff.forEach((diff) => {
+ if (diff.type === "-") {
+ delete newTags2[diff.key];
+ } else if (diff.type === "+") {
+ newTags2[diff.key] = diff.newVal;
+ }
+ });
+ return actionChangeTags(currEntity.id, newTags2)(graph2);
+ }
+ function addNotTag(graph2) {
+ const currEntity = graph2.hasEntity(entity.id);
+ if (!currEntity)
+ return graph2;
+ const item = nsiResult && nsiResult.matched;
+ if (!item)
+ return graph2;
+ let newTags2 = Object.assign({}, currEntity.tags);
+ const wd = item.mainTag;
+ const notwd = `not:${wd}`;
+ const qid = item.tags[wd];
+ newTags2[notwd] = qid;
+ if (newTags2[wd] === qid) {
+ const wp = item.mainTag.replace("wikidata", "wikipedia");
+ delete newTags2[wd];
+ delete newTags2[wp];
+ }
+ return actionChangeTags(currEntity.id, newTags2)(graph2);
+ }
+ function showMessage(context) {
+ const currEntity = context.hasEntity(entity.id);
+ if (!currEntity)
+ return "";
+ let messageID = `issues.outdated_tags.${prefix}message`;
+ if (subtype === "noncanonical_brand" && isOnlyAddingTags) {
+ messageID += "_incomplete";
+ }
+ return _t.append(messageID, {
+ feature: utilDisplayLabel(
+ currEntity,
+ context.graph(),
+ true
+ /* verbose */
+ )
+ });
+ }
+ function showReference(selection2) {
+ let enter = selection2.selectAll(".issue-reference").data([0]).enter();
+ enter.append("div").attr("class", "issue-reference").call(_t.append(`issues.outdated_tags.${prefix}reference`));
+ enter.append("strong").call(_t.append("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", (d) => {
+ let klass = d.type === "+" ? "add" : "remove";
+ return `tagDiff-cell tagDiff-cell-${klass}`;
+ }).html((d) => d.display);
+ }
+ }
+ function oldMultipolygonIssues(entity, graph) {
+ let 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 [];
+ }
+ if (!multipolygon || !outerWay)
+ return [];
+ return [new validationIssue({
+ type: type2,
+ subtype: "old_multipolygon",
+ severity: "warning",
+ message: showMessage,
+ reference: showReference,
+ entityIds: [outerWay.id, multipolygon.id],
+ dynamicFixes: () => {
+ return [
+ new validationIssueFix({
+ autoArgs: [doUpgrade, _t("issues.fix.move_tags.annotation")],
+ title: _t.append("issues.fix.move_tags.title"),
+ onClick: (context) => {
+ context.perform(doUpgrade, _t("issues.fix.move_tags.annotation"));
+ }
+ })
+ ];
+ }
+ })];
+ function doUpgrade(graph2) {
+ let currMultipolygon = graph2.hasEntity(multipolygon.id);
+ let currOuterWay = graph2.hasEntity(outerWay.id);
+ if (!currMultipolygon || !currOuterWay)
+ return graph2;
+ currMultipolygon = currMultipolygon.mergeTags(currOuterWay.tags);
+ graph2 = graph2.replace(currMultipolygon);
+ return actionChangeTags(currOuterWay.id, {})(graph2);
+ }
+ function showMessage(context) {
+ let currMultipolygon = context.hasEntity(multipolygon.id);
+ if (!currMultipolygon)
+ return "";
+ return _t.append(
+ "issues.old_multipolygon.message",
+ { multipolygon: utilDisplayLabel(
+ currMultipolygon,
+ context.graph(),
+ true
+ /* verbose */
+ ) }
+ );
+ }
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.old_multipolygon.reference"));
+ }
+ }
+ let validation = function checkOutdatedTags(entity, graph) {
+ let issues = oldMultipolygonIssues(entity, graph);
+ if (!issues.length)
+ issues = oldTagIssues(entity, graph);
+ return issues;
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/private_data.js
+ function validationPrivateData() {
+ var type2 = "private_data";
+ var privateBuildingValues = {
+ detached: true,
+ farm: true,
+ house: true,
+ houseboat: true,
+ residential: true,
+ semidetached_house: true,
+ static_caravan: true
+ };
+ var publicKeys = {
+ amenity: true,
+ craft: true,
+ historic: true,
+ leisure: true,
+ office: true,
+ shop: true,
+ tourism: true
+ };
+ var personalTags = {
+ "contact:email": true,
+ "contact:fax": true,
+ "contact:phone": true,
+ email: true,
+ fax: true,
+ phone: true
+ };
+ var validation = function checkPrivateData(entity) {
+ var tags = entity.tags;
+ if (!tags.building || !privateBuildingValues[tags.building])
+ return [];
+ var keepTags = {};
+ for (var k in tags) {
+ if (publicKeys[k])
+ return [];
+ if (!personalTags[k]) {
+ keepTags[k] = tags[k];
+ }
+ }
+ var tagDiff = utilTagDiff(tags, keepTags);
+ if (!tagDiff.length)
+ return [];
+ var fixID = tagDiff.length === 1 ? "remove_tag" : "remove_tags";
+ return [new validationIssue({
+ type: type2,
+ severity: "warning",
+ message: showMessage,
+ reference: showReference,
+ entityIds: [entity.id],
+ dynamicFixes: function() {
+ return [
+ new validationIssueFix({
+ icon: "iD-operation-delete",
+ title: _t.append("issues.fix." + fixID + ".title"),
+ onClick: function(context) {
+ context.perform(doUpgrade, _t("issues.fix.remove_tag.annotation"));
+ }
+ })
+ ];
+ }
+ })];
+ function doUpgrade(graph) {
+ var currEntity = graph.hasEntity(entity.id);
+ if (!currEntity)
+ return graph;
+ var newTags = Object.assign({}, currEntity.tags);
+ 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(this.entityIds[0]);
+ if (!currEntity)
+ return "";
+ return _t.append(
+ "issues.private_data.contact.message",
+ { feature: utilDisplayLabel(currEntity, context.graph()) }
+ );
+ }
+ function showReference(selection2) {
+ var enter = selection2.selectAll(".issue-reference").data([0]).enter();
+ enter.append("div").attr("class", "issue-reference").call(_t.append("issues.private_data.reference"));
+ enter.append("strong").call(_t.append("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;
+ });
+ }
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/suspicious_name.js
+ function validationSuspiciousName() {
+ const type2 = "suspicious_name";
+ const keysToTestForGenericValues = [
+ "aerialway",
+ "aeroway",
+ "amenity",
+ "building",
+ "craft",
+ "highway",
+ "leisure",
+ "railway",
+ "man_made",
+ "office",
+ "shop",
+ "tourism",
+ "waterway"
+ ];
+ let _waitingForNsi = false;
+ function isGenericMatchInNsi(tags) {
+ const nsi = services.nsi;
+ if (nsi) {
+ _waitingForNsi = nsi.status() === "loading";
+ if (!_waitingForNsi) {
+ return nsi.isGenericName(tags);
+ }
+ }
+ return false;
+ }
+ function nameMatchesRawTag(lowercaseName, tags) {
+ for (let i2 = 0; i2 < keysToTestForGenericValues.length; i2++) {
+ let key = keysToTestForGenericValues[i2];
+ let val = tags[key];
+ if (val) {
+ val = val.toLowerCase();
+ if (key === lowercaseName || val === lowercaseName || key.replace(/\_/g, " ") === lowercaseName || val.replace(/\_/g, " ") === lowercaseName) {
+ return true;
+ }
+ }
+ }
+ return false;
+ }
+ function isGenericName(name, tags) {
+ name = name.toLowerCase();
+ return nameMatchesRawTag(name, tags) || isGenericMatchInNsi(tags);
+ }
+ function makeGenericNameIssue(entityId, nameKey, genericName, langCode) {
+ return new validationIssue({
+ type: type2,
+ subtype: "generic_name",
+ severity: "warning",
+ message: function(context) {
+ let entity = context.hasEntity(this.entityIds[0]);
+ if (!entity)
+ return "";
+ let preset = _mainPresetIndex.match(entity, context.graph());
+ let langName = langCode && _mainLocalizer.languageName(langCode);
+ return _t.append(
+ "issues.generic_name.message" + (langName ? "_language" : ""),
+ { feature: preset.name(), name: genericName, language: langName }
+ );
+ },
+ reference: showReference,
+ entityIds: [entityId],
+ hash: `${nameKey}=${genericName}`,
+ dynamicFixes: function() {
+ return [
+ new validationIssueFix({
+ icon: "iD-operation-delete",
+ title: _t.append("issues.fix.remove_the_name.title"),
+ onClick: function(context) {
+ let entityId2 = this.issue.entityIds[0];
+ let entity = context.entity(entityId2);
+ let tags = Object.assign({}, entity.tags);
+ delete tags[nameKey];
+ context.perform(
+ actionChangeTags(entityId2, tags),
+ _t("issues.fix.remove_generic_name.annotation")
+ );
+ }
+ })
+ ];
+ }
+ });
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.generic_name.reference"));
+ }
+ }
+ function makeIncorrectNameIssue(entityId, nameKey, incorrectName, langCode) {
+ return new validationIssue({
+ type: type2,
+ subtype: "not_name",
+ severity: "warning",
+ message: function(context) {
+ const entity = context.hasEntity(this.entityIds[0]);
+ if (!entity)
+ return "";
+ const preset = _mainPresetIndex.match(entity, context.graph());
+ const langName = langCode && _mainLocalizer.languageName(langCode);
+ return _t.append(
+ "issues.incorrect_name.message" + (langName ? "_language" : ""),
+ { feature: preset.name(), name: incorrectName, language: langName }
+ );
+ },
+ reference: showReference,
+ entityIds: [entityId],
+ hash: `${nameKey}=${incorrectName}`,
+ dynamicFixes: function() {
+ return [
+ new validationIssueFix({
+ icon: "iD-operation-delete",
+ title: _t.append("issues.fix.remove_the_name.title"),
+ onClick: function(context) {
+ const entityId2 = this.issue.entityIds[0];
+ const entity = context.entity(entityId2);
+ let tags = Object.assign({}, entity.tags);
+ delete tags[nameKey];
+ context.perform(
+ actionChangeTags(entityId2, tags),
+ _t("issues.fix.remove_mistaken_name.annotation")
+ );
+ }
+ })
+ ];
+ }
+ });
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.generic_name.reference"));
+ }
+ }
+ let validation = function checkGenericName(entity) {
+ const tags = entity.tags;
+ const hasWikidata = !!tags.wikidata || !!tags["brand:wikidata"] || !!tags["operator:wikidata"];
+ if (hasWikidata)
+ return [];
+ let issues = [];
+ const notNames2 = (tags["not:name"] || "").split(";");
+ for (let key in tags) {
+ const m = key.match(/^name(?:(?::)([a-zA-Z_-]+))?$/);
+ if (!m)
+ continue;
+ const langCode = m.length >= 2 ? m[1] : null;
+ const value = tags[key];
+ if (notNames2.length) {
+ for (let i2 in notNames2) {
+ const notName = notNames2[i2];
+ if (notName && value === notName) {
+ issues.push(makeIncorrectNameIssue(entity.id, key, value, langCode));
+ continue;
+ }
+ }
+ }
+ if (isGenericName(value, tags)) {
+ issues.provisional = _waitingForNsi;
+ issues.push(makeGenericNameIssue(entity.id, key, value, langCode));
+ }
+ }
+ return issues;
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/unsquare_way.js
+ function validationUnsquareWay(context) {
+ var type2 = "unsquare_way";
+ var DEFAULT_DEG_THRESHOLD = 5;
+ var epsilon3 = 0.05;
+ var nodeThreshold = 10;
+ function isBuilding(entity, graph) {
+ if (entity.type !== "way" || entity.geometry(graph) !== "area")
+ return false;
+ return entity.tags.building && entity.tags.building !== "no";
+ }
+ var validation = function checkUnsquareWay(entity, graph) {
+ if (!isBuilding(entity, graph))
+ return [];
+ if (entity.tags.nonsquare === "yes")
+ return [];
+ var isClosed = entity.isClosed();
+ if (!isClosed)
+ return [];
+ var nodes = graph.childNodes(entity).slice();
+ if (nodes.length > nodeThreshold + 1)
+ return [];
+ var osm = services.osm;
+ if (!osm || nodes.some(function(node) {
+ return !osm.isDataLoaded(node.loc);
+ }))
+ return [];
+ 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";
+ });
+ });
+ });
+ if (hasConnectedSquarableWays)
+ return [];
+ var storedDegreeThreshold = corePreferences("validate-square-degrees");
+ var degreeThreshold = isFinite(storedDegreeThreshold) ? Number(storedDegreeThreshold) : DEFAULT_DEG_THRESHOLD;
+ var points = nodes.map(function(node) {
+ return context.projection(node.loc);
+ });
+ if (!geoOrthoCanOrthogonalize(points, isClosed, epsilon3, degreeThreshold, true))
+ return [];
+ var autoArgs;
+ if (!entity.tags.wikidata) {
+ var autoAction = actionOrthogonalize(entity.id, context.projection, void 0, degreeThreshold);
+ autoAction.transitionable = false;
+ autoArgs = [autoAction, _t("operations.orthogonalize.annotation.feature", { n: 1 })];
+ }
+ return [new validationIssue({
+ type: type2,
+ subtype: "building",
+ severity: "warning",
+ message: function(context2) {
+ var entity2 = context2.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.unsquare_way.message", {
+ feature: utilDisplayLabel(entity2, context2.graph())
+ }) : "";
+ },
+ reference: showReference,
+ entityIds: [entity.id],
+ hash: degreeThreshold,
+ dynamicFixes: function() {
+ return [
+ new validationIssueFix({
+ icon: "iD-operation-orthogonalize",
+ title: _t.append("issues.fix.square_feature.title"),
+ autoArgs,
+ onClick: function(context2, completionHandler) {
+ var entityId = this.issue.entityIds[0];
+ context2.perform(
+ actionOrthogonalize(entityId, context2.projection, void 0, degreeThreshold),
+ _t("operations.orthogonalize.annotation.feature", { n: 1 })
+ );
+ window.setTimeout(function() {
+ completionHandler();
+ }, 175);
+ }
+ })
+ /*
+ new validationIssueFix({
+ title: t.append('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')
+ );
+ }
+ })
+ */
+ ];
+ }
+ })];
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.unsquare_way.buildings.reference"));
+ }
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/core/validator.js
+ function coreValidator(context) {
+ let dispatch10 = dispatch_default("validated", "focusedIssue");
+ let validator = utilRebind({}, dispatch10, "on");
+ let _rules = {};
+ let _disabledRules = {};
+ let _ignoredIssueIDs = /* @__PURE__ */ new Set();
+ let _resolvedIssueIDs = /* @__PURE__ */ new Set();
+ let _baseCache = validationCache("base");
+ let _headCache = validationCache("head");
+ let _completeDiff = {};
+ let _headIsCurrent = false;
+ let _deferredRIC = {};
+ let _deferredST = /* @__PURE__ */ new Set();
+ let _headPromise;
+ const RETRY = 5e3;
+ const _errorOverrides = parseHashParam(context.initialHashParams.validationError);
+ const _warningOverrides = parseHashParam(context.initialHashParams.validationWarning);
+ const _disableOverrides = parseHashParam(context.initialHashParams.validationDisable);
+ function parseHashParam(param) {
+ let result = [];
+ let rules = (param || "").split(",");
+ rules.forEach((rule) => {
+ rule = rule.trim();
+ const parts = rule.split("/", 2);
+ const type2 = parts[0];
+ const subtype = parts[1] || "*";
+ if (!type2 || !subtype)
+ return;
+ result.push({ type: makeRegExp(type2), subtype: makeRegExp(subtype) });
+ });
+ return result;
+ function makeRegExp(str2) {
+ const escaped = str2.replace(/[-\/\\^$+?.()|[\]{}]/g, "\\$&").replace(/\*/g, ".*");
+ return new RegExp("^" + escaped + "$");
+ }
+ }
+ validator.init = () => {
+ Object.values(validations_exports).forEach((validation) => {
+ if (typeof validation !== "function")
+ return;
+ const fn = validation(context);
+ const key = fn.type;
+ _rules[key] = fn;
+ });
+ let disabledRules = corePreferences("validate-disabledRules");
+ if (disabledRules) {
+ disabledRules.split(",").forEach((k) => _disabledRules[k] = true);
+ }
+ };
+ function reset(resetIgnored) {
+ _baseCache.queue = [];
+ _headCache.queue = [];
+ Object.keys(_deferredRIC).forEach((key) => {
+ window.cancelIdleCallback(key);
+ _deferredRIC[key]();
+ });
+ _deferredRIC = {};
+ _deferredST.forEach(window.clearTimeout);
+ _deferredST.clear();
+ if (resetIgnored)
+ _ignoredIssueIDs.clear();
+ _resolvedIssueIDs.clear();
+ _baseCache = validationCache("base");
+ _headCache = validationCache("head");
+ _completeDiff = {};
+ _headIsCurrent = false;
+ }
+ validator.reset = () => {
+ reset(true);
+ };
+ validator.resetIgnoredIssues = () => {
+ _ignoredIssueIDs.clear();
+ dispatch10.call("validated");
+ };
+ validator.revalidateUnsquare = () => {
+ revalidateUnsquare(_headCache);
+ revalidateUnsquare(_baseCache);
+ dispatch10.call("validated");
+ };
+ function revalidateUnsquare(cache) {
+ const checkUnsquareWay = _rules.unsquare_way;
+ if (!cache.graph || typeof checkUnsquareWay !== "function")
+ return;
+ cache.uncacheIssuesOfType("unsquare_way");
+ const buildings = context.history().tree().intersects(geoExtent([-180, -90], [180, 90]), cache.graph).filter((entity) => entity.type === "way" && entity.tags.building && entity.tags.building !== "no");
+ buildings.forEach((entity) => {
+ const detected = checkUnsquareWay(entity, cache.graph);
+ if (!detected.length)
+ return;
+ cache.cacheIssues(detected);
+ });
+ }
+ validator.getIssues = (options2) => {
+ const opts = Object.assign({ what: "all", where: "all", includeIgnored: false, includeDisabledRules: false }, options2);
+ const view = context.map().extent();
+ let seen = /* @__PURE__ */ new Set();
+ let results = [];
+ if (_headCache.graph && _headCache.graph !== _baseCache.graph) {
+ Object.values(_headCache.issuesByIssueID).forEach((issue) => {
+ const userModified = (issue.entityIds || []).some((id2) => _completeDiff.hasOwnProperty(id2));
+ if (opts.what === "edited" && !userModified)
+ return;
+ if (!filter2(issue))
+ return;
+ seen.add(issue.id);
+ results.push(issue);
+ });
+ }
+ if (opts.what === "all") {
+ Object.values(_baseCache.issuesByIssueID).forEach((issue) => {
+ if (!filter2(issue))
+ return;
+ seen.add(issue.id);
+ results.push(issue);
+ });
+ }
+ return results;
+ function filter2(issue) {
+ 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;
+ if ((issue.entityIds || []).some((id2) => !context.hasEntity(id2)))
+ return false;
+ if (opts.where === "visible") {
+ const extent = issue.extent(context.graph());
+ if (!view.intersects(extent))
+ return false;
+ }
+ return true;
+ }
+ };
+ validator.getResolvedIssues = () => {
+ return Array.from(_resolvedIssueIDs).map((issueID) => _baseCache.issuesByIssueID[issueID]).filter(Boolean);
+ };
+ validator.focusIssue = (issue) => {
+ const graph = context.graph();
+ let selectID;
+ let focusCenter;
+ const issueExtent = issue.extent(graph);
+ if (issueExtent) {
+ focusCenter = issueExtent.center();
+ }
+ if (issue.entityIds && issue.entityIds.length) {
+ selectID = issue.entityIds[0];
+ if (selectID && selectID.charAt(0) === "r") {
+ const ids = utilEntityAndDeepMemberIDs([selectID], graph);
+ let nodeID = ids.find((id2) => id2.charAt(0) === "n" && graph.hasEntity(id2));
+ if (!nodeID) {
+ const wayID = ids.find((id2) => id2.charAt(0) === "w" && graph.hasEntity(id2));
+ if (wayID) {
+ nodeID = graph.entity(wayID).first();
+ }
+ }
+ if (nodeID) {
+ focusCenter = graph.entity(nodeID).loc;
+ }
+ }
+ }
+ if (focusCenter) {
+ const setZoom = Math.max(context.map().zoom(), 19);
+ context.map().unobscuredCenterZoomEase(focusCenter, setZoom);
+ }
+ if (selectID) {
+ window.setTimeout(() => {
+ context.enter(modeSelect(context, [selectID]));
+ dispatch10.call("focusedIssue", this, issue);
+ }, 250);
+ }
+ };
+ validator.getIssuesBySeverity = (options2) => {
+ let groups = utilArrayGroupBy(validator.getIssues(options2), "severity");
+ groups.error = groups.error || [];
+ groups.warning = groups.warning || [];
+ return groups;
+ };
+ validator.getSharedEntityIssues = (entityIDs, options2) => {
+ const orderedIssueTypes = [
+ // Show some issue types in a particular order:
+ "missing_tag",
+ "missing_role",
+ // - missing data first
+ "outdated_tags",
+ "mismatched_geometry",
+ // - identity issues
+ "crossing_ways",
+ "almost_junction",
+ // - geometry issues where fixing them might solve connectivity issues
+ "disconnected_way",
+ "impossible_oneway"
+ // - finally connectivity issues
+ ];
+ const allIssues = validator.getIssues(options2);
+ const forEntityIDs = new Set(entityIDs);
+ return allIssues.filter((issue) => (issue.entityIds || []).some((entityID) => forEntityIDs.has(entityID))).sort((issue1, issue2) => {
+ if (issue1.type === issue2.type) {
+ return issue1.id < issue2.id ? -1 : 1;
+ }
+ const index1 = orderedIssueTypes.indexOf(issue1.type);
+ const index2 = orderedIssueTypes.indexOf(issue2.type);
+ if (index1 !== -1 && index2 !== -1) {
+ return index1 - index2;
+ } else if (index1 === -1 && index2 === -1) {
+ return issue1.type < issue2.type ? -1 : 1;
+ } else {
+ return index1 !== -1 ? -1 : 1;
+ }
+ });
+ };
+ validator.getEntityIssues = (entityID, options2) => {
+ return validator.getSharedEntityIssues([entityID], options2);
+ };
+ validator.getRuleKeys = () => {
+ return Object.keys(_rules);
+ };
+ validator.isRuleEnabled = (key) => {
+ return !_disabledRules[key];
+ };
+ validator.toggleRule = (key) => {
+ if (_disabledRules[key]) {
+ delete _disabledRules[key];
+ } else {
+ _disabledRules[key] = true;
+ }
+ corePreferences("validate-disabledRules", Object.keys(_disabledRules).join(","));
+ validator.validate();
+ };
+ validator.disableRules = (keys) => {
+ _disabledRules = {};
+ keys.forEach((k) => _disabledRules[k] = true);
+ corePreferences("validate-disabledRules", Object.keys(_disabledRules).join(","));
+ validator.validate();
+ };
+ validator.ignoreIssue = (issueID) => {
+ _ignoredIssueIDs.add(issueID);
+ };
+ validator.validate = () => {
+ const baseGraph = context.history().base();
+ if (!_headCache.graph)
+ _headCache.graph = baseGraph;
+ if (!_baseCache.graph)
+ _baseCache.graph = baseGraph;
+ const prevGraph = _headCache.graph;
+ const currGraph = context.graph();
+ if (currGraph === prevGraph) {
+ _headIsCurrent = true;
+ dispatch10.call("validated");
+ return Promise.resolve();
+ }
+ if (_headPromise) {
+ _headIsCurrent = false;
+ return _headPromise;
+ }
+ _headCache.graph = currGraph;
+ _completeDiff = context.history().difference().complete();
+ const incrementalDiff = coreDifference(prevGraph, currGraph);
+ let entityIDs = Object.keys(incrementalDiff.complete());
+ entityIDs = _headCache.withAllRelatedEntities(entityIDs);
+ if (!entityIDs.size) {
+ dispatch10.call("validated");
+ return Promise.resolve();
+ }
+ _headPromise = validateEntitiesAsync(entityIDs, _headCache).then(() => updateResolvedIssues(entityIDs)).then(() => dispatch10.call("validated")).catch(() => {
+ }).then(() => {
+ _headPromise = null;
+ if (!_headIsCurrent) {
+ validator.validate();
+ }
+ });
+ return _headPromise;
+ };
+ context.history().on("restore.validator", validator.validate).on("undone.validator", validator.validate).on("redone.validator", validator.validate).on("reset.validator", () => {
+ reset(false);
+ validator.validate();
+ });
+ context.on("exit.validator", validator.validate);
+ context.history().on("merge.validator", (entities) => {
+ if (!entities)
+ return;
+ const baseGraph = context.history().base();
+ if (!_headCache.graph)
+ _headCache.graph = baseGraph;
+ if (!_baseCache.graph)
+ _baseCache.graph = baseGraph;
+ let entityIDs = entities.map((entity) => entity.id);
+ entityIDs = _baseCache.withAllRelatedEntities(entityIDs);
+ validateEntitiesAsync(entityIDs, _baseCache);
+ });
+ function validateEntity(entity, graph) {
+ let result = { issues: [], provisional: false };
+ Object.keys(_rules).forEach(runValidation);
+ return result;
+ function runValidation(key) {
+ const fn = _rules[key];
+ if (typeof fn !== "function") {
+ console.error("no such validation rule = " + key);
+ return;
+ }
+ let detected = fn(entity, graph);
+ if (detected.provisional) {
+ result.provisional = true;
+ }
+ detected = detected.filter(applySeverityOverrides);
+ result.issues = result.issues.concat(detected);
+ function applySeverityOverrides(issue) {
+ const type2 = issue.type;
+ const subtype = issue.subtype || "";
+ let i2;
+ for (i2 = 0; i2 < _errorOverrides.length; i2++) {
+ if (_errorOverrides[i2].type.test(type2) && _errorOverrides[i2].subtype.test(subtype)) {
+ issue.severity = "error";
+ return true;
+ }
+ }
+ for (i2 = 0; i2 < _warningOverrides.length; i2++) {
+ if (_warningOverrides[i2].type.test(type2) && _warningOverrides[i2].subtype.test(subtype)) {
+ issue.severity = "warning";
+ return true;
+ }
+ }
+ for (i2 = 0; i2 < _disableOverrides.length; i2++) {
+ if (_disableOverrides[i2].type.test(type2) && _disableOverrides[i2].subtype.test(subtype)) {
+ return false;
+ }
+ }
+ return true;
+ }
+ }
+ }
+ function updateResolvedIssues(entityIDs) {
+ entityIDs.forEach((entityID) => {
+ const baseIssues = _baseCache.issuesByEntityID[entityID];
+ if (!baseIssues)
+ return;
+ baseIssues.forEach((issueID) => {
+ const issue = _baseCache.issuesByIssueID[issueID];
+ const userModified = (issue.entityIds || []).some((id2) => _completeDiff.hasOwnProperty(id2));
+ if (userModified && !_headCache.issuesByIssueID[issueID]) {
+ _resolvedIssueIDs.add(issueID);
+ } else {
+ _resolvedIssueIDs.delete(issueID);
+ }
+ });
+ });
+ }
+ function validateEntitiesAsync(entityIDs, cache) {
+ const jobs = Array.from(entityIDs).map((entityID) => {
+ if (cache.queuedEntityIDs.has(entityID))
+ return null;
+ cache.queuedEntityIDs.add(entityID);
+ cache.uncacheEntityID(entityID);
+ return () => {
+ cache.queuedEntityIDs.delete(entityID);
+ const graph = cache.graph;
+ if (!graph)
+ return;
+ const entity = graph.hasEntity(entityID);
+ if (!entity)
+ return;
+ const result = validateEntity(entity, graph);
+ if (result.provisional) {
+ cache.provisionalEntityIDs.add(entityID);
+ }
+ cache.cacheIssues(result.issues);
+ };
+ }).filter(Boolean);
+ cache.queue = cache.queue.concat(utilArrayChunk(jobs, 100));
+ if (cache.queuePromise)
+ return cache.queuePromise;
+ cache.queuePromise = processQueue(cache).then(() => revalidateProvisionalEntities(cache)).catch(() => {
+ }).finally(() => cache.queuePromise = null);
+ return cache.queuePromise;
+ }
+ function revalidateProvisionalEntities(cache) {
+ if (!cache.provisionalEntityIDs.size)
+ return;
+ const handle = window.setTimeout(() => {
+ _deferredST.delete(handle);
+ if (!cache.provisionalEntityIDs.size)
+ return;
+ validateEntitiesAsync(Array.from(cache.provisionalEntityIDs), cache);
+ }, RETRY);
+ _deferredST.add(handle);
+ }
+ function processQueue(cache) {
+ if (!cache.queue.length)
+ return Promise.resolve();
+ const chunk = cache.queue.pop();
+ return new Promise((resolvePromise, rejectPromise) => {
+ const handle = window.requestIdleCallback(() => {
+ delete _deferredRIC[handle];
+ chunk.forEach((job) => job());
+ resolvePromise();
+ });
+ _deferredRIC[handle] = rejectPromise;
+ }).then(() => {
+ if (cache.queue.length % 25 === 0)
+ dispatch10.call("validated");
+ }).then(() => processQueue(cache));
+ }
+ return validator;
+ }
+ function validationCache(which) {
+ let cache = {
+ which,
+ graph: null,
+ queue: [],
+ queuePromise: null,
+ queuedEntityIDs: /* @__PURE__ */ new Set(),
+ provisionalEntityIDs: /* @__PURE__ */ new Set(),
+ issuesByIssueID: {},
+ // issue.id -> issue
+ issuesByEntityID: {}
+ // entity.id -> Set(issue.id)
+ };
+ cache.cacheIssue = (issue) => {
+ (issue.entityIds || []).forEach((entityID) => {
+ if (!cache.issuesByEntityID[entityID]) {
+ cache.issuesByEntityID[entityID] = /* @__PURE__ */ new Set();
+ }
+ cache.issuesByEntityID[entityID].add(issue.id);
+ });
+ cache.issuesByIssueID[issue.id] = issue;
+ };
+ cache.uncacheIssue = (issue) => {
+ (issue.entityIds || []).forEach((entityID) => {
+ if (cache.issuesByEntityID[entityID]) {
+ cache.issuesByEntityID[entityID].delete(issue.id);
+ }
+ });
+ delete cache.issuesByIssueID[issue.id];
+ };
+ cache.cacheIssues = (issues) => {
+ issues.forEach(cache.cacheIssue);
+ };
+ cache.uncacheIssues = (issues) => {
+ issues.forEach(cache.uncacheIssue);
+ };
+ cache.uncacheIssuesOfType = (type2) => {
+ const issuesOfType = Object.values(cache.issuesByIssueID).filter((issue) => issue.type === type2);
+ cache.uncacheIssues(issuesOfType);
+ };
+ cache.uncacheEntityID = (entityID) => {
+ const entityIssueIDs = cache.issuesByEntityID[entityID];
+ if (entityIssueIDs) {
+ entityIssueIDs.forEach((issueID) => {
+ const issue = cache.issuesByIssueID[issueID];
+ if (issue) {
+ cache.uncacheIssue(issue);
+ } else {
+ delete cache.issuesByIssueID[issueID];
+ }
+ });
+ }
+ delete cache.issuesByEntityID[entityID];
+ cache.provisionalEntityIDs.delete(entityID);
+ };
+ cache.withAllRelatedEntities = (entityIDs) => {
+ let result = /* @__PURE__ */ new Set();
+ (entityIDs || []).forEach((entityID) => {
+ result.add(entityID);
+ const entityIssueIDs = cache.issuesByEntityID[entityID];
+ if (entityIssueIDs) {
+ entityIssueIDs.forEach((issueID) => {
+ const issue = cache.issuesByIssueID[issueID];
+ if (issue) {
+ (issue.entityIds || []).forEach((relatedID) => result.add(relatedID));
+ } else {
+ delete cache.issuesByIssueID[issueID];
+ }
+ });
+ }
+ });
+ return result;
+ };
+ return cache;
+ }
+
+ // modules/core/uploader.js
+ function coreUploader(context) {
+ var dispatch10 = dispatch_default(
+ // 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 = [];
+ var _origChanges;
+ var _discardTags = {};
+ _mainFileFetcher.get("discarded").then(function(d) {
+ _discardTags = d;
+ }).catch(function() {
+ });
+ var uploader = utilRebind({}, dispatch10, "on");
+ uploader.isSaving = function() {
+ return _isSaving;
+ };
+ uploader.save = function(changeset, tryAgain, checkConflicts) {
+ if (_isSaving && !tryAgain) {
+ return;
+ }
+ var osm = context.connection();
+ if (!osm)
+ return;
+ if (!osm.authenticated()) {
+ osm.authenticate(function(err) {
+ if (!err) {
+ uploader.save(changeset, tryAgain, checkConflicts);
+ }
+ });
+ return;
+ }
+ if (!_isSaving) {
+ _isSaving = true;
+ dispatch10.call("saveStarted", this);
+ }
+ var history = context.history();
+ _conflicts = [];
+ _errors = [];
+ _origChanges = history.changes(actionDiscardTags(history.difference(), _discardTags));
+ if (!tryAgain) {
+ history.perform(actionNoop());
+ }
+ if (!checkConflicts) {
+ upload(changeset);
+ } else {
+ performFullConflictCheck(changeset);
+ }
+ };
+ 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 = [];
+ for (var i2 = 0; i2 < summary.length; i2++) {
+ var item = summary[i2];
+ if (item.changeType === "modified") {
+ _toCheck.push(item.entity.id);
+ }
+ }
+ var _toLoad = withChildNodes(_toCheck, localGraph);
+ var _loaded = {};
+ var _toLoadCount = 0;
+ var _toLoadTotal = _toLoad.length;
+ if (_toCheck.length) {
+ dispatch10.call("progressChanged", this, _toLoadCount, _toLoadTotal);
+ _toLoad.forEach(function(id2) {
+ _loaded[id2] = false;
+ });
+ osm.loadMultiple(_toLoad, loaded);
+ } else {
+ upload(changeset);
+ }
+ return;
+ function withChildNodes(ids, graph) {
+ var s = new Set(ids);
+ ids.forEach(function(id2) {
+ var entity = graph.entity(id2);
+ if (entity.type !== "way")
+ return;
+ graph.childNodes(entity).forEach(function(child) {
+ if (child.version !== void 0) {
+ s.add(child.id);
+ }
+ });
+ });
+ return Array.from(s);
+ }
+ 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 })]
+ });
+ 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;
+ var i3, id2;
+ if (entity.type === "way") {
+ for (i3 = 0; i3 < entity.nodes.length; i3++) {
+ id2 = entity.nodes[i3];
+ if (_loaded[id2] === void 0) {
+ _loaded[id2] = false;
+ loadMore.push(id2);
+ }
+ }
+ } else if (entity.type === "relation" && entity.isMultipolygon()) {
+ for (i3 = 0; i3 < entity.members.length; i3++) {
+ id2 = entity.members[i3].id;
+ if (_loaded[id2] === void 0) {
+ _loaded[id2] = false;
+ loadMore.push(id2);
+ }
+ }
+ }
+ });
+ _toLoadCount += result.data.length;
+ _toLoadTotal += loadMore.length;
+ dispatch10.call("progressChanged", this, _toLoadCount, _toLoadTotal);
+ if (loadMore.length) {
+ _toLoad.push.apply(_toLoad, loadMore);
+ osm.loadMultiple(loadMore, loaded);
+ }
+ if (!_toLoad.length) {
+ detectConflicts();
+ upload(changeset);
+ }
+ }
+ }
+ function detectConflicts() {
+ function choice(id2, text2, action) {
+ return {
+ id: id2,
+ text: text2,
+ action: function() {
+ history.replace(action);
+ }
+ };
+ }
+ function formatUser(d) {
+ return '<a href="' + osm.userURL(d) + '" target="_blank">' + escape_default(d) + "</a>";
+ }
+ function entityName(entity) {
+ return utilDisplayName(entity) || utilDisplayType(entity.id) + " " + entity.id;
+ }
+ function sameVersions(local, remote) {
+ if (local.version !== remote.version)
+ return false;
+ if (local.type === "way") {
+ var children2 = utilArrayUnion(local.nodes, remote.nodes);
+ for (var i3 = 0; i3 < children2.length; i3++) {
+ var a = localGraph.hasEntity(children2[i3]);
+ var b = remoteGraph.hasEntity(children2[i3]);
+ if (a && b && a.version !== b.version)
+ return false;
+ }
+ }
+ return true;
+ }
+ _toCheck.forEach(function(id2) {
+ var local = localGraph.entity(id2);
+ var remote = remoteGraph.entity(id2);
+ if (sameVersions(local, remote))
+ return;
+ var merge3 = actionMergeRemoteChanges(id2, localGraph, remoteGraph, _discardTags, formatUser);
+ history.replace(merge3);
+ var mergeConflicts = merge3.conflicts();
+ if (!mergeConflicts.length)
+ return;
+ var forceLocal = actionMergeRemoteChanges(id2, localGraph, remoteGraph, _discardTags).withOption("force_local");
+ var forceRemote = actionMergeRemoteChanges(id2, 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"));
+ _conflicts.push({
+ id: id2,
+ name: entityName(local),
+ details: mergeConflicts,
+ chosen: 1,
+ choices: [
+ choice(id2, keepMine, forceLocal),
+ choice(id2, keepTheirs, forceRemote)
+ ]
+ });
+ });
+ }
+ }
+ function upload(changeset) {
+ var osm = context.connection();
+ if (!osm) {
+ _errors.push({ msg: "No OSM Service" });
+ }
+ 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) {
+ dispatch10.call("willAttemptUpload", this);
+ osm.putChangeset(changeset, changes, uploadCallback);
+ } else {
+ didResultInNoChanges();
+ }
+ }
+ }
+ function uploadCallback(err, changeset) {
+ if (err) {
+ if (err.status === 409) {
+ uploader.save(changeset, true, true);
+ } else {
+ _errors.push({
+ msg: err.message || err.responseText,
+ details: [_t("save.status_code", { code: err.status })]
+ });
+ didResultInErrors();
+ }
+ } else {
+ didResultInSuccess(changeset);
+ }
+ }
+ function didResultInNoChanges() {
+ dispatch10.call("resultNoChanges", this);
+ endSave();
+ context.flush();
+ }
+ function didResultInErrors() {
+ context.history().pop();
+ dispatch10.call("resultErrors", this, _errors);
+ endSave();
+ }
+ function didResultInConflicts(changeset) {
+ _conflicts.sort(function(a, b) {
+ return b.id.localeCompare(a.id);
+ });
+ dispatch10.call("resultConflicts", this, changeset, _conflicts, _origChanges);
+ endSave();
+ }
+ function didResultInSuccess(changeset) {
+ context.history().clearSaved();
+ dispatch10.call("resultSuccess", this, changeset);
+ window.setTimeout(function() {
+ endSave();
+ context.flush();
+ }, 2500);
+ }
+ function endSave() {
+ _isSaving = false;
+ dispatch10.call("saveEnded", this);
+ }
+ uploader.cancelConflictResolution = function() {
+ context.history().pop();
+ };
+ uploader.processResolvedConflicts = function(changeset) {
+ var history = context.history();
+ for (var i2 = 0; i2 < _conflicts.length; i2++) {
+ if (_conflicts[i2].chosen === 1) {
+ var entity = context.hasEntity(_conflicts[i2].id);
+ if (entity && entity.type === "way") {
+ var children2 = utilArrayUniq(entity.nodes);
+ for (var j2 = 0; j2 < children2.length; j2++) {
+ history.replace(actionRevert(children2[j2]));
+ }
+ }
+ history.replace(actionRevert(_conflicts[i2].id));
+ }
+ }
+ uploader.save(changeset, true, false);
+ };
+ uploader.reset = function() {
+ };
+ return uploader;
+ }
+
+ // modules/renderer/background_source.js
+ var import_lodash2 = __toESM(require_lodash());
+
+ // modules/util/IntervalTasksQueue.js
+ var IntervalTasksQueue = class {
+ /**
+ * Interval in milliseconds inside which only 1 task can execute.
+ * e.g. if interval is 200ms, and 5 async tasks are unqueued,
+ * they will complete in ~1s if not cleared
+ * @param {number} intervalInMs
+ */
+ constructor(intervalInMs) {
+ this.intervalInMs = intervalInMs;
+ this.pendingHandles = [];
+ this.time = 0;
+ }
+ enqueue(task) {
+ let taskTimeout = this.time;
+ this.time += this.intervalInMs;
+ this.pendingHandles.push(setTimeout(() => {
+ this.time -= this.intervalInMs;
+ task();
+ }, taskTimeout));
+ }
+ clear() {
+ this.pendingHandles.forEach((timeoutHandle) => {
+ clearTimeout(timeoutHandle);
+ });
+ this.pendingHandles = [];
+ this.time = 0;
+ }
+ };
+
+ // modules/renderer/background_source.js
+ var isRetina = window.devicePixelRatio && window.devicePixelRatio >= 2;
+ window.matchMedia(`
+ (-webkit-min-device-pixel-ratio: 2), /* Safari */
+ (min-resolution: 2dppx), /* standard */
+ (min-resolution: 192dpi) /* fallback */
+ `).addListener(function() {
+ isRetina = window.devicePixelRatio && window.devicePixelRatio >= 2;
+ });
+ function localeDateString(s) {
+ if (!s)
+ return null;
+ var options2 = { day: "numeric", month: "short", year: "numeric" };
+ var d = new Date(s);
+ if (isNaN(d.getTime()))
+ return null;
+ return d.toLocaleDateString(_mainLocalizer.localeCode(), options2);
+ }
+ function vintageRange(vintage) {
+ var s;
+ if (vintage.start || vintage.end) {
+ s = vintage.start || "?";
+ if (vintage.start !== vintage.end) {
+ s += " - " + (vintage.end || "?");
+ }
+ }
+ return s;
+ }
+ function rendererBackgroundSource(data) {
+ var source = Object.assign({}, data);
+ 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;
+ 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;
+ };
+ source.nudge = function(val, zoomlevel) {
+ _offset[0] += val[0] / Math.pow(2, zoomlevel);
+ _offset[1] += val[1] / Math.pow(2, zoomlevel);
+ return source;
+ };
+ source.name = function() {
+ var id_safe = source.id.replace(/\./g, "<TX_DOT>");
+ return _t("imagery." + id_safe + ".name", { default: (0, import_lodash2.escape)(_name) });
+ };
+ source.label = function() {
+ var id_safe = source.id.replace(/\./g, "<TX_DOT>");
+ return _t.append("imagery." + id_safe + ".name", { default: (0, import_lodash2.escape)(_name) });
+ };
+ source.hasDescription = function() {
+ var id_safe = source.id.replace(/\./g, "<TX_DOT>");
+ var descriptionText = _mainLocalizer.tInfo("imagery." + id_safe + ".description", { default: (0, import_lodash2.escape)(_description) }).text;
+ return descriptionText !== "";
+ };
+ source.description = function() {
+ var id_safe = source.id.replace(/\./g, "<TX_DOT>");
+ return _t.append("imagery." + id_safe + ".description", { default: (0, import_lodash2.escape)(_description) });
+ };
+ source.best = function() {
+ return _best;
+ };
+ source.area = function() {
+ if (!data.polygon)
+ return Number.MAX_VALUE;
+ var area = area_default({ type: "MultiPolygon", coordinates: [data.polygon] });
+ return isNaN(area) ? 0 : area;
+ };
+ source.imageryUsed = function() {
+ return _name || source.id;
+ };
+ source.template = function(val) {
+ if (!arguments.length)
+ return _template;
+ if (source.id === "custom" || source.id === "Bing") {
+ _template = val;
+ }
+ return source;
+ };
+ source.url = function(coord2) {
+ var result = _template.replace(/#[\s\S]*/u, "");
+ if (result === "")
+ return result;
+ if (!source.type || source.id === "custom") {
+ if (/SERVICE=WMS|\{(proj|wkid|bbox)\}/.test(result)) {
+ source.type = "wms";
+ source.projection = "EPSG:3857";
+ } else if (/\{(x|y)\}/.test(result)) {
+ source.type = "tms";
+ } else if (/\{u\}/.test(result)) {
+ source.type = "bing";
+ }
+ }
+ if (source.type === "wms") {
+ var tileToProjectedCoords = function(x, y, z) {
+ var zoomSize = Math.pow(2, z);
+ var lon = x / zoomSize * Math.PI * 2 - Math.PI;
+ var lat = Math.atan(Math.sinh(Math.PI * (1 - 2 * y / zoomSize)));
+ switch (source.projection) {
+ case "EPSG:4326":
+ return {
+ x: lon * 180 / Math.PI,
+ y: lat * 180 / Math.PI
+ };
+ default:
+ var mercCoords = mercatorRaw(lon, lat);
+ return {
+ x: 2003750834e-2 / Math.PI * mercCoords[0],
+ y: 2003750834e-2 / Math.PI * mercCoords[1]
+ };
+ }
+ };
+ var tileSize = source.tileSize;
+ var projection2 = source.projection;
+ var minXmaxY = tileToProjectedCoords(coord2[0], coord2[1], coord2[2]);
+ var maxXminY = tileToProjectedCoords(coord2[0] + 1, coord2[1] + 1, coord2[2]);
+ result = result.replace(/\{(\w+)\}/g, function(token, key) {
+ switch (key) {
+ case "width":
+ case "height":
+ return tileSize;
+ case "proj":
+ return projection2;
+ case "wkid":
+ return projection2.replace(/^EPSG:/, "");
+ case "bbox":
+ if (projection2 === "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;
+ }
+ case "w":
+ return minXmaxY.x;
+ case "s":
+ return maxXminY.y;
+ case "n":
+ return maxXminY.x;
+ case "e":
+ return minXmaxY.y;
+ default:
+ return token;
+ }
+ });
+ } else if (source.type === "tms") {
+ result = result.replace("{x}", coord2[0]).replace("{y}", coord2[1]).replace(/\{[t-]y\}/, Math.pow(2, coord2[2]) - coord2[1] - 1).replace(/\{z(oom)?\}/, coord2[2]).replace(/\{@2x\}|\{r\}/, isRetina ? "@2x" : "");
+ } else if (source.type === "bing") {
+ result = result.replace("{u}", function() {
+ var u = "";
+ for (var zoom = coord2[2]; zoom > 0; zoom--) {
+ var b = 0;
+ var mask = 1 << zoom - 1;
+ if ((coord2[0] & mask) !== 0)
+ b++;
+ if ((coord2[1] & mask) !== 0)
+ b += 2;
+ u += b.toString();
+ }
+ return u;
+ });
+ }
+ result = result.replace(/\{switch:([^}]+)\}/, function(s, r) {
+ var subdomains = r.split(",");
+ return subdomains[(coord2[0] + coord2[1]) % subdomains.length];
+ });
+ return result;
+ };
+ source.validZoom = function(z) {
+ return source.zoomExtent[0] <= z && (source.overzoom || source.zoomExtent[1] > z);
+ };
+ source.isLocatorOverlay = function() {
+ return source.id === "mapbox_locator_overlay";
+ };
+ source.isHidden = function() {
+ return source.id === "DigitalGlobe-Premium-vintage" || source.id === "DigitalGlobe-Standard-vintage";
+ };
+ source.copyrightNotices = function() {
+ };
+ source.getMetadata = function(center, tileCoord, callback) {
+ var vintage = {
+ start: localeDateString(source.startDate),
+ end: localeDateString(source.endDate)
+ };
+ vintage.range = vintageRange(vintage);
+ var metadata = { vintage };
+ callback(null, metadata);
+ };
+ return source;
+ }
+ rendererBackgroundSource.Bing = function(data, dispatch10) {
+ data.template = "https://ecn.t{switch:0,1,2,3}.tiles.virtualearth.net/tiles/a{u}.jpeg?g=1&pr=odbl&n=z";
+ var bing = rendererBackgroundSource(data);
+ var key = utilAesDecrypt("5c875730b09c6b422433e807e1ff060b6536c791dbfffcffc4c6b18a1bdba1f14593d151adb50e19e1be1ab19aef813bf135d0f103475e5c724dec94389e45d0");
+ const strictParam = "n";
+ var url = "https://dev.virtualearth.net/REST/v1/Imagery/Metadata/AerialOSM?include=ImageryProviders&uriScheme=https&key=" + key;
+ var cache = {};
+ var inflight = {};
+ var providers = [];
+ var taskQueue = new IntervalTasksQueue(250);
+ var metadataLastZoom = -1;
+ json_default(url).then(function(json) {
+ let imageryResource = json.resourceSets[0].resources[0];
+ let template = imageryResource.imageUrl;
+ let subDomains = imageryResource.imageUrlSubdomains;
+ let subDomainNumbers = subDomains.map((subDomain) => {
+ return subDomain.substring(1);
+ }).join(",");
+ template = template.replace("{subdomain}", `t{switch:${subDomainNumbers}}`).replace("{quadkey}", "{u}");
+ if (!new URLSearchParams(template).has(strictParam)) {
+ template += `&${strictParam}=z`;
+ }
+ 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]])
+ };
+ })
+ };
+ });
+ dispatch10.call("change");
+ }).catch(function() {
+ });
+ 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(", ");
+ };
+ 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];
+ var url2 = "https://dev.virtualearth.net/REST/v1/Imagery/BasicMetadata/AerialOSM/" + centerPoint + "?zl=" + zoom + "&key=" + key;
+ if (inflight[tileID])
+ return;
+ if (!cache[tileID]) {
+ cache[tileID] = {};
+ }
+ if (cache[tileID] && cache[tileID].metadata) {
+ return callback(null, cache[tileID].metadata);
+ }
+ inflight[tileID] = true;
+ if (metadataLastZoom !== tileCoord[2]) {
+ metadataLastZoom = tileCoord[2];
+ taskQueue.clear();
+ }
+ taskQueue.enqueue(() => {
+ json_default(url2).then(function(result) {
+ delete inflight[tileID];
+ if (!result) {
+ throw new Error("Unknown Error");
+ }
+ 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 };
+ cache[tileID].metadata = metadata;
+ if (callback)
+ callback(null, metadata);
+ }).catch(function(err) {
+ delete inflight[tileID];
+ if (callback)
+ callback(err.message);
+ });
+ });
+ };
+ bing.terms_url = "https://blog.openstreetmap.org/2010/11/30/microsoft-imagery-details";
+ return bing;
+ };
+ rendererBackgroundSource.Esri = function(data) {
+ if (data.template.match(/blankTile/) === null) {
+ data.template = data.template + "?blankTile=false";
+ }
+ var esri = rendererBackgroundSource(data);
+ var cache = {};
+ var inflight = {};
+ var _prevCenter;
+ esri.fetchTilemap = function(center) {
+ if (_prevCenter && geoSphericalDistance(center, _prevCenter) < 5e3)
+ return;
+ _prevCenter = center;
+ var z = 20;
+ var dummyUrl = esri.url([1, 2, 3]);
+ 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));
+ var tilemapUrl = dummyUrl.replace(/tile\/[0-9]+\/[0-9]+\/[0-9]+\?blankTile=false/, "tilemap") + "/" + z + "/" + y + "/" + x + "/8/8";
+ json_default(tilemapUrl).then(function(tilemap) {
+ if (!tilemap) {
+ throw new Error("Unknown Error");
+ }
+ var hasTiles = true;
+ for (var i2 = 0; i2 < tilemap.data.length; i2++) {
+ if (!tilemap.data[i2]) {
+ hasTiles = false;
+ break;
+ }
+ }
+ esri.zoomExtent[1] = hasTiles ? 22 : 19;
+ }).catch(function() {
+ });
+ };
+ esri.getMetadata = function(center, tileCoord, callback) {
+ if (esri.id !== "EsriWorldImagery") {
+ return callback(null, {});
+ }
+ var tileID = tileCoord.slice(0, 3).join("/");
+ var zoom = Math.min(tileCoord[2], esri.zoomExtent[1]);
+ var centerPoint = center[0] + "," + center[1];
+ var unknown = _t("info_panels.background.unknown");
+ var vintage = {};
+ var metadata = {};
+ if (inflight[tileID])
+ return;
+ var url = "https://services.arcgisonline.com/arcgis/rest/services/World_Imagery/MapServer/4/query";
+ url += "?returnGeometry=false&geometry=" + centerPoint + "&inSR=4326&geometryType=esriGeometryPoint&outFields=*&f=json";
+ if (!cache[tileID]) {
+ cache[tileID] = {};
+ }
+ if (cache[tileID] && cache[tileID].metadata) {
+ return callback(null, cache[tileID].metadata);
+ }
+ inflight[tileID] = true;
+ json_default(url).then(function(result) {
+ delete inflight[tileID];
+ result = result.features.map((f2) => f2.attributes).filter((a) => a.MinMapLevel <= zoom && a.MaxMapLevel >= zoom)[0];
+ 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);
+ }
+ var captureDate = localeDateString(result.SRC_DATE2);
+ vintage = {
+ start: captureDate,
+ end: captureDate,
+ range: captureDate
+ };
+ metadata = {
+ vintage,
+ source: clean2(result.NICE_NAME),
+ description: clean2(result.NICE_DESC),
+ resolution: clean2(+Number(result.SRC_RES).toFixed(4)),
+ accuracy: clean2(+Number(result.SRC_ACC).toFixed(4))
+ };
+ if (isFinite(metadata.resolution)) {
+ metadata.resolution += " m";
+ }
+ if (isFinite(metadata.accuracy)) {
+ metadata.accuracy += " m";
+ }
+ cache[tileID].metadata = metadata;
+ if (callback)
+ callback(null, metadata);
+ }).catch(function(err) {
+ delete inflight[tileID];
+ if (callback)
+ callback(err.message);
+ });
+ function clean2(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.append("background.none");
+ };
+ source.imageryUsed = function() {
+ return null;
+ };
+ source.area = function() {
+ return -1;
+ };
+ return source;
+ };
+ rendererBackgroundSource.Custom = function(template) {
+ var source = rendererBackgroundSource({ id: "custom", template });
+ source.name = function() {
+ return _t("background.custom");
+ };
+ source.label = function() {
+ return _t.append("background.custom");
+ };
+ source.imageryUsed = function() {
+ var cleaned = source.template();
+ 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);
+ }
+ cleaned = cleaned.replace(/token\/(\w+)/, "token/{apikey}").replace(/key=(\w+)/, "key={apikey}");
+ return "Custom (" + cleaned + " )";
+ };
+ source.area = function() {
+ return -2;
+ };
+ return source;
+ };
+
+ // node_modules/@turf/helpers/dist/es/index.js
+ var earthRadius = 63710088e-1;
+ var factors = {
+ centimeters: earthRadius * 100,
+ centimetres: earthRadius * 100,
+ degrees: earthRadius / 111325,
+ feet: earthRadius * 3.28084,
+ inches: earthRadius * 39.37,
+ kilometers: earthRadius / 1e3,
+ kilometres: earthRadius / 1e3,
+ meters: earthRadius,
+ metres: earthRadius,
+ miles: earthRadius / 1609.344,
+ millimeters: earthRadius * 1e3,
+ millimetres: earthRadius * 1e3,
+ nauticalmiles: earthRadius / 1852,
+ radians: 1,
+ yards: earthRadius * 1.0936
+ };
+ var unitsFactors = {
+ centimeters: 100,
+ centimetres: 100,
+ degrees: 1 / 111325,
+ feet: 3.28084,
+ inches: 39.37,
+ kilometers: 1 / 1e3,
+ kilometres: 1 / 1e3,
+ meters: 1,
+ metres: 1,
+ miles: 1 / 1609.344,
+ millimeters: 1e3,
+ millimetres: 1e3,
+ nauticalmiles: 1 / 1852,
+ radians: 1 / earthRadius,
+ yards: 1.0936133
+ };
+ function feature2(geom, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
+ }
+ var feat = { type: "Feature" };
+ if (options2.id === 0 || options2.id) {
+ feat.id = options2.id;
+ }
+ if (options2.bbox) {
+ feat.bbox = options2.bbox;
+ }
+ feat.properties = properties || {};
+ feat.geometry = geom;
+ return feat;
+ }
+ function polygon(coordinates, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
+ }
+ for (var _i = 0, coordinates_1 = coordinates; _i < coordinates_1.length; _i++) {
+ var ring = coordinates_1[_i];
+ if (ring.length < 4) {
+ throw new Error("Each LinearRing of a Polygon must have 4 or more Positions.");
+ }
+ for (var j2 = 0; j2 < ring[ring.length - 1].length; j2++) {
+ if (ring[ring.length - 1][j2] !== ring[0][j2]) {
+ throw new Error("First and last Position are not equivalent.");
+ }
+ }
+ }
+ var geom = {
+ type: "Polygon",
+ coordinates
+ };
+ return feature2(geom, properties, options2);
+ }
+ function lineString(coordinates, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
+ }
+ if (coordinates.length < 2) {
+ throw new Error("coordinates must be an array of two or more positions");
+ }
+ var geom = {
+ type: "LineString",
+ coordinates
+ };
+ return feature2(geom, properties, options2);
+ }
+ function multiLineString(coordinates, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
+ }
+ var geom = {
+ type: "MultiLineString",
+ coordinates
+ };
+ return feature2(geom, properties, options2);
+ }
+ function multiPolygon(coordinates, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
+ }
+ var geom = {
+ type: "MultiPolygon",
+ coordinates
+ };
+ return feature2(geom, properties, options2);
+ }
+
+ // node_modules/@turf/invariant/dist/es/index.js
+ function getGeom(geojson) {
+ if (geojson.type === "Feature") {
+ return geojson.geometry;
+ }
+ return geojson;
+ }
+
+ // node_modules/@turf/bbox-clip/dist/es/lib/lineclip.js
+ function lineclip(points, bbox2, result) {
+ var len = points.length, codeA = bitCode(points[0], bbox2), part = [], i2, codeB, lastCode;
+ var a;
+ var b;
+ if (!result)
+ result = [];
+ for (i2 = 1; i2 < len; i2++) {
+ a = points[i2 - 1];
+ b = points[i2];
+ codeB = lastCode = bitCode(b, bbox2);
+ while (true) {
+ if (!(codeA | codeB)) {
+ part.push(a);
+ if (codeB !== lastCode) {
+ part.push(b);
+ if (i2 < len - 1) {
+ result.push(part);
+ part = [];
+ }
+ } else if (i2 === len - 1) {
+ part.push(b);
+ }
+ break;
+ } else if (codeA & codeB) {
+ break;
+ } else if (codeA) {
+ a = intersect(a, b, codeA, bbox2);
+ codeA = bitCode(a, bbox2);
+ } else {
+ b = intersect(a, b, codeB, bbox2);
+ codeB = bitCode(b, bbox2);
+ }
+ }
+ codeA = lastCode;
+ }
+ if (part.length)
+ result.push(part);
+ return result;
+ }
+ function polygonclip(points, bbox2) {
+ var result, edge, prev, prevInside, i2, p, inside;
+ for (edge = 1; edge <= 8; edge *= 2) {
+ result = [];
+ prev = points[points.length - 1];
+ prevInside = !(bitCode(prev, bbox2) & edge);
+ for (i2 = 0; i2 < points.length; i2++) {
+ p = points[i2];
+ inside = !(bitCode(p, bbox2) & edge);
+ if (inside !== prevInside)
+ result.push(intersect(prev, p, edge, bbox2));
+ if (inside)
+ result.push(p);
+ prev = p;
+ prevInside = inside;
+ }
+ points = result;
+ if (!points.length)
+ break;
+ }
+ return result;
+ }
+ function intersect(a, b, edge, bbox2) {
+ return edge & 8 ? [a[0] + (b[0] - a[0]) * (bbox2[3] - a[1]) / (b[1] - a[1]), bbox2[3]] : edge & 4 ? [a[0] + (b[0] - a[0]) * (bbox2[1] - a[1]) / (b[1] - a[1]), bbox2[1]] : edge & 2 ? [bbox2[2], a[1] + (b[1] - a[1]) * (bbox2[2] - a[0]) / (b[0] - a[0])] : edge & 1 ? [bbox2[0], a[1] + (b[1] - a[1]) * (bbox2[0] - a[0]) / (b[0] - a[0])] : null;
+ }
+ function bitCode(p, bbox2) {
+ var code = 0;
+ if (p[0] < bbox2[0])
+ code |= 1;
+ else if (p[0] > bbox2[2])
+ code |= 2;
+ if (p[1] < bbox2[1])
+ code |= 4;
+ else if (p[1] > bbox2[3])
+ code |= 8;
+ return code;
+ }
+
+ // node_modules/@turf/bbox-clip/dist/es/index.js
+ function bboxClip(feature3, bbox2) {
+ var geom = getGeom(feature3);
+ var type2 = geom.type;
+ var properties = feature3.type === "Feature" ? feature3.properties : {};
+ var coords = geom.coordinates;
+ switch (type2) {
+ case "LineString":
+ case "MultiLineString": {
+ var lines_1 = [];
+ if (type2 === "LineString") {
+ coords = [coords];
+ }
+ coords.forEach(function(line) {
+ lineclip(line, bbox2, lines_1);
+ });
+ if (lines_1.length === 1) {
+ return lineString(lines_1[0], properties);
+ }
+ return multiLineString(lines_1, properties);
+ }
+ case "Polygon":
+ return polygon(clipPolygon(coords, bbox2), properties);
+ case "MultiPolygon":
+ return multiPolygon(coords.map(function(poly) {
+ return clipPolygon(poly, bbox2);
+ }), properties);
+ default:
+ throw new Error("geometry " + type2 + " not supported");
+ }
+ }
+ function clipPolygon(rings, bbox2) {
+ var outRings = [];
+ for (var _i = 0, rings_1 = rings; _i < rings_1.length; _i++) {
+ var ring = rings_1[_i];
+ var clipped = polygonclip(ring, bbox2);
+ 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]);
+ }
+ if (clipped.length >= 4) {
+ outRings.push(clipped);
+ }
+ }
+ }
+ return outRings;
+ }
+
+ // node_modules/@turf/meta/dist/es/index.js
+ function coordEach(geojson, callback, excludeWrapCoord) {
+ if (geojson === null)
+ return;
+ var j2, k, l, geometry, stopG, coords, geometryMaybeCollection, wrapShrink = 0, coordIndex = 0, isGeometryCollection, type2 = geojson.type, isFeatureCollection = type2 === "FeatureCollection", isFeature = type2 === "Feature", stop = isFeatureCollection ? geojson.features.length : 1;
+ for (var featureIndex = 0; featureIndex < stop; featureIndex++) {
+ geometryMaybeCollection = isFeatureCollection ? geojson.features[featureIndex].geometry : isFeature ? geojson.geometry : geojson;
+ isGeometryCollection = geometryMaybeCollection ? geometryMaybeCollection.type === "GeometryCollection" : false;
+ stopG = isGeometryCollection ? geometryMaybeCollection.geometries.length : 1;
+ for (var geomIndex = 0; geomIndex < stopG; geomIndex++) {
+ var multiFeatureIndex = 0;
+ var geometryIndex = 0;
+ geometry = isGeometryCollection ? geometryMaybeCollection.geometries[geomIndex] : geometryMaybeCollection;
+ if (geometry === null)
+ continue;
+ coords = geometry.coordinates;
+ var geomType = geometry.type;
+ wrapShrink = excludeWrapCoord && (geomType === "Polygon" || geomType === "MultiPolygon") ? 1 : 0;
+ switch (geomType) {
+ case null:
+ break;
+ case "Point":
+ if (callback(
+ coords,
+ coordIndex,
+ featureIndex,
+ multiFeatureIndex,
+ geometryIndex
+ ) === false)
+ return false;
+ coordIndex++;
+ multiFeatureIndex++;
+ break;
+ case "LineString":
+ case "MultiPoint":
+ for (j2 = 0; j2 < coords.length; j2++) {
+ if (callback(
+ coords[j2],
+ coordIndex,
+ featureIndex,
+ multiFeatureIndex,
+ geometryIndex
+ ) === false)
+ return false;
+ coordIndex++;
+ if (geomType === "MultiPoint")
+ multiFeatureIndex++;
+ }
+ if (geomType === "LineString")
+ multiFeatureIndex++;
+ break;
+ case "Polygon":
+ case "MultiLineString":
+ for (j2 = 0; j2 < coords.length; j2++) {
+ for (k = 0; k < coords[j2].length - wrapShrink; k++) {
+ if (callback(
+ coords[j2][k],
+ coordIndex,
+ featureIndex,
+ multiFeatureIndex,
+ geometryIndex
+ ) === false)
+ return false;
+ coordIndex++;
+ }
+ if (geomType === "MultiLineString")
+ multiFeatureIndex++;
+ if (geomType === "Polygon")
+ geometryIndex++;
+ }
+ if (geomType === "Polygon")
+ multiFeatureIndex++;
+ break;
+ case "MultiPolygon":
+ for (j2 = 0; j2 < coords.length; j2++) {
+ geometryIndex = 0;
+ for (k = 0; k < coords[j2].length; k++) {
+ for (l = 0; l < coords[j2][k].length - wrapShrink; l++) {
+ if (callback(
+ coords[j2][k][l],
+ coordIndex,
+ featureIndex,
+ multiFeatureIndex,
+ geometryIndex
+ ) === false)
+ return false;
+ coordIndex++;
+ }
+ geometryIndex++;
+ }
+ multiFeatureIndex++;
+ }
+ break;
+ case "GeometryCollection":
+ for (j2 = 0; j2 < geometry.geometries.length; j2++)
+ if (coordEach(geometry.geometries[j2], callback, excludeWrapCoord) === false)
+ return false;
+ break;
+ default:
+ throw new Error("Unknown Geometry Type");
+ }
+ }
+ }
+ }
+
+ // node_modules/@turf/bbox/dist/es/index.js
+ function bbox(geojson) {
+ var result = [Infinity, Infinity, -Infinity, -Infinity];
+ coordEach(geojson, function(coord2) {
+ if (result[0] > coord2[0]) {
+ result[0] = coord2[0];
+ }
+ if (result[1] > coord2[1]) {
+ result[1] = coord2[1];
+ }
+ if (result[2] < coord2[0]) {
+ result[2] = coord2[0];
+ }
+ if (result[3] < coord2[1]) {
+ result[3] = coord2[1];
+ }
+ });
+ return result;
+ }
+ bbox["default"] = bbox;
+ var es_default = bbox;
+
+ // modules/renderer/background.js
+ var import_which_polygon4 = __toESM(require_which_polygon());
+
+ // modules/renderer/tile_layer.js
+ function rendererTileLayer(context) {
+ var transformProp = utilPrefixCSSProperty("Transform");
+ var tiler8 = utilTiler();
+ var _tileSize = 256;
+ var _projection;
+ var _cache4 = {};
+ var _tileOrigin;
+ var _zoom;
+ var _source;
+ function tileSizeAtZoom(d, z) {
+ var EPSILON = 2e-3;
+ return _tileSize * Math.pow(2, z - d[2]) / _tileSize + EPSILON;
+ }
+ function atZoom(t, distance) {
+ var power = Math.pow(2, distance);
+ return [
+ Math.floor(t[0] * power),
+ Math.floor(t[1] * power),
+ t[2] + distance
+ ];
+ }
+ function lookUp(d) {
+ for (var up = -1; up > -d[2]; up--) {
+ var tile = atZoom(d, up);
+ if (_cache4[_source.url(tile)] !== false) {
+ return tile;
+ }
+ }
+ }
+ function uniqueBy(a, n2) {
+ var o = [];
+ var seen = {};
+ for (var i2 = 0; i2 < a.length; i2++) {
+ if (seen[a[i2][n2]] === void 0) {
+ o.push(a[i2]);
+ seen[a[i2][n2]] = true;
+ }
+ }
+ return o;
+ }
+ function addSource(d) {
+ d.push(_source.url(d));
+ return d;
+ }
+ function background(selection2) {
+ _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 {
+ pixelOffset = [0, 0];
+ }
+ var translate = [
+ _projection.translate()[0] + pixelOffset[0],
+ _projection.translate()[1] + pixelOffset[1]
+ ];
+ tiler8.scale(_projection.scale() * 2 * Math.PI).translate(translate);
+ _tileOrigin = [
+ _projection.scale() * Math.PI - translate[0],
+ _projection.scale() * Math.PI - translate[1]
+ ];
+ render(selection2);
+ }
+ function render(selection2) {
+ if (!_source)
+ return;
+ var requests = [];
+ var showDebug = context.getDebug("tile") && !_source.overlay;
+ if (_source.validZoom(_zoom)) {
+ tiler8.skipNullIsland(!!_source.overlay);
+ tiler8().forEach(function(d) {
+ addSource(d);
+ if (d[3] === "")
+ return;
+ if (typeof d[3] !== "string")
+ return;
+ requests.push(d);
+ if (_cache4[d[3]] === false && lookUp(d)) {
+ requests.push(addSource(lookUp(d)));
+ }
+ });
+ requests = uniqueBy(requests, 3).filter(function(r) {
+ return _cache4[r[3]] !== false;
+ });
+ }
+ function load(d3_event, d) {
+ _cache4[d[3]] = true;
+ select_default2(this).on("error", null).on("load", null).classed("tile-loaded", true);
+ render(selection2);
+ }
+ function error(d3_event, d) {
+ _cache4[d[3]] = false;
+ select_default2(this).on("error", null).on("load", null).remove();
+ render(selection2);
+ }
+ function imageTransform(d) {
+ var ts = _tileSize * Math.pow(2, _zoom - d[2]);
+ var scale = tileSizeAtZoom(d, _zoom);
+ return "translate(" + (d[0] * ts - _tileOrigin[0]) + "px," + (d[1] * ts - _tileOrigin[1]) + "px) scale(" + scale + "," + scale + ")";
+ }
+ function tileCenter(d) {
+ var ts = _tileSize * Math.pow(2, _zoom - d[2]);
+ return [
+ d[0] * ts - _tileOrigin[0] + ts / 2,
+ d[1] * ts - _tileOrigin[1] + ts / 2
+ ];
+ }
+ function debugTransform(d) {
+ var coord2 = tileCenter(d);
+ return "translate(" + coord2[0] + "px," + coord2[1] + "px)";
+ }
+ var dims = tiler8.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);
+ if (dist < minDist) {
+ minDist = dist;
+ nearCenter = d;
+ }
+ });
+ var image = selection2.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_default2(this);
+ window.setTimeout(function() {
+ if (tile.classed("tile-removing")) {
+ tile.remove();
+ }
+ }, 300);
+ });
+ image.enter().append("img").attr("class", "tile").attr("alt", "").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 debug2 = selection2.selectAll(".tile-label-debug").data(showDebug ? requests : [], function(d) {
+ return d[3];
+ });
+ debug2.exit().remove();
+ if (showDebug) {
+ var debugEnter = debug2.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");
+ debug2 = debug2.merge(debugEnter);
+ debug2.style(transformProp, debugTransform);
+ debug2.selectAll(".tile-label-debug-coord").text(function(d) {
+ return d[2] + " / " + d[0] + " / " + d[1];
+ });
+ debug2.selectAll(".tile-label-debug-vintage").each(function(d) {
+ var span = select_default2(this);
+ var center = context.projection.invert(tileCenter(d));
+ _source.getMetadata(center, d, function(err, result) {
+ if (result && result.vintage && result.vintage.range) {
+ span.text(result.vintage.range);
+ } else {
+ span.text("");
+ span.call(_t.append("info_panels.background.vintage"));
+ span.append("span").text(": ");
+ span.call(_t.append("info_panels.background.unknown"));
+ }
+ });
+ });
+ }
+ }
+ background.projection = function(val) {
+ if (!arguments.length)
+ return _projection;
+ _projection = val;
+ return background;
+ };
+ background.dimensions = function(val) {
+ if (!arguments.length)
+ return tiler8.size();
+ tiler8.size(val);
+ return background;
+ };
+ background.source = function(val) {
+ if (!arguments.length)
+ return _source;
+ _source = val;
+ _tileSize = _source.tileSize;
+ _cache4 = {};
+ tiler8.tileSize(_source.tileSize).zoomExtent(_source.zoomExtent);
+ return background;
+ };
+ return background;
+ }
+
+ // modules/renderer/background.js
+ var _imageryIndex = null;
+ function rendererBackground(context) {
+ const dispatch10 = dispatch_default("change");
+ const baseLayer = rendererTileLayer(context).projection(context.projection);
+ let _checkedBlocklists = [];
+ let _isValid = true;
+ let _overlayLayers = [];
+ let _brightness = 1;
+ let _contrast = 1;
+ let _saturation = 1;
+ let _sharpness = 1;
+ function ensureImageryIndex() {
+ return _mainFileFetcher.get("imagery").then((sources) => {
+ if (_imageryIndex)
+ return _imageryIndex;
+ _imageryIndex = {
+ imagery: sources,
+ features: {}
+ };
+ const features = sources.map((source) => {
+ if (!source.polygon)
+ return null;
+ const rings = source.polygon.map((ring) => [ring]);
+ const feature3 = {
+ type: "Feature",
+ properties: { id: source.id },
+ geometry: { type: "MultiPolygon", coordinates: rings }
+ };
+ _imageryIndex.features[source.id] = feature3;
+ return feature3;
+ }).filter(Boolean);
+ _imageryIndex.query = (0, import_which_polygon4.default)({ type: "FeatureCollection", features });
+ _imageryIndex.backgrounds = sources.map((source) => {
+ if (source.type === "bing") {
+ return rendererBackgroundSource.Bing(source, dispatch10);
+ } else if (/^EsriWorldImagery/.test(source.id)) {
+ return rendererBackgroundSource.Esri(source);
+ } else {
+ return rendererBackgroundSource(source);
+ }
+ });
+ _imageryIndex.backgrounds.unshift(rendererBackgroundSource.None());
+ let template = corePreferences("background-custom-template") || "";
+ const custom = rendererBackgroundSource.Custom(template);
+ _imageryIndex.backgrounds.unshift(custom);
+ return _imageryIndex;
+ });
+ }
+ function background(selection2) {
+ const currSource = baseLayer.source();
+ if (context.map().zoom() > 18) {
+ if (currSource && /^EsriWorldImagery/.test(currSource.id)) {
+ const center = context.map().center();
+ currSource.fetchTilemap(center);
+ }
+ }
+ const sources = background.sources(context.map().extent());
+ const wasValid = _isValid;
+ _isValid = !!sources.filter((d) => d === currSource).length;
+ if (wasValid !== _isValid) {
+ background.updateImagery();
+ }
+ let baseFilter = "";
+ if (_brightness !== 1) {
+ baseFilter += ` brightness(${_brightness})`;
+ }
+ if (_contrast !== 1) {
+ baseFilter += ` contrast(${_contrast})`;
+ }
+ if (_saturation !== 1) {
+ baseFilter += ` saturate(${_saturation})`;
+ }
+ if (_sharpness < 1) {
+ const blur = number_default(0.5, 5)(1 - _sharpness);
+ baseFilter += ` blur(${blur}px)`;
+ }
+ let base = selection2.selectAll(".layer-background").data([0]);
+ base = base.enter().insert("div", ".layer-data").attr("class", "layer layer-background").merge(base);
+ base.style("filter", baseFilter || null);
+ let imagery = base.selectAll(".layer-imagery").data([0]);
+ imagery.enter().append("div").attr("class", "layer layer-imagery").merge(imagery).call(baseLayer);
+ let maskFilter = "";
+ let mixBlendMode = "";
+ if (_sharpness > 1) {
+ mixBlendMode = "overlay";
+ maskFilter = "saturate(0) blur(3px) invert(1)";
+ let contrast = _sharpness - 1;
+ maskFilter += ` contrast(${contrast})`;
+ let brightness = number_default(1, 0.85)(_sharpness - 1);
+ maskFilter += ` brightness(${brightness})`;
+ }
+ let mask = base.selectAll(".layer-unsharp-mask").data(_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);
+ let overlays = selection2.selectAll(".layer-overlay").data(_overlayLayers, (d) => d.source().name());
+ overlays.exit().remove();
+ overlays.enter().insert("div", ".layer-data").attr("class", "layer layer-overlay").merge(overlays).each((layer, i2, nodes) => select_default2(nodes[i2]).call(layer));
+ }
+ background.updateImagery = function() {
+ let currSource = baseLayer.source();
+ if (context.inIntro() || !currSource)
+ return;
+ let o = _overlayLayers.filter((d) => !d.source().isLocatorOverlay() && !d.source().isHidden()).map((d) => d.source().id).join(",");
+ const meters = geoOffsetToMeters(currSource.offset());
+ const EPSILON = 0.01;
+ const x = +meters[0].toFixed(2);
+ const y = +meters[1].toFixed(2);
+ let hash = utilStringQs(window.location.hash);
+ let id2 = currSource.id;
+ if (id2 === "custom") {
+ id2 = `custom:${currSource.template()}`;
+ }
+ if (id2) {
+ hash.background = id2;
+ } else {
+ delete hash.background;
+ }
+ if (o) {
+ hash.overlays = o;
+ } else {
+ delete hash.overlays;
+ }
+ if (Math.abs(x) > EPSILON || Math.abs(y) > EPSILON) {
+ hash.offset = `${x},${y}`;
+ } else {
+ delete hash.offset;
+ }
+ if (!window.mocha) {
+ window.location.replace("#" + utilQsString(hash, true));
+ }
+ let imageryUsed = [];
+ let photoOverlaysUsed = [];
+ const currUsed = currSource.imageryUsed();
+ if (currUsed && _isValid) {
+ imageryUsed.push(currUsed);
+ }
+ _overlayLayers.filter((d) => !d.source().isLocatorOverlay() && !d.source().isHidden()).forEach((d) => imageryUsed.push(d.source().imageryUsed()));
+ const dataLayer = context.layers().layer("data");
+ if (dataLayer && dataLayer.enabled() && dataLayer.hasData()) {
+ imageryUsed.push(dataLayer.getSrc());
+ }
+ const photoOverlayLayers = {
+ streetside: "Bing Streetside",
+ mapillary: "Mapillary Images",
+ "mapillary-map-features": "Mapillary Map Features",
+ "mapillary-signs": "Mapillary Signs",
+ kartaview: "KartaView Images"
+ };
+ for (let layerID in photoOverlayLayers) {
+ const layer = context.layers().layer(layerID);
+ if (layer && layer.enabled()) {
+ photoOverlaysUsed.push(layerID);
+ imageryUsed.push(photoOverlayLayers[layerID]);
+ }
+ }
+ context.history().imageryUsed(imageryUsed);
+ context.history().photoOverlaysUsed(photoOverlaysUsed);
+ };
+ background.sources = (extent, zoom, includeCurrent) => {
+ if (!_imageryIndex)
+ return [];
+ let visible = {};
+ (_imageryIndex.query.bbox(extent.rectangle(), true) || []).forEach((d) => visible[d.id] = true);
+ const currSource = baseLayer.source();
+ const osm = context.connection();
+ const blocklists = osm && osm.imageryBlocklists() || [];
+ const blocklistChanged = blocklists.length !== _checkedBlocklists.length || blocklists.some((regex, index) => String(regex) !== _checkedBlocklists[index]);
+ if (blocklistChanged) {
+ _imageryIndex.backgrounds.forEach((source) => {
+ source.isBlocked = blocklists.some((regex) => regex.test(source.template()));
+ });
+ _checkedBlocklists = blocklists.map((regex) => String(regex));
+ }
+ return _imageryIndex.backgrounds.filter((source) => {
+ if (includeCurrent && currSource === source)
+ return true;
+ if (source.isBlocked)
+ return false;
+ if (!source.polygon)
+ return true;
+ if (zoom && zoom < 6)
+ return false;
+ return visible[source.id];
+ });
+ };
+ background.dimensions = (val) => {
+ if (!val)
+ return;
+ baseLayer.dimensions(val);
+ _overlayLayers.forEach((layer) => layer.dimensions(val));
+ };
+ background.baseLayerSource = function(d) {
+ if (!arguments.length)
+ return baseLayer.source();
+ const osm = context.connection();
+ if (!osm)
+ return background;
+ const blocklists = osm.imageryBlocklists();
+ const template = d.template();
+ let fail = false;
+ let tested = 0;
+ let regex;
+ for (let i2 = 0; i2 < blocklists.length; i2++) {
+ regex = blocklists[i2];
+ fail = regex.test(template);
+ tested++;
+ if (fail)
+ break;
+ }
+ if (!tested) {
+ regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
+ fail = regex.test(template);
+ }
+ baseLayer.source(!fail ? d : background.findSource("none"));
+ dispatch10.call("change");
+ background.updateImagery();
+ return background;
+ };
+ background.findSource = (id2) => {
+ if (!id2 || !_imageryIndex)
+ return null;
+ return _imageryIndex.backgrounds.find((d) => d.id && d.id === id2);
+ };
+ background.bing = () => {
+ background.baseLayerSource(background.findSource("Bing"));
+ };
+ background.showsLayer = (d) => {
+ const currSource = baseLayer.source();
+ if (!d || !currSource)
+ return false;
+ return d.id === currSource.id || _overlayLayers.some((layer) => d.id === layer.source().id);
+ };
+ background.overlayLayerSources = () => {
+ return _overlayLayers.map((layer) => layer.source());
+ };
+ background.toggleOverlayLayer = (d) => {
+ let layer;
+ for (let i2 = 0; i2 < _overlayLayers.length; i2++) {
+ layer = _overlayLayers[i2];
+ if (layer.source() === d) {
+ _overlayLayers.splice(i2, 1);
+ dispatch10.call("change");
+ background.updateImagery();
+ return;
+ }
+ }
+ layer = rendererTileLayer(context).source(d).projection(context.projection).dimensions(
+ baseLayer.dimensions()
+ );
+ _overlayLayers.push(layer);
+ dispatch10.call("change");
+ background.updateImagery();
+ };
+ background.nudge = (d, zoom) => {
+ const currSource = baseLayer.source();
+ if (currSource) {
+ currSource.nudge(d, zoom);
+ dispatch10.call("change");
+ background.updateImagery();
+ }
+ return background;
+ };
+ background.offset = function(d) {
+ const currSource = baseLayer.source();
+ if (!arguments.length) {
+ return currSource && currSource.offset() || [0, 0];
+ }
+ if (currSource) {
+ currSource.offset(d);
+ dispatch10.call("change");
+ background.updateImagery();
+ }
+ return background;
+ };
+ background.brightness = function(d) {
+ if (!arguments.length)
+ return _brightness;
+ _brightness = d;
+ if (context.mode())
+ dispatch10.call("change");
+ return background;
+ };
+ background.contrast = function(d) {
+ if (!arguments.length)
+ return _contrast;
+ _contrast = d;
+ if (context.mode())
+ dispatch10.call("change");
+ return background;
+ };
+ background.saturation = function(d) {
+ if (!arguments.length)
+ return _saturation;
+ _saturation = d;
+ if (context.mode())
+ dispatch10.call("change");
+ return background;
+ };
+ background.sharpness = function(d) {
+ if (!arguments.length)
+ return _sharpness;
+ _sharpness = d;
+ if (context.mode())
+ dispatch10.call("change");
+ return background;
+ };
+ let _loadPromise;
+ background.ensureLoaded = () => {
+ if (_loadPromise)
+ return _loadPromise;
+ return _loadPromise = ensureImageryIndex();
+ };
+ background.init = () => {
+ const loadPromise = background.ensureLoaded();
+ const hash = utilStringQs(window.location.hash);
+ const requestedBackground = hash.background || hash.layer;
+ const lastUsedBackground = corePreferences("background-last-used");
+ return loadPromise.then((imageryIndex) => {
+ const extent = context.map().extent();
+ const validBackgrounds = background.sources(extent).filter((d) => d.id !== "none" && d.id !== "custom");
+ const first = validBackgrounds.length && validBackgrounds[0];
+ const isLastUsedValid = !!validBackgrounds.find((d) => d.id && d.id === lastUsedBackground);
+ let best;
+ if (!requestedBackground && extent) {
+ const viewArea = extent.area();
+ best = validBackgrounds.find((s) => {
+ if (!s.best() || s.overlay)
+ return false;
+ let bbox2 = es_default(bboxClip(
+ { type: "MultiPolygon", coordinates: [s.polygon || [extent.polygon()]] },
+ extent.rectangle()
+ ));
+ let area = geoExtent(bbox2.slice(0, 2), bbox2.slice(2, 4)).area();
+ return area / viewArea > 0.5;
+ });
+ }
+ if (requestedBackground && requestedBackground.indexOf("custom:") === 0) {
+ const template = requestedBackground.replace(/^custom:/, "");
+ const custom = background.findSource("custom");
+ background.baseLayerSource(custom.template(template));
+ corePreferences("background-custom-template", template);
+ } else {
+ background.baseLayerSource(
+ background.findSource(requestedBackground) || best || isLastUsedValid && background.findSource(lastUsedBackground) || background.findSource("Bing") || first || background.findSource("none")
+ );
+ }
+ const locator = imageryIndex.backgrounds.find((d) => d.overlay && d.default);
+ if (locator) {
+ background.toggleOverlayLayer(locator);
+ }
+ const overlays = (hash.overlays || "").split(",");
+ overlays.forEach((overlay) => {
+ overlay = background.findSource(overlay);
+ if (overlay) {
+ background.toggleOverlayLayer(overlay);
+ }
+ });
+ if (hash.gpx) {
+ const gpx2 = context.layers().layer("data");
+ if (gpx2) {
+ gpx2.url(hash.gpx, ".gpx");
+ }
+ }
+ if (hash.offset) {
+ const offset = hash.offset.replace(/;/g, ",").split(",").map((n2) => !isNaN(n2) && n2);
+ if (offset.length === 2) {
+ background.offset(geoMetersToOffset(offset));
+ }
+ }
+ }).catch((err) => {
+ console.error(err);
+ });
+ };
+ return utilRebind(background, dispatch10, "on");
+ }
+
+ // modules/renderer/features.js
+ function rendererFeatures(context) {
+ var dispatch10 = dispatch_default("change", "redraw");
+ var features = utilRebind({}, dispatch10, "on");
+ var _deferred2 = /* @__PURE__ */ new Set();
+ 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,
+ "busway": 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 _cullFactor = 1;
+ var _cache4 = {};
+ var _rules = {};
+ var _stats = {};
+ var _keys = [];
+ var _hidden = [];
+ var _forceVisible = {};
+ function update() {
+ if (!window.mocha) {
+ var hash = utilStringQs(window.location.hash);
+ var disabled = features.disabled();
+ if (disabled.length) {
+ hash.disable_features = disabled.join(",");
+ } else {
+ delete hash.disable_features;
+ }
+ window.location.replace("#" + utilQsString(hash, true));
+ corePreferences("disabled-features", disabled.join(","));
+ }
+ _hidden = features.hidden();
+ dispatch10.call("change");
+ dispatch10.call("redraw");
+ }
+ function defineRule(k, filter2, max3) {
+ var isEnabled = true;
+ _keys.push(k);
+ _rules[k] = {
+ filter: filter2,
+ enabled: isEnabled,
+ // whether the user wants it enabled..
+ count: 0,
+ currentMax: max3 || Infinity,
+ defaultMax: max3 || Infinity,
+ enable: function() {
+ this.enabled = true;
+ this.currentMax = this.defaultMax;
+ },
+ disable: function() {
+ this.enabled = false;
+ this.currentMax = 0;
+ },
+ hidden: function() {
+ return this.count === 0 && !this.enabled || this.count > this.currentMax * _cullFactor;
+ },
+ autoHidden: function() {
+ return this.hidden() && this.currentMax > 0;
+ }
+ };
+ }
+ 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, geometry) {
+ return (geometry === "line" && !!tags.boundary || geometry === "relation" && tags.type === "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;
+ });
+ 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);
+ for (var i2 = 0; i2 < strings.length; i2++) {
+ var s = strings[i2];
+ if (osmLifecyclePrefixes[s] || osmLifecyclePrefixes[tags[s]])
+ return true;
+ }
+ return false;
+ });
+ defineRule("others", function isOther(tags, geometry) {
+ return geometry === "line" || geometry === "area";
+ });
+ features.features = function() {
+ return _rules;
+ };
+ features.keys = function() {
+ return _keys;
+ };
+ features.enabled = function(k) {
+ if (!arguments.length) {
+ return _keys.filter(function(k2) {
+ return _rules[k2].enabled;
+ });
+ }
+ return _rules[k] && _rules[k].enabled;
+ };
+ features.disabled = function(k) {
+ if (!arguments.length) {
+ return _keys.filter(function(k2) {
+ return !_rules[k2].enabled;
+ });
+ }
+ return _rules[k] && !_rules[k].enabled;
+ };
+ features.hidden = function(k) {
+ if (!arguments.length) {
+ return _keys.filter(function(k2) {
+ return _rules[k2].hidden();
+ });
+ }
+ return _rules[k] && _rules[k].hidden();
+ };
+ features.autoHidden = function(k) {
+ if (!arguments.length) {
+ return _keys.filter(function(k2) {
+ return _rules[k2].autoHidden();
+ });
+ }
+ return _rules[k] && _rules[k].autoHidden();
+ };
+ features.enable = function(k) {
+ if (_rules[k] && !_rules[k].enabled) {
+ _rules[k].enable();
+ update();
+ }
+ };
+ features.enableAll = function() {
+ var didEnable = false;
+ for (var k in _rules) {
+ if (!_rules[k].enabled) {
+ didEnable = true;
+ _rules[k].enable();
+ }
+ }
+ if (didEnable)
+ update();
+ };
+ features.disable = function(k) {
+ if (_rules[k] && _rules[k].enabled) {
+ _rules[k].disable();
+ update();
+ }
+ };
+ features.disableAll = function() {
+ var didDisable = false;
+ for (var k in _rules) {
+ if (_rules[k].enabled) {
+ didDisable = true;
+ _rules[k].disable();
+ }
+ }
+ if (didDisable)
+ update();
+ };
+ features.toggle = function(k) {
+ if (_rules[k]) {
+ (function(f2) {
+ return f2.enabled ? f2.disable() : f2.enable();
+ })(_rules[k]);
+ update();
+ }
+ };
+ features.resetStats = function() {
+ for (var i2 = 0; i2 < _keys.length; i2++) {
+ _rules[_keys[i2]].count = 0;
+ }
+ dispatch10.call("change");
+ };
+ 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, i2, j2;
+ for (i2 = 0; i2 < _keys.length; i2++) {
+ _rules[_keys[i2]].count = 0;
+ }
+ _cullFactor = dimensions[0] * dimensions[1] / 1e6;
+ for (i2 = 0; i2 < entities.length; i2++) {
+ geometry = entities[i2].geometry(resolver);
+ matches = Object.keys(features.getMatches(entities[i2], resolver, geometry));
+ for (j2 = 0; j2 < matches.length; j2++) {
+ _rules[matches[j2]].count++;
+ }
+ }
+ currHidden = features.hidden();
+ if (currHidden !== _hidden) {
+ _hidden = currHidden;
+ needsRedraw = true;
+ dispatch10.call("change");
+ }
+ return needsRedraw;
+ };
+ features.stats = function() {
+ for (var i2 = 0; i2 < _keys.length; i2++) {
+ _stats[_keys[i2]] = _rules[_keys[i2]].count;
+ }
+ return _stats;
+ };
+ features.clear = function(d) {
+ for (var i2 = 0; i2 < d.length; i2++) {
+ features.clearEntity(d[i2]);
+ }
+ };
+ features.clearEntity = function(entity) {
+ delete _cache4[osmEntity.key(entity)];
+ };
+ features.reset = function() {
+ Array.from(_deferred2).forEach(function(handle) {
+ window.cancelIdleCallback(handle);
+ _deferred2.delete(handle);
+ });
+ _cache4 = {};
+ };
+ function relationShouldBeChecked(relation) {
+ return relation.tags.type === "boundary";
+ }
+ features.getMatches = function(entity, resolver, geometry) {
+ if (geometry === "vertex" || geometry === "relation" && !relationShouldBeChecked(entity))
+ return {};
+ var ent = osmEntity.key(entity);
+ if (!_cache4[ent]) {
+ _cache4[ent] = {};
+ }
+ if (!_cache4[ent].matches) {
+ var matches = {};
+ var hasMatch = false;
+ for (var i2 = 0; i2 < _keys.length; i2++) {
+ if (_keys[i2] === "others") {
+ if (hasMatch)
+ continue;
+ if (entity.type === "way") {
+ var parents = features.getParents(entity, resolver, geometry);
+ 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";
+ })) {
+ var pkey = osmEntity.key(parents[0]);
+ if (_cache4[pkey] && _cache4[pkey].matches) {
+ matches = Object.assign({}, _cache4[pkey].matches);
+ continue;
+ }
+ }
+ }
+ }
+ if (_rules[_keys[i2]].filter(entity.tags, geometry)) {
+ matches[_keys[i2]] = hasMatch = true;
+ }
+ }
+ _cache4[ent].matches = matches;
+ }
+ return _cache4[ent].matches;
+ };
+ features.getParents = function(entity, resolver, geometry) {
+ if (geometry === "point")
+ return [];
+ var ent = osmEntity.key(entity);
+ if (!_cache4[ent]) {
+ _cache4[ent] = {};
+ }
+ if (!_cache4[ent].parents) {
+ var parents = [];
+ if (geometry === "vertex") {
+ parents = resolver.parentWays(entity);
+ } else {
+ parents = resolver.parentRelations(entity);
+ }
+ _cache4[ent].parents = parents;
+ }
+ return _cache4[ent].parents;
+ };
+ features.isHiddenPreset = function(preset, geometry) {
+ if (!_hidden.length)
+ return false;
+ if (!preset.tags)
+ return false;
+ var test = preset.setTags({}, geometry);
+ for (var key in _rules) {
+ if (_rules[key].filter(test, geometry)) {
+ if (_hidden.indexOf(key) !== -1) {
+ return key;
+ }
+ return false;
+ }
+ }
+ 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);
+ });
+ };
+ 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;
+ for (var i2 = 0; i2 < parents.length; i2++) {
+ if (!features.isHidden(parents[i2], resolver, parents[i2].geometry(resolver))) {
+ return false;
+ }
+ }
+ return true;
+ };
+ features.hasHiddenConnections = function(entity, resolver) {
+ if (!_hidden.length)
+ return false;
+ var childNodes, connections;
+ 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));
+ }
+ 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));
+ });
+ };
+ 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);
+ };
+ features.filter = function(d, resolver) {
+ if (!_hidden.length)
+ return d;
+ var result = [];
+ for (var i2 = 0; i2 < d.length; i2++) {
+ var entity = d[i2];
+ if (!features.isHidden(entity, resolver, entity.geometry(resolver))) {
+ result.push(entity);
+ }
+ }
+ return result;
+ };
+ features.forceVisible = function(entityIDs) {
+ if (!arguments.length)
+ return Object.keys(_forceVisible);
+ _forceVisible = {};
+ for (var i2 = 0; i2 < entityIDs.length; i2++) {
+ _forceVisible[entityIDs[i2]] = true;
+ var entity = context.hasEntity(entityIDs[i2]);
+ if (entity && entity.type === "relation") {
+ for (var j2 in entity.members) {
+ _forceVisible[entity.members[j2].id] = true;
+ }
+ }
+ }
+ return features;
+ };
+ features.init = function() {
+ var storage = corePreferences("disabled-features");
+ if (storage) {
+ var storageDisabled = storage.replace(/;/g, ",").split(",");
+ storageDisabled.forEach(features.disable);
+ }
+ var hash = utilStringQs(window.location.hash);
+ if (hash.disable_features) {
+ var hashDisabled = hash.disable_features.replace(/;/g, ",").split(",");
+ hashDisabled.forEach(features.disable);
+ }
+ };
+ context.history().on("merge.features", function(newEntities) {
+ if (!newEntities)
+ return;
+ var handle = window.requestIdleCallback(function() {
+ var graph = context.graph();
+ var types = utilArrayGroupBy(newEntities, "type");
+ var entities = [].concat(types.relation || [], types.way || [], types.node || []);
+ for (var i2 = 0; i2 < entities.length; i2++) {
+ var geometry = entities[i2].geometry(graph);
+ features.getMatches(entities[i2], graph, geometry);
+ }
+ });
+ _deferred2.add(handle);
+ });
+ return features;
+ }
+
+ // modules/svg/areas.js
+ var import_fast_deep_equal5 = __toESM(require_fast_deep_equal());
+
+ // modules/svg/helpers.js
+ function svgPassiveVertex(node, graph, activeID) {
+ if (!activeID)
+ return 1;
+ if (activeID === node.id)
+ return 0;
+ var parents = graph.parentWays(node);
+ var i2, j2, nodes, isClosed, ix1, ix2, ix3, ix4, max3;
+ for (i2 = 0; i2 < parents.length; i2++) {
+ nodes = parents[i2].nodes;
+ isClosed = parents[i2].isClosed();
+ for (j2 = 0; j2 < nodes.length; j2++) {
+ if (nodes[j2] === node.id) {
+ ix1 = j2 - 2;
+ ix2 = j2 - 1;
+ ix3 = j2 + 1;
+ ix4 = j2 + 2;
+ if (isClosed) {
+ max3 = nodes.length - 1;
+ if (ix1 < 0)
+ ix1 = max3 + ix1;
+ if (ix2 < 0)
+ ix2 = max3 + ix2;
+ if (ix3 > max3)
+ ix3 = ix3 - max3;
+ if (ix4 > max3)
+ ix4 = ix4 - max3;
+ }
+ if (nodes[ix1] === activeID)
+ return 0;
+ else if (nodes[ix2] === activeID)
+ return 2;
+ else if (nodes[ix3] === activeID)
+ return 2;
+ else if (nodes[ix4] === activeID)
+ return 0;
+ else if (isClosed && nodes.indexOf(activeID) !== -1)
+ return 0;
+ }
+ }
+ }
+ return 1;
+ }
+ function svgMarkerSegments(projection2, graph, dt, shouldReverse, bothDirections) {
+ return function(entity) {
+ var i2 = 0;
+ var offset = dt;
+ var segments = [];
+ var clip = identity_default2().clipExtent(projection2.clipExtent()).stream;
+ var coordinates = graph.childNodes(entity).map(function(n2) {
+ return n2.loc;
+ });
+ var a, b;
+ if (shouldReverse(entity)) {
+ coordinates.reverse();
+ }
+ stream_default({
+ type: "LineString",
+ coordinates
+ }, projection2.stream(clip({
+ lineStart: function() {
+ },
+ lineEnd: function() {
+ a = null;
+ },
+ point: function(x, y) {
+ b = [x, y];
+ if (a) {
+ var span = geoVecLength(a, b) - offset;
+ 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)
+ ];
+ var coord2 = [a, p];
+ for (span -= dt; span >= 0; span -= dt) {
+ p = geoVecAdd(p, [dx, dy]);
+ coord2.push(p);
+ }
+ coord2.push(b);
+ var segment = "";
+ var j2;
+ for (j2 = 0; j2 < coord2.length; j2++) {
+ segment += (j2 === 0 ? "M" : "L") + coord2[j2][0] + "," + coord2[j2][1];
+ }
+ segments.push({ id: entity.id, index: i2++, d: segment });
+ if (bothDirections(entity)) {
+ segment = "";
+ for (j2 = coord2.length - 1; j2 >= 0; j2--) {
+ segment += (j2 === coord2.length - 1 ? "M" : "L") + coord2[j2][0] + "," + coord2[j2][1];
+ }
+ segments.push({ id: entity.id, index: i2++, d: segment });
+ }
+ }
+ offset = -span;
+ }
+ a = b;
+ }
+ })));
+ return segments;
+ };
+ }
+ function svgPath(projection2, graph, isArea) {
+ var cache = {};
+ var padding = isArea ? 65 : 5;
+ var viewport = projection2.clipExtent();
+ var paddedExtent = [
+ [viewport[0][0] - padding, viewport[0][1] - padding],
+ [viewport[1][0] + padding, viewport[1][1] + padding]
+ ];
+ var clip = identity_default2().clipExtent(paddedExtent).stream;
+ var project = projection2.stream;
+ var path = path_default().projection({ stream: function(output) {
+ return project(clip(output));
+ } });
+ var svgpath = function(entity) {
+ if (entity.id in cache) {
+ return cache[entity.id];
+ } else {
+ return cache[entity.id] = path(entity.asGeoJSON(graph));
+ }
+ };
+ svgpath.geojson = function(d) {
+ if (d.__featurehash__ !== void 0) {
+ if (d.__featurehash__ in cache) {
+ return cache[d.__featurehash__];
+ } else {
+ return cache[d.__featurehash__] = path(d);
+ }
+ } else {
+ return path(d);
+ }
+ };
+ return svgpath;
+ }
+ function svgPointTransform(projection2) {
+ var svgpoint = function(entity) {
+ var pt = projection2(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 type2 = relation.tags.type;
+ if (type2 === "multipolygon" && shouldCopyMultipolygonTags || type2 === "boundary") {
+ tags = Object.assign({}, relation.tags, tags);
+ }
+ });
+ return tags;
+ };
+ }
+ function svgSegmentWay(way, graph, activeID) {
+ if (activeID === void 0) {
+ return graph.transient(way, "waySegments", getWaySegments);
+ } else {
+ return getWaySegments();
+ }
+ function getWaySegments() {
+ var isActiveWay = way.nodes.indexOf(activeID) !== -1;
+ var features = { passive: [], active: [] };
+ var start2 = {};
+ var end = {};
+ var node, type2;
+ for (var i2 = 0; i2 < way.nodes.length; i2++) {
+ node = graph.entity(way.nodes[i2]);
+ type2 = svgPassiveVertex(node, graph, activeID);
+ end = { node, type: type2 };
+ if (start2.type !== void 0) {
+ if (start2.node.id === activeID || end.node.id === activeID) {
+ } else if (isActiveWay && (start2.type === 2 || end.type === 2)) {
+ pushActive(start2, end, i2);
+ } else if (start2.type === 0 && end.type === 0) {
+ pushActive(start2, end, i2);
+ } else {
+ pushPassive(start2, end, i2);
+ }
+ }
+ start2 = end;
+ }
+ return features;
+ function pushActive(start3, end2, index) {
+ features.active.push({
+ type: "Feature",
+ id: way.id + "-" + index + "-nope",
+ properties: {
+ nope: true,
+ target: true,
+ entity: way,
+ nodes: [start3.node, end2.node],
+ index
+ },
+ geometry: {
+ type: "LineString",
+ coordinates: [start3.node.loc, end2.node.loc]
+ }
+ });
+ }
+ function pushPassive(start3, end2, index) {
+ features.passive.push({
+ type: "Feature",
+ id: way.id + "-" + index,
+ properties: {
+ target: true,
+ entity: way,
+ nodes: [start3.node, end2.node],
+ index
+ },
+ geometry: {
+ type: "LineString",
+ coordinates: [start3.node.loc, end2.node.loc]
+ }
+ });
+ }
+ }
+ }
+
+ // modules/svg/tag_classes.js
+ 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 = Object.keys(osmLifecyclePrefixes);
+ var secondaries = [
+ "oneway",
+ "bridge",
+ "tunnel",
+ "embankment",
+ "cutting",
+ "barrier",
+ "surface",
+ "tracktype",
+ "footway",
+ "crossing",
+ "service",
+ "sport",
+ "public_transport",
+ "location",
+ "parking",
+ "golf",
+ "type",
+ "leisure",
+ "man_made",
+ "indoor",
+ "construction",
+ "proposed"
+ ];
+ var _tags = function(entity) {
+ return entity.tags;
+ };
+ var tagClasses = function(selection2) {
+ selection2.each(function tagClassesEach(entity) {
+ var value = this.className;
+ if (value.baseVal !== void 0) {
+ value = value.baseVal;
+ }
+ var t = _tags(entity);
+ var computed = tagClasses.getClassesString(t, value);
+ if (computed !== value) {
+ select_default2(this).attr("class", computed);
+ }
+ });
+ };
+ tagClasses.getClassesString = function(t, value) {
+ var primary, status;
+ var i2, j2, k, v;
+ var overrideGeometry;
+ if (/\bstroke\b/.test(value)) {
+ if (!!t.barrier && t.barrier !== "no") {
+ overrideGeometry = "line";
+ }
+ }
+ var classes = value.trim().split(/\s+/).filter(function(klass) {
+ return klass.length && !/^tag-/.test(klass);
+ }).map(function(klass) {
+ return klass === "line" || klass === "area" ? overrideGeometry || klass : klass;
+ });
+ for (i2 = 0; i2 < primaries.length; i2++) {
+ k = primaries[i2];
+ v = t[k];
+ if (!v || v === "no")
+ continue;
+ if (k === "piste:type") {
+ k = "piste";
+ } else if (k === "building:part") {
+ k = "building_part";
+ }
+ primary = k;
+ if (statuses.indexOf(v) !== -1) {
+ status = v;
+ classes.push("tag-" + k);
+ } else {
+ classes.push("tag-" + k);
+ classes.push("tag-" + k + "-" + v);
+ }
+ break;
+ }
+ if (!primary) {
+ for (i2 = 0; i2 < statuses.length; i2++) {
+ for (j2 = 0; j2 < primaries.length; j2++) {
+ k = statuses[i2] + ":" + primaries[j2];
+ v = t[k];
+ if (!v || v === "no")
+ continue;
+ status = statuses[i2];
+ break;
+ }
+ }
+ }
+ if (!status) {
+ for (i2 = 0; i2 < statuses.length; i2++) {
+ k = statuses[i2];
+ v = t[k];
+ if (!v || v === "no")
+ continue;
+ if (v === "yes") {
+ status = k;
+ } else if (primary && primary === v) {
+ status = k;
+ } else if (!primary && primaries.indexOf(v) !== -1) {
+ status = k;
+ primary = v;
+ classes.push("tag-" + v);
+ }
+ if (status)
+ break;
+ }
+ }
+ if (status) {
+ classes.push("tag-status");
+ classes.push("tag-status-" + status);
+ }
+ for (i2 = 0; i2 < secondaries.length; i2++) {
+ k = secondaries[i2];
+ v = t[k];
+ if (!v || v === "no" || k === primary)
+ continue;
+ classes.push("tag-" + k);
+ classes.push("tag-" + k + "-" + v);
+ }
+ 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";
+ }
+ }
+ classes.push("tag-" + surface);
+ }
+ var qid = t.wikidata || t["flag:wikidata"] || t["brand:wikidata"] || t["network:wikidata"] || t["operator:wikidata"];
+ if (qid) {
+ classes.push("tag-wikidata");
+ }
+ return classes.join(" ").trim();
+ };
+ tagClasses.tags = function(val) {
+ if (!arguments.length)
+ return _tags;
+ _tags = val;
+ return tagClasses;
+ };
+ return tagClasses;
+ }
+
+ // modules/svg/tag_pattern.js
+ 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: "grass",
+ landfill: "landfill",
+ meadow: "meadow",
+ military: "construction",
+ orchard: "orchard",
+ quarry: "quarry",
+ vineyard: "vineyard"
+ },
+ leisure: {
+ horse_riding: "farmyard"
+ },
+ 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'
+ ]
+ },
+ golf: {
+ green: "golf_green",
+ tee: "grass",
+ fairway: "grass",
+ rough: "scrub"
+ },
+ surface: {
+ grass: "grass",
+ sand: "beach"
+ }
+ };
+ function svgTagPattern(tags) {
+ if (tags.building && tags.building !== "no") {
+ return null;
+ }
+ for (var tag in patterns) {
+ var entityValue = tags[tag];
+ if (!entityValue)
+ continue;
+ if (typeof patterns[tag] === "string") {
+ return "pattern-" + patterns[tag];
+ } else {
+ var values = patterns[tag];
+ for (var value in values) {
+ if (entityValue !== value)
+ continue;
+ var rules = values[value];
+ if (typeof rules === "string") {
+ return "pattern-" + rules;
+ }
+ for (var ruleKey in rules) {
+ var rule = rules[ruleKey];
+ var pass = true;
+ for (var criterion in rule) {
+ if (criterion !== "pattern") {
+ var v = tags[criterion];
+ if (!v || v !== rule[criterion]) {
+ pass = false;
+ break;
+ }
+ }
+ }
+ if (pass) {
+ return "pattern-" + rule.pattern;
+ }
+ }
+ }
+ }
+ }
+ return null;
+ }
+
+ // modules/svg/areas.js
+ function svgAreas(projection2, context) {
+ function getPatternStyle(tags) {
+ var imageID = svgTagPattern(tags);
+ if (imageID) {
+ return 'url("#ideditor-' + imageID + '")';
+ }
+ return "";
+ }
+ function drawTargets(selection2, graph, entities, filter2) {
+ var targetClass = context.getDebug("target") ? "pink " : "nocolor ";
+ var nopeClass = context.getDebug("target") ? "red " : "nocolor ";
+ var getPath = svgPath(projection2).geojson;
+ var activeID = context.activeID();
+ var base = context.history().base();
+ 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);
+ });
+ var targetData = data.targets.filter(getPath);
+ var targets = selection2.selectAll(".area.target-allowed").filter(function(d) {
+ return filter2(d.properties.entity);
+ }).data(targetData, function key(d) {
+ return d.id;
+ });
+ targets.exit().remove();
+ var segmentWasEdited = function(d) {
+ var wayID = d.properties.entity.id;
+ if (!base.entities[wayID] || !(0, import_fast_deep_equal5.default)(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
+ return false;
+ }
+ return d.properties.nodes.some(function(n2) {
+ return !base.entities[n2.id] || !(0, import_fast_deep_equal5.default)(graph.entities[n2.id].loc, base.entities[n2.id].loc);
+ });
+ };
+ 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);
+ var nopeData = data.nopes.filter(getPath);
+ var nopes = selection2.selectAll(".area.target-nope").filter(function(d) {
+ return filter2(d.properties.entity);
+ }).data(nopeData, function key(d) {
+ return d.id;
+ });
+ nopes.exit().remove();
+ 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);
+ }
+ function drawAreas(selection2, graph, entities, filter2) {
+ var path = svgPath(projection2, graph, true);
+ var areas = {};
+ var multipolygon;
+ var base = context.history().base();
+ for (var i2 = 0; i2 < entities.length; i2++) {
+ var entity = entities[i2];
+ if (entity.geometry(graph) !== "area")
+ continue;
+ multipolygon = osmIsOldMultipolygonOuterMember(entity, graph);
+ 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,
+ area: Math.abs(entity.area(graph))
+ };
+ }
+ }
+ 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(filter2).data(data.clip, osmEntity.key);
+ clipPaths.exit().remove();
+ var clipPathsEnter = clipPaths.enter().append("clipPath").attr("class", "clipPath-osm").attr("id", function(entity2) {
+ return "ideditor-" + entity2.id + "-clippath";
+ });
+ clipPathsEnter.append("path");
+ clipPaths.merge(clipPathsEnter).selectAll("path").attr("d", path);
+ var drawLayer = selection2.selectAll(".layer-osm.areas");
+ var touchLayer = selection2.selectAll(".layer-touch.areas");
+ 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(filter2).data(function(layer) {
+ return data[layer];
+ }, osmEntity.key);
+ paths.exit().remove();
+ var fillpaths = selection2.selectAll(".area-fill path.area").nodes();
+ var bisect = bisector(function(node) {
+ return -node.__data__.area(graph);
+ }).left;
+ function sortedByArea(entity2) {
+ if (this._parent.__data__ === "fill") {
+ return fillpaths[bisect(fillpaths, -entity2.area(graph))];
+ }
+ }
+ paths = paths.enter().insert("path", sortedByArea).merge(paths).each(function(entity2) {
+ var layer = this.parentNode.__data__;
+ this.setAttribute("class", entity2.type + " area " + layer + " " + entity2.id);
+ if (layer === "fill") {
+ this.setAttribute("clip-path", "url(#ideditor-" + entity2.id + "-clippath)");
+ this.style.fill = this.style.stroke = getPatternStyle(entity2.tags);
+ }
+ }).classed("added", function(d) {
+ return !base.entities[d.id];
+ }).classed("geometry-edited", function(d) {
+ return graph.entities[d.id] && base.entities[d.id] && !(0, import_fast_deep_equal5.default)(graph.entities[d.id].nodes, base.entities[d.id].nodes);
+ }).classed("retagged", function(d) {
+ return graph.entities[d.id] && base.entities[d.id] && !(0, import_fast_deep_equal5.default)(graph.entities[d.id].tags, base.entities[d.id].tags);
+ }).call(svgTagClasses()).attr("d", path);
+ touchLayer.call(drawTargets, graph, data.stroke, filter2);
+ }
+ return drawAreas;
+ }
+
+ // modules/svg/data.js
+ var import_fast_json_stable_stringify = __toESM(require_fast_json_stable_stringify());
+
+ // node_modules/@tmcw/togeojson/dist/togeojson.es.mjs
+ function $(element, tagName) {
+ return Array.from(element.getElementsByTagName(tagName));
+ }
+ function normalizeId(id2) {
+ return id2[0] === "#" ? id2 : `#${id2}`;
+ }
+ function $ns(element, tagName, ns) {
+ return Array.from(element.getElementsByTagNameNS(ns, tagName));
+ }
+ function nodeVal(node) {
+ node?.normalize();
+ return node && node.textContent || "";
+ }
+ function get1(node, tagName, callback) {
+ const n2 = node.getElementsByTagName(tagName);
+ const result = n2.length ? n2[0] : null;
+ if (result && callback)
+ callback(result);
+ return result;
+ }
+ function get3(node, tagName, callback) {
+ const properties = {};
+ if (!node)
+ return properties;
+ const n2 = node.getElementsByTagName(tagName);
+ const result = n2.length ? n2[0] : null;
+ if (result && callback) {
+ return callback(result, properties);
+ }
+ return properties;
+ }
+ function val1(node, tagName, callback) {
+ const val = nodeVal(get1(node, tagName));
+ if (val && callback)
+ return callback(val) || {};
+ return {};
+ }
+ function $num(node, tagName, callback) {
+ const val = parseFloat(nodeVal(get1(node, tagName)));
+ if (isNaN(val))
+ return void 0;
+ if (val && callback)
+ return callback(val) || {};
+ return {};
+ }
+ function num1(node, tagName, callback) {
+ const val = parseFloat(nodeVal(get1(node, tagName)));
+ if (isNaN(val))
+ return void 0;
+ if (val && callback)
+ callback(val);
+ return val;
+ }
+ function getMulti(node, propertyNames) {
+ const properties = {};
+ for (const property of propertyNames) {
+ val1(node, property, (val) => {
+ properties[property] = val;
+ });
+ }
+ return properties;
+ }
+ function isElement(node) {
+ return node?.nodeType === 1;
+ }
+ function getLineStyle(node) {
+ return get3(node, "line", (lineStyle) => {
+ const val = Object.assign({}, val1(lineStyle, "color", (color2) => {
+ return { stroke: `#${color2}` };
+ }), $num(lineStyle, "opacity", (opacity) => {
+ return { "stroke-opacity": opacity };
+ }), $num(lineStyle, "width", (width) => {
+ return { "stroke-width": width * 96 / 25.4 };
+ }));
+ return val;
+ });
+ }
+ function getExtensions(node) {
+ let values = [];
+ if (node === null)
+ return values;
+ for (const child of Array.from(node.childNodes)) {
+ if (!isElement(child))
+ continue;
+ const name = abbreviateName(child.nodeName);
+ if (name === "gpxtpx:TrackPointExtension") {
+ values = values.concat(getExtensions(child));
+ } else {
+ const val = nodeVal(child);
+ values.push([name, parseNumeric(val)]);
+ }
+ }
+ return values;
+ }
+ function abbreviateName(name) {
+ return ["heart", "gpxtpx:hr", "hr"].includes(name) ? "heart" : name;
+ }
+ function parseNumeric(val) {
+ const num = parseFloat(val);
+ return isNaN(num) ? val : num;
+ }
+ function coordPair$1(node) {
+ const ll = [
+ parseFloat(node.getAttribute("lon") || ""),
+ parseFloat(node.getAttribute("lat") || "")
+ ];
+ if (isNaN(ll[0]) || isNaN(ll[1])) {
+ return null;
+ }
+ num1(node, "ele", (val) => {
+ ll.push(val);
+ });
+ const time = get1(node, "time");
+ return {
+ coordinates: ll,
+ time: time ? nodeVal(time) : null,
+ extendedValues: getExtensions(get1(node, "extensions"))
+ };
+ }
+ function extractProperties(node) {
+ const properties = getMulti(node, [
+ "name",
+ "cmt",
+ "desc",
+ "type",
+ "time",
+ "keywords"
+ ]);
+ const extensions = Array.from(node.getElementsByTagNameNS("http://www.garmin.com/xmlschemas/GpxExtensions/v3", "*"));
+ for (const child of extensions) {
+ if (child.parentNode?.parentNode === node) {
+ properties[child.tagName.replace(":", "_")] = nodeVal(child);
+ }
+ }
+ const links = $(node, "link");
+ if (links.length) {
+ properties.links = links.map((link2) => Object.assign({ href: link2.getAttribute("href") }, getMulti(link2, ["text", "type"])));
+ }
+ return properties;
+ }
+ function getPoints$1(node, pointname) {
+ const pts = $(node, pointname);
+ const line = [];
+ const times = [];
+ const extendedValues = {};
+ for (let i2 = 0; i2 < pts.length; i2++) {
+ const c = coordPair$1(pts[i2]);
+ if (!c) {
+ continue;
+ }
+ line.push(c.coordinates);
+ if (c.time)
+ times.push(c.time);
+ for (const [name, val] of c.extendedValues) {
+ const plural = name === "heart" ? name : name.replace("gpxtpx:", "") + "s";
+ if (!extendedValues[plural]) {
+ extendedValues[plural] = Array(pts.length).fill(null);
+ }
+ extendedValues[plural][i2] = val;
+ }
+ }
+ if (line.length < 2)
+ return;
+ return {
+ line,
+ times,
+ extendedValues
+ };
+ }
+ function getRoute(node) {
+ const line = getPoints$1(node, "rtept");
+ if (!line)
+ return;
+ return {
+ type: "Feature",
+ properties: Object.assign({ _gpxType: "rte" }, extractProperties(node), getLineStyle(get1(node, "extensions"))),
+ geometry: {
+ type: "LineString",
+ coordinates: line.line
+ }
+ };
+ }
+ function getTrack(node) {
+ const segments = $(node, "trkseg");
+ const track = [];
+ const times = [];
+ const extractedLines = [];
+ for (const segment of segments) {
+ const line = getPoints$1(segment, "trkpt");
+ if (line) {
+ extractedLines.push(line);
+ if (line.times && line.times.length)
+ times.push(line.times);
+ }
+ }
+ if (extractedLines.length === 0)
+ return null;
+ const multi = extractedLines.length > 1;
+ const properties = Object.assign({ _gpxType: "trk" }, extractProperties(node), getLineStyle(get1(node, "extensions")), times.length ? {
+ coordinateProperties: {
+ times: multi ? times : times[0]
+ }
+ } : {});
+ for (const line of extractedLines) {
+ track.push(line.line);
+ if (!properties.coordinateProperties) {
+ properties.coordinateProperties = {};
+ }
+ const props = properties.coordinateProperties;
+ const entries = Object.entries(line.extendedValues);
+ for (let i2 = 0; i2 < entries.length; i2++) {
+ const [name, val] = entries[i2];
+ if (multi) {
+ if (!props[name]) {
+ props[name] = extractedLines.map((line2) => new Array(line2.line.length).fill(null));
+ }
+ props[name][i2] = val;
+ } else {
+ props[name] = val;
+ }
+ }
+ }
+ return {
+ type: "Feature",
+ properties,
+ geometry: multi ? {
+ type: "MultiLineString",
+ coordinates: track
+ } : {
+ type: "LineString",
+ coordinates: track[0]
+ }
+ };
+ }
+ function getPoint(node) {
+ const properties = Object.assign(extractProperties(node), getMulti(node, ["sym"]));
+ const pair2 = coordPair$1(node);
+ if (!pair2)
+ return null;
+ return {
+ type: "Feature",
+ properties,
+ geometry: {
+ type: "Point",
+ coordinates: pair2.coordinates
+ }
+ };
+ }
+ function* gpxGen(node) {
+ for (const track of $(node, "trk")) {
+ const feature3 = getTrack(track);
+ if (feature3)
+ yield feature3;
+ }
+ for (const route of $(node, "rte")) {
+ const feature3 = getRoute(route);
+ if (feature3)
+ yield feature3;
+ }
+ for (const waypoint of $(node, "wpt")) {
+ const point2 = getPoint(waypoint);
+ if (point2)
+ yield point2;
+ }
+ }
+ function gpx(node) {
+ return {
+ type: "FeatureCollection",
+ features: Array.from(gpxGen(node))
+ };
+ }
+ function fixColor(v, prefix) {
+ const properties = {};
+ const colorProp = prefix == "stroke" || prefix === "fill" ? prefix : prefix + "-color";
+ if (v[0] === "#") {
+ v = v.substring(1);
+ }
+ if (v.length === 6 || v.length === 3) {
+ properties[colorProp] = "#" + v;
+ } else if (v.length === 8) {
+ properties[prefix + "-opacity"] = parseInt(v.substring(0, 2), 16) / 255;
+ properties[colorProp] = "#" + v.substring(6, 8) + v.substring(4, 6) + v.substring(2, 4);
+ }
+ return properties;
+ }
+ function numericProperty(node, source, target) {
+ const properties = {};
+ num1(node, source, (val) => {
+ properties[target] = val;
+ });
+ return properties;
+ }
+ function getColor(node, output) {
+ return get3(node, "color", (elem) => fixColor(nodeVal(elem), output));
+ }
+ function extractIcon(node) {
+ return get3(node, "IconStyle", (iconStyle) => {
+ return Object.assign(getColor(iconStyle, "icon"), numericProperty(iconStyle, "scale", "icon-scale"), numericProperty(iconStyle, "heading", "icon-heading"), get3(iconStyle, "hotSpot", (hotspot) => {
+ const left = parseFloat(hotspot.getAttribute("x") || "");
+ const top = parseFloat(hotspot.getAttribute("y") || "");
+ const xunits = hotspot.getAttribute("xunits") || "";
+ const yunits = hotspot.getAttribute("yunits") || "";
+ if (!isNaN(left) && !isNaN(top))
+ return {
+ "icon-offset": [left, top],
+ "icon-offset-units": [xunits, yunits]
+ };
+ return {};
+ }), get3(iconStyle, "Icon", (icon2, properties) => {
+ val1(icon2, "href", (href) => {
+ properties.icon = href;
+ });
+ return properties;
+ }));
+ });
+ }
+ function extractLabel(node) {
+ return get3(node, "LabelStyle", (labelStyle) => {
+ return Object.assign(getColor(labelStyle, "label"), numericProperty(labelStyle, "scale", "label-scale"));
+ });
+ }
+ function extractLine(node) {
+ return get3(node, "LineStyle", (lineStyle) => {
+ return Object.assign(getColor(lineStyle, "stroke"), numericProperty(lineStyle, "width", "stroke-width"));
+ });
+ }
+ function extractPoly(node) {
+ return get3(node, "PolyStyle", (polyStyle, properties) => {
+ return Object.assign(properties, get3(polyStyle, "color", (elem) => fixColor(nodeVal(elem), "fill")), val1(polyStyle, "fill", (fill) => {
+ if (fill === "0")
+ return { "fill-opacity": 0 };
+ }), val1(polyStyle, "outline", (outline) => {
+ if (outline === "0")
+ return { "stroke-opacity": 0 };
+ }));
+ });
+ }
+ function extractStyle(node) {
+ return Object.assign({}, extractPoly(node), extractLine(node), extractLabel(node), extractIcon(node));
+ }
+ var removeSpace = /\s*/g;
+ var trimSpace = /^\s*|\s*$/g;
+ var splitSpace = /\s+/;
+ function coord1(value) {
+ return value.replace(removeSpace, "").split(",").map(parseFloat).filter((num) => !isNaN(num)).slice(0, 3);
+ }
+ function coord(value) {
+ return value.replace(trimSpace, "").split(splitSpace).map(coord1).filter((coord2) => {
+ return coord2.length >= 2;
+ });
+ }
+ function gxCoords(node) {
+ let elems = $(node, "coord");
+ if (elems.length === 0) {
+ elems = $ns(node, "coord", "*");
+ }
+ const coordinates = elems.map((elem) => {
+ return nodeVal(elem).split(" ").map(parseFloat);
+ });
+ if (coordinates.length === 0) {
+ return null;
+ }
+ return {
+ geometry: coordinates.length > 2 ? {
+ type: "LineString",
+ coordinates
+ } : {
+ type: "Point",
+ coordinates: coordinates[0]
+ },
+ times: $(node, "when").map((elem) => nodeVal(elem))
+ };
+ }
+ function fixRing(ring) {
+ if (ring.length === 0)
+ return ring;
+ const first = ring[0];
+ const last = ring[ring.length - 1];
+ let equal = true;
+ for (let i2 = 0; i2 < Math.max(first.length, last.length); i2++) {
+ if (first[i2] !== last[i2]) {
+ equal = false;
+ break;
+ }
+ }
+ if (!equal) {
+ return ring.concat([ring[0]]);
+ }
+ return ring;
+ }
+ var GEO_TYPES = [
+ "Polygon",
+ "LineString",
+ "Point",
+ "Track",
+ "gx:Track"
+ ];
+ function getCoordinates(node) {
+ return nodeVal(get1(node, "coordinates"));
+ }
+ function getGeometry(node) {
+ const geometries = [];
+ const coordTimes = [];
+ for (const t of ["MultiGeometry", "MultiTrack", "gx:MultiTrack"]) {
+ const elem = get1(node, t);
+ if (elem) {
+ return getGeometry(elem);
+ }
+ }
+ for (const geoType of GEO_TYPES) {
+ for (const geomNode of $(node, geoType)) {
+ switch (geoType) {
+ case "Point": {
+ const coordinates = coord1(getCoordinates(geomNode));
+ if (coordinates.length >= 2) {
+ geometries.push({
+ type: "Point",
+ coordinates
+ });
+ }
+ break;
+ }
+ case "LineString": {
+ const coordinates = coord(getCoordinates(geomNode));
+ if (coordinates.length >= 2) {
+ geometries.push({
+ type: "LineString",
+ coordinates
+ });
+ }
+ break;
+ }
+ case "Polygon": {
+ const coords = [];
+ for (const linearRing of $(geomNode, "LinearRing")) {
+ const ring = fixRing(coord(getCoordinates(linearRing)));
+ if (ring.length >= 4) {
+ coords.push(ring);
+ }
+ }
+ if (coords.length) {
+ geometries.push({
+ type: "Polygon",
+ coordinates: coords
+ });
+ }
+ break;
+ }
+ case "Track":
+ case "gx:Track": {
+ const gx = gxCoords(geomNode);
+ if (!gx)
+ break;
+ const { times, geometry } = gx;
+ geometries.push(geometry);
+ if (times.length)
+ coordTimes.push(times);
+ break;
+ }
+ }
+ }
+ }
+ return {
+ geometries,
+ coordTimes
+ };
+ }
+ function extractExtendedData(node) {
+ return get3(node, "ExtendedData", (extendedData, properties) => {
+ for (const data of $(extendedData, "Data")) {
+ properties[data.getAttribute("name") || ""] = nodeVal(get1(data, "value"));
+ }
+ for (const simpleData of $(extendedData, "SimpleData")) {
+ properties[simpleData.getAttribute("name") || ""] = nodeVal(simpleData);
+ }
+ return properties;
+ });
+ }
+ function geometryListToGeometry(geometries) {
+ return geometries.length === 0 ? null : geometries.length === 1 ? geometries[0] : {
+ type: "GeometryCollection",
+ geometries
+ };
+ }
+ function extractTimeSpan(node) {
+ return get3(node, "TimeSpan", (timeSpan) => {
+ return {
+ timespan: {
+ begin: nodeVal(get1(timeSpan, "begin")),
+ end: nodeVal(get1(timeSpan, "end"))
+ }
+ };
+ });
+ }
+ function extractTimeStamp(node) {
+ return get3(node, "TimeStamp", (timeStamp) => {
+ return { timestamp: nodeVal(get1(timeStamp, "when")) };
+ });
+ }
+ function extractCascadedStyle(node, styleMap) {
+ return val1(node, "styleUrl", (styleUrl) => {
+ styleUrl = normalizeId(styleUrl);
+ if (styleMap[styleUrl]) {
+ return Object.assign({ styleUrl }, styleMap[styleUrl]);
+ }
+ return { styleUrl };
+ });
+ }
+ function getMaybeHTMLDescription(node) {
+ const descriptionNode = get1(node, "description");
+ for (const c of Array.from(descriptionNode?.childNodes || [])) {
+ if (c.nodeType === 4) {
+ return {
+ description: {
+ "@type": "html",
+ value: nodeVal(c)
+ }
+ };
+ }
+ }
+ return {};
+ }
+ function getPlacemark(node, styleMap) {
+ const { coordTimes, geometries } = getGeometry(node);
+ const feature3 = {
+ type: "Feature",
+ geometry: geometryListToGeometry(geometries),
+ properties: Object.assign(getMulti(node, [
+ "name",
+ "address",
+ "visibility",
+ "open",
+ "phoneNumber",
+ "description"
+ ]), getMaybeHTMLDescription(node), extractCascadedStyle(node, styleMap), extractStyle(node), extractExtendedData(node), extractTimeSpan(node), extractTimeStamp(node), coordTimes.length ? {
+ coordinateProperties: {
+ times: coordTimes.length === 1 ? coordTimes[0] : coordTimes
+ }
+ } : {})
+ };
+ if (feature3.properties?.visibility !== void 0) {
+ feature3.properties.visibility = feature3.properties.visibility !== "0";
+ }
+ const id2 = node.getAttribute("id");
+ if (id2 !== null && id2 !== "")
+ feature3.id = id2;
+ return feature3;
+ }
+ function getStyleId(style) {
+ let id2 = style.getAttribute("id");
+ const parentNode = style.parentNode;
+ if (!id2 && isElement(parentNode) && parentNode.localName === "CascadingStyle") {
+ id2 = parentNode.getAttribute("kml:id") || parentNode.getAttribute("id");
+ }
+ return normalizeId(id2 || "");
+ }
+ function buildStyleMap(node) {
+ const styleMap = {};
+ for (const style of $(node, "Style")) {
+ styleMap[getStyleId(style)] = extractStyle(style);
+ }
+ for (const map2 of $(node, "StyleMap")) {
+ const id2 = normalizeId(map2.getAttribute("id") || "");
+ val1(map2, "styleUrl", (styleUrl) => {
+ styleUrl = normalizeId(styleUrl);
+ if (styleMap[styleUrl]) {
+ styleMap[id2] = styleMap[styleUrl];
+ }
+ });
+ }
+ return styleMap;
+ }
+ function* kmlGen(node) {
+ const styleMap = buildStyleMap(node);
+ for (const placemark of $(node, "Placemark")) {
+ const feature3 = getPlacemark(placemark, styleMap);
+ if (feature3)
+ yield feature3;
+ }
+ }
+ function kml(node) {
+ return {
+ type: "FeatureCollection",
+ features: Array.from(kmlGen(node))
+ };
+ }
+
+ // modules/svg/data.js
+ var _initialized = false;
+ var _enabled = false;
+ var _geojson;
+ function svgData(projection2, context, dispatch10) {
+ var throttledRedraw = throttle_default(function() {
+ dispatch10.call("change");
+ }, 1e3);
+ var _showLabels = true;
+ var detected = utilDetect();
+ var layer = select_default2(null);
+ var _vtService;
+ var _fileList;
+ var _template;
+ var _src;
+ function init2() {
+ if (_initialized)
+ return;
+ _geojson = {};
+ _enabled = true;
+ 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;
+ }
+ function getService() {
+ if (services.vectorTile && !_vtService) {
+ _vtService = services.vectorTile;
+ _vtService.event.on("loadedData", throttledRedraw);
+ } else if (!services.vectorTile && _vtService) {
+ _vtService = null;
+ }
+ return _vtService;
+ }
+ function showLayer() {
+ layerOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
+ dispatch10.call("change");
+ });
+ }
+ function hideLayer() {
+ throttledRedraw.cancel();
+ layer.transition().duration(250).style("opacity", 0).on("end", layerOff);
+ }
+ function layerOn() {
+ layer.style("display", "block");
+ }
+ function layerOff() {
+ layer.selectAll(".viewfield-group").remove();
+ layer.style("display", "none");
+ }
+ function ensureIDs(gj) {
+ if (!gj)
+ return null;
+ if (gj.type === "FeatureCollection") {
+ for (var i2 = 0; i2 < gj.features.length; i2++) {
+ ensureFeatureID(gj.features[i2]);
+ }
+ } else {
+ ensureFeatureID(gj);
+ }
+ return gj;
+ }
+ function ensureFeatureID(feature3) {
+ if (!feature3)
+ return;
+ feature3.__featurehash__ = utilHashcode((0, import_fast_json_stable_stringify.default)(feature3));
+ return feature3;
+ }
+ function getFeatures(gj) {
+ if (!gj)
+ return [];
+ if (gj.type === "FeatureCollection") {
+ return gj.features;
+ } else {
+ return [gj];
+ }
+ }
+ function featureKey(d) {
+ return d.__featurehash__;
+ }
+ function isPolygon(d) {
+ return d.geometry.type === "Polygon" || d.geometry.type === "MultiPolygon";
+ }
+ function clipPathID(d) {
+ return "ideditor-data-" + d.__featurehash__ + "-clippath";
+ }
+ function featureClasses(d) {
+ return [
+ "data" + d.__featurehash__,
+ d.geometry.type,
+ isPolygon(d) ? "area" : "",
+ d.__layerID__ || ""
+ ].filter(Boolean).join(" ");
+ }
+ function drawData(selection2) {
+ var vtService = getService();
+ var getPath = svgPath(projection2).geojson;
+ var getAreaPath = svgPath(projection2, null, true).geojson;
+ var hasData = drawData.hasData();
+ layer = selection2.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;
+ var geoData, polygonData;
+ if (_template && vtService) {
+ var sourceID = _template;
+ vtService.loadTiles(sourceID, _template, projection2);
+ geoData = vtService.data(sourceID, projection2);
+ } else {
+ geoData = getFeatures(_geojson);
+ }
+ geoData = geoData.filter(getPath);
+ polygonData = geoData.filter(isPolygon);
+ 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);
+ 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);
+ var pathData = {
+ fill: polygonData,
+ shadow: geoData,
+ stroke: geoData
+ };
+ var paths = datagroups.selectAll("path").data(function(layer2) {
+ return pathData[layer2];
+ }, featureKey);
+ paths.exit().remove();
+ 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);
+ });
+ layer.call(drawLabels, "label-halo", geoData).call(drawLabels, "label", geoData);
+ function drawLabels(selection3, textClass, data) {
+ var labelPath = path_default(projection2);
+ var labelData = data.filter(function(d) {
+ return _showLabels && d.properties && (d.properties.desc || d.properties.name);
+ });
+ var labels = selection3.selectAll("text." + textClass).data(labelData, featureKey);
+ labels.exit().remove();
+ 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];
+ });
+ }
+ }
+ function getExtension(fileName) {
+ if (!fileName)
+ return;
+ var re2 = /\.(gpx|kml|(geo)?json)$/i;
+ var match = fileName.toLowerCase().match(re2);
+ return match && match.length && match[0];
+ }
+ function xmlToDom(textdata) {
+ return new DOMParser().parseFromString(textdata, "text/xml");
+ }
+ function stringifyGeojsonProperties(feature3) {
+ const properties = feature3.properties;
+ for (const key in properties) {
+ const property = properties[key];
+ if (typeof property === "number" || typeof property === "boolean" || Array.isArray(property)) {
+ properties[key] = property.toString();
+ } else if (property === null) {
+ properties[key] = "null";
+ } else if (typeof property === "object") {
+ properties[key] = JSON.stringify(property);
+ }
+ }
+ }
+ drawData.setFile = function(extension, data) {
+ _template = null;
+ _fileList = null;
+ _geojson = null;
+ _src = null;
+ var gj;
+ switch (extension) {
+ case ".gpx":
+ gj = gpx(xmlToDom(data));
+ break;
+ case ".kml":
+ gj = kml(xmlToDom(data));
+ break;
+ case ".geojson":
+ case ".json":
+ gj = JSON.parse(data);
+ if (gj.type === "FeatureCollection") {
+ gj.features.forEach(stringifyGeojsonProperties);
+ } else if (gj.type === "Feature") {
+ stringifyGeojsonProperties(gj);
+ }
+ break;
+ }
+ gj = gj || {};
+ if (Object.keys(gj).length) {
+ _geojson = ensureIDs(gj);
+ _src = extension + " data file";
+ this.fitZoom();
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawData.showLabels = function(val) {
+ if (!arguments.length)
+ return _showLabels;
+ _showLabels = val;
+ return this;
+ };
+ drawData.enabled = function(val) {
+ if (!arguments.length)
+ return _enabled;
+ _enabled = val;
+ if (_enabled) {
+ showLayer();
+ } else {
+ hideLayer();
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawData.hasData = function() {
+ var gj = _geojson || {};
+ return !!(_template || Object.keys(gj).length);
+ };
+ drawData.template = function(val, src) {
+ if (!arguments.length)
+ return _template;
+ var osm = context.connection();
+ if (osm) {
+ var blocklists = osm.imageryBlocklists();
+ var fail = false;
+ var tested = 0;
+ var regex;
+ for (var i2 = 0; i2 < blocklists.length; i2++) {
+ regex = blocklists[i2];
+ fail = regex.test(val);
+ tested++;
+ if (fail)
+ break;
+ }
+ if (!tested) {
+ regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
+ fail = regex.test(val);
+ }
+ }
+ _template = val;
+ _fileList = null;
+ _geojson = null;
+ _src = src || "vectortile:" + val.split(/[?#]/)[0];
+ dispatch10.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";
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawData.fileList = function(fileList) {
+ if (!arguments.length)
+ return _fileList;
+ _template = null;
+ _fileList = fileList;
+ _geojson = null;
+ _src = null;
+ if (!fileList || !fileList.length)
+ return this;
+ var f2 = fileList[0];
+ var extension = getExtension(f2.name);
+ var reader = new FileReader();
+ reader.onload = function() {
+ return function(e) {
+ drawData.setFile(extension, e.target.result);
+ };
+ }(f2);
+ reader.readAsText(f2);
+ return this;
+ };
+ drawData.url = function(url, defaultExtension) {
+ _template = null;
+ _fileList = null;
+ _geojson = null;
+ _src = null;
+ var testUrl = url.split(/[?#]/)[0];
+ var extension = getExtension(testUrl) || defaultExtension;
+ if (extension) {
+ _template = null;
+ text_default3(url).then(function(data) {
+ drawData.setFile(extension, data);
+ }).catch(function() {
+ });
+ } else {
+ drawData.template(url);
+ }
+ return this;
+ };
+ drawData.getSrc = function() {
+ return _src || "";
+ };
+ drawData.fitZoom = function() {
+ var features = getFeatures(_geojson);
+ if (!features.length)
+ return;
+ var map2 = context.map();
+ var viewport = map2.trimmedExtent().polygon();
+ var coords = features.reduce(function(coords2, feature3) {
+ var geom = feature3.geometry;
+ if (!geom)
+ return coords2;
+ var c = geom.coordinates;
+ switch (geom.type) {
+ case "Point":
+ c = [c];
+ case "MultiPoint":
+ case "LineString":
+ break;
+ case "MultiPolygon":
+ c = utilArrayFlatten(c);
+ case "Polygon":
+ case "MultiLineString":
+ c = utilArrayFlatten(c);
+ break;
+ }
+ return utilArrayUnion(coords2, c);
+ }, []);
+ if (!geoPolygonIntersectsPolygon(viewport, coords, true)) {
+ var extent = geoExtent(bounds_default({ type: "LineString", coordinates: coords }));
+ map2.centerZoom(extent.center(), map2.trimmedExtentZoom(extent));
+ }
+ return this;
+ };
+ init2();
+ return drawData;
+ }
+
+ // modules/svg/debug.js
+ function svgDebug(projection2, context) {
+ function drawDebug(selection2) {
+ const showTile = context.getDebug("tile");
+ const showCollision = context.getDebug("collision");
+ const showImagery = context.getDebug("imagery");
+ const showTouchTargets = context.getDebug("target");
+ const showDownloaded = context.getDebug("downloaded");
+ let debugData = [];
+ if (showTile) {
+ debugData.push({ class: "red", label: "tile" });
+ }
+ if (showCollision) {
+ debugData.push({ class: "yellow", label: "collision" });
+ }
+ if (showImagery) {
+ debugData.push({ class: "orange", label: "imagery" });
+ }
+ if (showTouchTargets) {
+ debugData.push({ class: "pink", label: "touchTargets" });
+ }
+ if (showDownloaded) {
+ debugData.push({ class: "purple", label: "downloaded" });
+ }
+ let 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);
+ let legendItems = legend.selectAll(".debug-legend-item").data(debugData, (d) => d.label);
+ legendItems.exit().remove();
+ legendItems.enter().append("span").attr("class", (d) => `debug-legend-item ${d.class}`).text((d) => d.label);
+ let layer = selection2.selectAll(".layer-debug").data(showImagery || showDownloaded ? [0] : []);
+ layer.exit().remove();
+ layer = layer.enter().append("g").attr("class", "layer-debug").merge(layer);
+ const extent = context.map().extent();
+ _mainFileFetcher.get("imagery").then((d) => {
+ const hits = showImagery && d.query.bbox(extent.rectangle(), true) || [];
+ const features = hits.map((d2) => d2.features[d2.id]);
+ let imagery = layer.selectAll("path.debug-imagery").data(features);
+ imagery.exit().remove();
+ imagery.enter().append("path").attr("class", "debug-imagery debug orange");
+ }).catch(() => {
+ });
+ const osm = context.connection();
+ let dataDownloaded = [];
+ if (osm && showDownloaded) {
+ const rtree = osm.caches("get").tile.rtree;
+ dataDownloaded = rtree.all().map((bbox2) => {
+ return {
+ type: "Feature",
+ properties: { id: bbox2.id },
+ geometry: {
+ type: "Polygon",
+ coordinates: [[
+ [bbox2.minX, bbox2.minY],
+ [bbox2.minX, bbox2.maxY],
+ [bbox2.maxX, bbox2.maxY],
+ [bbox2.maxX, bbox2.minY],
+ [bbox2.minX, bbox2.minY]
+ ]]
+ }
+ };
+ });
+ }
+ let downloaded = layer.selectAll("path.debug-downloaded").data(showDownloaded ? dataDownloaded : []);
+ downloaded.exit().remove();
+ downloaded.enter().append("path").attr("class", "debug-downloaded debug purple");
+ layer.selectAll("path").attr("d", svgPath(projection2).geojson);
+ }
+ 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;
+ }
+ };
+ return drawDebug;
+ }
+
+ // modules/svg/defs.js
+ function svgDefs(context) {
+ var _defsSelection = select_default2(null);
+ var _spritesheetIds = [
+ "iD-sprite",
+ "maki-sprite",
+ "temaki-sprite",
+ "fa-sprite",
+ "community-sprite"
+ ];
+ function drawDefs(selection2) {
+ _defsSelection = selection2.append("defs");
+ _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");
+ function addSidedMarker(name, color2, 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", color2);
+ }
+ addSidedMarker("natural", "rgb(170, 170, 170)", 0);
+ addSidedMarker("coastline", "#77dede", 1);
+ addSidedMarker("waterway", "#77dede", 1);
+ 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");
+ _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");
+ var patterns2 = _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");
+ patterns2.append("rect").attr("x", 0).attr("y", 0).attr("width", 32).attr("height", 32).attr("class", function(d) {
+ return "pattern-color-" + d[0];
+ });
+ patterns2.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");
+ });
+ _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;
+ });
+ addSprites(_spritesheetIds, true);
+ }
+ function addSprites(ids, overrideColors) {
+ _spritesheetIds = utilArrayUniq(_spritesheetIds.concat(ids));
+ var spritesheets = _defsSelection.selectAll(".spritesheet").data(_spritesheetIds);
+ spritesheets.enter().append("g").attr("class", function(d) {
+ return "spritesheet spritesheet-" + d;
+ }).each(function(d) {
+ var url = context.imagePath(d + ".svg");
+ var node = select_default2(this).node();
+ svg(url).then(function(svg2) {
+ node.appendChild(
+ select_default2(svg2.documentElement).attr("id", "ideditor-" + d).node()
+ );
+ if (overrideColors && d !== "iD-sprite") {
+ select_default2(node).selectAll("path").attr("fill", "currentColor");
+ }
+ }).catch(function() {
+ });
+ });
+ spritesheets.exit().remove();
+ }
+ drawDefs.addSprites = addSprites;
+ return drawDefs;
+ }
+
+ // modules/svg/keepRight.js
+ var _layerEnabled = false;
+ var _qaService;
+ function svgKeepRight(projection2, context, dispatch10) {
+ const throttledRedraw = throttle_default(() => dispatch10.call("change"), 1e3);
+ const minZoom3 = 12;
+ let touchLayer = select_default2(null);
+ let drawLayer = select_default2(null);
+ let layerVisible = false;
+ function markerPath(selection2, klass) {
+ selection2.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");
+ }
+ function getService() {
+ if (services.keepRight && !_qaService) {
+ _qaService = services.keepRight;
+ _qaService.on("loaded", throttledRedraw);
+ } else if (!services.keepRight && _qaService) {
+ _qaService = null;
+ }
+ return _qaService;
+ }
+ function editOn() {
+ if (!layerVisible) {
+ layerVisible = true;
+ drawLayer.style("display", "block");
+ }
+ }
+ function editOff() {
+ if (layerVisible) {
+ layerVisible = false;
+ drawLayer.style("display", "none");
+ drawLayer.selectAll(".qaItem.keepRight").remove();
+ touchLayer.selectAll(".qaItem.keepRight").remove();
+ }
+ }
+ function layerOn() {
+ editOn();
+ drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch10.call("change"));
+ }
+ function layerOff() {
+ throttledRedraw.cancel();
+ drawLayer.interrupt();
+ touchLayer.selectAll(".qaItem.keepRight").remove();
+ drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
+ editOff();
+ dispatch10.call("change");
+ });
+ }
+ function updateMarkers() {
+ if (!layerVisible || !_layerEnabled)
+ return;
+ const service = getService();
+ const selectedID = context.selectedErrorID();
+ const data = service ? service.getItems(projection2) : [];
+ const getTransform = svgPointTransform(projection2);
+ const markers = drawLayer.selectAll(".qaItem.keepRight").data(data, (d) => d.id);
+ markers.exit().remove();
+ const markersEnter = markers.enter().append("g").attr("class", (d) => `qaItem ${d.service} itemId-${d.id} itemType-${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");
+ markers.merge(markersEnter).sort(sortY).classed("selected", (d) => d.id === selectedID).attr("transform", getTransform);
+ if (touchLayer.empty())
+ return;
+ const fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ const targets = touchLayer.selectAll(".qaItem.keepRight").data(data, (d) => d.id);
+ targets.exit().remove();
+ targets.enter().append("rect").attr("width", "20px").attr("height", "20px").attr("x", "-8px").attr("y", "-22px").merge(targets).sort(sortY).attr("class", (d) => `qaItem ${d.service} target ${fillClass} itemId-${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];
+ }
+ }
+ function drawKeepRight(selection2) {
+ const service = getService();
+ const surface = context.surface();
+ if (surface && !surface.empty()) {
+ touchLayer = surface.selectAll(".data-layer.touch .layer-touch.markers");
+ }
+ drawLayer = selection2.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);
+ if (_layerEnabled) {
+ if (service && ~~context.map().zoom() >= minZoom3) {
+ editOn();
+ service.loadIssues(projection2);
+ updateMarkers();
+ } else {
+ editOff();
+ }
+ }
+ }
+ drawKeepRight.enabled = function(val) {
+ if (!arguments.length)
+ return _layerEnabled;
+ _layerEnabled = val;
+ if (_layerEnabled) {
+ layerOn();
+ } else {
+ layerOff();
+ if (context.selectedErrorID()) {
+ context.enter(modeBrowse(context));
+ }
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawKeepRight.supported = () => !!getService();
+ return drawKeepRight;
+ }
+
+ // modules/svg/geolocate.js
+ function svgGeolocate(projection2) {
+ var layer = select_default2(null);
+ var _position;
+ function init2() {
+ if (svgGeolocate.initialized)
+ return;
+ svgGeolocate.enabled = false;
+ svgGeolocate.initialized = true;
+ }
+ function showLayer() {
+ layer.style("display", "block");
+ }
+ function hideLayer() {
+ layer.transition().duration(250).style("opacity", 0);
+ }
+ function layerOn() {
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1);
+ }
+ function layerOff() {
+ layer.style("display", "none");
+ }
+ function transform2(d) {
+ return svgPointTransform(projection2)(d);
+ }
+ function accuracy(accuracy2, loc) {
+ var degreesRadius = geoMetersToLat(accuracy2), tangentLoc = [loc[0], loc[1] + degreesRadius], projectedTangent = projection2(tangentLoc), projectedLoc = projection2([loc[0], loc[1]]);
+ 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", transform2);
+ layer.select(".geolocate-radius").attr("r", accuracy(_position.coords.accuracy, geolocation.loc));
+ }
+ function drawLocation(selection2) {
+ var enabled = svgGeolocate.enabled;
+ layer = selection2.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 (enabled) {
+ update();
+ } else {
+ layerOff();
+ }
+ }
+ drawLocation.enabled = function(position, enabled) {
+ if (!arguments.length)
+ return svgGeolocate.enabled;
+ _position = position;
+ svgGeolocate.enabled = enabled;
+ if (svgGeolocate.enabled) {
+ showLayer();
+ layerOn();
+ } else {
+ hideLayer();
+ }
+ return this;
+ };
+ init2();
+ return drawLocation;
+ }
+
+ // modules/svg/labels.js
+ var import_rbush7 = __toESM(require_rbush_min());
+ function svgLabels(projection2, context) {
+ var path = path_default(projection2);
+ var detected = utilDetect();
+ var baselineHack = detected.ie || detected.browser.toLowerCase() === "edge" || detected.browser.toLowerCase() === "firefox" && detected.version >= 70;
+ var _rdrawn = new import_rbush7.default();
+ var _rskipped = new import_rbush7.default();
+ var _textWidthCache = {};
+ var _entitybboxes = {};
+ 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", "ref", "*", 12],
+ ["area", "ref", "*", 12],
+ ["point", "ref", "*", 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;
+ });
+ }
+ function get4(array2, prop) {
+ return function(d, i2) {
+ return array2[i2][prop];
+ };
+ }
+ function textWidth(text2, size, elem) {
+ var c = _textWidthCache[size];
+ if (!c)
+ c = _textWidthCache[size] = {};
+ if (c[text2]) {
+ return c[text2];
+ } else if (elem) {
+ c[text2] = elem.getComputedTextLength();
+ return c[text2];
+ } else {
+ var str2 = encodeURIComponent(text2).match(/%[CDEFcdef]/g);
+ if (str2 === null) {
+ return size / 3 * 2 * text2.length;
+ } else {
+ return size / 3 * (2 * text2.length + str2.length);
+ }
+ }
+ }
+ function drawLinePaths(selection2, entities, filter2, classes, labels) {
+ var paths = selection2.selectAll("path").filter(filter2).data(entities, osmEntity.key);
+ paths.exit().remove();
+ paths.enter().append("path").style("stroke-width", get4(labels, "font-size")).attr("id", function(d) {
+ return "ideditor-labelpath-" + d.id;
+ }).attr("class", classes).merge(paths).attr("d", get4(labels, "lineString"));
+ }
+ function drawLineLabels(selection2, entities, filter2, classes, labels) {
+ var texts = selection2.selectAll("text." + classes).filter(filter2).data(entities, osmEntity.key);
+ texts.exit().remove();
+ texts.enter().append("text").attr("class", function(d, i2) {
+ return classes + " " + labels[i2].classes + " " + d.id;
+ }).attr("dy", baselineHack ? "0.35em" : null).append("textPath").attr("class", "textpath");
+ selection2.selectAll("text." + classes).selectAll(".textpath").filter(filter2).data(entities, osmEntity.key).attr("startOffset", "50%").attr("xlink:href", function(d) {
+ return "#ideditor-labelpath-" + d.id;
+ }).text(utilDisplayNameForPath);
+ }
+ function drawPointLabels(selection2, entities, filter2, classes, labels) {
+ var texts = selection2.selectAll("text." + classes).filter(filter2).data(entities, osmEntity.key);
+ texts.exit().remove();
+ texts.enter().append("text").attr("class", function(d, i2) {
+ return classes + " " + labels[i2].classes + " " + d.id;
+ }).merge(texts).attr("x", get4(labels, "x")).attr("y", get4(labels, "y")).style("text-anchor", get4(labels, "textAnchor")).text(utilDisplayName).each(function(d, i2) {
+ textWidth(utilDisplayName(d), labels[i2].height, this);
+ });
+ }
+ function drawAreaLabels(selection2, entities, filter2, classes, labels) {
+ entities = entities.filter(hasText);
+ labels = labels.filter(hasText);
+ drawPointLabels(selection2, entities, filter2, classes, labels);
+ function hasText(d, i2) {
+ return labels[i2].hasOwnProperty("x") && labels[i2].hasOwnProperty("y");
+ }
+ }
+ function drawAreaIcons(selection2, entities, filter2, classes, labels) {
+ var icons = selection2.selectAll("use." + classes).filter(filter2).data(entities, osmEntity.key);
+ icons.exit().remove();
+ icons.enter().append("use").attr("class", "icon " + classes).attr("width", "17px").attr("height", "17px").merge(icons).attr("transform", get4(labels, "transform")).attr("xlink:href", function(d) {
+ var preset = _mainPresetIndex.match(d, context.graph());
+ var picon = preset && preset.icon;
+ return picon ? "#" + picon : "";
+ });
+ }
+ function drawCollisionBoxes(selection2, 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]
+ ]] };
+ });
+ }
+ var boxes = selection2.selectAll("." + which).data(gj);
+ boxes.exit().remove();
+ boxes.enter().append("path").attr("class", classes).merge(boxes).attr("d", path_default());
+ }
+ function drawLabels(selection2, graph, entities, filter2, dimensions, fullRedraw) {
+ var wireframe = context.surface().classed("fill-wireframe");
+ var zoom = geoScaleToZoom(projection2.scale());
+ var labelable = [];
+ var renderNodeAs = {};
+ var i2, j2, k, entity, geometry;
+ for (i2 = 0; i2 < labelStack.length; i2++) {
+ labelable.push([]);
+ }
+ if (fullRedraw) {
+ _rdrawn.clear();
+ _rskipped.clear();
+ _entitybboxes = {};
+ } else {
+ for (i2 = 0; i2 < entities.length; i2++) {
+ entity = entities[i2];
+ var toRemove = [].concat(_entitybboxes[entity.id] || []).concat(_entitybboxes[entity.id + "I"] || []);
+ for (j2 = 0; j2 < toRemove.length; j2++) {
+ _rdrawn.remove(toRemove[j2]);
+ _rskipped.remove(toRemove[j2]);
+ }
+ }
+ }
+ for (i2 = 0; i2 < entities.length; i2++) {
+ entity = entities[i2];
+ geometry = entity.geometry(graph);
+ if (geometry === "point" || geometry === "vertex" && isInterestingVertex(entity)) {
+ var hasDirections = entity.directions(graph, projection2).length;
+ var markerPadding;
+ if (!wireframe && geometry === "point" && !(zoom >= 18 && hasDirections)) {
+ renderNodeAs[entity.id] = "point";
+ markerPadding = 20;
+ } else {
+ renderNodeAs[entity.id] = "vertex";
+ markerPadding = 0;
+ }
+ var coord2 = projection2(entity.loc);
+ var nodePadding = 10;
+ var bbox2 = {
+ minX: coord2[0] - nodePadding,
+ minY: coord2[1] - nodePadding - markerPadding,
+ maxX: coord2[0] + nodePadding,
+ maxY: coord2[1] + nodePadding
+ };
+ doInsert(bbox2, entity.id + "P");
+ }
+ if (geometry === "vertex") {
+ geometry = "point";
+ }
+ var preset = geometry === "area" && _mainPresetIndex.match(entity, graph);
+ var icon2 = preset && !shouldSkipIcon(preset) && preset.icon;
+ if (!icon2 && !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;
+ }
+ }
+ }
+ var positions = {
+ point: [],
+ line: [],
+ area: []
+ };
+ var labelled = {
+ point: [],
+ line: [],
+ area: []
+ };
+ for (k = 0; k < labelable.length; k++) {
+ var fontSize = labelStack[k][3];
+ for (i2 = 0; i2 < labelable[k].length; i2++) {
+ entity = labelable[k][i2];
+ 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") {
+ 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);
+ }
+ if (p) {
+ if (geometry === "vertex") {
+ geometry = "point";
+ }
+ p.classes = geometry + " tag-" + labelStack[k][1];
+ positions[geometry].push(p);
+ labelled[geometry].push(entity);
+ }
+ }
+ }
+ function isInterestingVertex(entity2) {
+ var selectedIDs = context.selectedIDs();
+ return entity2.hasInterestingTags() || entity2.isEndpoint(graph) || entity2.isConnected(graph) || selectedIDs.indexOf(entity2.id) !== -1 || graph.parentWays(entity2).some(function(parent) {
+ return selectedIDs.indexOf(parent.id) !== -1;
+ });
+ }
+ function getPointLabel(entity2, width2, height, geometry2) {
+ var y = geometry2 === "point" ? -12 : 0;
+ var pointOffsets = {
+ ltr: [15, y, "start"],
+ rtl: [-15, y, "end"]
+ };
+ var textDirection = _mainLocalizer.textDirection();
+ var coord3 = projection2(entity2.loc);
+ var textPadding = 2;
+ var offset = pointOffsets[textDirection];
+ var p2 = {
+ height,
+ width: width2,
+ x: coord3[0] + offset[0],
+ y: coord3[1] + offset[1],
+ textAnchor: offset[2]
+ };
+ var bbox3;
+ if (textDirection === "rtl") {
+ bbox3 = {
+ minX: p2.x - width2 - textPadding,
+ minY: p2.y - height / 2 - textPadding,
+ maxX: p2.x + textPadding,
+ maxY: p2.y + height / 2 + textPadding
+ };
+ } else {
+ bbox3 = {
+ minX: p2.x - textPadding,
+ minY: p2.y - height / 2 - textPadding,
+ maxX: p2.x + width2 + textPadding,
+ maxY: p2.y + height / 2 + textPadding
+ };
+ }
+ if (tryInsert([bbox3], entity2.id, true)) {
+ return p2;
+ }
+ }
+ function getLineLabel(entity2, width2, height) {
+ var viewport = geoExtent(context.projection.clipExtent()).polygon();
+ var points = graph.childNodes(entity2).map(function(node) {
+ return projection2(node.loc);
+ });
+ var length = geoPathLength(points);
+ if (length < width2 + 20)
+ return;
+ 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 i3 = 0; i3 < lineOffsets.length; i3++) {
+ var offset = lineOffsets[i3];
+ var middle = offset / 100 * length;
+ var start2 = middle - width2 / 2;
+ if (start2 < 0 || start2 + width2 > length)
+ continue;
+ var sub = subpath(points, start2, start2 + width2);
+ if (!sub || !geoPolygonIntersectsPolygon(viewport, sub, true)) {
+ continue;
+ }
+ var isReverse = reverse(sub);
+ if (isReverse) {
+ sub = sub.reverse();
+ }
+ var bboxes = [];
+ var boxsize = (height + 2) / 2;
+ for (var j3 = 0; j3 < sub.length - 1; j3++) {
+ var a = sub[j3];
+ var b = sub[j3 + 1];
+ var num = Math.max(1, Math.floor(geoVecLength(a, b) / boxsize / 2));
+ for (var box = 0; box < num; box++) {
+ var p2 = geoVecInterp(a, b, box / num);
+ var x05 = p2[0] - boxsize - padding;
+ var y05 = p2[1] - boxsize - padding;
+ var x12 = p2[0] + boxsize + padding;
+ var y12 = p2[1] + boxsize + padding;
+ bboxes.push({
+ minX: Math.min(x05, x12),
+ minY: Math.min(y05, y12),
+ maxX: Math.max(x05, x12),
+ maxY: Math.max(y05, y12)
+ });
+ }
+ }
+ if (tryInsert(bboxes, entity2.id, false)) {
+ return {
+ "font-size": height + 2,
+ lineString: lineString2(sub),
+ startOffset: offset + "%"
+ };
+ }
+ }
+ function reverse(p3) {
+ var angle2 = Math.atan2(p3[1][1] - p3[0][1], p3[1][0] - p3[0][0]);
+ return !(p3[0][0] < p3[p3.length - 1][0] && angle2 < Math.PI / 2 && angle2 > -Math.PI / 2);
+ }
+ function lineString2(points2) {
+ return "M" + points2.join("L");
+ }
+ function subpath(points2, from, to) {
+ var sofar = 0;
+ var start3, end, i0, i1;
+ for (var i4 = 0; i4 < points2.length - 1; i4++) {
+ var a2 = points2[i4];
+ var b2 = points2[i4 + 1];
+ var current = geoVecLength(a2, b2);
+ var portion;
+ if (!start3 && sofar + current >= from) {
+ portion = (from - sofar) / current;
+ start3 = [
+ a2[0] + portion * (b2[0] - a2[0]),
+ a2[1] + portion * (b2[1] - a2[1])
+ ];
+ i0 = i4 + 1;
+ }
+ if (!end && sofar + current >= to) {
+ portion = (to - sofar) / current;
+ end = [
+ a2[0] + portion * (b2[0] - a2[0]),
+ a2[1] + portion * (b2[1] - a2[1])
+ ];
+ i1 = i4 + 1;
+ }
+ sofar += current;
+ }
+ var result = points2.slice(i0, i1);
+ result.unshift(start3);
+ result.push(end);
+ return result;
+ }
+ }
+ function getAreaLabel(entity2, width2, height) {
+ var centroid = path.centroid(entity2.asGeoJSON(graph));
+ var extent = entity2.extent(graph);
+ var areaWidth = projection2(extent[1])[0] - projection2(extent[0])[0];
+ if (isNaN(centroid[0]) || areaWidth < 20)
+ return;
+ var preset2 = _mainPresetIndex.match(entity2, context.graph());
+ var picon = preset2 && preset2.icon;
+ var iconSize = 17;
+ var padding = 2;
+ var p2 = {};
+ if (picon) {
+ if (addIcon()) {
+ addLabel(iconSize + padding);
+ return p2;
+ }
+ } else {
+ if (addLabel(0)) {
+ return p2;
+ }
+ }
+ function addIcon() {
+ var iconX = centroid[0] - iconSize / 2;
+ var iconY = centroid[1] - iconSize / 2;
+ var bbox3 = {
+ minX: iconX,
+ minY: iconY,
+ maxX: iconX + iconSize,
+ maxY: iconY + iconSize
+ };
+ if (tryInsert([bbox3], entity2.id + "I", true)) {
+ p2.transform = "translate(" + iconX + "," + iconY + ")";
+ return true;
+ }
+ return false;
+ }
+ function addLabel(yOffset) {
+ if (width2 && areaWidth >= width2 + 20) {
+ var labelX = centroid[0];
+ var labelY = centroid[1] + yOffset;
+ var bbox3 = {
+ minX: labelX - width2 / 2 - padding,
+ minY: labelY - height / 2 - padding,
+ maxX: labelX + width2 / 2 + padding,
+ maxY: labelY + height / 2 + padding
+ };
+ if (tryInsert([bbox3], entity2.id, true)) {
+ p2.x = labelX;
+ p2.y = labelY;
+ p2.textAnchor = "middle";
+ p2.height = height;
+ return true;
+ }
+ }
+ return false;
+ }
+ }
+ function doInsert(bbox3, id2) {
+ bbox3.id = id2;
+ var oldbox = _entitybboxes[id2];
+ if (oldbox) {
+ _rdrawn.remove(oldbox);
+ }
+ _entitybboxes[id2] = bbox3;
+ _rdrawn.insert(bbox3);
+ }
+ function tryInsert(bboxes, id2, saveSkipped) {
+ var skipped = false;
+ for (var i3 = 0; i3 < bboxes.length; i3++) {
+ var bbox3 = bboxes[i3];
+ bbox3.id = id2;
+ if (bbox3.minX < 0 || bbox3.minY < 0 || bbox3.maxX > dimensions[0] || bbox3.maxY > dimensions[1]) {
+ skipped = true;
+ break;
+ }
+ if (_rdrawn.collides(bbox3)) {
+ skipped = true;
+ break;
+ }
+ }
+ _entitybboxes[id2] = bboxes;
+ if (skipped) {
+ if (saveSkipped) {
+ _rskipped.load(bboxes);
+ }
+ } else {
+ _rdrawn.load(bboxes);
+ }
+ return !skipped;
+ }
+ var layer = selection2.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 debug2 = layer.selectAll(".labels-group.debug");
+ drawPointLabels(label, labelled.point, filter2, "pointlabel", positions.point);
+ drawPointLabels(halo, labelled.point, filter2, "pointlabel-halo", positions.point);
+ drawLinePaths(layer, labelled.line, filter2, "", positions.line);
+ drawLineLabels(label, labelled.line, filter2, "linelabel", positions.line);
+ drawLineLabels(halo, labelled.line, filter2, "linelabel-halo", positions.line);
+ drawAreaLabels(label, labelled.area, filter2, "arealabel", positions.area);
+ drawAreaLabels(halo, labelled.area, filter2, "arealabel-halo", positions.area);
+ drawAreaIcons(label, labelled.area, filter2, "areaicon", positions.area);
+ drawAreaIcons(halo, labelled.area, filter2, "areaicon-halo", positions.area);
+ drawCollisionBoxes(debug2, _rskipped, "debug-skipped");
+ drawCollisionBoxes(debug2, _rdrawn, "debug-drawn");
+ layer.call(filterLabels);
+ }
+ function filterLabels(selection2) {
+ var drawLayer = selection2.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 pad2, bbox2;
+ if (mouse) {
+ pad2 = 20;
+ bbox2 = { minX: mouse[0] - pad2, minY: mouse[1] - pad2, maxX: mouse[0] + pad2, maxY: mouse[1] + pad2 };
+ var nearMouse = _rdrawn.search(bbox2).map(function(entity2) {
+ return entity2.id;
+ });
+ ids.push.apply(ids, nearMouse);
+ }
+ for (var i2 = 0; i2 < selectedIDs.length; i2++) {
+ var entity = graph.hasEntity(selectedIDs[i2]);
+ if (entity && entity.type === "node") {
+ ids.push(selectedIDs[i2]);
+ }
+ }
+ layers.selectAll(utilEntitySelector(ids)).classed("nolabel", true);
+ var debug2 = selection2.selectAll(".labels-group.debug");
+ var gj = [];
+ if (context.getDebug("collision")) {
+ gj = bbox2 ? [{
+ type: "Polygon",
+ coordinates: [[
+ [bbox2.minX, bbox2.minY],
+ [bbox2.maxX, bbox2.minY],
+ [bbox2.maxX, bbox2.maxY],
+ [bbox2.minX, bbox2.maxY],
+ [bbox2.minX, bbox2.minY]
+ ]]
+ }] : [];
+ }
+ var box = debug2.selectAll(".debug-mouse").data(gj);
+ box.exit().remove();
+ box.enter().append("path").attr("class", "debug debug-mouse yellow").merge(box).attr("d", path_default());
+ }
+ var throttleFilterLabels = throttle_default(filterLabels, 100);
+ drawLabels.observe = function(selection2) {
+ var listener = function() {
+ throttleFilterLabels(selection2);
+ };
+ selection2.on("mousemove.hidelabels", listener);
+ context.on("enter.hidelabels", listener);
+ };
+ drawLabels.off = function(selection2) {
+ throttleFilterLabels.cancel();
+ selection2.on("mousemove.hidelabels", null);
+ context.on("enter.hidelabels", null);
+ };
+ return drawLabels;
+ }
+
+ // modules/svg/improveOSM.js
+ var _layerEnabled2 = false;
+ var _qaService2;
+ function svgImproveOSM(projection2, context, dispatch10) {
+ const throttledRedraw = throttle_default(() => dispatch10.call("change"), 1e3);
+ const minZoom3 = 12;
+ let touchLayer = select_default2(null);
+ let drawLayer = select_default2(null);
+ let layerVisible = false;
+ function markerPath(selection2, klass) {
+ selection2.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");
+ }
+ function getService() {
+ if (services.improveOSM && !_qaService2) {
+ _qaService2 = services.improveOSM;
+ _qaService2.on("loaded", throttledRedraw);
+ } else if (!services.improveOSM && _qaService2) {
+ _qaService2 = null;
+ }
+ return _qaService2;
+ }
+ function editOn() {
+ if (!layerVisible) {
+ layerVisible = true;
+ drawLayer.style("display", "block");
+ }
+ }
+ function editOff() {
+ if (layerVisible) {
+ layerVisible = false;
+ drawLayer.style("display", "none");
+ drawLayer.selectAll(".qaItem.improveOSM").remove();
+ touchLayer.selectAll(".qaItem.improveOSM").remove();
+ }
+ }
+ function layerOn() {
+ editOn();
+ drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch10.call("change"));
+ }
+ function layerOff() {
+ throttledRedraw.cancel();
+ drawLayer.interrupt();
+ touchLayer.selectAll(".qaItem.improveOSM").remove();
+ drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
+ editOff();
+ dispatch10.call("change");
+ });
+ }
+ function updateMarkers() {
+ if (!layerVisible || !_layerEnabled2)
+ return;
+ const service = getService();
+ const selectedID = context.selectedErrorID();
+ const data = service ? service.getItems(projection2) : [];
+ const getTransform = svgPointTransform(projection2);
+ const markers = drawLayer.selectAll(".qaItem.improveOSM").data(data, (d) => d.id);
+ markers.exit().remove();
+ const markersEnter = markers.enter().append("g").attr("class", (d) => `qaItem ${d.service} itemId-${d.id} itemType-${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("class", "icon-annotation").attr("transform", "translate(-6, -22)").attr("width", "12px").attr("height", "12px").attr("xlink:href", (d) => d.icon ? "#" + d.icon : "");
+ markers.merge(markersEnter).sort(sortY).classed("selected", (d) => d.id === selectedID).attr("transform", getTransform);
+ if (touchLayer.empty())
+ return;
+ const fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ const targets = touchLayer.selectAll(".qaItem.improveOSM").data(data, (d) => d.id);
+ targets.exit().remove();
+ targets.enter().append("rect").attr("width", "20px").attr("height", "30px").attr("x", "-10px").attr("y", "-28px").merge(targets).sort(sortY).attr("class", (d) => `qaItem ${d.service} target ${fillClass} itemId-${d.id}`).attr("transform", getTransform);
+ function sortY(a, b) {
+ return a.id === selectedID ? 1 : b.id === selectedID ? -1 : b.loc[1] - a.loc[1];
+ }
+ }
+ function drawImproveOSM(selection2) {
+ const service = getService();
+ const surface = context.surface();
+ if (surface && !surface.empty()) {
+ touchLayer = surface.selectAll(".data-layer.touch .layer-touch.markers");
+ }
+ drawLayer = selection2.selectAll(".layer-improveOSM").data(service ? [0] : []);
+ drawLayer.exit().remove();
+ drawLayer = drawLayer.enter().append("g").attr("class", "layer-improveOSM").style("display", _layerEnabled2 ? "block" : "none").merge(drawLayer);
+ if (_layerEnabled2) {
+ if (service && ~~context.map().zoom() >= minZoom3) {
+ editOn();
+ service.loadIssues(projection2);
+ updateMarkers();
+ } else {
+ editOff();
+ }
+ }
+ }
+ drawImproveOSM.enabled = function(val) {
+ if (!arguments.length)
+ return _layerEnabled2;
+ _layerEnabled2 = val;
+ if (_layerEnabled2) {
+ layerOn();
+ } else {
+ layerOff();
+ if (context.selectedErrorID()) {
+ context.enter(modeBrowse(context));
+ }
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawImproveOSM.supported = () => !!getService();
+ return drawImproveOSM;
+ }
+
+ // modules/svg/osmose.js
+ var _layerEnabled3 = false;
+ var _qaService3;
+ function svgOsmose(projection2, context, dispatch10) {
+ const throttledRedraw = throttle_default(() => dispatch10.call("change"), 1e3);
+ const minZoom3 = 12;
+ let touchLayer = select_default2(null);
+ let drawLayer = select_default2(null);
+ let layerVisible = false;
+ function markerPath(selection2, klass) {
+ selection2.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");
+ }
+ function getService() {
+ if (services.osmose && !_qaService3) {
+ _qaService3 = services.osmose;
+ _qaService3.on("loaded", throttledRedraw);
+ } else if (!services.osmose && _qaService3) {
+ _qaService3 = null;
+ }
+ return _qaService3;
+ }
+ function editOn() {
+ if (!layerVisible) {
+ layerVisible = true;
+ drawLayer.style("display", "block");
+ }
+ }
+ function editOff() {
+ if (layerVisible) {
+ layerVisible = false;
+ drawLayer.style("display", "none");
+ drawLayer.selectAll(".qaItem.osmose").remove();
+ touchLayer.selectAll(".qaItem.osmose").remove();
+ }
+ }
+ function layerOn() {
+ editOn();
+ drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch10.call("change"));
+ }
+ function layerOff() {
+ throttledRedraw.cancel();
+ drawLayer.interrupt();
+ touchLayer.selectAll(".qaItem.osmose").remove();
+ drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
+ editOff();
+ dispatch10.call("change");
+ });
+ }
+ function updateMarkers() {
+ if (!layerVisible || !_layerEnabled3)
+ return;
+ const service = getService();
+ const selectedID = context.selectedErrorID();
+ const data = service ? service.getItems(projection2) : [];
+ const getTransform = svgPointTransform(projection2);
+ const markers = drawLayer.selectAll(".qaItem.osmose").data(data, (d) => d.id);
+ markers.exit().remove();
+ const markersEnter = markers.enter().append("g").attr("class", (d) => `qaItem ${d.service} itemId-${d.id} itemType-${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", (d) => service.getColor(d.item)).call(markerPath, "qaItem-fill");
+ markersEnter.append("use").attr("class", "icon-annotation").attr("transform", "translate(-6, -22)").attr("width", "12px").attr("height", "12px").attr("xlink:href", (d) => d.icon ? "#" + d.icon : "");
+ markers.merge(markersEnter).sort(sortY).classed("selected", (d) => d.id === selectedID).attr("transform", getTransform);
+ if (touchLayer.empty())
+ return;
+ const fillClass = context.getDebug("target") ? "pink" : "nocolor";
+ const targets = touchLayer.selectAll(".qaItem.osmose").data(data, (d) => d.id);
+ targets.exit().remove();
+ targets.enter().append("rect").attr("width", "20px").attr("height", "30px").attr("x", "-10px").attr("y", "-28px").merge(targets).sort(sortY).attr("class", (d) => `qaItem ${d.service} target ${fillClass} itemId-${d.id}`).attr("transform", getTransform);
+ function sortY(a, b) {
+ return a.id === selectedID ? 1 : b.id === selectedID ? -1 : b.loc[1] - a.loc[1];
+ }
+ }
+ function drawOsmose(selection2) {
+ const service = getService();
+ const surface = context.surface();
+ if (surface && !surface.empty()) {
+ touchLayer = surface.selectAll(".data-layer.touch .layer-touch.markers");
+ }
+ drawLayer = selection2.selectAll(".layer-osmose").data(service ? [0] : []);
+ drawLayer.exit().remove();
+ drawLayer = drawLayer.enter().append("g").attr("class", "layer-osmose").style("display", _layerEnabled3 ? "block" : "none").merge(drawLayer);
+ if (_layerEnabled3) {
+ if (service && ~~context.map().zoom() >= minZoom3) {
+ editOn();
+ service.loadIssues(projection2);
+ updateMarkers();
+ } else {
+ editOff();
+ }
+ }
+ }
+ drawOsmose.enabled = function(val) {
+ if (!arguments.length)
+ return _layerEnabled3;
+ _layerEnabled3 = val;
+ if (_layerEnabled3) {
+ getService().loadStrings().then(layerOn).catch((err) => {
+ console.log(err);
+ });
+ } else {
+ layerOff();
+ if (context.selectedErrorID()) {
+ context.enter(modeBrowse(context));
+ }
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawOsmose.supported = () => !!getService();
+ return drawOsmose;
+ }
+
+ // modules/svg/streetside.js
+ function svgStreetside(projection2, context, dispatch10) {
+ var throttledRedraw = throttle_default(function() {
+ dispatch10.call("change");
+ }, 1e3);
+ var minZoom3 = 14;
+ var minMarkerZoom = 16;
+ var minViewfieldZoom = 18;
+ var layer = select_default2(null);
+ var _viewerYaw = 0;
+ var _selectedSequence = null;
+ var _streetside;
+ function init2() {
+ if (svgStreetside.initialized)
+ return;
+ svgStreetside.enabled = false;
+ svgStreetside.initialized = true;
+ }
+ function getService() {
+ if (services.streetside && !_streetside) {
+ _streetside = services.streetside;
+ _streetside.event.on("viewerChanged.svgStreetside", viewerChanged).on("loadedImages.svgStreetside", throttledRedraw);
+ } else if (!services.streetside && _streetside) {
+ _streetside = null;
+ }
+ return _streetside;
+ }
+ function showLayer() {
+ var service = getService();
+ if (!service)
+ return;
+ editOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
+ dispatch10.call("change");
+ });
+ }
+ 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, d) {
+ var service = getService();
+ if (!service)
+ return;
+ if (d.sequenceKey !== _selectedSequence) {
+ _viewerYaw = 0;
+ }
+ _selectedSequence = d.sequenceKey;
+ service.ensureViewerLoaded(context).then(function() {
+ service.selectImage(context, d.key).yaw(_viewerYaw).showViewer(context);
+ });
+ context.map().centerEase(d.loc);
+ }
+ function mouseover(d3_event, d) {
+ var service = getService();
+ if (service)
+ service.setStyles(context, d);
+ }
+ function mouseout() {
+ var service = getService();
+ if (service)
+ service.setStyles(context, null);
+ }
+ function transform2(d) {
+ var t = svgPointTransform(projection2)(d);
+ var rot = d.ca + _viewerYaw;
+ if (rot) {
+ t += " rotate(" + Math.floor(rot) + ",0,0)";
+ }
+ return t;
+ }
+ function viewerChanged() {
+ var service = getService();
+ if (!service)
+ return;
+ var viewer = service.viewer();
+ if (!viewer)
+ return;
+ _viewerYaw = viewer.getYaw();
+ if (context.map().isTransformed())
+ return;
+ layer.selectAll(".viewfield-group.currentView").attr("transform", transform2);
+ }
+ 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;
+ });
+ }
+ if (toDate) {
+ var toTimestamp = new Date(toDate).getTime();
+ bubbles = bubbles.filter(function(bubble) {
+ return new Date(bubble.captured_at).getTime() <= toTimestamp;
+ });
+ }
+ if (usernames) {
+ bubbles = bubbles.filter(function(bubble) {
+ return usernames.indexOf(bubble.captured_by) !== -1;
+ });
+ }
+ 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(sequences2) {
+ return new Date(sequences2.properties.captured_at).getTime() >= fromTimestamp;
+ });
+ }
+ if (toDate) {
+ var toTimestamp = new Date(toDate).getTime();
+ sequences = sequences.filter(function(sequences2) {
+ return new Date(sequences2.properties.captured_at).getTime() <= toTimestamp;
+ });
+ }
+ if (usernames) {
+ sequences = sequences.filter(function(sequences2) {
+ return usernames.indexOf(sequences2.properties.captured_by) !== -1;
+ });
+ }
+ return sequences;
+ }
+ function update() {
+ var viewer = context.container().select(".photoviewer");
+ var selected = viewer.empty() ? void 0 : viewer.datum();
+ var z = ~~context.map().zoom();
+ var showMarkers = z >= minMarkerZoom;
+ var showViewfields = z >= minViewfieldZoom;
+ var service = getService();
+ var sequences = [];
+ var bubbles = [];
+ if (context.photos().showsPanoramic()) {
+ sequences = service ? service.sequences(projection2) : [];
+ bubbles = service && showMarkers ? service.bubbles(projection2) : [];
+ sequences = filterSequences(sequences);
+ bubbles = filterBubbles(bubbles);
+ }
+ var traces = layer.selectAll(".sequences").selectAll(".sequence").data(sequences, function(d) {
+ return d.properties.key;
+ });
+ traces.exit().remove();
+ traces = traces.enter().append("path").attr("class", "sequence").merge(traces).attr("d", svgPath(projection2).geojson);
+ var groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(bubbles, function(d) {
+ return d.key + (d.sequenceKey ? "v1" : "v0");
+ });
+ groups.exit().remove();
+ 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");
+ var markers = groups.merge(groupsEnter).sort(function(a, b) {
+ return a === selected ? 1 : b === selected ? -1 : b.loc[1] - a.loc[1];
+ }).attr("transform", transform2).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", viewfieldPath);
+ function viewfieldPath() {
+ var d = this.parentNode.__data__;
+ 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";
+ }
+ }
+ }
+ function drawImages(selection2) {
+ var enabled = svgStreetside.enabled;
+ var service = getService();
+ layer = selection2.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);
+ if (enabled) {
+ if (service && ~~context.map().zoom() >= minZoom3) {
+ editOn();
+ update();
+ service.loadBubbles(projection2);
+ } else {
+ editOff();
+ }
+ }
+ }
+ drawImages.enabled = function(_) {
+ if (!arguments.length)
+ return svgStreetside.enabled;
+ svgStreetside.enabled = _;
+ if (svgStreetside.enabled) {
+ showLayer();
+ context.photos().on("change.streetside", update);
+ } else {
+ hideLayer();
+ context.photos().on("change.streetside", null);
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawImages.supported = function() {
+ return !!getService();
+ };
+ init2();
+ return drawImages;
+ }
+
+ // modules/svg/mapillary_images.js
+ function svgMapillaryImages(projection2, context, dispatch10) {
+ const throttledRedraw = throttle_default(function() {
+ dispatch10.call("change");
+ }, 1e3);
+ const minZoom3 = 12;
+ const minMarkerZoom = 16;
+ const minViewfieldZoom = 18;
+ let layer = select_default2(null);
+ let _mapillary;
+ function init2() {
+ if (svgMapillaryImages.initialized)
+ return;
+ svgMapillaryImages.enabled = false;
+ svgMapillaryImages.initialized = true;
+ }
+ function getService() {
+ if (services.mapillary && !_mapillary) {
+ _mapillary = services.mapillary;
+ _mapillary.event.on("loadedImages", throttledRedraw);
+ } else if (!services.mapillary && _mapillary) {
+ _mapillary = null;
+ }
+ return _mapillary;
+ }
+ function showLayer() {
+ const service = getService();
+ if (!service)
+ return;
+ editOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
+ dispatch10.call("change");
+ });
+ }
+ 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) {
+ const service = getService();
+ if (!service)
+ return;
+ service.ensureViewerLoaded(context).then(function() {
+ service.selectImage(context, image.id).showViewer(context);
+ });
+ context.map().centerEase(image.loc);
+ }
+ function mouseover(d3_event, image) {
+ const service = getService();
+ if (service)
+ service.setStyles(context, image);
+ }
+ function mouseout() {
+ const service = getService();
+ if (service)
+ service.setStyles(context, null);
+ }
+ function transform2(d) {
+ let t = svgPointTransform(projection2)(d);
+ if (d.ca) {
+ t += " rotate(" + Math.floor(d.ca) + ",0,0)";
+ }
+ return t;
+ }
+ function filterImages(images) {
+ const showsPano = context.photos().showsPanoramic();
+ const showsFlat = context.photos().showsFlat();
+ const fromDate = context.photos().fromDate();
+ const toDate = context.photos().toDate();
+ if (!showsPano || !showsFlat) {
+ images = images.filter(function(image) {
+ if (image.is_pano)
+ return showsPano;
+ return showsFlat;
+ });
+ }
+ if (fromDate) {
+ images = images.filter(function(image) {
+ return new Date(image.captured_at).getTime() >= new Date(fromDate).getTime();
+ });
+ }
+ if (toDate) {
+ images = images.filter(function(image) {
+ return new Date(image.captured_at).getTime() <= new Date(toDate).getTime();
+ });
+ }
+ return images;
+ }
+ function filterSequences(sequences) {
+ const showsPano = context.photos().showsPanoramic();
+ const showsFlat = context.photos().showsFlat();
+ const fromDate = context.photos().fromDate();
+ const 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;
+ }
+ return false;
+ });
+ }
+ 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();
+ });
+ }
+ return sequences;
+ }
+ function update() {
+ const z = ~~context.map().zoom();
+ const showMarkers = z >= minMarkerZoom;
+ const showViewfields = z >= minViewfieldZoom;
+ const service = getService();
+ let sequences = service ? service.sequences(projection2) : [];
+ let images = service && showMarkers ? service.images(projection2) : [];
+ images = filterImages(images);
+ sequences = filterSequences(sequences, service);
+ service.filterViewer(context);
+ let traces = layer.selectAll(".sequences").selectAll(".sequence").data(sequences, function(d) {
+ return d.properties.id;
+ });
+ traces.exit().remove();
+ traces = traces.enter().append("path").attr("class", "sequence").merge(traces).attr("d", svgPath(projection2).geojson);
+ const groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(images, function(d) {
+ return d.id;
+ });
+ groups.exit().remove();
+ const 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");
+ const markers = groups.merge(groupsEnter).sort(function(a, b) {
+ return b.loc[1] - a.loc[1];
+ }).attr("transform", transform2).select(".viewfield-scale");
+ markers.selectAll("circle").data([0]).enter().append("circle").attr("dx", "0").attr("dy", "0").attr("r", "6");
+ const 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__.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 drawImages(selection2) {
+ const enabled = svgMapillaryImages.enabled;
+ const service = getService();
+ layer = selection2.selectAll(".layer-mapillary").data(service ? [0] : []);
+ layer.exit().remove();
+ const 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() >= minZoom3) {
+ editOn();
+ update();
+ service.loadImages(projection2);
+ } else {
+ editOff();
+ }
+ }
+ }
+ 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);
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawImages.supported = function() {
+ return !!getService();
+ };
+ init2();
+ return drawImages;
+ }
+
+ // modules/svg/mapillary_position.js
+ function svgMapillaryPosition(projection2, context) {
+ const throttledRedraw = throttle_default(function() {
+ update();
+ }, 1e3);
+ const minZoom3 = 12;
+ const minViewfieldZoom = 18;
+ let layer = select_default2(null);
+ let _mapillary;
+ let viewerCompassAngle;
+ function init2() {
+ if (svgMapillaryPosition.initialized)
+ return;
+ svgMapillaryPosition.initialized = true;
+ }
+ function getService() {
+ if (services.mapillary && !_mapillary) {
+ _mapillary = services.mapillary;
+ _mapillary.event.on("imageChanged", throttledRedraw);
+ _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", transform2);
+ });
+ } else if (!services.mapillary && _mapillary) {
+ _mapillary = null;
+ }
+ return _mapillary;
+ }
+ function editOn() {
+ layer.style("display", "block");
+ }
+ function editOff() {
+ layer.selectAll(".viewfield-group").remove();
+ layer.style("display", "none");
+ }
+ function transform2(d) {
+ let t = svgPointTransform(projection2)(d);
+ 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 t;
+ }
+ function update() {
+ const z = ~~context.map().zoom();
+ const showViewfields = z >= minViewfieldZoom;
+ const service = getService();
+ const image = service && service.getActiveImage();
+ const groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(image ? [image] : [], function(d) {
+ return d.id;
+ });
+ groups.exit().remove();
+ const groupsEnter = groups.enter().append("g").attr("class", "viewfield-group currentView highlighted");
+ groupsEnter.append("g").attr("class", "viewfield-scale");
+ const markers = groups.merge(groupsEnter).attr("transform", transform2).select(".viewfield-scale");
+ markers.selectAll("circle").data([0]).enter().append("circle").attr("dx", "0").attr("dy", "0").attr("r", "6");
+ const 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(selection2) {
+ const service = getService();
+ layer = selection2.selectAll(".layer-mapillary-position").data(service ? [0] : []);
+ layer.exit().remove();
+ const 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() >= minZoom3) {
+ editOn();
+ update();
+ } else {
+ editOff();
+ }
+ }
+ drawImages.enabled = function() {
+ update();
+ return this;
+ };
+ drawImages.supported = function() {
+ return !!getService();
+ };
+ init2();
+ return drawImages;
+ }
+
+ // modules/svg/mapillary_signs.js
+ function svgMapillarySigns(projection2, context, dispatch10) {
+ const throttledRedraw = throttle_default(function() {
+ dispatch10.call("change");
+ }, 1e3);
+ const minZoom3 = 12;
+ let layer = select_default2(null);
+ let _mapillary;
+ function init2() {
+ if (svgMapillarySigns.initialized)
+ return;
+ 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;
+ }
+ return _mapillary;
+ }
+ function showLayer() {
+ const 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) {
+ const service = getService();
+ if (!service)
+ return;
+ context.map().centerEase(d.loc);
+ const selectedImageId = service.getActiveImage() && service.getActiveImage().id;
+ service.getDetections(d.id).then((detections) => {
+ if (detections.length) {
+ const 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);
+ });
+ }
+ }
+ });
+ }
+ 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(feature3) {
+ return new Date(feature3.last_seen_at).getTime() >= fromTimestamp;
+ });
+ }
+ if (toDate) {
+ var toTimestamp = new Date(toDate).getTime();
+ detectedFeatures = detectedFeatures.filter(function(feature3) {
+ return new Date(feature3.first_seen_at).getTime() <= toTimestamp;
+ });
+ }
+ return detectedFeatures;
+ }
+ function update() {
+ const service = getService();
+ let data = service ? service.signs(projection2) : [];
+ data = filterData(data);
+ const transform2 = svgPointTransform(projection2);
+ const signs = layer.selectAll(".icon-sign").data(data, function(d) {
+ return d.id;
+ });
+ signs.exit().remove();
+ const 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");
+ signs.merge(enter).attr("transform", transform2);
+ }
+ function drawSigns(selection2) {
+ const enabled = svgMapillarySigns.enabled;
+ const service = getService();
+ layer = selection2.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() >= minZoom3) {
+ editOn();
+ update();
+ service.loadSigns(projection2);
+ service.showSignDetections(true);
+ } else {
+ editOff();
+ }
+ } else if (service) {
+ service.showSignDetections(false);
+ }
+ }
+ drawSigns.enabled = function(_) {
+ if (!arguments.length)
+ return svgMapillarySigns.enabled;
+ svgMapillarySigns.enabled = _;
+ if (svgMapillarySigns.enabled) {
+ showLayer();
+ context.photos().on("change.mapillary_signs", update);
+ } else {
+ hideLayer();
+ context.photos().on("change.mapillary_signs", null);
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawSigns.supported = function() {
+ return !!getService();
+ };
+ init2();
+ return drawSigns;
+ }
+
+ // modules/svg/mapillary_map_features.js
+ function svgMapillaryMapFeatures(projection2, context, dispatch10) {
+ const throttledRedraw = throttle_default(function() {
+ dispatch10.call("change");
+ }, 1e3);
+ const minZoom3 = 12;
+ let layer = select_default2(null);
+ let _mapillary;
+ function init2() {
+ if (svgMapillaryMapFeatures.initialized)
+ return;
+ svgMapillaryMapFeatures.enabled = false;
+ svgMapillaryMapFeatures.initialized = true;
+ }
+ function getService() {
+ if (services.mapillary && !_mapillary) {
+ _mapillary = services.mapillary;
+ _mapillary.event.on("loadedMapFeatures", throttledRedraw);
+ } else if (!services.mapillary && _mapillary) {
+ _mapillary = null;
+ }
+ return _mapillary;
+ }
+ function showLayer() {
+ const service = getService();
+ if (!service)
+ return;
+ service.loadObjectResources(context);
+ editOn();
+ }
+ function hideLayer() {
+ throttledRedraw.cancel();
+ editOff();
+ }
+ function editOn() {
+ layer.style("display", "block");
+ }
+ function editOff() {
+ layer.selectAll(".icon-map-feature").remove();
+ layer.style("display", "none");
+ }
+ function click(d3_event, d) {
+ const service = getService();
+ if (!service)
+ return;
+ context.map().centerEase(d.loc);
+ const selectedImageId = service.getActiveImage() && service.getActiveImage().id;
+ service.getDetections(d.id).then((detections) => {
+ if (detections.length) {
+ const 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);
+ });
+ }
+ }
+ });
+ }
+ function filterData(detectedFeatures) {
+ const fromDate = context.photos().fromDate();
+ const toDate = context.photos().toDate();
+ if (fromDate) {
+ detectedFeatures = detectedFeatures.filter(function(feature3) {
+ return new Date(feature3.last_seen_at).getTime() >= new Date(fromDate).getTime();
+ });
+ }
+ if (toDate) {
+ detectedFeatures = detectedFeatures.filter(function(feature3) {
+ return new Date(feature3.first_seen_at).getTime() <= new Date(toDate).getTime();
+ });
+ }
+ return detectedFeatures;
+ }
+ function update() {
+ const service = getService();
+ let data = service ? service.mapFeatures(projection2) : [];
+ data = filterData(data);
+ const transform2 = svgPointTransform(projection2);
+ const mapFeatures = layer.selectAll(".icon-map-feature").data(data, function(d) {
+ return d.id;
+ });
+ mapFeatures.exit().remove();
+ const enter = mapFeatures.enter().append("g").attr("class", "icon-map-feature icon-detected").on("click", click);
+ enter.append("title").text(function(d) {
+ var id2 = d.value.replace(/--/g, ".").replace(/-/g, "_");
+ return _t("mapillary_map_features." + id2);
+ });
+ 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") {
+ return "#object--sign--advertisement";
+ }
+ return "#" + d.value;
+ });
+ enter.append("rect").attr("width", "24px").attr("height", "24px").attr("x", "-12px").attr("y", "-12px");
+ mapFeatures.merge(enter).attr("transform", transform2);
+ }
+ function drawMapFeatures(selection2) {
+ const enabled = svgMapillaryMapFeatures.enabled;
+ const service = getService();
+ layer = selection2.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() >= minZoom3) {
+ editOn();
+ update();
+ service.loadMapFeatures(projection2);
+ service.showFeatureDetections(true);
+ } else {
+ editOff();
+ }
+ } 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);
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawMapFeatures.supported = function() {
+ return !!getService();
+ };
+ init2();
+ return drawMapFeatures;
+ }
+
+ // modules/svg/kartaview_images.js
+ function svgKartaviewImages(projection2, context, dispatch10) {
+ var throttledRedraw = throttle_default(function() {
+ dispatch10.call("change");
+ }, 1e3);
+ var minZoom3 = 12;
+ var minMarkerZoom = 16;
+ var minViewfieldZoom = 18;
+ var layer = select_default2(null);
+ var _kartaview;
+ function init2() {
+ if (svgKartaviewImages.initialized)
+ return;
+ svgKartaviewImages.enabled = false;
+ svgKartaviewImages.initialized = true;
+ }
+ function getService() {
+ if (services.kartaview && !_kartaview) {
+ _kartaview = services.kartaview;
+ _kartaview.event.on("loadedImages", throttledRedraw);
+ } else if (!services.kartaview && _kartaview) {
+ _kartaview = null;
+ }
+ return _kartaview;
+ }
+ function showLayer() {
+ var service = getService();
+ if (!service)
+ return;
+ editOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
+ dispatch10.call("change");
+ });
+ }
+ 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, 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 mouseover(d3_event, d) {
+ var service = getService();
+ if (service)
+ service.setStyles(context, d);
+ }
+ function mouseout() {
+ var service = getService();
+ if (service)
+ service.setStyles(context, null);
+ }
+ function transform2(d) {
+ var t = svgPointTransform(projection2)(d);
+ if (d.ca) {
+ t += " rotate(" + Math.floor(d.ca) + ",0,0)";
+ }
+ return t;
+ }
+ function filterImages(images) {
+ var fromDate = context.photos().fromDate();
+ var toDate = context.photos().toDate();
+ var usernames = context.photos().usernames();
+ if (fromDate) {
+ var fromTimestamp = new Date(fromDate).getTime();
+ images = images.filter(function(item) {
+ return new Date(item.captured_at).getTime() >= fromTimestamp;
+ });
+ }
+ if (toDate) {
+ var toTimestamp = new Date(toDate).getTime();
+ images = images.filter(function(item) {
+ return new Date(item.captured_at).getTime() <= toTimestamp;
+ });
+ }
+ if (usernames) {
+ images = images.filter(function(item) {
+ return usernames.indexOf(item.captured_by) !== -1;
+ });
+ }
+ return images;
+ }
+ 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(image) {
+ return new Date(image.properties.captured_at).getTime() >= fromTimestamp;
+ });
+ }
+ if (toDate) {
+ var toTimestamp = new Date(toDate).getTime();
+ sequences = sequences.filter(function(image) {
+ return new Date(image.properties.captured_at).getTime() <= toTimestamp;
+ });
+ }
+ if (usernames) {
+ sequences = sequences.filter(function(image) {
+ return usernames.indexOf(image.properties.captured_by) !== -1;
+ });
+ }
+ return sequences;
+ }
+ function update() {
+ var viewer = context.container().select(".photoviewer");
+ var selected = viewer.empty() ? void 0 : viewer.datum();
+ var z = ~~context.map().zoom();
+ var showMarkers = z >= minMarkerZoom;
+ var showViewfields = z >= minViewfieldZoom;
+ var service = getService();
+ var sequences = [];
+ var images = [];
+ if (context.photos().showsFlat()) {
+ sequences = service ? service.sequences(projection2) : [];
+ images = service && showMarkers ? service.images(projection2) : [];
+ sequences = filterSequences(sequences);
+ images = filterImages(images);
+ }
+ var traces = layer.selectAll(".sequences").selectAll(".sequence").data(sequences, function(d) {
+ return d.properties.key;
+ });
+ traces.exit().remove();
+ traces = traces.enter().append("path").attr("class", "sequence").merge(traces).attr("d", svgPath(projection2).geojson);
+ var groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(images, function(d) {
+ return d.key;
+ });
+ groups.exit().remove();
+ 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");
+ var markers = groups.merge(groupsEnter).sort(function(a, b) {
+ return a === selected ? 1 : b === selected ? -1 : b.loc[1] - a.loc[1];
+ }).attr("transform", transform2).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(selection2) {
+ var enabled = svgKartaviewImages.enabled, service = getService();
+ layer = selection2.selectAll(".layer-kartaview").data(service ? [0] : []);
+ layer.exit().remove();
+ var layerEnter = layer.enter().append("g").attr("class", "layer-kartaview").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() >= minZoom3) {
+ editOn();
+ update();
+ service.loadImages(projection2);
+ } else {
+ editOff();
+ }
+ }
+ }
+ drawImages.enabled = function(_) {
+ if (!arguments.length)
+ return svgKartaviewImages.enabled;
+ svgKartaviewImages.enabled = _;
+ if (svgKartaviewImages.enabled) {
+ showLayer();
+ context.photos().on("change.kartaview_images", update);
+ } else {
+ hideLayer();
+ context.photos().on("change.kartaview_images", null);
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ drawImages.supported = function() {
+ return !!getService();
+ };
+ init2();
+ return drawImages;
+ }
+
+ // modules/svg/osm.js
+ function svgOsm(projection2, context, dispatch10) {
+ var enabled = true;
+ function drawOsm(selection2) {
+ selection2.selectAll(".layer-osm").data(["covered", "areas", "lines", "points", "labels"]).enter().append("g").attr("class", function(d) {
+ return "layer-osm " + d;
+ });
+ selection2.selectAll(".layer-osm.points").selectAll(".points-group").data(["points", "midpoints", "vertices", "turns"]).enter().append("g").attr("class", function(d) {
+ return "points-group " + d;
+ });
+ }
+ 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() {
+ dispatch10.call("change");
+ });
+ }
+ 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);
+ dispatch10.call("change");
+ });
+ }
+ drawOsm.enabled = function(val) {
+ if (!arguments.length)
+ return enabled;
+ enabled = val;
+ if (enabled) {
+ showLayer();
+ } else {
+ hideLayer();
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ return drawOsm;
+ }
+
+ // modules/svg/notes.js
+ var _notesEnabled = false;
+ var _osmService;
+ function svgNotes(projection2, context, dispatch10) {
+ if (!dispatch10) {
+ dispatch10 = dispatch_default("change");
+ }
+ var throttledRedraw = throttle_default(function() {
+ dispatch10.call("change");
+ }, 1e3);
+ var minZoom3 = 12;
+ var touchLayer = select_default2(null);
+ var drawLayer = select_default2(null);
+ var _notesVisible = false;
+ function markerPath(selection2, klass) {
+ selection2.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");
+ }
+ function getService() {
+ if (services.osm && !_osmService) {
+ _osmService = services.osm;
+ _osmService.on("loadedNotes", throttledRedraw);
+ } else if (!services.osm && _osmService) {
+ _osmService = null;
+ }
+ return _osmService;
+ }
+ function editOn() {
+ if (!_notesVisible) {
+ _notesVisible = true;
+ drawLayer.style("display", "block");
+ }
+ }
+ function editOff() {
+ if (_notesVisible) {
+ _notesVisible = false;
+ drawLayer.style("display", "none");
+ drawLayer.selectAll(".note").remove();
+ touchLayer.selectAll(".note").remove();
+ }
+ }
+ function layerOn() {
+ editOn();
+ drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", function() {
+ dispatch10.call("change");
+ });
+ }
+ function layerOff() {
+ throttledRedraw.cancel();
+ drawLayer.interrupt();
+ touchLayer.selectAll(".note").remove();
+ drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", function() {
+ editOff();
+ dispatch10.call("change");
+ });
+ }
+ function updateMarkers() {
+ if (!_notesVisible || !_notesEnabled)
+ return;
+ var service = getService();
+ var selectedID = context.selectedNoteID();
+ var data = service ? service.notes(projection2) : [];
+ var getTransform = svgPointTransform(projection2);
+ var notes = drawLayer.selectAll(".note").data(data, function(d) {
+ return d.status + d.id;
+ });
+ notes.exit().remove();
+ 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";
+ });
+ notes.merge(notesEnter).sort(sortY).classed("selected", function(d) {
+ var mode = context.mode();
+ var isMoving = mode && mode.id === "drag-note";
+ return !isMoving && d.id === selectedID;
+ }).attr("transform", getTransform);
+ if (touchLayer.empty())
+ return;
+ var fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ var targets = touchLayer.selectAll(".note").data(data, function(d) {
+ return d.id;
+ });
+ targets.exit().remove();
+ 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);
+ function sortY(a, b) {
+ if (a.id === selectedID)
+ return 1;
+ if (b.id === selectedID)
+ return -1;
+ return b.loc[1] - a.loc[1];
+ }
+ }
+ function drawNotes(selection2) {
+ var service = getService();
+ var surface = context.surface();
+ if (surface && !surface.empty()) {
+ touchLayer = surface.selectAll(".data-layer.touch .layer-touch.markers");
+ }
+ drawLayer = selection2.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);
+ if (_notesEnabled) {
+ if (service && ~~context.map().zoom() >= minZoom3) {
+ editOn();
+ service.loadNotes(projection2);
+ updateMarkers();
+ } else {
+ editOff();
+ }
+ }
+ }
+ drawNotes.enabled = function(val) {
+ if (!arguments.length)
+ return _notesEnabled;
+ _notesEnabled = val;
+ if (_notesEnabled) {
+ layerOn();
+ } else {
+ layerOff();
+ if (context.selectedNoteID()) {
+ context.enter(modeBrowse(context));
+ }
+ }
+ dispatch10.call("change");
+ return this;
+ };
+ return drawNotes;
+ }
+
+ // modules/svg/touch.js
+ function svgTouch() {
+ function drawTouch(selection2) {
+ selection2.selectAll(".layer-touch").data(["areas", "lines", "points", "turns", "markers"]).enter().append("g").attr("class", function(d) {
+ return "layer-touch " + d;
+ });
+ }
+ return drawTouch;
+ }
+
+ // modules/util/dimensions.js
+ function refresh(selection2, node) {
+ var cr = node.getBoundingClientRect();
+ var prop = [cr.width, cr.height];
+ selection2.property("__dimensions__", prop);
+ return prop;
+ }
+ function utilGetDimensions(selection2, force) {
+ if (!selection2 || selection2.empty()) {
+ return [0, 0];
+ }
+ var node = selection2.node(), cached = selection2.property("__dimensions__");
+ return !cached || force ? refresh(selection2, node) : cached;
+ }
+ function utilSetDimensions(selection2, dimensions) {
+ if (!selection2 || selection2.empty()) {
+ return selection2;
+ }
+ var node = selection2.node();
+ if (dimensions === null) {
+ refresh(selection2, node);
+ return selection2;
+ }
+ return selection2.property("__dimensions__", [dimensions[0], dimensions[1]]).attr("width", dimensions[0]).attr("height", dimensions[1]);
+ }
+
+ // modules/svg/layers.js
+ function svgLayers(projection2, context) {
+ var dispatch10 = dispatch_default("change");
+ var svg2 = select_default2(null);
+ var _layers = [
+ { id: "osm", layer: svgOsm(projection2, context, dispatch10) },
+ { id: "notes", layer: svgNotes(projection2, context, dispatch10) },
+ { id: "data", layer: svgData(projection2, context, dispatch10) },
+ { id: "keepRight", layer: svgKeepRight(projection2, context, dispatch10) },
+ { id: "improveOSM", layer: svgImproveOSM(projection2, context, dispatch10) },
+ { id: "osmose", layer: svgOsmose(projection2, context, dispatch10) },
+ { id: "streetside", layer: svgStreetside(projection2, context, dispatch10) },
+ { id: "mapillary", layer: svgMapillaryImages(projection2, context, dispatch10) },
+ { id: "mapillary-position", layer: svgMapillaryPosition(projection2, context, dispatch10) },
+ { id: "mapillary-map-features", layer: svgMapillaryMapFeatures(projection2, context, dispatch10) },
+ { id: "mapillary-signs", layer: svgMapillarySigns(projection2, context, dispatch10) },
+ { id: "kartaview", layer: svgKartaviewImages(projection2, context, dispatch10) },
+ { id: "debug", layer: svgDebug(projection2, context, dispatch10) },
+ { id: "geolocate", layer: svgGeolocate(projection2, context, dispatch10) },
+ { id: "touch", layer: svgTouch(projection2, context, dispatch10) }
+ ];
+ function drawLayers(selection2) {
+ svg2 = selection2.selectAll(".surface").data([0]);
+ svg2 = svg2.enter().append("svg").attr("class", "surface").merge(svg2);
+ var defs = svg2.selectAll(".surface-defs").data([0]);
+ defs.enter().append("defs").attr("class", "surface-defs");
+ var groups = svg2.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_default2(this).call(d.layer);
+ });
+ }
+ drawLayers.all = function() {
+ return _layers;
+ };
+ drawLayers.layer = function(id2) {
+ var obj = _layers.find(function(o) {
+ return o.id === id2;
+ });
+ return obj && obj.layer;
+ };
+ drawLayers.only = function(what) {
+ var arr = [].concat(what);
+ var all = _layers.map(function(layer) {
+ return layer.id;
+ });
+ return drawLayers.remove(utilArrayDifference(all, arr));
+ };
+ drawLayers.remove = function(what) {
+ var arr = [].concat(what);
+ arr.forEach(function(id2) {
+ _layers = _layers.filter(function(o) {
+ return o.id !== id2;
+ });
+ });
+ dispatch10.call("change");
+ return this;
+ };
+ drawLayers.add = function(what) {
+ var arr = [].concat(what);
+ arr.forEach(function(obj) {
+ if ("id" in obj && "layer" in obj) {
+ _layers.push(obj);
+ }
+ });
+ dispatch10.call("change");
+ return this;
+ };
+ drawLayers.dimensions = function(val) {
+ if (!arguments.length)
+ return utilGetDimensions(svg2);
+ utilSetDimensions(svg2, val);
+ return this;
+ };
+ return utilRebind(drawLayers, dispatch10, "on");
+ }
+
+ // modules/svg/lines.js
+ var import_fast_deep_equal6 = __toESM(require_fast_deep_equal());
+ function svgLines(projection2, 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(selection2, graph, entities, filter2) {
+ var targetClass = context.getDebug("target") ? "pink " : "nocolor ";
+ var nopeClass = context.getDebug("target") ? "red " : "nocolor ";
+ var getPath = svgPath(projection2).geojson;
+ var activeID = context.activeID();
+ var base = context.history().base();
+ 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);
+ });
+ var targetData = data.targets.filter(getPath);
+ var targets = selection2.selectAll(".line.target-allowed").filter(function(d) {
+ return filter2(d.properties.entity);
+ }).data(targetData, function key(d) {
+ return d.id;
+ });
+ targets.exit().remove();
+ var segmentWasEdited = function(d) {
+ var wayID = d.properties.entity.id;
+ if (!base.entities[wayID] || !(0, import_fast_deep_equal6.default)(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
+ return false;
+ }
+ return d.properties.nodes.some(function(n2) {
+ return !base.entities[n2.id] || !(0, import_fast_deep_equal6.default)(graph.entities[n2.id].loc, base.entities[n2.id].loc);
+ });
+ };
+ 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);
+ var nopeData = data.nopes.filter(getPath);
+ var nopes = selection2.selectAll(".line.target-nope").filter(function(d) {
+ return filter2(d.properties.entity);
+ }).data(nopeData, function key(d) {
+ return d.id;
+ });
+ nopes.exit().remove();
+ 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);
+ }
+ function drawLines(selection2, graph, entities, filter2) {
+ var base = context.history().base();
+ 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;
+ if (a.tags.highway) {
+ scoreA -= highway_stack[a.tags.highway];
+ }
+ if (b.tags.highway) {
+ scoreB -= highway_stack[b.tags.highway];
+ }
+ return scoreA - scoreB;
+ }
+ function drawLineGroup(selection3, klass, isSelected) {
+ var mode = context.mode();
+ var isDrawing = mode && /^draw/.test(mode.id);
+ var selectedClass = !isDrawing && isSelected ? "selected " : "";
+ var lines = selection3.selectAll("path").filter(filter2).data(getPathData(isSelected), osmEntity.key);
+ lines.exit().remove();
+ lines.enter().append("path").attr("class", function(d) {
+ var prefix = "way line";
+ if (!d.hasInterestingTags()) {
+ var parentRelations = graph.parentRelations(d);
+ var parentMultipolygons = parentRelations.filter(function(relation) {
+ return relation.isMultipolygon();
+ });
+ if (parentMultipolygons.length > 0 && // and only multipolygon relations
+ parentRelations.length === parentMultipolygons.length) {
+ prefix = "relation area";
+ }
+ }
+ 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] && !(0, import_fast_deep_equal6.default)(graph.entities[d.id].nodes, base.entities[d.id].nodes);
+ }).classed("retagged", function(d) {
+ return graph.entities[d.id] && base.entities[d.id] && !(0, import_fast_deep_equal6.default)(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 selection3;
+ }
+ 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;
+ }
+ });
+ };
+ }
+ 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(filter2).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;
+ });
+ if (detected.ie) {
+ markers.each(function() {
+ this.parentNode.insertBefore(this, this);
+ });
+ }
+ }
+ var getPath = svgPath(projection2, graph);
+ var ways = [];
+ var onewaydata = {};
+ var sideddata = {};
+ var oldMultiPolygonOuters = {};
+ for (var i2 = 0; i2 < entities.length; i2++) {
+ var entity = entities[i2];
+ var outer = osmOldMultipolygonOuterMember(entity, graph);
+ if (outer) {
+ ways.push(entity.mergeTags(outer.tags));
+ oldMultiPolygonOuters[outer.id] = true;
+ } else if (entity.geometry(graph) === "line" || entity.geometry(graph) === "area" && entity.sidednessIdentifier && entity.sidednessIdentifier() === "coastline") {
+ ways.push(entity);
+ }
+ }
+ 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(
+ projection2,
+ graph,
+ 35,
+ function shouldReverse(entity2) {
+ return entity2.tags.oneway === "-1";
+ },
+ function bothDirections(entity2) {
+ return entity2.tags.oneway === "reversible" || entity2.tags.oneway === "alternating";
+ }
+ );
+ onewaydata[k] = utilArrayFlatten(onewayArr.map(onewaySegments));
+ var sidedArr = v.filter(function(d) {
+ return d.isSided();
+ });
+ var sidedSegments = svgMarkerSegments(
+ projection2,
+ graph,
+ 30,
+ function shouldReverse() {
+ return false;
+ },
+ function bothDirections() {
+ return false;
+ }
+ );
+ sideddata[k] = utilArrayFlatten(sidedArr.map(sidedSegments));
+ });
+ var covered = selection2.selectAll(".layer-osm.covered");
+ var uncovered = selection2.selectAll(".layer-osm.lines");
+ var touchLayer = selection2.selectAll(".layer-touch.lines");
+ [covered, uncovered].forEach(function(selection3) {
+ var range3 = selection3 === covered ? range(-10, 0) : range(0, 11);
+ var layergroup = selection3.selectAll("g.layergroup").data(range3);
+ 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 + ")";
+ }
+ );
+ });
+ touchLayer.call(drawTargets, graph, ways, filter2);
+ }
+ return drawLines;
+ }
+
+ // modules/svg/midpoints.js
+ function svgMidpoints(projection2, context) {
+ var targetRadius = 8;
+ function drawTargets(selection2, graph, entities, filter2) {
+ var fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ var getTransform = svgPointTransform(projection2).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 = selection2.selectAll(".midpoint.target").filter(function(d) {
+ return filter2(d.properties.entity);
+ }).data(data, function key(d) {
+ return d.id;
+ });
+ targets.exit().remove();
+ targets.enter().append("circle").attr("r", targetRadius).merge(targets).attr("class", function(d) {
+ return "node midpoint target " + fillClass + d.id;
+ }).attr("transform", getTransform);
+ }
+ function drawMidpoints(selection2, graph, entities, filter2, extent) {
+ var drawLayer = selection2.selectAll(".layer-osm.points .points-group.midpoints");
+ var touchLayer = selection2.selectAll(".layer-touch.points");
+ var mode = context.mode();
+ if (mode && mode.id !== "select" || !context.map().withinEditableZoom()) {
+ drawLayer.selectAll(".midpoint").remove();
+ touchLayer.selectAll(".midpoint.target").remove();
+ return;
+ }
+ var poly = extent.polygon();
+ var midpoints = {};
+ for (var i2 = 0; i2 < entities.length; i2++) {
+ var entity = entities[i2];
+ if (entity.type !== "way")
+ continue;
+ if (!filter2(entity))
+ continue;
+ if (context.selectedIDs().indexOf(entity.id) < 0)
+ continue;
+ var nodes = graph.childNodes(entity);
+ for (var j2 = 0; j2 < nodes.length - 1; j2++) {
+ var a = nodes[j2];
+ var b = nodes[j2 + 1];
+ var id2 = [a.id, b.id].sort().join("-");
+ if (midpoints[id2]) {
+ midpoints[id2].parents.push(entity);
+ } else if (geoVecLength(projection2(a.loc), projection2(b.loc)) > 40) {
+ var point2 = geoVecInterp(a.loc, b.loc, 0.5);
+ var loc = null;
+ if (extent.intersects(point2)) {
+ loc = point2;
+ } else {
+ for (var k = 0; k < 4; k++) {
+ point2 = geoLineIntersection([a.loc, b.loc], [poly[k], poly[k + 1]]);
+ if (point2 && geoVecLength(projection2(a.loc), projection2(point2)) > 20 && geoVecLength(projection2(b.loc), projection2(point2)) > 20) {
+ loc = point2;
+ break;
+ }
+ }
+ }
+ if (loc) {
+ midpoints[id2] = {
+ type: "midpoint",
+ id: id2,
+ loc,
+ edge: [a.id, b.id],
+ parents: [entity]
+ };
+ }
+ }
+ }
+ }
+ function midpointFilter(d) {
+ if (midpoints[d.id])
+ return true;
+ for (var i3 = 0; i3 < d.parents.length; i3++) {
+ if (filter2(d.parents[i3])) {
+ return true;
+ }
+ }
+ return false;
+ }
+ 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(projection2);
+ var a2 = graph.entity(d.edge[0]);
+ var b2 = graph.entity(d.edge[1]);
+ var angle2 = geoAngle(a2, b2, projection2) * (180 / Math.PI);
+ return translate(d) + " rotate(" + angle2 + ")";
+ }).call(svgTagClasses().tags(
+ function(d) {
+ return d.parents[0].tags;
+ }
+ ));
+ groups.select("polygon.shadow");
+ groups.select("polygon.fill");
+ touchLayer.call(drawTargets, graph, Object.values(midpoints), midpointFilter);
+ }
+ return drawMidpoints;
+ }
+
+ // modules/svg/points.js
+ var import_fast_deep_equal7 = __toESM(require_fast_deep_equal());
+ function svgPoints(projection2, context) {
+ function markerPath(selection2, klass) {
+ selection2.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];
+ }
+ 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);
+ }
+ function drawTargets(selection2, graph, entities, filter2) {
+ var fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ var getTransform = svgPointTransform(projection2).geojson;
+ var activeID = context.activeID();
+ var data = [];
+ entities.forEach(function(node) {
+ if (activeID === node.id)
+ return;
+ data.push({
+ type: "Feature",
+ id: node.id,
+ properties: {
+ target: true,
+ entity: node
+ },
+ geometry: node.asGeoJSON()
+ });
+ });
+ var targets = selection2.selectAll(".point.target").filter(function(d) {
+ return filter2(d.properties.entity);
+ }).data(data, function key(d) {
+ return d.id;
+ });
+ targets.exit().remove();
+ 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 drawPoints(selection2, graph, entities, filter2) {
+ var wireframe = context.surface().classed("fill-wireframe");
+ var zoom = geoScaleToZoom(projection2.scale());
+ var base = context.history().base();
+ function renderAsPoint(entity) {
+ return entity.geometry(graph) === "point" && !(zoom >= 18 && entity.directions(graph, projection2).length);
+ }
+ var points = wireframe ? [] : entities.filter(renderAsPoint);
+ points.sort(sortY);
+ var drawLayer = selection2.selectAll(".layer-osm.points .points-group.points");
+ var touchLayer = selection2.selectAll(".layer-touch.points");
+ var groups = drawLayer.selectAll("g.point").filter(filter2).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.5, -20)").attr("class", "icon").attr("width", "12px").attr("height", "12px");
+ groups = groups.merge(enter).attr("transform", svgPointTransform(projection2)).classed("added", function(d) {
+ return !base.entities[d.id];
+ }).classed("moved", function(d) {
+ return base.entities[d.id] && !(0, import_fast_deep_equal7.default)(graph.entities[d.id].loc, base.entities[d.id].loc);
+ }).classed("retagged", function(d) {
+ return base.entities[d.id] && !(0, import_fast_deep_equal7.default)(graph.entities[d.id].tags, base.entities[d.id].tags);
+ }).call(svgTagClasses());
+ groups.select(".shadow");
+ groups.select(".stroke");
+ groups.select(".icon").attr("xlink:href", function(entity) {
+ var preset = _mainPresetIndex.match(entity, graph);
+ var picon = preset && preset.icon;
+ return picon ? "#" + picon : "";
+ });
+ touchLayer.call(drawTargets, graph, points, filter2);
+ }
+ return drawPoints;
+ }
+
+ // modules/svg/turns.js
+ function svgTurns(projection2, context) {
+ function icon2(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 drawTurns(selection2, graph, turns) {
+ function turnTransform(d) {
+ var pxRadius = 50;
+ var toWay = graph.entity(d.to.way);
+ var toPoints = graph.childNodes(toWay).map(function(n2) {
+ return n2.loc;
+ }).map(projection2);
+ var toLength = geoPathLength(toPoints);
+ var mid = toLength / 2;
+ var toNode = graph.entity(d.to.node);
+ var toVertex = graph.entity(d.to.vertex);
+ var a = geoAngle(toVertex, toNode, projection2);
+ var o = projection2(toVertex.loc);
+ var r = d.u ? 0 : !toWay.__via ? pxRadius : Math.min(mid, pxRadius);
+ return "translate(" + (r * Math.cos(a) + o[0]) + "," + (r * Math.sin(a) + o[1]) + ") rotate(" + a * 180 / Math.PI + ")";
+ }
+ var drawLayer = selection2.selectAll(".layer-osm.points .points-group.turns");
+ var touchLayer = selection2.selectAll(".layer-touch.turns");
+ var groups = drawLayer.selectAll("g.turn").data(turns, function(d) {
+ return d.key;
+ });
+ groups.exit().remove();
+ 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");
+ groups = groups.merge(groupsEnter).attr("opacity", function(d) {
+ return d.direct === false ? "0.7" : null;
+ }).attr("transform", turnTransform);
+ groups.select("use").attr("xlink:href", icon2);
+ groups.select("rect");
+ groups.select("circle");
+ var fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ groups = touchLayer.selectAll("g.turn").data(turns, function(d) {
+ return d.key;
+ });
+ groups.exit().remove();
+ 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");
+ groups = groups.merge(groupsEnter).attr("transform", turnTransform);
+ groups.select("rect");
+ groups.select("circle");
+ return this;
+ }
+ return drawTurns;
+ }
+
+ // modules/svg/vertices.js
+ var import_fast_deep_equal8 = __toESM(require_fast_deep_equal());
+ function svgVertices(projection2, 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 _currHoverTarget;
+ var _currPersistent = {};
+ var _currHover = {};
+ var _prevHover = {};
+ var _currSelected = {};
+ var _prevSelected = {};
+ var _radii = {};
+ function sortY(a, b) {
+ return b.loc[1] - a.loc[1];
+ }
+ 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);
+ }
+ function draw(selection2, graph, vertices, sets2, filter2) {
+ sets2 = sets2 || { selected: {}, important: {}, hovered: {} };
+ var icons = {};
+ var directions = {};
+ var wireframe = context.surface().classed("fill-wireframe");
+ var zoom = geoScaleToZoom(projection2.scale());
+ var z = zoom < 17 ? 0 : zoom < 18 ? 1 : 2;
+ var activeID = context.activeID();
+ var base = context.history().base();
+ function getIcon(d) {
+ 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];
+ }
+ function getDirections(entity) {
+ if (entity.id in directions)
+ return directions[entity.id];
+ var angles = entity.directions(graph, projection2);
+ directions[entity.id] = angles.length ? angles : false;
+ return angles;
+ }
+ function updateAttributes(selection3) {
+ ["shadow", "stroke", "fill"].forEach(function(klass) {
+ var rads = radiuses[klass];
+ selection3.selectAll("." + klass).each(function(entity) {
+ var i2 = z && getIcon(entity);
+ var r = rads[i2 ? 3 : z];
+ if (entity.id !== activeID && entity.isEndpoint(graph) && !entity.isConnected(graph)) {
+ r += 1.5;
+ }
+ if (klass === "shadow") {
+ _radii[entity.id] = r;
+ }
+ select_default2(this).attr("r", r).attr("visibility", i2 && klass === "fill" ? "hidden" : null);
+ });
+ });
+ }
+ vertices.sort(sortY);
+ var groups = selection2.selectAll("g.vertex").filter(filter2).data(vertices, fastEntityKey);
+ groups.exit().remove();
+ 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");
+ enter.filter(function(d) {
+ return d.hasInterestingTags();
+ }).append("circle").attr("class", "fill");
+ groups = groups.merge(enter).attr("transform", svgPointTransform(projection2)).classed("sibling", function(d) {
+ return d.id in sets2.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];
+ }).classed("moved", function(d) {
+ return base.entities[d.id] && !(0, import_fast_deep_equal8.default)(graph.entities[d.id].loc, base.entities[d.id].loc);
+ }).classed("retagged", function(d) {
+ return base.entities[d.id] && !(0, import_fast_deep_equal8.default)(graph.entities[d.id].tags, base.entities[d.id].tags);
+ }).call(updateAttributes);
+ var iconUse = groups.selectAll(".icon").data(function data(d) {
+ return zoom >= 17 && getIcon(d) ? [d] : [];
+ }, fastEntityKey);
+ iconUse.exit().remove();
+ iconUse.enter().append("use").attr("class", "icon").attr("width", "12px").attr("height", "12px").attr("transform", "translate(-6, -6)").attr("xlink:href", function(d) {
+ var picon = getIcon(d);
+ return picon ? "#" + picon : "";
+ });
+ var dgroups = groups.selectAll(".viewfieldgroup").data(function data(d) {
+ return zoom >= 18 && getDirections(d) ? [d] : [];
+ }, fastEntityKey);
+ dgroups.exit().remove();
+ 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);
+ });
+ viewfields.exit().remove();
+ 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 drawTargets(selection2, graph, entities, filter2) {
+ var targetClass = context.getDebug("target") ? "pink " : "nocolor ";
+ var nopeClass = context.getDebug("target") ? "red " : "nocolor ";
+ var getTransform = svgPointTransform(projection2).geojson;
+ var activeID = context.activeID();
+ var data = { targets: [], nopes: [] };
+ entities.forEach(function(node) {
+ if (activeID === node.id)
+ return;
+ var vertexType = svgPassiveVertex(node, graph, activeID);
+ if (vertexType !== 0) {
+ 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()
+ });
+ }
+ });
+ var targets = selection2.selectAll(".vertex.target-allowed").filter(function(d) {
+ return filter2(d.properties.entity);
+ }).data(data.targets, function key(d) {
+ return d.id;
+ });
+ targets.exit().remove();
+ 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);
+ var nopes = selection2.selectAll(".vertex.target-nope").filter(function(d) {
+ return filter2(d.properties.entity);
+ }).data(data.nopes, function key(d) {
+ return d.id;
+ });
+ nopes.exit().remove();
+ 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);
+ }
+ function renderAsVertex(entity, graph, wireframe, zoom) {
+ var geometry = entity.geometry(graph);
+ return geometry === "vertex" || geometry === "point" && (wireframe || zoom >= 18 && entity.directions(graph, projection2).length);
+ }
+ function isEditedNode(node, base, head) {
+ var baseNode = base.entities[node.id];
+ var headNode = head.entities[node.id];
+ return !headNode || !baseNode || !(0, import_fast_deep_equal8.default)(headNode.tags, baseNode.tags) || !(0, import_fast_deep_equal8.default)(headNode.loc, baseNode.loc);
+ }
+ function getSiblingAndChildVertices(ids, graph, wireframe, zoom) {
+ var results = {};
+ var seenIds = {};
+ function addChildVertices(entity) {
+ if (seenIds[entity.id])
+ return;
+ seenIds[entity.id] = true;
+ var geometry = entity.geometry(graph);
+ if (!context.features().isHiddenFeature(entity, graph, geometry)) {
+ var i2;
+ if (entity.type === "way") {
+ for (i2 = 0; i2 < entity.nodes.length; i2++) {
+ var child = graph.hasEntity(entity.nodes[i2]);
+ if (child) {
+ addChildVertices(child);
+ }
+ }
+ } else if (entity.type === "relation") {
+ for (i2 = 0; i2 < entity.members.length; i2++) {
+ var member = graph.hasEntity(entity.members[i2].id);
+ if (member) {
+ addChildVertices(member);
+ }
+ }
+ } else if (renderAsVertex(entity, graph, wireframe, zoom)) {
+ results[entity.id] = entity;
+ }
+ }
+ }
+ ids.forEach(function(id2) {
+ var entity = graph.hasEntity(id2);
+ if (!entity)
+ return;
+ if (entity.type === "node") {
+ if (renderAsVertex(entity, graph, wireframe, zoom)) {
+ results[entity.id] = entity;
+ graph.parentWays(entity).forEach(function(entity2) {
+ addChildVertices(entity2);
+ });
+ }
+ } else {
+ addChildVertices(entity);
+ }
+ });
+ return results;
+ }
+ function drawVertices(selection2, graph, entities, filter2, extent, fullRedraw) {
+ var wireframe = context.surface().classed("fill-wireframe");
+ var visualDiff = context.surface().classed("highlight-edited");
+ var zoom = geoScaleToZoom(projection2.scale());
+ var mode = context.mode();
+ var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
+ var base = context.history().base();
+ var drawLayer = selection2.selectAll(".layer-osm.points .points-group.vertices");
+ var touchLayer = selection2.selectAll(".layer-touch.points");
+ if (fullRedraw) {
+ _currPersistent = {};
+ _radii = {};
+ }
+ for (var i2 = 0; i2 < entities.length; i2++) {
+ var entity = entities[i2];
+ var geometry = entity.geometry(graph);
+ var keep = false;
+ if (geometry === "point" && renderAsVertex(entity, graph, wireframe, zoom)) {
+ _currPersistent[entity.id] = entity;
+ keep = true;
+ } else if (geometry === "vertex" && (entity.hasInterestingTags() || entity.isEndpoint(graph) || entity.isConnected(graph) || visualDiff && isEditedNode(entity, base, graph))) {
+ _currPersistent[entity.id] = entity;
+ keep = true;
+ }
+ if (!keep && !fullRedraw) {
+ delete _currPersistent[entity.id];
+ }
+ }
+ var sets2 = {
+ 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 all = Object.assign({}, isMoving ? _currHover : {}, _currSelected, _currPersistent);
+ var filterRendered = function(d) {
+ return d.id in _currPersistent || d.id in _currSelected || d.id in _currHover || filter2(d);
+ };
+ drawLayer.call(draw, graph, currentVisible(all), sets2, filterRendered);
+ var filterTouch = function(d) {
+ return isMoving ? true : filterRendered(d);
+ };
+ touchLayer.call(drawTargets, graph, currentVisible(all), filterTouch);
+ function currentVisible(which) {
+ return Object.keys(which).map(graph.hasEntity, graph).filter(function(entity2) {
+ return entity2 && entity2.intersects(extent, graph);
+ });
+ }
+ }
+ drawVertices.drawSelected = function(selection2, graph, extent) {
+ var wireframe = context.surface().classed("fill-wireframe");
+ var zoom = geoScaleToZoom(projection2.scale());
+ _prevSelected = _currSelected || {};
+ if (context.map().isInWideSelection()) {
+ _currSelected = {};
+ context.selectedIDs().forEach(function(id2) {
+ var entity = graph.hasEntity(id2);
+ if (!entity)
+ return;
+ if (entity.type === "node") {
+ if (renderAsVertex(entity, graph, wireframe, zoom)) {
+ _currSelected[entity.id] = entity;
+ }
+ }
+ });
+ } else {
+ _currSelected = getSiblingAndChildVertices(context.selectedIDs(), graph, wireframe, zoom);
+ }
+ var filter2 = function(d) {
+ return d.id in _prevSelected;
+ };
+ drawVertices(selection2, graph, Object.values(_prevSelected), filter2, extent, false);
+ };
+ drawVertices.drawHover = function(selection2, graph, target, extent) {
+ if (target === _currHoverTarget)
+ return;
+ var wireframe = context.surface().classed("fill-wireframe");
+ var zoom = geoScaleToZoom(projection2.scale());
+ _prevHover = _currHover || {};
+ _currHoverTarget = target;
+ var entity = target && target.properties && target.properties.entity;
+ if (entity) {
+ _currHover = getSiblingAndChildVertices([entity.id], graph, wireframe, zoom);
+ } else {
+ _currHover = {};
+ }
+ var filter2 = function(d) {
+ return d.id in _prevHover;
+ };
+ drawVertices(selection2, graph, Object.values(_prevHover), filter2, extent, false);
+ };
+ return drawVertices;
+ }
+
+ // modules/util/bind_once.js
+ function utilBindOnce(target, type2, listener, capture) {
+ var typeOnce = type2 + ".once";
+ function one2() {
+ target.on(typeOnce, null);
+ listener.apply(this, arguments);
+ }
+ target.on(typeOnce, one2, capture);
+ return this;
+ }
+
+ // modules/util/zoom_pan.js
+ function defaultFilter3(d3_event) {
+ return !d3_event.ctrlKey && !d3_event.button;
+ }
+ function defaultExtent2() {
+ var e = this;
+ if (e instanceof SVGElement) {
+ e = e.ownerSVGElement || e;
+ if (e.hasAttribute("viewBox")) {
+ e = e.viewBox.baseVal;
+ return [[e.x, e.y], [e.x + e.width, e.y + e.height]];
+ }
+ return [[0, 0], [e.width.baseVal.value, e.height.baseVal.value]];
+ }
+ return [[0, 0], [e.clientWidth, e.clientHeight]];
+ }
+ function defaultWheelDelta2(d3_event) {
+ return -d3_event.deltaY * (d3_event.deltaMode === 1 ? 0.05 : d3_event.deltaMode ? 1 : 2e-3);
+ }
+ function defaultConstrain2(transform2, extent, translateExtent) {
+ var dx0 = transform2.invertX(extent[0][0]) - translateExtent[0][0], dx1 = transform2.invertX(extent[1][0]) - translateExtent[1][0], dy0 = transform2.invertY(extent[0][1]) - translateExtent[0][1], dy1 = transform2.invertY(extent[1][1]) - translateExtent[1][1];
+ return transform2.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)
+ );
+ }
+ function utilZoomPan() {
+ var filter2 = defaultFilter3, extent = defaultExtent2, constrain = defaultConstrain2, wheelDelta = defaultWheelDelta2, scaleExtent = [0, Infinity], translateExtent = [[-Infinity, -Infinity], [Infinity, Infinity]], interpolate = zoom_default, dispatch10 = dispatch_default("start", "zoom", "end"), _wheelDelay = 150, _transform = identity2, _activeGesture;
+ function zoom(selection2) {
+ selection2.on("pointerdown.zoom", pointerdown).on("wheel.zoom", wheeled).style("touch-action", "none").style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
+ select_default2(window).on("pointermove.zoompan", pointermove).on("pointerup.zoompan pointercancel.zoompan", pointerup);
+ }
+ zoom.transform = function(collection, transform2, point2) {
+ var selection2 = collection.selection ? collection.selection() : collection;
+ if (collection !== selection2) {
+ schedule(collection, transform2, point2);
+ } else {
+ selection2.interrupt().each(function() {
+ gesture(this, arguments).start(null).zoom(null, null, typeof transform2 === "function" ? transform2.apply(this, arguments) : transform2).end(null);
+ });
+ }
+ };
+ zoom.scaleBy = function(selection2, k, p) {
+ zoom.scaleTo(selection2, function() {
+ var k0 = _transform.k, k1 = typeof k === "function" ? k.apply(this, arguments) : k;
+ return k0 * k1;
+ }, p);
+ };
+ zoom.scaleTo = function(selection2, k, p) {
+ zoom.transform(selection2, function() {
+ var e = extent.apply(this, arguments), t0 = _transform, p02 = !p ? centroid(e) : typeof p === "function" ? p.apply(this, arguments) : p, p1 = t0.invert(p02), k1 = typeof k === "function" ? k.apply(this, arguments) : k;
+ return constrain(translate(scale(t0, k1), p02, p1), e, translateExtent);
+ }, p);
+ };
+ zoom.translateBy = function(selection2, x, y) {
+ zoom.transform(selection2, 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);
+ });
+ };
+ zoom.translateTo = function(selection2, x, y, p) {
+ zoom.transform(selection2, function() {
+ var e = extent.apply(this, arguments), t = _transform, p02 = !p ? centroid(e) : typeof p === "function" ? p.apply(this, arguments) : p;
+ return constrain(identity2.translate(p02[0], p02[1]).scale(t.k).translate(
+ typeof x === "function" ? -x.apply(this, arguments) : -x,
+ typeof y === "function" ? -y.apply(this, arguments) : -y
+ ), e, translateExtent);
+ }, p);
+ };
+ function scale(transform2, k) {
+ k = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k));
+ return k === transform2.k ? transform2 : new Transform(k, transform2.x, transform2.y);
+ }
+ function translate(transform2, p02, p1) {
+ var x = p02[0] - p1[0] * transform2.k, y = p02[1] - p1[1] * transform2.k;
+ return x === transform2.x && y === transform2.y ? transform2 : new Transform(transform2.k, x, y);
+ }
+ function centroid(extent2) {
+ return [(+extent2[0][0] + +extent2[1][0]) / 2, (+extent2[0][1] + +extent2[1][1]) / 2];
+ }
+ function schedule(transition2, transform2, point2) {
+ transition2.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 = !point2 ? centroid(e) : typeof point2 === "function" ? point2.apply(that, args) : point2, w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]), a = _transform, b = typeof transform2 === "function" ? transform2.apply(that, args) : transform2, i2 = interpolate(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));
+ return function(t) {
+ if (t === 1) {
+ t = b;
+ } else {
+ var l = i2(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);
+ };
+ });
+ }
+ function gesture(that, args, clean2) {
+ return !clean2 && _activeGesture || new Gesture(that, args);
+ }
+ function Gesture(that, args) {
+ this.that = that;
+ this.args = args;
+ this.active = 0;
+ this.extent = extent.apply(that, args);
+ }
+ Gesture.prototype = {
+ start: function(d3_event) {
+ if (++this.active === 1) {
+ _activeGesture = this;
+ dispatch10.call("start", this, d3_event);
+ }
+ return this;
+ },
+ zoom: function(d3_event, key, transform2) {
+ if (this.mouse && key !== "mouse")
+ this.mouse[1] = transform2.invert(this.mouse[0]);
+ if (this.pointer0 && key !== "touch")
+ this.pointer0[1] = transform2.invert(this.pointer0[0]);
+ if (this.pointer1 && key !== "touch")
+ this.pointer1[1] = transform2.invert(this.pointer1[0]);
+ _transform = transform2;
+ dispatch10.call("zoom", this, d3_event, key, transform2);
+ return this;
+ },
+ end: function(d3_event) {
+ if (--this.active === 0) {
+ _activeGesture = null;
+ dispatch10.call("end", this, d3_event);
+ }
+ return this;
+ }
+ };
+ function wheeled(d3_event) {
+ if (!filter2.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 (g.wheel) {
+ if (g.mouse[0][0] !== p[0] || g.mouse[0][1] !== p[1]) {
+ g.mouse[1] = t.invert(g.mouse[0] = p);
+ }
+ clearTimeout(g.wheel);
+ } else {
+ g.mouse = [p, t.invert(p)];
+ interrupt_default(this);
+ g.start(d3_event);
+ }
+ 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));
+ function wheelidled() {
+ g.wheel = null;
+ g.end(d3_event);
+ }
+ }
+ var _downPointerIDs = /* @__PURE__ */ new Set();
+ var _pointerLocGetter;
+ function pointerdown(d3_event) {
+ _downPointerIDs.add(d3_event.pointerId);
+ if (!filter2.apply(this, arguments))
+ return;
+ var g = gesture(this, arguments, _downPointerIDs.size === 1);
+ var started;
+ d3_event.stopImmediatePropagation();
+ _pointerLocGetter = utilFastMouse(this);
+ var loc = _pointerLocGetter(d3_event);
+ var p = [loc, _transform.invert(loc), d3_event.pointerId];
+ if (!g.pointer0) {
+ g.pointer0 = p;
+ started = true;
+ } else if (!g.pointer1 && g.pointer0[2] !== p[2]) {
+ g.pointer1 = p;
+ }
+ if (started) {
+ interrupt_default(this);
+ g.start(d3_event);
+ }
+ }
+ 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;
+ if ((isPointer0 || isPointer1) && "buttons" in d3_event && !d3_event.buttons) {
+ if (g.pointer0)
+ _downPointerIDs.delete(g.pointer0[2]);
+ if (g.pointer1)
+ _downPointerIDs.delete(g.pointer1[2]);
+ g.end(d3_event);
+ return;
+ }
+ d3_event.preventDefault();
+ d3_event.stopImmediatePropagation();
+ var loc = _pointerLocGetter(d3_event);
+ var t, p, l;
+ if (isPointer0)
+ g.pointer0[0] = loc;
+ else if (isPointer1)
+ g.pointer1[0] = loc;
+ t = _transform;
+ if (g.pointer1) {
+ var p02 = g.pointer0[0], l0 = g.pointer0[1], p1 = g.pointer1[0], l1 = g.pointer1[1], dp = (dp = p1[0] - p02[0]) * dp + (dp = p1[1] - p02[1]) * dp, dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;
+ t = scale(t, Math.sqrt(dp / dl));
+ p = [(p02[0] + p1[0]) / 2, (p02[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;
+ }
+ g.zoom(d3_event, "touch", constrain(translate(t, p, l), g.extent, translateExtent));
+ }
+ function pointerup(d3_event) {
+ if (!_downPointerIDs.has(d3_event.pointerId))
+ return;
+ _downPointerIDs.delete(d3_event.pointerId);
+ 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;
+ 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);
+ }
+ }
+ zoom.wheelDelta = function(_) {
+ return arguments.length ? (wheelDelta = utilFunctor(+_), zoom) : wheelDelta;
+ };
+ zoom.filter = function(_) {
+ return arguments.length ? (filter2 = utilFunctor(!!_), zoom) : filter2;
+ };
+ zoom.extent = function(_) {
+ return arguments.length ? (extent = utilFunctor([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
+ };
+ zoom.scaleExtent = function(_) {
+ return arguments.length ? (scaleExtent[0] = +_[0], scaleExtent[1] = +_[1], zoom) : [scaleExtent[0], scaleExtent[1]];
+ };
+ 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]]];
+ };
+ zoom.constrain = function(_) {
+ return arguments.length ? (constrain = _, zoom) : constrain;
+ };
+ zoom.interpolate = function(_) {
+ return arguments.length ? (interpolate = _, zoom) : interpolate;
+ };
+ zoom._transform = function(_) {
+ return arguments.length ? (_transform = _, zoom) : _transform;
+ };
+ return utilRebind(zoom, dispatch10, "on");
+ }
+
+ // modules/util/double_up.js
+ function utilDoubleUp() {
+ var dispatch10 = dispatch_default("doubleUp");
+ var _maxTimespan = 500;
+ var _maxDistance = 20;
+ var _pointer;
+ function pointerIsValidFor(loc) {
+ 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;
+ }
+ function pointerdown(d3_event) {
+ if (d3_event.ctrlKey || d3_event.button === 2)
+ return;
+ var loc = [d3_event.clientX, d3_event.clientY];
+ if (_pointer && !pointerIsValidFor(loc)) {
+ _pointer = void 0;
+ }
+ if (!_pointer) {
+ _pointer = {
+ startLoc: loc,
+ startTime: new Date().getTime(),
+ upCount: 0,
+ pointerId: d3_event.pointerId
+ };
+ } else {
+ _pointer.pointerId = d3_event.pointerId;
+ }
+ }
+ function pointerup(d3_event) {
+ if (d3_event.ctrlKey || d3_event.button === 2)
+ return;
+ if (!_pointer || _pointer.pointerId !== d3_event.pointerId)
+ return;
+ _pointer.upCount += 1;
+ if (_pointer.upCount === 2) {
+ var loc = [d3_event.clientX, d3_event.clientY];
+ if (pointerIsValidFor(loc)) {
+ var locInThis = utilFastMouse(this)(d3_event);
+ dispatch10.call("doubleUp", this, d3_event, locInThis);
+ }
+ _pointer = void 0;
+ }
+ }
+ function doubleUp(selection2) {
+ if ("PointerEvent" in window) {
+ selection2.on("pointerdown.doubleUp", pointerdown).on("pointerup.doubleUp", pointerup);
+ } else {
+ selection2.on("dblclick.doubleUp", function(d3_event) {
+ dispatch10.call("doubleUp", this, d3_event, utilFastMouse(this)(d3_event));
+ });
+ }
+ }
+ doubleUp.off = function(selection2) {
+ selection2.on("pointerdown.doubleUp", null).on("pointerup.doubleUp", null).on("dblclick.doubleUp", null);
+ };
+ return utilRebind(doubleUp, dispatch10, "on");
+ }
+
+ // modules/renderer/map.js
+ var TILESIZE = 256;
+ var minZoom2 = 2;
+ var maxZoom = 24;
+ var kMin = geoZoomToScale(minZoom2, TILESIZE);
+ var kMax = geoZoomToScale(maxZoom, TILESIZE);
+ function clamp(num, min3, max3) {
+ return Math.max(min3, Math.min(num, max3));
+ }
+ function rendererMap(context) {
+ var dispatch10 = dispatch_default(
+ "move",
+ "drawn",
+ "crossEditableZoom",
+ "hitMinZoom",
+ "changeHighlighting",
+ "changeAreaFill"
+ );
+ var projection2 = context.projection;
+ var curtainProjection = context.curtainProjection;
+ var drawLayers;
+ var drawPoints;
+ var drawVertices;
+ var drawLines;
+ var drawAreas;
+ var drawMidpoints;
+ var drawLabels;
+ var _selection = select_default2(null);
+ var supersurface = select_default2(null);
+ var wrapper = select_default2(null);
+ var surface = select_default2(null);
+ var _dimensions = [1, 1];
+ var _dblClickZoomEnabled = true;
+ var _redrawEnabled = true;
+ var _gestureTransformStart;
+ var _transformStart = projection2.transform();
+ var _transformLast;
+ var _isTransformed = false;
+ var _minzoom = 0;
+ var _getMouseCoords;
+ var _lastPointerEvent;
+ var _lastWithinEditableZoom;
+ var _pointerDown = false;
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ var _zoomerPannerFunction = "PointerEvent" in window ? utilZoomPan : zoom_default2;
+ var _zoomerPanner = _zoomerPannerFunction().scaleExtent([kMin, kMax]).interpolate(value_default).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;
+ });
+ var _doubleUpHandler = utilDoubleUp();
+ var scheduleRedraw = throttle_default(redraw, 750);
+ function cancelPendingRedraw() {
+ scheduleRedraw.cancel();
+ }
+ function map2(selection2) {
+ _selection = selection2;
+ context.on("change.map", immediateRedraw);
+ var osm = context.connection();
+ if (osm) {
+ osm.on("change.map", immediateRedraw);
+ }
+ function didUndoOrRedo(targetTransform) {
+ var mode = context.mode().id;
+ if (mode !== "browse" && mode !== "select")
+ return;
+ if (targetTransform) {
+ map2.transformEase(targetTransform);
+ }
+ }
+ 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();
+ });
+ selection2.on("wheel.map mousewheel.map", function(d3_event) {
+ d3_event.preventDefault();
+ }).call(_zoomerPanner).call(_zoomerPanner.transform, projection2.transform()).on("dblclick.zoom", null);
+ map2.supersurface = supersurface = selection2.append("div").attr("class", "supersurface").call(utilSetTransform, 0, 0);
+ wrapper = supersurface.append("div").attr("class", "layer layer-data");
+ map2.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;
+ if (resetTransform()) {
+ immediateRedraw();
+ }
+ }).on(_pointerPrefix + "move.map", function(d3_event) {
+ _lastPointerEvent = d3_event;
+ }).on(_pointerPrefix + "over.vertices", function(d3_event) {
+ if (map2.editableDataEnabled() && !_isTransformed) {
+ var hover = d3_event.target.__data__;
+ surface.call(drawVertices.drawHover, context.graph(), hover, map2.extent());
+ dispatch10.call("drawn", this, { full: false });
+ }
+ }).on(_pointerPrefix + "out.vertices", function(d3_event) {
+ if (map2.editableDataEnabled() && !_isTransformed) {
+ var hover = d3_event.relatedTarget && d3_event.relatedTarget.__data__;
+ surface.call(drawVertices.drawHover, context.graph(), hover, map2.extent());
+ dispatch10.call("drawn", this, { full: false });
+ }
+ });
+ var detected = utilDetect();
+ 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) {
+ surface.on("gesturestart.surface", function(d3_event) {
+ d3_event.preventDefault();
+ _gestureTransformStart = projection2.transform();
+ }).on("gesturechange.surface", gestureChange);
+ }
+ updateAreaFill();
+ _doubleUpHandler.on("doubleUp.map", function(d3_event, p02) {
+ if (!_dblClickZoomEnabled)
+ return;
+ if (typeof d3_event.target.__data__ === "object" && // or area fills
+ !select_default2(d3_event.target).classed("fill"))
+ return;
+ var zoomOut2 = d3_event.shiftKey;
+ var t = projection2.transform();
+ var p1 = t.invert(p02);
+ t = t.scale(zoomOut2 ? 0.5 : 2);
+ t.x = p02[0] - p1[0] * t.k;
+ t.y = p02[1] - p1[1] * t.k;
+ map2.transformEase(t);
+ });
+ context.on("enter.map", function() {
+ if (!map2.editableDataEnabled(
+ true
+ /* skip zoom check */
+ ))
+ return;
+ if (_isTransformed)
+ return;
+ var graph = context.graph();
+ var selectedAndParents = {};
+ context.selectedIDs().forEach(function(id2) {
+ var entity = graph.hasEntity(id2);
+ if (entity) {
+ selectedAndParents[entity.id] = entity;
+ if (entity.type === "node") {
+ graph.parentWays(entity).forEach(function(parent) {
+ selectedAndParents[parent.id] = parent;
+ });
+ }
+ }
+ });
+ var data = Object.values(selectedAndParents);
+ var filter2 = function(d) {
+ return d.id in selectedAndParents;
+ };
+ data = context.features().filter(data, graph);
+ surface.call(drawVertices.drawSelected, graph, map2.extent()).call(drawLines, graph, data, filter2).call(drawAreas, graph, data, filter2).call(drawMidpoints, graph, data, filter2, map2.trimmedExtent());
+ dispatch10.call("drawn", this, { full: false });
+ scheduleRedraw();
+ });
+ map2.dimensions(utilGetDimensions(selection2));
+ }
+ function zoomEventFilter(d3_event) {
+ if (d3_event.type === "mousedown") {
+ var hasOrphan = false;
+ var listeners = window.__on;
+ for (var i2 = 0; i2 < listeners.length; i2++) {
+ var listener = listeners[i2];
+ if (listener.name === "zoom" && listener.type === "mouseup") {
+ hasOrphan = true;
+ break;
+ }
+ }
+ if (hasOrphan) {
+ var event = window.CustomEvent;
+ if (event) {
+ event = new event("mouseup");
+ } else {
+ event = window.document.createEvent("Event");
+ event.initEvent("mouseup", false, false);
+ }
+ event.view = window;
+ window.dispatchEvent(event);
+ }
+ }
+ return d3_event.button !== 2;
+ }
+ 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(map2.extent());
+ var fullRedraw = false;
+ var data;
+ var set3;
+ var filter2;
+ var applyFeatureLayerFilters = true;
+ if (map2.isInWideSelection()) {
+ data = [];
+ utilEntityAndDeepMemberIDs(mode.selectedIDs(), context.graph()).forEach(function(id2) {
+ var entity = context.hasEntity(id2);
+ if (entity)
+ data.push(entity);
+ });
+ fullRedraw = true;
+ filter2 = utilFunctor(true);
+ applyFeatureLayerFilters = false;
+ } else if (difference) {
+ var complete = difference.complete(map2.extent());
+ data = Object.values(complete).filter(Boolean);
+ set3 = new Set(Object.keys(complete));
+ filter2 = function(d) {
+ return set3.has(d.id);
+ };
+ features.clear(data);
+ } else {
+ if (features.gatherStats(all, graph, _dimensions)) {
+ extent = void 0;
+ }
+ if (extent) {
+ data = context.history().intersects(map2.extent().intersection(extent));
+ set3 = new Set(data.map(function(entity) {
+ return entity.id;
+ }));
+ filter2 = function(d) {
+ return set3.has(d.id);
+ };
+ } else {
+ data = all;
+ fullRedraw = true;
+ filter2 = utilFunctor(true);
+ }
+ }
+ if (applyFeatureLayerFilters) {
+ data = features.filter(data, graph);
+ } else {
+ context.features().resetStats();
+ }
+ if (mode && mode.id === "select") {
+ surface.call(drawVertices.drawSelected, graph, map2.extent());
+ }
+ surface.call(drawVertices, graph, data, filter2, map2.extent(), fullRedraw).call(drawLines, graph, data, filter2).call(drawAreas, graph, data, filter2).call(drawMidpoints, graph, data, filter2, map2.trimmedExtent()).call(drawLabels, graph, data, filter2, _dimensions, fullRedraw).call(drawPoints, graph, data, filter2);
+ dispatch10.call("drawn", this, { full: true });
+ }
+ map2.init = function() {
+ drawLayers = svgLayers(projection2, context);
+ drawPoints = svgPoints(projection2, context);
+ drawVertices = svgVertices(projection2, context);
+ drawLines = svgLines(projection2, context);
+ drawAreas = svgAreas(projection2, context);
+ drawMidpoints = svgMidpoints(projection2, context);
+ drawLabels = svgLabels(projection2, context);
+ };
+ 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));
+ }
+ dispatch10.call("drawn", this, { full: true });
+ }
+ function gestureChange(d3_event) {
+ 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 e22 = new WheelEvent("wheel", props);
+ e22._scale = e.scale;
+ e22._rotation = e.rotation;
+ _selection.node().dispatchEvent(e22);
+ }
+ function zoomPan(event, key, transform2) {
+ var source = event && event.sourceEvent || event;
+ var eventTransform = transform2 || event && event.transform;
+ var x = eventTransform.x;
+ var y = eventTransform.y;
+ var k = eventTransform.k;
+ if (source && source.type === "wheel") {
+ 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, p02, p1;
+ if (source.deltaMode === 1) {
+ var lines = Math.abs(source.deltaY);
+ var sign2 = source.deltaY > 0 ? 1 : -1;
+ dY = sign2 * clamp(
+ Math.exp((lines - 1) * 0.75) * 4.000244140625,
+ 4.000244140625,
+ // min
+ 350.000244140625
+ // max
+ );
+ if (detected.os !== "mac") {
+ dY *= 5;
+ }
+ t0 = _isTransformed ? _transformLast : _transformStart;
+ p02 = _getMouseCoords(source);
+ p1 = t0.invert(p02);
+ k2 = t0.k * Math.pow(2, -dY / 500);
+ k2 = clamp(k2, kMin, kMax);
+ x2 = p02[0] - p1[0] * k2;
+ y2 = p02[1] - p1[1] * k2;
+ } else if (source._scale) {
+ t0 = _gestureTransformStart;
+ p02 = _getMouseCoords(source);
+ p1 = t0.invert(p02);
+ k2 = t0.k * source._scale;
+ k2 = clamp(k2, kMin, kMax);
+ x2 = p02[0] - p1[0] * k2;
+ y2 = p02[1] - p1[1] * k2;
+ } else if (source.ctrlKey && !isInteger(dY)) {
+ dY *= 6;
+ t0 = _isTransformed ? _transformLast : _transformStart;
+ p02 = _getMouseCoords(source);
+ p1 = t0.invert(p02);
+ k2 = t0.k * Math.pow(2, -dY / 500);
+ k2 = clamp(k2, kMin, kMax);
+ x2 = p02[0] - p1[0] * k2;
+ y2 = p02[1] - p1[1] * k2;
+ } else if ((source.altKey || source.shiftKey) && isInteger(dY)) {
+ t0 = _isTransformed ? _transformLast : _transformStart;
+ p02 = _getMouseCoords(source);
+ p1 = t0.invert(p02);
+ k2 = t0.k * Math.pow(2, -dY / 500);
+ k2 = clamp(k2, kMin, kMax);
+ x2 = p02[0] - p1[0] * k2;
+ y2 = p02[1] - p1[1] * k2;
+ } else if (detected.os === "mac" && detected.browser !== "Firefox" && !source.ctrlKey && isInteger(dX) && isInteger(dY)) {
+ p1 = projection2.translate();
+ x2 = p1[0] - dX;
+ y2 = p1[1] - dY;
+ k2 = projection2.scale();
+ k2 = clamp(k2, kMin, kMax);
+ }
+ if (x2 !== x || y2 !== y || k2 !== k) {
+ x = x2;
+ y = y2;
+ k = k2;
+ eventTransform = identity2.translate(x2, y2).scale(k2);
+ if (_zoomerPanner._transform) {
+ _zoomerPanner._transform(eventTransform);
+ } else {
+ _selection.node().__zoom = eventTransform;
+ }
+ }
+ }
+ if (_transformStart.x === x && _transformStart.y === y && _transformStart.k === k) {
+ return;
+ }
+ if (geoScaleToZoom(k, TILESIZE) < _minzoom) {
+ surface.interrupt();
+ dispatch10.call("hitMinZoom", this, map2);
+ setCenterZoom(map2.center(), context.minEditableZoom(), 0, true);
+ scheduleRedraw();
+ dispatch10.call("move", this, map2);
+ return;
+ }
+ projection2.transform(eventTransform);
+ var withinEditableZoom = map2.withinEditableZoom();
+ if (_lastWithinEditableZoom !== withinEditableZoom) {
+ if (_lastWithinEditableZoom !== void 0) {
+ dispatch10.call("crossEditableZoom", this, withinEditableZoom);
+ }
+ _lastWithinEditableZoom = withinEditableZoom;
+ }
+ var scale = k / _transformStart.k;
+ var tX = (x / scale - _transformStart.x) * scale;
+ var tY = (y / scale - _transformStart.y) * scale;
+ if (context.inIntro()) {
+ curtainProjection.transform({
+ x: x - tX,
+ y: y - tY,
+ k
+ });
+ }
+ if (source) {
+ _lastPointerEvent = event;
+ }
+ _isTransformed = true;
+ _transformLast = eventTransform;
+ utilSetTransform(supersurface, tX, tY, scale);
+ scheduleRedraw();
+ dispatch10.call("move", this, map2);
+ function isInteger(val) {
+ return typeof val === "number" && isFinite(val) && Math.floor(val) === val;
+ }
+ }
+ function resetTransform() {
+ if (!_isTransformed)
+ return false;
+ utilSetTransform(supersurface, 0, 0);
+ _isTransformed = false;
+ if (context.inIntro()) {
+ curtainProjection.transform(projection2.transform());
+ }
+ return true;
+ }
+ function redraw(difference, extent) {
+ if (surface.empty() || !_redrawEnabled)
+ return;
+ if (resetTransform()) {
+ difference = extent = void 0;
+ }
+ var zoom = map2.zoom();
+ var z = String(~~zoom);
+ if (surface.attr("data-zoom") !== z) {
+ surface.attr("data-zoom", z);
+ }
+ var lat = map2.center()[1];
+ var lowzoom = linear3().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);
+ }
+ if (map2.editableDataEnabled() || map2.isInWideSelection()) {
+ context.loadTiles(projection2);
+ drawEditable(difference, extent);
+ } else {
+ editOff();
+ }
+ _transformStart = projection2.transform();
+ return map2;
+ }
+ var immediateRedraw = function(difference, extent) {
+ if (!difference && !extent)
+ cancelPendingRedraw();
+ redraw(difference, extent);
+ };
+ map2.lastPointerEvent = function() {
+ return _lastPointerEvent;
+ };
+ map2.mouse = function(d3_event) {
+ var event = d3_event || _lastPointerEvent;
+ if (event) {
+ var s;
+ while (s = event.sourceEvent) {
+ event = s;
+ }
+ return _getMouseCoords(event);
+ }
+ return null;
+ };
+ map2.mouseCoordinates = function() {
+ var coord2 = map2.mouse() || pxCenter();
+ return projection2.invert(coord2);
+ };
+ map2.dblclickZoomEnable = function(val) {
+ if (!arguments.length)
+ return _dblClickZoomEnabled;
+ _dblClickZoomEnabled = val;
+ return map2;
+ };
+ map2.redrawEnable = function(val) {
+ if (!arguments.length)
+ return _redrawEnabled;
+ _redrawEnabled = val;
+ return map2;
+ };
+ map2.isTransformed = function() {
+ return _isTransformed;
+ };
+ function setTransform(t2, duration, force) {
+ var t = projection2.transform();
+ if (!force && t2.k === t.k && t2.x === t.x && t2.y === t.y)
+ return false;
+ if (duration) {
+ _selection.transition().duration(duration).on("start", function() {
+ map2.startEase();
+ }).call(_zoomerPanner.transform, identity2.translate(t2.x, t2.y).scale(t2.k));
+ } else {
+ projection2.transform(t2);
+ _transformStart = t2;
+ _selection.call(_zoomerPanner.transform, _transformStart);
+ }
+ return true;
+ }
+ function setCenterZoom(loc2, z2, duration, force) {
+ var c = map2.center();
+ var z = map2.zoom();
+ if (loc2[0] === c[0] && loc2[1] === c[1] && z2 === z && !force)
+ return false;
+ var proj = geoRawMercator().transform(projection2.transform());
+ var k2 = clamp(geoZoomToScale(z2, TILESIZE), kMin, kMax);
+ proj.scale(k2);
+ var t = proj.translate();
+ var point2 = proj(loc2);
+ var center = pxCenter();
+ t[0] += center[0] - point2[0];
+ t[1] += center[1] - point2[1];
+ return setTransform(identity2.translate(t[0], t[1]).scale(k2), duration, force);
+ }
+ map2.pan = function(delta, duration) {
+ var t = projection2.translate();
+ var k = projection2.scale();
+ t[0] += delta[0];
+ t[1] += delta[1];
+ if (duration) {
+ _selection.transition().duration(duration).on("start", function() {
+ map2.startEase();
+ }).call(_zoomerPanner.transform, identity2.translate(t[0], t[1]).scale(k));
+ } else {
+ projection2.translate(t);
+ _transformStart = projection2.transform();
+ _selection.call(_zoomerPanner.transform, _transformStart);
+ dispatch10.call("move", this, map2);
+ immediateRedraw();
+ }
+ return map2;
+ };
+ map2.dimensions = function(val) {
+ if (!arguments.length)
+ return _dimensions;
+ _dimensions = val;
+ drawLayers.dimensions(_dimensions);
+ context.background().dimensions(_dimensions);
+ projection2.clipExtent([[0, 0], _dimensions]);
+ _getMouseCoords = utilFastMouse(supersurface.node());
+ scheduleRedraw();
+ return map2;
+ };
+ function zoomIn(delta) {
+ setCenterZoom(map2.center(), ~~map2.zoom() + delta, 250, true);
+ }
+ function zoomOut(delta) {
+ setCenterZoom(map2.center(), ~~map2.zoom() - delta, 250, true);
+ }
+ map2.zoomIn = function() {
+ zoomIn(1);
+ };
+ map2.zoomInFurther = function() {
+ zoomIn(4);
+ };
+ map2.canZoomIn = function() {
+ return map2.zoom() < maxZoom;
+ };
+ map2.zoomOut = function() {
+ zoomOut(1);
+ };
+ map2.zoomOutFurther = function() {
+ zoomOut(4);
+ };
+ map2.canZoomOut = function() {
+ return map2.zoom() > minZoom2;
+ };
+ map2.center = function(loc2) {
+ if (!arguments.length) {
+ return projection2.invert(pxCenter());
+ }
+ if (setCenterZoom(loc2, map2.zoom())) {
+ dispatch10.call("move", this, map2);
+ }
+ scheduleRedraw();
+ return map2;
+ };
+ map2.unobscuredCenterZoomEase = function(loc, zoom) {
+ var offset = map2.unobscuredOffsetPx();
+ var proj = geoRawMercator().transform(projection2.transform());
+ proj.scale(geoZoomToScale(zoom, TILESIZE));
+ var locPx = proj(loc);
+ var offsetLocPx = [locPx[0] + offset[0], locPx[1] + offset[1]];
+ var offsetLoc = proj.invert(offsetLocPx);
+ map2.centerZoomEase(offsetLoc, zoom);
+ };
+ map2.unobscuredOffsetPx = function() {
+ var openPane = context.container().select(".map-panes .map-pane.shown");
+ if (!openPane.empty()) {
+ return [openPane.node().offsetWidth / 2, 0];
+ }
+ return [0, 0];
+ };
+ map2.zoom = function(z2) {
+ if (!arguments.length) {
+ return Math.max(geoScaleToZoom(projection2.scale(), TILESIZE), 0);
+ }
+ if (z2 < _minzoom) {
+ surface.interrupt();
+ dispatch10.call("hitMinZoom", this, map2);
+ z2 = context.minEditableZoom();
+ }
+ if (setCenterZoom(map2.center(), z2)) {
+ dispatch10.call("move", this, map2);
+ }
+ scheduleRedraw();
+ return map2;
+ };
+ map2.centerZoom = function(loc2, z2) {
+ if (setCenterZoom(loc2, z2)) {
+ dispatch10.call("move", this, map2);
+ }
+ scheduleRedraw();
+ return map2;
+ };
+ map2.zoomTo = function(entity) {
+ var extent = entity.extent(context.graph());
+ if (!isFinite(extent.area()))
+ return map2;
+ var z2 = clamp(map2.trimmedExtentZoom(extent), 0, 20);
+ return map2.centerZoom(extent.center(), z2);
+ };
+ map2.centerEase = function(loc2, duration) {
+ duration = duration || 250;
+ setCenterZoom(loc2, map2.zoom(), duration);
+ return map2;
+ };
+ map2.zoomEase = function(z2, duration) {
+ duration = duration || 250;
+ setCenterZoom(map2.center(), z2, duration, false);
+ return map2;
+ };
+ map2.centerZoomEase = function(loc2, z2, duration) {
+ duration = duration || 250;
+ setCenterZoom(loc2, z2, duration, false);
+ return map2;
+ };
+ map2.transformEase = function(t2, duration) {
+ duration = duration || 250;
+ setTransform(
+ t2,
+ duration,
+ false
+ /* don't force */
+ );
+ return map2;
+ };
+ map2.zoomToEase = function(obj, duration) {
+ var extent;
+ if (Array.isArray(obj)) {
+ obj.forEach(function(entity) {
+ var entityExtent = entity.extent(context.graph());
+ if (!extent) {
+ extent = entityExtent;
+ } else {
+ extent = extent.extend(entityExtent);
+ }
+ });
+ } else {
+ extent = obj.extent(context.graph());
+ }
+ if (!isFinite(extent.area()))
+ return map2;
+ var z2 = clamp(map2.trimmedExtentZoom(extent), 0, 20);
+ return map2.centerZoomEase(extent.center(), z2, duration);
+ };
+ map2.startEase = function() {
+ utilBindOnce(surface, _pointerPrefix + "down.ease", function() {
+ map2.cancelEase();
+ });
+ return map2;
+ };
+ map2.cancelEase = function() {
+ _selection.interrupt();
+ return map2;
+ };
+ map2.extent = function(val) {
+ if (!arguments.length) {
+ return new geoExtent(
+ projection2.invert([0, _dimensions[1]]),
+ projection2.invert([_dimensions[0], 0])
+ );
+ } else {
+ var extent = geoExtent(val);
+ map2.centerZoom(extent.center(), map2.extentZoom(extent));
+ }
+ };
+ map2.trimmedExtent = function(val) {
+ if (!arguments.length) {
+ var headerY = 71;
+ var footerY = 30;
+ var pad2 = 10;
+ return new geoExtent(
+ projection2.invert([pad2, _dimensions[1] - footerY - pad2]),
+ projection2.invert([_dimensions[0] - pad2, headerY + pad2])
+ );
+ } else {
+ var extent = geoExtent(val);
+ map2.centerZoom(extent.center(), map2.trimmedExtentZoom(extent));
+ }
+ };
+ function calcExtentZoom(extent, dim) {
+ var tl = projection2([extent[0][0], extent[1][1]]);
+ var br = projection2([extent[1][0], extent[0][1]]);
+ 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 = map2.zoom() - Math.max(hZoomDiff, vZoomDiff);
+ return newZoom;
+ }
+ map2.extentZoom = function(val) {
+ return calcExtentZoom(geoExtent(val), _dimensions);
+ };
+ map2.trimmedExtentZoom = function(val) {
+ var trimY = 120;
+ var trimX = 40;
+ var trimmed = [_dimensions[0] - trimX, _dimensions[1] - trimY];
+ return calcExtentZoom(geoExtent(val), trimmed);
+ };
+ map2.withinEditableZoom = function() {
+ return map2.zoom() >= context.minEditableZoom();
+ };
+ map2.isInWideSelection = function() {
+ return !map2.withinEditableZoom() && context.selectedIDs().length;
+ };
+ map2.editableDataEnabled = function(skipZoomCheck) {
+ var layer = context.layers().layer("osm");
+ if (!layer || !layer.enabled())
+ return false;
+ return skipZoomCheck || map2.withinEditableZoom();
+ };
+ map2.notesEditable = function() {
+ var layer = context.layers().layer("notes");
+ if (!layer || !layer.enabled())
+ return false;
+ return map2.withinEditableZoom();
+ };
+ map2.minzoom = function(val) {
+ if (!arguments.length)
+ return _minzoom;
+ _minzoom = val;
+ return map2;
+ };
+ map2.toggleHighlightEdited = function() {
+ surface.classed("highlight-edited", !surface.classed("highlight-edited"));
+ map2.pan([0, 0]);
+ dispatch10.call("changeHighlighting", this);
+ };
+ map2.areaFillOptions = ["wireframe", "partial", "full"];
+ map2.activeAreaFill = function(val) {
+ if (!arguments.length)
+ return corePreferences("area-fill") || "partial";
+ corePreferences("area-fill", val);
+ if (val !== "wireframe") {
+ corePreferences("area-fill-toggle", val);
+ }
+ updateAreaFill();
+ map2.pan([0, 0]);
+ dispatch10.call("changeAreaFill", this);
+ return map2;
+ };
+ map2.toggleWireframe = function() {
+ var activeFill = map2.activeAreaFill();
+ if (activeFill === "wireframe") {
+ activeFill = corePreferences("area-fill-toggle") || "partial";
+ } else {
+ activeFill = "wireframe";
+ }
+ map2.activeAreaFill(activeFill);
+ };
+ function updateAreaFill() {
+ var activeFill = map2.activeAreaFill();
+ map2.areaFillOptions.forEach(function(opt) {
+ surface.classed("fill-" + opt, Boolean(opt === activeFill));
+ });
+ }
+ map2.layers = () => drawLayers;
+ map2.doubleUpHandler = function() {
+ return _doubleUpHandler;
+ };
+ return utilRebind(map2, dispatch10, "on");
+ }
+
+ // modules/renderer/photos.js
+ function rendererPhotos(context) {
+ var dispatch10 = dispatch_default("change");
+ var _layerIDs = ["streetside", "mapillary", "mapillary-map-features", "mapillary-signs", "kartaview"];
+ var _allPhotoTypes = ["flat", "panoramic"];
+ var _shownPhotoTypes = _allPhotoTypes.slice();
+ var _dateFilters = ["fromDate", "toDate"];
+ var _fromDate;
+ var _toDate;
+ var _usernames;
+ function photos() {
+ }
+ 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 {
+ delete hash.photo_overlay;
+ }
+ window.location.replace("#" + utilQsString(hash, true));
+ }
+ photos.overlayLayerIDs = function() {
+ return _layerIDs;
+ };
+ photos.allPhotoTypes = function() {
+ return _allPhotoTypes;
+ };
+ photos.dateFilters = function() {
+ return _dateFilters;
+ };
+ photos.dateFilterValue = function(val) {
+ return val === _dateFilters[0] ? _fromDate : _toDate;
+ };
+ photos.setDateFilter = function(type2, val, updateUrl) {
+ var date = val && new Date(val);
+ if (date && !isNaN(date)) {
+ val = date.toISOString().slice(0, 10);
+ } else {
+ val = null;
+ }
+ if (type2 === _dateFilters[0]) {
+ _fromDate = val;
+ if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
+ _toDate = _fromDate;
+ }
+ }
+ if (type2 === _dateFilters[1]) {
+ _toDate = val;
+ if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
+ _fromDate = _toDate;
+ }
+ }
+ dispatch10.call("change", this);
+ if (updateUrl) {
+ var rangeString;
+ if (_fromDate || _toDate) {
+ rangeString = (_fromDate || "") + "_" + (_toDate || "");
+ }
+ setUrlFilterValue("photo_dates", rangeString);
+ }
+ };
+ photos.setUsernameFilter = function(val, updateUrl) {
+ if (val && typeof val === "string")
+ val = val.replace(/;/g, ",").split(",");
+ if (val) {
+ val = val.map((d) => d.trim()).filter(Boolean);
+ if (!val.length) {
+ val = null;
+ }
+ }
+ _usernames = val;
+ dispatch10.call("change", this);
+ if (updateUrl) {
+ var hashString;
+ if (_usernames) {
+ hashString = _usernames.join(",");
+ }
+ setUrlFilterValue("photo_username", hashString);
+ }
+ };
+ function setUrlFilterValue(property, val) {
+ if (!window.mocha) {
+ var hash = utilStringQs(window.location.hash);
+ if (val) {
+ if (hash[property] === val)
+ return;
+ hash[property] = val;
+ } else {
+ if (!(property in hash))
+ return;
+ delete hash[property];
+ }
+ window.location.replace("#" + utilQsString(hash, true));
+ }
+ }
+ function showsLayer(id2) {
+ var layer = context.layers().layer(id2);
+ return layer && layer.supported() && layer.enabled();
+ }
+ photos.shouldFilterByDate = function() {
+ return showsLayer("mapillary") || showsLayer("kartaview") || showsLayer("streetside");
+ };
+ photos.shouldFilterByPhotoType = function() {
+ return showsLayer("mapillary") || showsLayer("streetside") && showsLayer("kartaview");
+ };
+ photos.shouldFilterByUsername = function() {
+ return !showsLayer("mapillary") && showsLayer("kartaview") && !showsLayer("streetside");
+ };
+ photos.showsPhotoType = function(val) {
+ if (!photos.shouldFilterByPhotoType())
+ return true;
+ return _shownPhotoTypes.indexOf(val) !== -1;
+ };
+ photos.showsFlat = function() {
+ return photos.showsPhotoType("flat");
+ };
+ photos.showsPanoramic = function() {
+ return photos.showsPhotoType("panoramic");
+ };
+ photos.fromDate = function() {
+ return _fromDate;
+ };
+ photos.toDate = function() {
+ return _toDate;
+ };
+ photos.togglePhotoType = function(val) {
+ var index = _shownPhotoTypes.indexOf(val);
+ if (index !== -1) {
+ _shownPhotoTypes.splice(index, 1);
+ } else {
+ _shownPhotoTypes.push(val);
+ }
+ dispatch10.call("change", this);
+ return photos;
+ };
+ photos.usernames = function() {
+ return _usernames;
+ };
+ photos.init = function() {
+ var hash = utilStringQs(window.location.hash);
+ if (hash.photo_dates) {
+ 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);
+ }
+ if (hash.photo_username) {
+ this.setUsernameFilter(hash.photo_username, false);
+ }
+ if (hash.photo_overlay) {
+ var hashOverlayIDs = hash.photo_overlay.replace(/;/g, ",").split(",");
+ hashOverlayIDs.forEach(function(id2) {
+ if (id2 === "openstreetcam")
+ id2 = "kartaview";
+ var layer2 = _layerIDs.indexOf(id2) !== -1 && context.layers().layer(id2);
+ if (layer2 && !layer2.enabled())
+ layer2.enabled(true);
+ });
+ }
+ if (hash.photo) {
+ var photoIds = hash.photo.replace(/;/g, ",").split(",");
+ var photoId = photoIds.length && photoIds[0].trim();
+ var results = /(.*)\/(.*)/g.exec(photoId);
+ if (results && results.length >= 3) {
+ var serviceId = results[1];
+ if (serviceId === "openstreetcam")
+ serviceId = "kartaview";
+ var photoKey = results[2];
+ var service = services[serviceId];
+ if (service && service.ensureViewerLoaded) {
+ 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() {
+ if (Date.now() - baselineTime > 45e3) {
+ service.on("loadedImages.rendererPhotos", null);
+ return;
+ }
+ if (!service.cachedImage(photoKey))
+ return;
+ service.on("loadedImages.rendererPhotos", null);
+ service.ensureViewerLoaded(context).then(function() {
+ service.selectImage(context, photoKey).showViewer(context);
+ });
+ });
+ }
+ }
+ }
+ context.layers().on("change.rendererPhotos", updateStorage);
+ };
+ return utilRebind(photos, dispatch10, "on");
+ }
+
+ // modules/ui/account.js
+ function uiAccount(context) {
+ const osm = context.connection();
+ function updateUserDetails(selection2) {
+ if (!osm)
+ return;
+ if (!osm.authenticated()) {
+ render(selection2, null);
+ } else {
+ osm.userDetails((err, user) => render(selection2, user));
+ }
+ }
+ function render(selection2, user) {
+ let userInfo = selection2.select(".userInfo");
+ let loginLogout = selection2.select(".loginLogout");
+ if (user) {
+ userInfo.html("").classed("hide", false);
+ let userLink = userInfo.append("a").attr("href", osm.userURL(user.display_name)).attr("target", "_blank");
+ if (user.image_url) {
+ userLink.append("img").attr("class", "icon pre-text user-icon").attr("src", user.image_url);
+ } else {
+ userLink.call(svgIcon("#iD-icon-avatar", "pre-text light"));
+ }
+ userLink.append("span").attr("class", "label").html(user.display_name);
+ loginLogout.classed("hide", false).select("a").text(_t("logout")).on("click", (e) => {
+ e.preventDefault();
+ osm.logout();
+ tryLogout();
+ });
+ } else {
+ userInfo.html("").classed("hide", true);
+ loginLogout.classed("hide", false).select("a").text(_t("login")).on("click", (e) => {
+ e.preventDefault();
+ osm.authenticate();
+ });
+ }
+ }
+ function tryLogout() {
+ if (!osm)
+ return;
+ const url = osm.getUrlRoot() + "/logout?referer=%2Flogin";
+ const w = 600;
+ const h = 550;
+ const settings = [
+ ["width", w],
+ ["height", h],
+ ["left", window.screen.width / 2 - w / 2],
+ ["top", window.screen.height / 2 - h / 2]
+ ].map((x) => x.join("=")).join(",");
+ window.open(url, "_blank", settings);
+ }
+ return function(selection2) {
+ if (!osm)
+ return;
+ selection2.append("li").attr("class", "userInfo").classed("hide", true);
+ selection2.append("li").attr("class", "loginLogout").classed("hide", true).append("a").attr("href", "#");
+ osm.on("change.account", () => updateUserDetails(selection2));
+ updateUserDetails(selection2);
+ };
+ }
+
+ // modules/ui/attribution.js
+ function uiAttribution(context) {
+ let _selection = select_default2(null);
+ function render(selection2, data, klass) {
+ let div = selection2.selectAll(`.${klass}`).data([0]);
+ div = div.enter().append("div").attr("class", klass).merge(div);
+ let attributions = div.selectAll(".attribution").data(data, (d) => d.id);
+ attributions.exit().remove();
+ attributions = attributions.enter().append("span").attr("class", "attribution").each((d, i2, nodes) => {
+ let attribution = select_default2(nodes[i2]);
+ if (d.terms_html) {
+ attribution.html(d.terms_html);
+ return;
+ }
+ if (d.terms_url) {
+ attribution = attribution.append("a").attr("href", d.terms_url).attr("target", "_blank");
+ }
+ const sourceID = d.id.replace(/\./g, "<TX_DOT>");
+ const terms_text = _t(
+ `imagery.${sourceID}.attribution.text`,
+ { default: d.terms_text || d.id || d.name() }
+ );
+ if (d.icon && !d.overlay) {
+ attribution.append("img").attr("class", "source-image").attr("src", d.icon);
+ }
+ attribution.append("span").attr("class", "attribution-text").text(terms_text);
+ }).merge(attributions);
+ let copyright = attributions.selectAll(".copyright-notice").data((d) => {
+ let 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.text(String);
+ }
+ function update() {
+ let baselayer = context.background().baseLayerSource();
+ _selection.call(render, baselayer ? [baselayer] : [], "base-layer-attribution");
+ const z = context.map().zoom();
+ let overlays = context.background().overlayLayerSources() || [];
+ _selection.call(render, overlays.filter((s) => s.validZoom(z)), "overlay-layer-attribution");
+ }
+ return function(selection2) {
+ _selection = selection2;
+ context.background().on("change.attribution", update);
+ context.map().on("move.attribution", throttle_default(update, 400, { leading: false }));
+ update();
+ };
+ }
+
+ // modules/ui/contributors.js
+ function uiContributors(context) {
+ var osm = context.connection(), debouncedUpdate = debounce_default(function() {
+ update();
+ }, 1e3), limit = 4, hidden = false, wrap2 = select_default2(null);
+ 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);
+ wrap2.html("").call(svgIcon("#iD-icon-nearby", "pre-text light"));
+ var userList = select_default2(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").text(String);
+ if (u.length > limit) {
+ var count = select_default2(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());
+ }).text(othersNum);
+ wrap2.append("span").html(_t.html("contributors.truncated_list", { n: othersNum, users: { html: userList.html() }, count: { html: count.html() } }));
+ } else {
+ wrap2.append("span").html(_t.html("contributors.list", { users: { html: userList.html() } }));
+ }
+ if (!u.length) {
+ hidden = true;
+ wrap2.transition().style("opacity", 0);
+ } else if (hidden) {
+ wrap2.transition().style("opacity", 1);
+ }
+ }
+ return function(selection2) {
+ if (!osm)
+ return;
+ wrap2 = selection2;
+ update();
+ osm.on("loaded.contributors", debouncedUpdate);
+ context.map().on("move.contributors", debouncedUpdate);
+ };
+ }
+
+ // modules/ui/popover.js
+ var _popoverID = 0;
+ function uiPopover(klass) {
+ var _id = _popoverID++;
+ var _anchorSelection = select_default2(null);
+ var popover = function(selection2) {
+ _anchorSelection = selection2;
+ selection2.each(setup);
+ };
+ var _animation = utilFunctor(false);
+ var _placement = utilFunctor("top");
+ var _alignment = utilFunctor("center");
+ var _scrollContainer = utilFunctor(select_default2(null));
+ var _content;
+ var _displayType = utilFunctor("");
+ var _hasArrow = utilFunctor(true);
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ popover.displayType = function(val) {
+ if (arguments.length) {
+ _displayType = utilFunctor(val);
+ return popover;
+ } else {
+ return _displayType;
+ }
+ };
+ popover.hasArrow = function(val) {
+ if (arguments.length) {
+ _hasArrow = utilFunctor(val);
+ return popover;
+ } else {
+ return _hasArrow;
+ }
+ };
+ 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;
+ }
+ };
+ popover.scrollContainer = function(val) {
+ if (arguments.length) {
+ _scrollContainer = utilFunctor(val);
+ return popover;
+ } else {
+ return _scrollContainer;
+ }
+ };
+ popover.content = function(val) {
+ if (arguments.length) {
+ _content = val;
+ return popover;
+ } else {
+ return _content;
+ }
+ };
+ popover.isShown = function() {
+ var popoverSelection = _anchorSelection.select(".popover-" + _id);
+ return !popoverSelection.empty() && popoverSelection.classed("in");
+ };
+ popover.show = function() {
+ _anchorSelection.each(show);
+ };
+ popover.updateContent = function() {
+ _anchorSelection.each(updateContent);
+ };
+ popover.hide = function() {
+ _anchorSelection.each(hide);
+ };
+ popover.toggle = function() {
+ _anchorSelection.each(toggle);
+ };
+ popover.destroy = function(selection2, selector) {
+ selector = selector || ".popover-" + _id;
+ selection2.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();
+ };
+ popover.destroyAny = function(selection2) {
+ selection2.call(popover.destroy, ".popover");
+ };
+ function setup() {
+ var anchor = select_default2(this);
+ 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);
+ if (animate) {
+ popoverSelection.classed("fade", true);
+ }
+ var display = _displayType.apply(this, arguments);
+ if (display === "hover") {
+ var _lastNonMouseEnterTime;
+ anchor.on(_pointerPrefix + "enter.popover", function(d3_event) {
+ if (d3_event.pointerType) {
+ if (d3_event.pointerType !== "mouse") {
+ _lastNonMouseEnterTime = d3_event.timeStamp;
+ return;
+ } else if (_lastNonMouseEnterTime && d3_event.timeStamp - _lastNonMouseEnterTime < 1500) {
+ return;
+ }
+ }
+ if (d3_event.buttons !== 0)
+ return;
+ show.apply(this, arguments);
+ }).on(_pointerPrefix + "leave.popover", function() {
+ hide.apply(this, arguments);
+ }).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.attr("tabindex", 0).on("blur.popover", function() {
+ anchor.each(function() {
+ hide.apply(this, arguments);
+ });
+ });
+ }
+ }
+ function show() {
+ var anchor = select_default2(this);
+ var popoverSelection = anchor.selectAll(".popover-" + _id);
+ if (popoverSelection.empty()) {
+ anchor.call(popover.destroy);
+ anchor.each(setup);
+ popoverSelection = anchor.selectAll(".popover-" + _id);
+ }
+ popoverSelection.classed("in", true);
+ var displayType = _displayType.apply(this, arguments);
+ if (displayType === "clickFocus") {
+ anchor.classed("active", true);
+ popoverSelection.node().focus();
+ }
+ anchor.each(updateContent);
+ }
+ function updateContent() {
+ var anchor = select_default2(this);
+ if (_content) {
+ anchor.selectAll(".popover-" + _id + " > .popover-inner").call(_content.apply(this, arguments));
+ }
+ updatePosition.apply(this, arguments);
+ updatePosition.apply(this, arguments);
+ updatePosition.apply(this, arguments);
+ }
+ function updatePosition() {
+ var anchor = select_default2(this);
+ var popoverSelection = anchor.selectAll(".popover-" + _id);
+ var scrollContainer = _scrollContainer && _scrollContainer.apply(this, arguments);
+ 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);
+ popoverSelection.classed("left", false).classed("right", false).classed("top", false).classed("bottom", false).classed(placement, true);
+ var alignment = _alignment.apply(this, arguments);
+ var alignFactor = 0.5;
+ if (alignment === "leading") {
+ alignFactor = 0;
+ } else if (alignment === "trailing") {
+ alignFactor = 1;
+ }
+ 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;
+ case "bottom":
+ position = {
+ x: anchorFrame.x + (anchorFrame.w - popoverFrame.w) * alignFactor,
+ y: anchorFrame.y + anchorFrame.h
+ };
+ break;
+ case "left":
+ position = {
+ x: anchorFrame.x - popoverFrame.w,
+ y: anchorFrame.y + (anchorFrame.h - popoverFrame.h) * alignFactor
+ };
+ break;
+ case "right":
+ position = {
+ x: anchorFrame.x + anchorFrame.w,
+ y: anchorFrame.y + (anchorFrame.h - popoverFrame.h) * alignFactor
+ };
+ break;
+ }
+ 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;
+ }
+ var arrow = anchor.selectAll(".popover-" + _id + " > .popover-arrow");
+ var arrowPosX = Math.min(Math.max(popoverFrame.w / 2 - (position.x - initialPosX), 10), popoverFrame.w - 10);
+ arrow.style("left", ~~arrowPosX + "px");
+ }
+ popoverSelection.style("left", ~~position.x + "px").style("top", ~~position.y + "px");
+ } else {
+ popoverSelection.style("left", null).style("top", null);
+ }
+ function getFrame(node) {
+ var positionStyle = select_default2(node).style("position");
+ 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
+ };
+ }
+ }
+ }
+ function hide() {
+ var anchor = select_default2(this);
+ if (_displayType.apply(this, arguments) === "clickFocus") {
+ anchor.classed("active", false);
+ }
+ anchor.selectAll(".popover-" + _id).classed("in", false);
+ }
+ function toggle() {
+ if (select_default2(this).select(".popover-" + _id).classed("in")) {
+ hide.apply(this, arguments);
+ } else {
+ show.apply(this, arguments);
+ }
+ }
+ return popover;
+ }
+
+ // modules/ui/tooltip.js
+ function uiTooltip(klass) {
+ var tooltip = uiPopover((klass || "") + " tooltip").displayType("hover");
+ var _title = function() {
+ var title = this.getAttribute("data-original-title");
+ if (title) {
+ return title;
+ } else {
+ title = this.getAttribute("title");
+ this.removeAttribute("title");
+ this.setAttribute("data-original-title", title);
+ }
+ return title;
+ };
+ var _heading = utilFunctor(null);
+ var _keys = utilFunctor(null);
+ tooltip.title = function(val) {
+ if (!arguments.length)
+ return _title;
+ _title = utilFunctor(val);
+ return tooltip;
+ };
+ tooltip.heading = function(val) {
+ if (!arguments.length)
+ return _heading;
+ _heading = utilFunctor(val);
+ return tooltip;
+ };
+ tooltip.keys = function(val) {
+ if (!arguments.length)
+ return _keys;
+ _keys = utilFunctor(val);
+ return tooltip;
+ };
+ tooltip.content(function() {
+ var heading = _heading.apply(this, arguments);
+ var text2 = _title.apply(this, arguments);
+ var keys = _keys.apply(this, arguments);
+ var headingCallback = typeof heading === "function" ? heading : (s) => s.text(heading);
+ var textCallback = typeof text2 === "function" ? text2 : (s) => s.text(text2);
+ return function(selection2) {
+ var headingSelect = selection2.selectAll(".tooltip-heading").data(heading ? [heading] : []);
+ headingSelect.exit().remove();
+ headingSelect.enter().append("div").attr("class", "tooltip-heading").merge(headingSelect).text("").call(headingCallback);
+ var textSelect = selection2.selectAll(".tooltip-text").data(text2 ? [text2] : []);
+ textSelect.exit().remove();
+ textSelect.enter().append("div").attr("class", "tooltip-text").merge(textSelect).text("").call(textCallback);
+ var keyhintWrap = selection2.selectAll(".keyhint-wrap").data(keys && keys.length ? [0] : []);
+ keyhintWrap.exit().remove();
+ var keyhintWrapEnter = keyhintWrap.enter().append("div").attr("class", "keyhint-wrap");
+ keyhintWrapEnter.append("span").call(_t.append("tooltip_keyhint"));
+ keyhintWrap = keyhintWrapEnter.merge(keyhintWrap);
+ keyhintWrap.selectAll("kbd.shortcut").data(keys && keys.length ? keys : []).enter().append("kbd").attr("class", "shortcut").text(function(d) {
+ return d;
+ });
+ };
+ });
+ return tooltip;
+ }
+
+ // modules/ui/edit_menu.js
+ function uiEditMenu(context) {
+ var dispatch10 = dispatch_default("toggled");
+ var _menu = select_default2(null);
+ var _operations = [];
+ var _anchorLoc = [0, 0];
+ var _anchorLocLonLat = [0, 0];
+ var _triggerType = "";
+ var _vpTopMargin = 85;
+ var _vpBottomMargin = 45;
+ var _vpSideMargin = 35;
+ var _menuTop = false;
+ var _menuHeight;
+ var _menuWidth;
+ var _verticalPadding = 4;
+ var _tooltipWidth = 210;
+ var _menuSideMargin = 10;
+ var _tooltips = [];
+ var editMenu = function(selection2) {
+ var isTouchMenu = _triggerType.includes("touch") || _triggerType.includes("pen");
+ var ops = _operations.filter(function(op) {
+ return !isTouchMenu || !op.mouseOnly;
+ });
+ if (!ops.length)
+ return;
+ _tooltips = [];
+ _menuTop = isTouchMenu;
+ var showLabels = isTouchMenu;
+ var buttonHeight = showLabels ? 32 : 34;
+ if (showLabels) {
+ _menuWidth = 52 + Math.min(120, 6 * Math.max.apply(Math, ops.map(function(op) {
+ return op.title.length;
+ })));
+ } else {
+ _menuWidth = 44;
+ }
+ _menuHeight = _verticalPadding * 2 + ops.length * buttonHeight;
+ _menu = selection2.append("div").attr("class", "edit-menu").classed("touch-menu", isTouchMenu).style("padding", _verticalPadding + "px 0");
+ var buttons = _menu.selectAll(".edit-menu-item").data(ops);
+ 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).on("pointerup", pointerup).on("pointerdown mousedown", function pointerdown(d3_event) {
+ d3_event.stopPropagation();
+ }).on("mouseenter.highlight", function(d3_event, d) {
+ if (!d.relatedEntityIds || select_default2(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]]);
+ _tooltips.push(tooltip);
+ select_default2(this).call(tooltip).append("div").attr("class", "icon-wrap").call(svgIcon(d.icon && d.icon() || "#iD-operation-" + d.id, "operation"));
+ });
+ if (showLabels) {
+ buttonsEnter.append("span").attr("class", "label").html(function(d) {
+ return d.title;
+ });
+ }
+ 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;
+ function pointerup(d3_event) {
+ lastPointerUpType = d3_event.pointerType;
+ }
+ function click(d3_event, operation) {
+ d3_event.stopPropagation();
+ if (operation.relatedEntityIds) {
+ utilHighlightEntities(operation.relatedEntityIds(), false, context);
+ }
+ if (operation.disabled()) {
+ if (lastPointerUpType === "touch" || lastPointerUpType === "pen") {
+ context.ui().flash.duration(4e3).iconName("#iD-operation-" + operation.id).iconClass("operation disabled").label(operation.tooltip())();
+ }
+ } else {
+ if (lastPointerUpType === "touch" || lastPointerUpType === "pen") {
+ context.ui().flash.duration(2e3).iconName("#iD-operation-" + operation.id).iconClass("operation").label(operation.annotation() || operation.title)();
+ }
+ operation();
+ editMenu.close();
+ }
+ lastPointerUpType = null;
+ }
+ dispatch10.call("toggled", this, true);
+ };
+ function updatePosition() {
+ if (!_menu || _menu.empty())
+ return;
+ var anchorLoc = context.projection(_anchorLocLonLat);
+ var viewport = context.surfaceRect();
+ if (anchorLoc[0] < 0 || anchorLoc[0] > viewport.width || anchorLoc[1] < 0 || anchorLoc[1] > viewport.height) {
+ editMenu.close();
+ return;
+ }
+ var menuLeft = displayOnLeft(viewport);
+ var offset = [0, 0];
+ offset[0] = menuLeft ? -1 * (_menuSideMargin + _menuWidth) : _menuSideMargin;
+ if (_menuTop) {
+ if (anchorLoc[1] - _menuHeight < _vpTopMargin) {
+ offset[1] = -anchorLoc[1] + _vpTopMargin;
+ } else {
+ offset[1] = -_menuHeight;
+ }
+ } else {
+ if (anchorLoc[1] + _menuHeight > viewport.height - _vpBottomMargin) {
+ offset[1] = -anchorLoc[1] - _menuHeight + viewport.height - _vpBottomMargin;
+ } else {
+ offset[1] = 0;
+ }
+ }
+ var origin = geoVecAdd(anchorLoc, offset);
+ _menu.style("left", origin[0] + "px").style("top", origin[1] + "px");
+ var tooltipSide = tooltipPosition(viewport, menuLeft);
+ _tooltips.forEach(function(tooltip) {
+ tooltip.placement(tooltipSide);
+ });
+ function displayOnLeft(viewport2) {
+ if (_mainLocalizer.textDirection() === "ltr") {
+ if (anchorLoc[0] + _menuSideMargin + _menuWidth > viewport2.width - _vpSideMargin) {
+ return true;
+ }
+ return false;
+ } else {
+ if (anchorLoc[0] - _menuSideMargin - _menuWidth < _vpSideMargin) {
+ return false;
+ }
+ return true;
+ }
+ }
+ function tooltipPosition(viewport2, menuLeft2) {
+ if (_mainLocalizer.textDirection() === "ltr") {
+ if (menuLeft2) {
+ return "left";
+ }
+ if (anchorLoc[0] + _menuSideMargin + _menuWidth + _tooltipWidth > viewport2.width - _vpSideMargin) {
+ return "left";
+ }
+ return "right";
+ } else {
+ if (!menuLeft2) {
+ return "right";
+ }
+ if (anchorLoc[0] - _menuSideMargin - _menuWidth - _tooltipWidth < _vpSideMargin) {
+ return "right";
+ }
+ return "left";
+ }
+ }
+ }
+ editMenu.close = function() {
+ context.map().on("move.edit-menu", null).on("drawn.edit-menu", null);
+ _menu.remove();
+ _tooltips = [];
+ dispatch10.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, dispatch10, "on");
+ }
+
+ // modules/ui/feature_info.js
+ function uiFeatureInfo(context) {
+ function update(selection2) {
+ 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.append("inspector.title_count", {
+ title: _t("feature." + k + ".description"),
+ count: stats[k]
+ });
+ }
+ return null;
+ }).filter(Boolean);
+ selection2.text("");
+ if (hiddenList.length) {
+ var tooltipBehavior = uiTooltip().placement("top").title(function() {
+ return (selection3) => {
+ hiddenList.forEach((hiddenFeature) => {
+ selection3.append("div").call(hiddenFeature);
+ });
+ };
+ });
+ selection2.append("a").attr("class", "chip").attr("href", "#").call(_t.append("feature_info.hidden_warning", { count })).call(tooltipBehavior).on("click", function(d3_event) {
+ tooltipBehavior.hide();
+ d3_event.preventDefault();
+ context.ui().togglePanes(context.container().select(".map-panes .map-data-pane"));
+ });
+ }
+ selection2.classed("hide", !hiddenList.length);
+ }
+ return function(selection2) {
+ update(selection2);
+ context.features().on("change.feature_info", function() {
+ update(selection2);
+ });
+ };
+ }
+
+ // modules/ui/flash.js
+ function uiFlash(context) {
+ var _flashTimer;
+ var _duration = 2e3;
+ var _iconName = "#iD-icon-no";
+ var _iconClass = "disabled";
+ var _label = (s) => s.text("");
+ function flash() {
+ if (_flashTimer) {
+ _flashTimer.stop();
+ }
+ 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]);
+ 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");
+ 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").call(_label);
+ _flashTimer = timeout_default(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;
+ }
+ flash.duration = function(_) {
+ if (!arguments.length)
+ return _duration;
+ _duration = _;
+ return flash;
+ };
+ flash.label = function(_) {
+ if (!arguments.length)
+ return _label;
+ if (typeof _ !== "function") {
+ _label = (selection2) => selection2.text(_);
+ } else {
+ _label = (selection2) => selection2.text("").call(_);
+ }
+ return flash;
+ };
+ flash.iconName = function(_) {
+ if (!arguments.length)
+ return _iconName;
+ _iconName = _;
+ return flash;
+ };
+ flash.iconClass = function(_) {
+ if (!arguments.length)
+ return _iconClass;
+ _iconClass = _;
+ return flash;
+ };
+ return flash;
+ }
+
+ // modules/ui/full_screen.js
+ function uiFullScreen(context) {
+ var element = context.container().node();
+ 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;
+ }
+ }
+ function isFullScreen() {
+ return document.fullscreenElement || document.mozFullScreenElement || document.webkitFullscreenElement || document.msFullscreenElement;
+ }
+ function isSupported() {
+ return !!getFullScreenFn();
+ }
+ function fullScreen(d3_event) {
+ d3_event.preventDefault();
+ if (!isFullScreen()) {
+ getFullScreenFn().apply(element);
+ } else {
+ getExitFullScreenFn().apply(document);
+ }
+ }
+ return function() {
+ if (!isSupported())
+ return;
+ var detected = utilDetect();
+ var keys = detected.os === "mac" ? [uiCmd("\u2303\u2318F"), "f11"] : ["f11"];
+ context.keybinding().on(keys, fullScreen);
+ };
+ }
+
+ // modules/ui/geolocate.js
+ function uiGeolocate(context) {
+ var _geolocationOptions = {
+ // prioritize speed and power usage over precision
+ enableHighAccuracy: false,
+ // don't hang indefinitely getting the location
+ timeout: 6e3
+ // 6sec
+ };
+ var _locating = uiLoading(context).message(_t.html("geolocate.locating")).blocking(true);
+ var _layer = context.layers().layer("geolocate");
+ var _position;
+ var _extent;
+ var _timeoutID;
+ var _button = select_default2(null);
+ function click() {
+ if (context.inIntro())
+ return;
+ if (!_layer.enabled() && !_locating.isShown()) {
+ _timeoutID = setTimeout(
+ error,
+ 1e4
+ /* 10sec */
+ );
+ context.container().call(_locating);
+ navigator.geolocation.getCurrentPosition(success, error, _geolocationOptions);
+ } else {
+ _locating.close();
+ _layer.enabled(null, false);
+ updateButtonState();
+ }
+ }
+ function zoomTo() {
+ context.enter(modeBrowse(context));
+ var map2 = context.map();
+ _layer.enabled(_position, true);
+ updateButtonState();
+ map2.centerZoomEase(_extent.center(), Math.min(20, map2.extentZoom(_extent)));
+ }
+ function success(geolocation) {
+ _position = geolocation;
+ var coords = _position.coords;
+ _extent = geoExtent([coords.longitude, coords.latitude]).padByMeters(coords.accuracy);
+ zoomTo();
+ finish();
+ }
+ function error() {
+ if (_position) {
+ zoomTo();
+ } else {
+ context.ui().flash.label(_t.append("geolocate.location_unavailable")).iconName("#iD-icon-geolocate")();
+ }
+ finish();
+ }
+ function finish() {
+ _locating.close();
+ if (_timeoutID) {
+ clearTimeout(_timeoutID);
+ }
+ _timeoutID = void 0;
+ }
+ function updateButtonState() {
+ _button.classed("active", _layer.enabled());
+ _button.attr("aria-pressed", _layer.enabled());
+ }
+ return function(selection2) {
+ if (!navigator.geolocation || !navigator.geolocation.getCurrentPosition)
+ return;
+ _button = selection2.append("button").on("click", click).attr("aria-pressed", false).call(svgIcon("#iD-icon-geolocate", "light")).call(
+ uiTooltip().placement(_mainLocalizer.textDirection() === "rtl" ? "right" : "left").title(() => _t.append("geolocate.title")).keys([_t("geolocate.key")])
+ );
+ context.keybinding().on(_t("geolocate.key"), click);
+ };
+ }
+
+ // modules/ui/panels/background.js
+ function uiPanelBackground(context) {
+ var background = context.background();
+ var _currSourceName = null;
+ var _metadata = {};
+ var _metadataKeys = [
+ "zoom",
+ "vintage",
+ "source",
+ "description",
+ "resolution",
+ "accuracy"
+ ];
+ var debouncedRedraw = debounce_default(redraw, 250);
+ function redraw(selection2) {
+ var source = background.baseLayerSource();
+ if (!source)
+ return;
+ var isDG = source.id.match(/^DigitalGlobe/i) !== null;
+ var sourceLabel = source.label();
+ if (_currSourceName !== sourceLabel) {
+ _currSourceName = sourceLabel;
+ _metadata = {};
+ }
+ selection2.text("");
+ var list = selection2.append("ul").attr("class", "background-info");
+ list.append("li").call(_currSourceName);
+ _metadataKeys.forEach(function(k) {
+ if (isDG && k === "vintage")
+ return;
+ list.append("li").attr("class", "background-info-list-" + k).classed("hide", !_metadata[k]).call(_t.append("info_panels.background." + k, { suffix: ":" })).append("span").attr("class", "background-info-span-" + k).text(_metadata[k]);
+ });
+ debouncedGetMetadata(selection2);
+ var toggleTiles = context.getDebug("tile") ? "hide_tiles" : "show_tiles";
+ selection2.append("a").call(_t.append("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"));
+ selection2.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";
+ selection2.append("a").call(_t.append("info_panels.background." + toggleVintage)).attr("href", "#").attr("class", "button button-toggle-vintage").on("click", function(d3_event) {
+ d3_event.preventDefault();
+ context.background().toggleOverlayLayer(sourceVintage);
+ selection2.call(redraw);
+ });
+ }
+ ["DigitalGlobe-Premium", "DigitalGlobe-Standard"].forEach(function(layerId) {
+ if (source.id !== layerId) {
+ var key2 = layerId + "-vintage";
+ var sourceVintage2 = context.background().findSource(key2);
+ if (context.background().showsLayer(sourceVintage2)) {
+ context.background().toggleOverlayLayer(sourceVintage2);
+ }
+ }
+ });
+ }
+ var debouncedGetMetadata = debounce_default(getMetadata, 250);
+ function getMetadata(selection2) {
+ var tile = context.container().select(".layer-background img.tile-center");
+ 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();
+ _metadata.zoom = String(zoom);
+ selection2.selectAll(".background-info-list-zoom").classed("hide", false).selectAll(".background-info-span-zoom").text(_metadata.zoom);
+ if (!d || !d.length >= 3)
+ return;
+ background.baseLayerSource().getMetadata(center, d, function(err, result) {
+ if (err || _currSourceName !== sourceName)
+ return;
+ var vintage = result.vintage;
+ _metadata.vintage = vintage && vintage.range || _t("info_panels.background.unknown");
+ selection2.selectAll(".background-info-list-vintage").classed("hide", false).selectAll(".background-info-span-vintage").text(_metadata.vintage);
+ _metadataKeys.forEach(function(k) {
+ if (k === "zoom" || k === "vintage")
+ return;
+ var val = result[k];
+ _metadata[k] = val;
+ selection2.selectAll(".background-info-list-" + k).classed("hide", !val).selectAll(".background-info-span-" + k).text(val);
+ });
+ });
+ }
+ var panel = function(selection2) {
+ selection2.call(redraw);
+ context.map().on("drawn.info-background", function() {
+ selection2.call(debouncedRedraw);
+ }).on("move.info-background", function() {
+ selection2.call(debouncedGetMetadata);
+ });
+ };
+ panel.off = function() {
+ context.map().on("drawn.info-background", null).on("move.info-background", null);
+ };
+ panel.id = "background";
+ panel.label = _t.append("info_panels.background.title");
+ panel.key = _t("info_panels.background.key");
+ return panel;
+ }
+
+ // modules/ui/panels/history.js
+ function uiPanelHistory(context) {
+ var osm;
+ function displayTimestamp(timestamp) {
+ if (!timestamp)
+ return _t("info_panels.history.unknown");
+ var options2 = {
+ 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(), options2);
+ }
+ function displayUser(selection2, userName) {
+ if (!userName) {
+ selection2.append("span").call(_t.append("info_panels.history.unknown"));
+ return;
+ }
+ selection2.append("span").attr("class", "user-name").text(userName);
+ var links = selection2.append("div").attr("class", "links");
+ if (osm) {
+ links.append("a").attr("class", "user-osm-link").attr("href", osm.userURL(userName)).attr("target", "_blank").call(_t.append("info_panels.history.profile_link"));
+ }
+ links.append("a").attr("class", "user-hdyc-link").attr("href", "https://hdyc.neis-one.org/?" + userName).attr("target", "_blank").attr("tabindex", -1).text("HDYC");
+ }
+ function displayChangeset(selection2, changeset) {
+ if (!changeset) {
+ selection2.append("span").call(_t.append("info_panels.history.unknown"));
+ return;
+ }
+ selection2.append("span").attr("class", "changeset-id").text(changeset);
+ var links = selection2.append("div").attr("class", "links");
+ if (osm) {
+ links.append("a").attr("class", "changeset-osm-link").attr("href", osm.changesetURL(changeset)).attr("target", "_blank").call(_t.append("info_panels.history.changeset_link"));
+ }
+ links.append("a").attr("class", "changeset-osmcha-link").attr("href", "https://osmcha.org/changesets/" + changeset).attr("target", "_blank").text("OSMCha");
+ links.append("a").attr("class", "changeset-achavi-link").attr("href", "https://overpass-api.de/achavi/?changeset=" + changeset).attr("target", "_blank").text("Achavi");
+ }
+ function redraw(selection2) {
+ var selectedNoteID = context.selectedNoteID();
+ osm = context.connection();
+ var selected, note, entity;
+ if (selectedNoteID && osm) {
+ selected = [_t.html("note.note") + " " + selectedNoteID];
+ note = osm.getNote(selectedNoteID);
+ } else {
+ selected = context.selectedIDs().filter(function(e) {
+ return context.hasEntity(e);
+ });
+ if (selected.length) {
+ entity = context.entity(selected[0]);
+ }
+ }
+ var singular = selected.length === 1 ? selected[0] : null;
+ selection2.html("");
+ if (singular) {
+ selection2.append("h4").attr("class", "history-heading").html(singular);
+ } else {
+ selection2.append("h4").attr("class", "history-heading").call(_t.append("info_panels.selected", { n: selected.length }));
+ }
+ if (!singular)
+ return;
+ if (entity) {
+ selection2.call(redrawEntity, entity);
+ } else if (note) {
+ selection2.call(redrawNote, note);
+ }
+ }
+ function redrawNote(selection2, note) {
+ if (!note || note.isNew()) {
+ selection2.append("div").call(_t.append("info_panels.history.note_no_history"));
+ return;
+ }
+ var list = selection2.append("ul");
+ list.append("li").call(_t.append("info_panels.history.note_comments", { suffix: ":" })).append("span").text(note.comments.length);
+ if (note.comments.length) {
+ list.append("li").call(_t.append("info_panels.history.note_created_date", { suffix: ":" })).append("span").text(displayTimestamp(note.comments[0].date));
+ list.append("li").call(_t.append("info_panels.history.note_created_user", { suffix: ":" })).call(displayUser, note.comments[0].user);
+ }
+ if (osm) {
+ selection2.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").call(_t.append("info_panels.history.note_link_text"));
+ }
+ }
+ function redrawEntity(selection2, entity) {
+ if (!entity || entity.isNew()) {
+ selection2.append("div").call(_t.append("info_panels.history.no_history"));
+ return;
+ }
+ var links = selection2.append("div").attr("class", "links");
+ if (osm) {
+ links.append("a").attr("class", "view-history-on-osm").attr("href", osm.historyURL(entity)).attr("target", "_blank").call(_t.append("info_panels.history.history_link"));
+ }
+ 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).text("PeWu");
+ var list = selection2.append("ul");
+ list.append("li").call(_t.append("info_panels.history.version", { suffix: ":" })).append("span").text(entity.version);
+ list.append("li").call(_t.append("info_panels.history.last_edit", { suffix: ":" })).append("span").text(displayTimestamp(entity.timestamp));
+ list.append("li").call(_t.append("info_panels.history.edited_by", { suffix: ":" })).call(displayUser, entity.user);
+ list.append("li").call(_t.append("info_panels.history.changeset", { suffix: ":" })).call(displayChangeset, entity.changeset);
+ }
+ var panel = function(selection2) {
+ selection2.call(redraw);
+ context.map().on("drawn.info-history", function() {
+ selection2.call(redraw);
+ });
+ context.on("enter.info-history", function() {
+ selection2.call(redraw);
+ });
+ };
+ panel.off = function() {
+ context.map().on("drawn.info-history", null);
+ context.on("enter.info-history", null);
+ };
+ panel.id = "history";
+ panel.label = _t.append("info_panels.history.title");
+ panel.key = _t("info_panels.history.key");
+ return panel;
+ }
+
+ // modules/util/units.js
+ var OSM_PRECISION = 7;
+ function displayLength(m, isImperial) {
+ var d = m * (isImperial ? 3.28084 : 1);
+ var unit2;
+ if (isImperial) {
+ if (d >= 5280) {
+ d /= 5280;
+ unit2 = "miles";
+ } else {
+ unit2 = "feet";
+ }
+ } else {
+ if (d >= 1e3) {
+ d /= 1e3;
+ unit2 = "kilometers";
+ } else {
+ unit2 = "meters";
+ }
+ }
+ return _t("units." + unit2, {
+ quantity: d.toLocaleString(_mainLocalizer.localeCode(), {
+ maximumSignificantDigits: 4
+ })
+ });
+ }
+ function displayArea(m2, isImperial) {
+ var locale2 = _mainLocalizer.localeCode();
+ var d = m2 * (isImperial ? 10.7639111056 : 1);
+ var d1, d2, area;
+ var unit1 = "";
+ var unit2 = "";
+ if (isImperial) {
+ if (d >= 6969600) {
+ d1 = d / 27878400;
+ unit1 = "square_miles";
+ } else {
+ d1 = d;
+ unit1 = "square_feet";
+ }
+ if (d > 4356 && d < 4356e4) {
+ d2 = d / 43560;
+ unit2 = "acres";
+ }
+ } else {
+ if (d >= 25e4) {
+ d1 = d / 1e6;
+ unit1 = "square_kilometers";
+ } else {
+ d1 = d;
+ unit1 = "square_meters";
+ }
+ if (d > 1e3 && d < 1e7) {
+ d2 = d / 1e4;
+ unit2 = "hectares";
+ }
+ }
+ area = _t("units." + unit1, {
+ quantity: d1.toLocaleString(locale2, {
+ maximumSignificantDigits: 4
+ })
+ });
+ if (d2) {
+ return _t("units.area_pair", {
+ area1: area,
+ area2: _t("units." + unit2, {
+ quantity: d2.toLocaleString(locale2, {
+ maximumSignificantDigits: 2
+ })
+ })
+ });
+ } else {
+ return area;
+ }
+ }
+ function wrap(x, min3, max3) {
+ var d = max3 - min3;
+ return ((x - min3) % d + d) % d + min3;
+ }
+ function clamp2(x, min3, max3) {
+ return Math.max(min3, Math.min(x, max3));
+ }
+ function displayCoordinate(deg, pos, neg) {
+ var locale2 = _mainLocalizer.localeCode();
+ var min3 = (Math.abs(deg) - Math.floor(Math.abs(deg))) * 60;
+ var sec = (min3 - Math.floor(min3)) * 60;
+ var displayDegrees = _t("units.arcdegrees", {
+ quantity: Math.floor(Math.abs(deg)).toLocaleString(locale2)
+ });
+ var displayCoordinate2;
+ if (Math.floor(sec) > 0) {
+ displayCoordinate2 = displayDegrees + _t("units.arcminutes", {
+ quantity: Math.floor(min3).toLocaleString(locale2)
+ }) + _t("units.arcseconds", {
+ quantity: Math.round(sec).toLocaleString(locale2)
+ });
+ } else if (Math.floor(min3) > 0) {
+ displayCoordinate2 = displayDegrees + _t("units.arcminutes", {
+ quantity: Math.round(min3).toLocaleString(locale2)
+ });
+ } else {
+ displayCoordinate2 = _t("units.arcdegrees", {
+ quantity: Math.round(Math.abs(deg)).toLocaleString(locale2)
+ });
+ }
+ if (deg === 0) {
+ return displayCoordinate2;
+ } else {
+ return _t("units.coordinate", {
+ coordinate: displayCoordinate2,
+ direction: _t("units." + (deg > 0 ? pos : neg))
+ });
+ }
+ }
+ function dmsCoordinatePair(coord2) {
+ return _t("units.coordinate_pair", {
+ latitude: displayCoordinate(clamp2(coord2[1], -90, 90), "north", "south"),
+ longitude: displayCoordinate(wrap(coord2[0], -180, 180), "east", "west")
+ });
+ }
+ function decimalCoordinatePair(coord2) {
+ return _t("units.coordinate_pair", {
+ latitude: clamp2(coord2[1], -90, 90).toFixed(OSM_PRECISION),
+ longitude: wrap(coord2[0], -180, 180).toFixed(OSM_PRECISION)
+ });
+ }
+
+ // modules/ui/panels/location.js
+ function uiPanelLocation(context) {
+ var currLocation = "";
+ function redraw(selection2) {
+ selection2.html("");
+ var list = selection2.append("ul");
+ var coord2 = context.map().mouseCoordinates();
+ if (coord2.some(isNaN)) {
+ coord2 = context.map().center();
+ }
+ list.append("li").text(dmsCoordinatePair(coord2)).append("li").text(decimalCoordinatePair(coord2));
+ selection2.append("div").attr("class", "location-info").text(currLocation || " ");
+ debouncedGetLocation(selection2, coord2);
+ }
+ var debouncedGetLocation = debounce_default(getLocation, 250);
+ function getLocation(selection2, coord2) {
+ if (!services.geocoder) {
+ currLocation = _t("info_panels.location.unknown_location");
+ selection2.selectAll(".location-info").text(currLocation);
+ } else {
+ services.geocoder.reverse(coord2, function(err, result) {
+ currLocation = result ? result.display_name : _t("info_panels.location.unknown_location");
+ selection2.selectAll(".location-info").text(currLocation);
+ });
+ }
+ }
+ var panel = function(selection2) {
+ selection2.call(redraw);
+ context.surface().on(("PointerEvent" in window ? "pointer" : "mouse") + "move.info-location", function() {
+ selection2.call(redraw);
+ });
+ };
+ panel.off = function() {
+ context.surface().on(".info-location", null);
+ };
+ panel.id = "location";
+ panel.label = _t.append("info_panels.location.title");
+ panel.key = _t("info_panels.location.key");
+ return panel;
+ }
+
+ // modules/ui/panels/measurement.js
+ function uiPanelMeasurement(context) {
+ function radiansToMeters(r) {
+ return r * 63710071809e-4;
+ }
+ function steradiansToSqmeters(r) {
+ return r / (4 * Math.PI) * 510065621724e3;
+ }
+ function toLineString(feature3) {
+ if (feature3.type === "LineString")
+ return feature3;
+ var result = { type: "LineString", coordinates: [] };
+ if (feature3.type === "Polygon") {
+ result.coordinates = feature3.coordinates[0];
+ } else if (feature3.type === "MultiPolygon") {
+ result.coordinates = feature3.coordinates[0][0];
+ }
+ return result;
+ }
+ var _isImperial = !_mainLocalizer.usesMetric();
+ function redraw(selection2) {
+ 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 (selectedNoteID && osm) {
+ var note = osm.getNote(selectedNoteID);
+ heading = _t.html("note.note") + " " + selectedNoteID;
+ location = note.loc;
+ geometry = "note";
+ } else {
+ var selectedIDs = context.selectedIDs().filter(function(id2) {
+ return context.hasEntity(id2);
+ });
+ var selected = selectedIDs.map(function(id2) {
+ return context.entity(id2);
+ });
+ heading = selected.length === 1 ? selected[0].id : _t.html("info_panels.selected", { n: selected.length });
+ if (selected.length) {
+ var extent = geoExtent();
+ for (var i2 in selected) {
+ var entity = selected[i2];
+ extent._extend(entity.extent(graph));
+ geometry = entity.geometry(graph);
+ if (geometry === "line" || geometry === "area") {
+ closed = entity.type === "relation" || entity.isClosed() && !entity.isDegenerate();
+ var feature3 = entity.asGeoJSON(graph);
+ length += radiansToMeters(length_default(toLineString(feature3)));
+ centroid = path_default(context.projection).centroid(entity.asGeoJSON(graph));
+ centroid = centroid && context.projection.invert(centroid);
+ if (!centroid || !isFinite(centroid[0]) || !isFinite(centroid[1])) {
+ centroid = entity.extent(graph).center();
+ }
+ if (closed) {
+ area += steradiansToSqmeters(entity.area(graph));
+ }
+ }
+ }
+ if (selected.length > 1) {
+ geometry = null;
+ closed = null;
+ centroid = null;
+ }
+ if (selected.length === 2 && selected[0].type === "node" && selected[1].type === "node") {
+ distance = geoSphericalDistance(selected[0].loc, selected[1].loc);
+ }
+ 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();
+ }
+ }
+ }
+ selection2.html("");
+ if (heading) {
+ selection2.append("h4").attr("class", "measurement-heading").html(heading);
+ }
+ var list = selection2.append("ul");
+ var coordItem;
+ if (geometry) {
+ list.append("li").call(_t.append("info_panels.measurement.geometry", { suffix: ":" })).append("span").html(
+ closed ? _t.html("info_panels.measurement.closed_" + geometry) : _t.html("geometry." + geometry)
+ );
+ }
+ if (totalNodeCount) {
+ list.append("li").call(_t.append("info_panels.measurement.node_count", { suffix: ":" })).append("span").text(totalNodeCount.toLocaleString(localeCode));
+ }
+ if (area) {
+ list.append("li").call(_t.append("info_panels.measurement.area", { suffix: ":" })).append("span").text(displayArea(area, _isImperial));
+ }
+ if (length) {
+ list.append("li").call(_t.append("info_panels.measurement." + (closed ? "perimeter" : "length"), { suffix: ":" })).append("span").text(displayLength(length, _isImperial));
+ }
+ if (typeof distance === "number") {
+ list.append("li").call(_t.append("info_panels.measurement.distance", { suffix: ":" })).append("span").text(displayLength(distance, _isImperial));
+ }
+ if (location) {
+ coordItem = list.append("li").call(_t.append("info_panels.measurement.location", { suffix: ":" }));
+ coordItem.append("span").text(dmsCoordinatePair(location));
+ coordItem.append("span").text(decimalCoordinatePair(location));
+ }
+ if (centroid) {
+ coordItem = list.append("li").call(_t.append("info_panels.measurement.centroid", { suffix: ":" }));
+ coordItem.append("span").text(dmsCoordinatePair(centroid));
+ coordItem.append("span").text(decimalCoordinatePair(centroid));
+ }
+ if (center) {
+ coordItem = list.append("li").call(_t.append("info_panels.measurement.center", { suffix: ":" }));
+ coordItem.append("span").text(dmsCoordinatePair(center));
+ coordItem.append("span").text(decimalCoordinatePair(center));
+ }
+ if (length || area || typeof distance === "number") {
+ var toggle = _isImperial ? "imperial" : "metric";
+ selection2.append("a").call(_t.append("info_panels.measurement." + toggle)).attr("href", "#").attr("class", "button button-toggle-units").on("click", function(d3_event) {
+ d3_event.preventDefault();
+ _isImperial = !_isImperial;
+ selection2.call(redraw);
+ });
+ }
+ }
+ var panel = function(selection2) {
+ selection2.call(redraw);
+ context.map().on("drawn.info-measurement", function() {
+ selection2.call(redraw);
+ });
+ context.on("enter.info-measurement", function() {
+ selection2.call(redraw);
+ });
+ };
+ panel.off = function() {
+ context.map().on("drawn.info-measurement", null);
+ context.on("enter.info-measurement", null);
+ };
+ panel.id = "measurement";
+ panel.label = _t.append("info_panels.measurement.title");
+ panel.key = _t("info_panels.measurement.key");
+ return panel;
+ }
+
+ // modules/ui/panels/index.js
+ var uiInfoPanels = {
+ background: uiPanelBackground,
+ history: uiPanelHistory,
+ location: uiPanelLocation,
+ measurement: uiPanelMeasurement
+ };
+
+ // modules/ui/info.js
+ function uiInfo(context) {
+ var ids = Object.keys(uiInfoPanels);
+ var wasActive = ["measurement"];
+ var panels = {};
+ var active = {};
+ ids.forEach(function(k) {
+ if (!panels[k]) {
+ panels[k] = uiInfoPanels[k](context);
+ active[k] = false;
+ }
+ });
+ function info(selection2) {
+ 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_default2(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").each(function(d) {
+ return panels[d].label(select_default2(this));
+ });
+ title.append("button").attr("class", "close").attr("title", _t("icons.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;
+ });
+ infoPanels.selectAll(".panel-content").each(function(d) {
+ select_default2(this).call(panels[d]);
+ });
+ }
+ info.toggle = function(which) {
+ var activeids = ids.filter(function(k) {
+ return active[k];
+ });
+ if (which) {
+ active[which] = !active[which];
+ if (activeids.length === 1 && activeids[0] === which) {
+ wasActive = [which];
+ }
+ context.container().select("." + which + "-panel-toggle-item").classed("active", active[which]).select("input").property("checked", active[which]);
+ } else {
+ if (activeids.length) {
+ wasActive = activeids;
+ activeids.forEach(function(k) {
+ active[k] = false;
+ });
+ } else {
+ wasActive.forEach(function(k) {
+ active[k] = true;
+ });
+ }
+ }
+ redraw();
+ };
+ var infoPanels = selection2.selectAll(".info-panels").data([0]);
+ infoPanels = infoPanels.enter().append("div").attr("class", "info-panels").merge(infoPanels);
+ redraw();
+ context.keybinding().on(uiCmd("\u2318" + _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("\u2318\u21E7" + key), function(d3_event) {
+ d3_event.stopImmediatePropagation();
+ d3_event.preventDefault();
+ info.toggle(k);
+ });
+ });
+ }
+ return info;
+ }
+
+ // modules/ui/intro/helper.js
+ function pointBox(loc, context) {
+ var rect = context.surfaceRect();
+ var point2 = context.curtainProjection(loc);
+ return {
+ left: point2[0] + rect.left - 40,
+ top: point2[1] + rect.top - 60,
+ width: 80,
+ height: 90
+ };
+ }
+ function pad(locOrBox, padding, context) {
+ var box;
+ if (locOrBox instanceof Array) {
+ var rect = context.surfaceRect();
+ var point2 = context.curtainProjection(locOrBox);
+ box = {
+ left: point2[0] + rect.left,
+ top: point2[1] + rect.top
+ };
+ } else {
+ box = locOrBox;
+ }
+ 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;
+ function helpHtml(id2, replacements) {
+ 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("\u21E7"),
+ alt: uiCmd.display("\u2325"),
+ return: uiCmd.display("\u21B5"),
+ 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")
+ };
+ for (var key in helpStringReplacements) {
+ helpStringReplacements[key] = { html: helpStringReplacements[key] };
+ }
+ }
+ var reps;
+ if (replacements) {
+ reps = Object.assign(replacements, helpStringReplacements);
+ } else {
+ reps = helpStringReplacements;
+ }
+ return _t.html(id2, reps).replace(/\`(.*?)\`/g, "<kbd>$1</kbd>");
+ }
+ function slugify(text2) {
+ return text2.toString().toLowerCase().replace(/\s+/g, "-").replace(/[^\w\-]+/g, "").replace(/\-\-+/g, "-").replace(/^-+/, "").replace(/-+$/, "");
+ }
+ var missingStrings = {};
+ function checkKey(key, text2) {
+ if (_t(key, { default: void 0 }) === void 0) {
+ if (missingStrings.hasOwnProperty(key))
+ return;
+ missingStrings[key] = text2;
+ var missing = key + ": " + text2;
+ if (typeof console !== "undefined")
+ console.log(missing);
+ }
+ }
+ function localize(obj) {
+ var key;
+ var name = obj.tags && obj.tags.name;
+ if (name) {
+ key = "intro.graph.name." + slugify(name);
+ obj.tags.name = _t(key, { default: name });
+ checkKey(key, name);
+ }
+ var street = obj.tags && obj.tags["addr:street"];
+ if (street) {
+ key = "intro.graph.name." + slugify(street);
+ obj.tags["addr:street"] = _t(key, { default: street });
+ checkKey(key, street);
+ var addrTags = [
+ "block_number",
+ "city",
+ "county",
+ "district",
+ "hamlet",
+ "neighbourhood",
+ "postcode",
+ "province",
+ "quarter",
+ "state",
+ "subdistrict",
+ "suburb"
+ ];
+ addrTags.forEach(function(k) {
+ var key2 = "intro.graph." + k;
+ var tag = "addr:" + k;
+ var val = obj.tags && obj.tags[tag];
+ var str2 = _t(key2, { default: val });
+ if (str2) {
+ if (str2.match(/^<.*>$/) !== null) {
+ delete obj.tags[tag];
+ } else {
+ obj.tags[tag] = str2;
+ }
+ }
+ });
+ }
+ return obj;
+ }
+ function isMostlySquare(points) {
+ var threshold = 15;
+ var lowerBound = Math.cos((90 - threshold) * Math.PI / 180);
+ var upperBound = Math.cos(threshold * Math.PI / 180);
+ for (var i2 = 0; i2 < points.length; i2++) {
+ var a = points[(i2 - 1 + points.length) % points.length];
+ var origin = points[i2];
+ var b = points[(i2 + 1) % points.length];
+ var dotp = geoVecNormalizedDot(a, b, origin);
+ var mag = Math.abs(dotp);
+ 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 1e3;
+ }
+ }
+
+ // modules/ui/toggle.js
+ function uiToggle(show, callback) {
+ return function(selection2) {
+ selection2.style("opacity", show ? 0 : 1).classed("hide", false).transition().style("opacity", show ? 1 : 0).on("end", function() {
+ select_default2(this).classed("hide", !show).style("opacity", null);
+ if (callback)
+ callback.apply(this);
+ });
+ };
+ }
+
+ // modules/ui/curtain.js
+ function uiCurtain(containerNode) {
+ var surface = select_default2(null), tooltip = select_default2(null), darkness = select_default2(null);
+ function curtain(selection2) {
+ surface = selection2.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_default2(window).on("resize.curtain", resize);
+ tooltip = selection2.append("div").attr("class", "tooltip");
+ tooltip.append("div").attr("class", "popover-arrow");
+ tooltip.append("div").attr("class", "popover-inner");
+ resize();
+ function resize() {
+ surface.attr("width", containerNode.clientWidth).attr("height", containerNode.clientHeight);
+ curtain.cut(darkness.datum());
+ }
+ }
+ curtain.reveal = function(box, html2, options2) {
+ options2 = options2 || {};
+ if (typeof box === "string") {
+ box = select_default2(box).node();
+ }
+ if (box && box.getBoundingClientRect) {
+ box = copyBox(box.getBoundingClientRect());
+ var containerRect = containerNode.getBoundingClientRect();
+ box.top -= containerRect.top;
+ box.left -= containerRect.left;
+ }
+ if (box && options2.padding) {
+ box.top -= options2.padding;
+ box.left -= options2.padding;
+ box.bottom += options2.padding;
+ box.right += options2.padding;
+ box.height += options2.padding * 2;
+ box.width += options2.padding * 2;
+ }
+ var tooltipBox;
+ if (options2.tooltipBox) {
+ tooltipBox = options2.tooltipBox;
+ if (typeof tooltipBox === "string") {
+ tooltipBox = select_default2(tooltipBox).node();
+ }
+ if (tooltipBox && tooltipBox.getBoundingClientRect) {
+ tooltipBox = copyBox(tooltipBox.getBoundingClientRect());
+ }
+ } else {
+ tooltipBox = box;
+ }
+ if (tooltipBox && html2) {
+ if (html2.indexOf("**") !== -1) {
+ if (html2.indexOf("<span") === 0) {
+ html2 = html2.replace(/^(<span.*?>)(.+?)(\*\*)/, "$1<span>$2</span>$3");
+ } else {
+ html2 = html2.replace(/^(.+?)(\*\*)/, "<span>$1</span>$2");
+ }
+ html2 = html2.replace(/\*\*(.*?)\*\*/g, '<span class="instruction">$1</span>');
+ }
+ html2 = html2.replace(/\*(.*?)\*/g, "<em>$1</em>");
+ html2 = html2.replace(/\{br\}/g, "<br/><br/>");
+ if (options2.buttonText && options2.buttonCallback) {
+ html2 += '<div class="button-section"><button href="#" class="button action">' + options2.buttonText + "</button></div>";
+ }
+ var classes = "curtain-tooltip popover tooltip arrowed in " + (options2.tooltipClass || "");
+ tooltip.classed(classes, true).selectAll(".popover-inner").html(html2);
+ if (options2.buttonText && options2.buttonCallback) {
+ var button = tooltip.selectAll(".button-section .button.action");
+ button.on("click", function(d3_event) {
+ d3_event.preventDefault();
+ options2.buttonCallback();
+ });
+ }
+ var tip = copyBox(tooltip.node().getBoundingClientRect()), w = containerNode.clientWidth, h = containerNode.clientHeight, tooltipWidth = 200, tooltipArrow = 5, side, pos;
+ if (options2.tooltipClass === "intro-mouse") {
+ tip.height += 80;
+ }
+ if (tooltipBox.top + tooltipBox.height > h) {
+ tooltipBox.height -= tooltipBox.top + tooltipBox.height - h;
+ }
+ if (tooltipBox.left + tooltipBox.width > w) {
+ tooltipBox.width -= tooltipBox.left + tooltipBox.width - w;
+ }
+ if (tooltipBox.top + tooltipBox.height < 100) {
+ side = "bottom";
+ pos = [
+ tooltipBox.left + tooltipBox.width / 2 - tip.width / 2,
+ tooltipBox.top + tooltipBox.height
+ ];
+ } else if (tooltipBox.top > h - 140) {
+ side = "top";
+ pos = [
+ tooltipBox.left + tooltipBox.width / 2 - tip.width / 2,
+ tooltipBox.top - tip.height
+ ];
+ } else {
+ var tipY = tooltipBox.top + tooltipBox.height / 2 - tip.height / 2;
+ 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 (options2.duration !== 0 || !tooltip.classed(side)) {
+ tooltip.call(uiToggle(true));
+ }
+ tooltip.style("top", pos[1] + "px").style("left", pos[0] + "px").attr("class", classes + " " + side);
+ var shiftY = 0;
+ 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;
+ }
+ }
+ tooltip.selectAll(".popover-inner").style("top", shiftY + "px");
+ } else {
+ tooltip.classed("in", false).call(uiToggle(false));
+ }
+ curtain.cut(box, options2.duration);
+ return tooltip;
+ };
+ curtain.cut = function(datum2, duration) {
+ darkness.datum(datum2).interrupt();
+ var selection2;
+ if (duration === 0) {
+ selection2 = darkness;
+ } else {
+ selection2 = darkness.transition().duration(duration || 600).ease(linear2);
+ }
+ selection2.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";
+ });
+ };
+ curtain.remove = function() {
+ surface.remove();
+ tooltip.remove();
+ select_default2(window).on("resize.curtain", null);
+ };
+ function copyBox(src) {
+ return {
+ top: src.top,
+ right: src.right,
+ bottom: src.bottom,
+ left: src.left,
+ width: src.width,
+ height: src.height
+ };
+ }
+ return curtain;
+ }
+
+ // modules/ui/intro/welcome.js
+ function uiIntroWelcome(context, reveal) {
+ var dispatch10 = dispatch_default("done");
+ var chapter = {
+ title: "intro.welcome.title"
+ };
+ 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 }
+ );
+ }
+ function words() {
+ reveal(
+ ".intro-nav-wrap .chapter-welcome",
+ helpHtml("intro.welcome.words"),
+ { buttonText: _t.html("intro.ok"), buttonCallback: chapters }
+ );
+ }
+ function chapters() {
+ dispatch10.call("done");
+ reveal(
+ ".intro-nav-wrap .chapter-navigation",
+ helpHtml("intro.welcome.chapters", { next: _t("intro.navigation.title") })
+ );
+ }
+ chapter.enter = function() {
+ welcome();
+ };
+ chapter.exit = function() {
+ context.container().select(".curtain-tooltip.intro-mouse").selectAll(".counter").remove();
+ };
+ chapter.restart = function() {
+ chapter.exit();
+ chapter.enter();
+ };
+ return utilRebind(chapter, dispatch10, "on");
+ }
+
+ // modules/ui/intro/navigation.js
+ function uiIntroNavigation(context, reveal) {
+ var dispatch10 = dispatch_default("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 timeout2(f2, t) {
+ timeouts.push(window.setTimeout(f2, t));
+ }
+ function eventCancel(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ }
+ function isTownHallSelected() {
+ var ids = context.selectedIDs();
+ return ids.length === 1 && ids[0] === hallId;
+ }
+ function dragMap() {
+ context.enter(modeBrowse(context));
+ context.history().reset("initial");
+ var msec = transitionTime(townHall, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(townHall, 19, msec);
+ timeout2(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 (centerStart[0] !== centerNow[0] || centerStart[1] !== centerNow[1]) {
+ context.map().on("move.intro", null);
+ timeout2(function() {
+ continueTo(zoomMap);
+ }, 3e3);
+ }
+ });
+ }, msec + 100);
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ nextStep();
+ }
+ }
+ 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);
+ timeout2(function() {
+ continueTo(features);
+ }, 3e3);
+ }
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ nextStep();
+ }
+ }
+ function features() {
+ var onClick = function() {
+ continueTo(pointsLinesAreas);
+ };
+ 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();
+ }
+ }
+ function pointsLinesAreas() {
+ var onClick = function() {
+ continueTo(nodesWays);
+ };
+ 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 }
+ );
+ });
+ function continueTo(nextStep) {
+ context.map().on("drawn.intro", null);
+ nextStep();
+ }
+ }
+ function nodesWays() {
+ var onClick = function() {
+ continueTo(clickTownHall);
+ };
+ 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 }
+ );
+ });
+ function continueTo(nextStep) {
+ context.map().on("drawn.intro", null);
+ nextStep();
+ }
+ }
+ 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);
+ timeout2(function() {
+ var entity2 = context.hasEntity(hallId);
+ if (!entity2)
+ return;
+ var box = pointBox(entity2.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 entity3 = context.hasEntity(hallId);
+ if (!entity3)
+ return;
+ var box2 = pointBox(entity3.loc, context);
+ reveal(box2, helpHtml("intro.navigation." + textId), { duration: 0 });
+ });
+ context.on("enter.intro", function() {
+ if (isTownHallSelected())
+ continueTo(selectedTownHall);
+ });
+ }, 550);
+ context.history().on("change.intro", function() {
+ if (!context.hasEntity(hallId)) {
+ continueTo(clickTownHall);
+ }
+ });
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function selectedTownHall() {
+ if (!isTownHallSelected())
+ return clickTownHall();
+ var entity = context.hasEntity(hallId);
+ if (!entity)
+ return clickTownHall();
+ var box = pointBox(entity.loc, context);
+ var onClick = function() {
+ continueTo(editorTownHall);
+ };
+ reveal(
+ box,
+ helpHtml("intro.navigation.selected_townhall"),
+ { buttonText: _t.html("intro.ok"), buttonCallback: onClick }
+ );
+ context.map().on("move.intro drawn.intro", function() {
+ var entity2 = context.hasEntity(hallId);
+ if (!entity2)
+ return;
+ var box2 = pointBox(entity2.loc, context);
+ reveal(
+ box2,
+ 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 continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function editorTownHall() {
+ if (!isTownHallSelected())
+ return clickTownHall();
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ var onClick = function() {
+ continueTo(presetTownHall);
+ };
+ 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 continueTo(nextStep) {
+ context.on("exit.intro", null);
+ context.history().on("change.intro", null);
+ context.container().select(".inspector-wrap").on("wheel.intro", null);
+ nextStep();
+ }
+ }
+ function presetTownHall() {
+ if (!isTownHallSelected())
+ return clickTownHall();
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ var entity = context.entity(context.selectedIDs()[0]);
+ var preset = _mainPresetIndex.match(entity, context.graph());
+ var onClick = function() {
+ 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);
+ }
+ });
+ 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 fieldsTownHall() {
+ if (!isTownHallSelected())
+ return clickTownHall();
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ var onClick = function() {
+ continueTo(closeTownHall);
+ };
+ 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 continueTo(nextStep) {
+ context.on("exit.intro", null);
+ context.history().on("change.intro", null);
+ context.container().select(".inspector-wrap").on("wheel.intro", null);
+ nextStep();
+ }
+ }
+ function closeTownHall() {
+ if (!isTownHallSelected())
+ return clickTownHall();
+ var selector = ".entity-editor-pane button.close svg use";
+ var href = select_default2(selector).attr("href") || "#iD-icon-close";
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.navigation.close_townhall", { button: { html: icon(href, "inline") } })
+ );
+ context.on("exit.intro", function() {
+ continueTo(searchStreet);
+ });
+ context.history().on("change.intro", function() {
+ var selector2 = ".entity-editor-pane button.close svg use";
+ var href2 = select_default2(selector2).attr("href") || "#iD-icon-close";
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.navigation.close_townhall", { button: { html: icon(href2, "inline") } }),
+ { duration: 0 }
+ );
+ });
+ 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");
+ var msec = transitionTime(springStreet, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(springStreet, 19, msec);
+ timeout2(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);
+ }
+ function checkSearchResult() {
+ var first = context.container().select(".feature-list-item:nth-child(0n+2)");
+ 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 }),
+ { duration: 300 }
+ );
+ context.on("exit.intro", function() {
+ continueTo(selectedStreet);
+ });
+ context.container().select(".search-header input").on("keydown.intro", eventCancel, true).on("keyup.intro", null);
+ }
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ context.container().select(".search-header input").on("keydown.intro", null).on("keyup.intro", null);
+ nextStep();
+ }
+ }
+ function selectedStreet() {
+ if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
+ return searchStreet();
+ }
+ var onClick = function() {
+ continueTo(editorStreet);
+ };
+ 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 }
+ );
+ timeout2(function() {
+ context.map().on("move.intro drawn.intro", function() {
+ var entity2 = context.hasEntity(springStreetEndId);
+ if (!entity2)
+ return;
+ var box2 = pointBox(entity2.loc, context);
+ box2.height = 500;
+ reveal(
+ box2,
+ helpHtml("intro.navigation.selected_street", { name: _t("intro.graph.name.spring-street") }),
+ { duration: 0, buttonText: _t.html("intro.ok"), buttonCallback: onClick }
+ );
+ });
+ }, 600);
+ context.on("enter.intro", function(mode) {
+ if (!context.hasEntity(springStreetId)) {
+ return continueTo(searchStreet);
+ }
+ var ids = context.selectedIDs();
+ if (mode.id !== "select" || !ids.length || ids[0] !== springStreetId) {
+ context.enter(modeSelect(context, [springStreetId]));
+ }
+ });
+ context.history().on("change.intro", function() {
+ if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
+ timeout2(function() {
+ continueTo(searchStreet);
+ }, 300);
+ }
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function editorStreet() {
+ var selector = ".entity-editor-pane button.close svg use";
+ var href = select_default2(selector).attr("href") || "#iD-icon-close";
+ reveal(".entity-editor-pane", helpHtml("intro.navigation.street_different_fields") + "{br}" + helpHtml("intro.navigation.editor_street", {
+ button: { html: icon(href, "inline") },
+ field1: onewayField.title(),
+ field2: maxspeedField.title()
+ }));
+ context.on("exit.intro", function() {
+ continueTo(play);
+ });
+ context.history().on("change.intro", function() {
+ var selector2 = ".entity-editor-pane button.close svg use";
+ var href2 = select_default2(selector2).attr("href") || "#iD-icon-close";
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.navigation.street_different_fields") + "{br}" + helpHtml("intro.navigation.editor_street", {
+ button: { html: icon(href2, "inline") },
+ field1: onewayField.title(),
+ field2: maxspeedField.title()
+ }),
+ { duration: 0 }
+ );
+ });
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function play() {
+ dispatch10.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() {
+ reveal(".ideditor");
+ }
+ }
+ );
+ }
+ chapter.enter = function() {
+ dragMap();
+ };
+ 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);
+ };
+ chapter.restart = function() {
+ chapter.exit();
+ chapter.enter();
+ };
+ return utilRebind(chapter, dispatch10, "on");
+ }
+
+ // modules/ui/intro/point.js
+ function uiIntroPoint(context, reveal) {
+ var dispatch10 = dispatch_default("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"
+ };
+ function timeout2(f2, t) {
+ timeouts.push(window.setTimeout(f2, t));
+ }
+ function eventCancel(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ }
+ function addPoint() {
+ context.enter(modeBrowse(context));
+ context.history().reset("initial");
+ var msec = transitionTime(intersection, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(intersection, 19, msec);
+ timeout2(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);
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function placePoint() {
+ if (context.mode().id !== "add-point") {
+ return chapter.restart();
+ }
+ var pointBox2 = pad(building, 150, context);
+ var textId = context.lastPointerType() === "mouse" ? "place_point" : "place_point_touch";
+ reveal(pointBox2, helpHtml("intro.points." + textId));
+ context.map().on("move.intro drawn.intro", function() {
+ pointBox2 = pad(building, 150, context);
+ reveal(pointBox2, 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();
+ }
+ }
+ function searchPreset() {
+ if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
+ return addPoint();
+ }
+ 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();
+ if (mode.id !== "select" || !ids.length || ids[0] !== _pointID) {
+ context.enter(modeSelect(context, [_pointID]));
+ 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);
+ }
+ });
+ function checkPresetSearch() {
+ var first = context.container().select(".preset-list-item:first-child");
+ 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);
+ });
+ }
+ }
+ 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();
+ }
+ timeout2(function() {
+ reveal(".entity-editor-pane", helpHtml("intro.points.feature_editor"), {
+ tooltipClass: "intro-points-describe",
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ continueTo(addName);
+ }
+ });
+ }, 400);
+ context.on("exit.intro", function() {
+ continueTo(reselectPoint);
+ });
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ function addName() {
+ if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
+ return addPoint();
+ }
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ var addNameString = helpHtml("intro.points.fields_info") + "{br}" + helpHtml("intro.points.add_name");
+ timeout2(function() {
+ var entity = context.entity(_pointID);
+ if (entity.tags.name) {
+ var tooltip = reveal(".entity-editor-pane", addNameString, {
+ tooltipClass: "intro-points-describe",
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ 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() {
+ continueTo(reselectPoint);
+ });
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function addCloseEditor() {
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ var selector = ".entity-editor-pane button.close svg use";
+ var href = select_default2(selector).attr("href") || "#iD-icon-close";
+ context.on("exit.intro", function() {
+ continueTo(reselectPoint);
+ });
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.points.add_close", { button: { html: icon(href, "inline") } })
+ );
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ function reselectPoint() {
+ if (!_pointID)
+ return chapter.restart();
+ var entity = context.hasEntity(_pointID);
+ if (!entity)
+ return chapter.restart();
+ 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 });
+ }
+ context.map().centerEase(entity.loc, msec);
+ timeout2(function() {
+ var box = pointBox(entity.loc, context);
+ reveal(box, helpHtml("intro.points.reselect"), { duration: 600 });
+ timeout2(function() {
+ context.map().on("move.intro drawn.intro", function() {
+ var entity2 = context.hasEntity(_pointID);
+ if (!entity2)
+ return chapter.restart();
+ var box2 = pointBox(entity2.loc, context);
+ reveal(box2, helpHtml("intro.points.reselect"), { duration: 0 });
+ });
+ }, 600);
+ 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();
+ }
+ }
+ function updatePoint() {
+ if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
+ return continueTo(reselectPoint);
+ }
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ context.on("exit.intro", function() {
+ continueTo(reselectPoint);
+ });
+ context.history().on("change.intro", function() {
+ continueTo(updateCloseEditor);
+ });
+ timeout2(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 updateCloseEditor() {
+ if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
+ return continueTo(reselectPoint);
+ }
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ context.on("exit.intro", function() {
+ continueTo(rightClickPoint);
+ });
+ timeout2(function() {
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.points.update_close", { button: { html: icon("#iD-icon-close", "inline") } })
+ );
+ }, 500);
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ 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 });
+ timeout2(function() {
+ context.map().on("move.intro", function() {
+ var entity2 = context.hasEntity(_pointID);
+ if (!entity2)
+ return chapter.restart();
+ var box2 = pointBox(entity2.loc, context);
+ reveal(box2, helpHtml("intro.points." + textId), { duration: 0 });
+ });
+ }, 600);
+ context.on("enter.intro", function(mode) {
+ if (mode.id !== "select")
+ return;
+ var ids = context.selectedIDs();
+ if (ids.length !== 1 || ids[0] !== _pointID)
+ return;
+ timeout2(function() {
+ var node = selectMenuItem(context, "delete").node();
+ if (!node)
+ return;
+ continueTo(enterDelete);
+ }, 50);
+ });
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ context.map().on("move.intro", null);
+ nextStep();
+ }
+ }
+ function enterDelete() {
+ if (!_pointID)
+ return chapter.restart();
+ var entity = context.hasEntity(_pointID);
+ if (!entity)
+ return chapter.restart();
+ var node = selectMenuItem(context, "delete").node();
+ if (!node) {
+ return continueTo(rightClickPoint);
+ }
+ reveal(
+ ".edit-menu",
+ helpHtml("intro.points.delete"),
+ { padding: 50 }
+ );
+ timeout2(function() {
+ context.map().on("move.intro", function() {
+ reveal(
+ ".edit-menu",
+ helpHtml("intro.points.delete"),
+ { duration: 0, padding: 50 }
+ );
+ });
+ }, 300);
+ context.on("exit.intro", function() {
+ if (!_pointID)
+ return chapter.restart();
+ var entity2 = context.hasEntity(_pointID);
+ if (entity2)
+ return continueTo(rightClickPoint);
+ });
+ 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();
+ }
+ }
+ function undo() {
+ context.history().on("change.intro", function() {
+ continueTo(play);
+ });
+ reveal(
+ ".top-toolbar button.undo-button",
+ helpHtml("intro.points.undo")
+ );
+ function continueTo(nextStep) {
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function play() {
+ dispatch10.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() {
+ reveal(".ideditor");
+ }
+ }
+ );
+ }
+ chapter.enter = function() {
+ addPoint();
+ };
+ 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();
+ };
+ return utilRebind(chapter, dispatch10, "on");
+ }
+
+ // modules/ui/intro/area.js
+ function uiIntroArea(context, reveal) {
+ var dispatch10 = dispatch_default("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 = [];
+ var _areaID;
+ var chapter = {
+ title: "intro.areas.title"
+ };
+ function timeout2(f2, t) {
+ timeouts.push(window.setTimeout(f2, t));
+ }
+ function eventCancel(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ }
+ function revealPlayground(center, text2, options2) {
+ var padding = 180 * Math.pow(2, context.map().zoom() - 19.5);
+ var box = pad(center, padding, context);
+ reveal(box, text2, options2);
+ }
+ function addArea() {
+ context.enter(modeBrowse(context));
+ context.history().reset("initial");
+ _areaID = null;
+ var msec = transitionTime(playground, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(playground, 19, msec);
+ timeout2(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);
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function startPlayground() {
+ if (context.mode().id !== "add-area") {
+ return chapter.restart();
+ }
+ _areaID = null;
+ context.map().zoomEase(19.5, 500);
+ timeout2(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 }
+ );
+ timeout2(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);
+ }, 550);
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function continuePlayground() {
+ if (context.mode().id !== "draw-area") {
+ return chapter.restart();
+ }
+ _areaID = null;
+ revealPlayground(
+ playground,
+ helpHtml("intro.areas.continue_playground"),
+ { duration: 250 }
+ );
+ timeout2(function() {
+ context.map().on("move.intro drawn.intro", function() {
+ revealPlayground(
+ playground,
+ helpHtml("intro.areas.continue_playground"),
+ { duration: 0 }
+ );
+ });
+ }, 250);
+ context.on("enter.intro", function(mode) {
+ if (mode.id === "draw-area") {
+ var entity = context.hasEntity(context.selectedIDs()[0]);
+ 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();
+ }
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ 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 }
+ );
+ timeout2(function() {
+ context.map().on("move.intro drawn.intro", function() {
+ revealPlayground(
+ playground,
+ finishString,
+ { duration: 0 }
+ );
+ });
+ }, 250);
+ 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();
+ }
+ }
+ function searchPresets() {
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return addArea();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _areaID) {
+ context.enter(modeSelect(context, [_areaID]));
+ }
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ timeout2(function() {
+ 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);
+ context.on("enter.intro", function(mode) {
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return continueTo(addArea);
+ }
+ var ids2 = context.selectedIDs();
+ if (mode.id !== "select" || !ids2.length || ids2[0] !== _areaID) {
+ context.enter(modeSelect(context, [_areaID]));
+ context.container().select(".inspector-wrap .panewrap").style("right", "-100%");
+ 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);
+ }
+ });
+ 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);
+ });
+ }
+ }
+ 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();
+ }
+ }
+ function clickAddField() {
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return addArea();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _areaID) {
+ return searchPresets();
+ }
+ if (!context.container().select(".form-field-description").empty()) {
+ return continueTo(describePlayground);
+ }
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ timeout2(function() {
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ var entity = context.entity(_areaID);
+ if (entity.tags.description) {
+ return continueTo(play);
+ }
+ var box = context.container().select(".more-fields").node().getBoundingClientRect();
+ if (box.top > 300) {
+ var pane = context.container().select(".entity-editor-pane .inspector-body");
+ var start2 = pane.node().scrollTop;
+ var end = start2 + (box.top - 300);
+ pane.transition().duration(250).tween("scroll.inspector", function() {
+ var node = this;
+ var i2 = number_default(start2, end);
+ return function(t) {
+ node.scrollTop = i2(t);
+ };
+ });
+ }
+ timeout2(function() {
+ reveal(
+ ".more-fields .combobox-input",
+ helpHtml("intro.areas.add_field", {
+ name: nameField.title(),
+ description: descriptionField.title()
+ }),
+ { duration: 300 }
+ );
+ context.container().select(".more-fields .combobox-input").on("click.intro", function() {
+ var watcher;
+ watcher = window.setInterval(function() {
+ if (!context.container().select("div.combobox").empty()) {
+ window.clearInterval(watcher);
+ continueTo(chooseDescriptionField);
+ }
+ }, 300);
+ });
+ }, 300);
+ }, 400);
+ context.on("exit.intro", function() {
+ return continueTo(searchPresets);
+ });
+ 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();
+ }
+ }
+ function chooseDescriptionField() {
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return addArea();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _areaID) {
+ return searchPresets();
+ }
+ if (!context.container().select(".form-field-description").empty()) {
+ return continueTo(describePlayground);
+ }
+ if (context.container().select("div.combobox").empty()) {
+ return continueTo(clickAddField);
+ }
+ var watcher;
+ watcher = window.setInterval(function() {
+ if (context.container().select("div.combobox").empty()) {
+ window.clearInterval(watcher);
+ timeout2(function() {
+ if (context.container().select(".form-field-description").empty()) {
+ continueTo(retryChooseDescription);
+ } else {
+ continueTo(describePlayground);
+ }
+ }, 300);
+ }
+ }, 300);
+ reveal(
+ "div.combobox",
+ helpHtml("intro.areas.choose_field", { field: descriptionField.title() }),
+ { 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();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _areaID) {
+ return searchPresets();
+ }
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ if (context.container().select(".form-field-description").empty()) {
+ return continueTo(retryChooseDescription);
+ }
+ context.on("exit.intro", function() {
+ continueTo(play);
+ });
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.areas.describe_playground", { button: { html: icon("#iD-icon-close", "inline") } }),
+ { duration: 300 }
+ );
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ function retryChooseDescription() {
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return addArea();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _areaID) {
+ return searchPresets();
+ }
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.areas.retry_add_field", { field: descriptionField.title() }),
+ {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ continueTo(clickAddField);
+ }
+ }
+ );
+ context.on("exit.intro", function() {
+ return continueTo(searchPresets);
+ });
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ function play() {
+ dispatch10.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() {
+ reveal(".ideditor");
+ }
+ }
+ );
+ }
+ chapter.enter = function() {
+ addArea();
+ };
+ 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();
+ };
+ return utilRebind(chapter, dispatch10, "on");
+ }
+
+ // modules/ui/intro/line.js
+ function uiIntroLine(context, reveal) {
+ var dispatch10 = dispatch_default("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 timeout2(f2, t) {
+ timeouts.push(window.setTimeout(f2, t));
+ }
+ function eventCancel(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ }
+ function addLine() {
+ context.enter(modeBrowse(context));
+ context.history().reset("initial");
+ var msec = transitionTime(tulipRoadStart, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(tulipRoadStart, 18.5, msec);
+ timeout2(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 });
+ });
+ 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();
+ }
+ }
+ function drawLine() {
+ if (context.mode().id !== "draw-line")
+ return chapter.restart();
+ _tulipRoadID = context.mode().selectedIDs()[0];
+ context.map().centerEase(tulipRoadMidpoint, 500);
+ timeout2(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);
+ 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();
+ }
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function isLineConnected() {
+ var entity = _tulipRoadID && context.hasEntity(_tulipRoadID);
+ 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 retryIntersect() {
+ select_default2(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") })
+ );
+ timeout2(chapter.restart, 3e3);
+ }
+ function continueLine() {
+ if (context.mode().id !== "draw-line")
+ return chapter.restart();
+ var entity = _tulipRoadID && context.hasEntity(_tulipRoadID);
+ 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();
+ }
+ }
+ 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();
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ timeout2(function() {
+ 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);
+ 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 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);
+ });
+ timeout2(function() {
+ reveal(
+ subgrid.node(),
+ helpHtml("intro.lines.choose_preset_residential", { preset: residentialPreset.name() }),
+ { tooltipBox: ".preset-highway-residential .preset-list-button", duration: 300 }
+ );
+ }, 300);
+ function continueTo(nextStep) {
+ context.container().select(".preset-list-button").on("click.intro", null);
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ function retryPresetResidential() {
+ if (context.mode().id !== "select")
+ return chapter.restart();
+ context.on("exit.intro", function() {
+ return chapter.restart();
+ });
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ timeout2(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 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);
+ });
+ timeout2(function() {
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.lines.name_road", { button: { html: icon("#iD-icon-close", "inline") } }),
+ { tooltipClass: "intro-lines-name_road" }
+ );
+ }, 500);
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ function didNameRoad() {
+ context.history().checkpoint("doneAddLine");
+ timeout2(function() {
+ reveal(".surface", helpHtml("intro.lines.did_name_road"), {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ continueTo(updateLine);
+ }
+ });
+ }, 500);
+ function continueTo(nextStep) {
+ nextStep();
+ }
+ }
+ function updateLine() {
+ context.history().reset("doneAddLine");
+ if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+ return chapter.restart();
+ }
+ var msec = transitionTime(woodRoadDragMidpoint, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(woodRoadDragMidpoint, 19, msec);
+ timeout2(function() {
+ var padding = 250 * Math.pow(2, context.map().zoom() - 19);
+ var box = pad(woodRoadDragMidpoint, padding, context);
+ var advance = function() {
+ continueTo(addNode);
+ };
+ reveal(
+ box,
+ helpHtml("intro.lines.update_line"),
+ { buttonText: _t.html("intro.ok"), buttonCallback: advance }
+ );
+ context.map().on("move.intro drawn.intro", function() {
+ var padding2 = 250 * Math.pow(2, context.map().zoom() - 19);
+ var box2 = pad(woodRoadDragMidpoint, padding2, context);
+ reveal(
+ box2,
+ helpHtml("intro.lines.update_line"),
+ { duration: 0, buttonText: _t.html("intro.ok"), buttonCallback: advance }
+ );
+ });
+ }, msec + 100);
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ nextStep();
+ }
+ }
+ function addNode() {
+ context.history().reset("doneAddLine");
+ if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+ return chapter.restart();
+ }
+ 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 padding2 = 40 * Math.pow(2, context.map().zoom() - 19);
+ var box2 = pad(woodRoadAddNode, padding2, context);
+ reveal(box2, addNodeString, { duration: 0 });
+ });
+ context.history().on("change.intro", function(changed) {
+ if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+ return continueTo(updateLine);
+ }
+ if (changed.created().length === 1) {
+ timeout2(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();
+ }
+ }
+ function startDragEndpoint() {
+ 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);
+ 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);
+ }
+ var padding2 = 100 * Math.pow(2, context.map().zoom() - 19);
+ var box2 = pad(woodRoadDragEndpoint, padding2, context);
+ reveal(box2, startDragString, { duration: 0 });
+ var entity = context.entity(woodRoadEndID);
+ if (geoSphericalDistance(entity.loc, woodRoadDragEndpoint) <= 4) {
+ continueTo(finishDragEndpoint);
+ }
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ nextStep();
+ }
+ }
+ function finishDragEndpoint() {
+ 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);
+ 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 padding2 = 100 * Math.pow(2, context.map().zoom() - 19);
+ var box2 = pad(woodRoadDragEndpoint, padding2, context);
+ reveal(box2, finishDragString, { duration: 0 });
+ var entity = context.entity(woodRoadEndID);
+ if (geoSphericalDistance(entity.loc, woodRoadDragEndpoint) > 4) {
+ continueTo(startDragEndpoint);
+ }
+ });
+ context.on("enter.intro", function() {
+ continueTo(startDragMidpoint);
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function startDragMidpoint() {
+ if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+ return continueTo(updateLine);
+ }
+ if (context.selectedIDs().indexOf(woodRoadID) === -1) {
+ context.enter(modeSelect(context, [woodRoadID]));
+ }
+ 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);
+ }
+ var padding2 = 80 * Math.pow(2, context.map().zoom() - 19);
+ var box2 = pad(woodRoadDragMidpoint, padding2, context);
+ reveal(box2, 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") {
+ context.enter(modeSelect(context, [woodRoadID]));
+ }
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function continueDragMidpoint() {
+ 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;
+ var advance = function() {
+ context.history().checkpoint("doneUpdateLine");
+ continueTo(deleteLines);
+ };
+ 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 padding2 = 100 * Math.pow(2, context.map().zoom() - 19);
+ var box2 = pad(woodRoadDragEndpoint, padding2, context);
+ box2.height += 400;
+ reveal(
+ box2,
+ 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();
+ }
+ }
+ function deleteLines() {
+ context.history().reset("doneUpdateLine");
+ context.enter(modeBrowse(context));
+ if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return chapter.restart();
+ }
+ var msec = transitionTime(deleteLinesLoc, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(deleteLinesLoc, 18, msec);
+ timeout2(function() {
+ var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+ var box = pad(deleteLinesLoc, padding, context);
+ box.top -= 200;
+ box.height += 400;
+ var advance = function() {
+ continueTo(rightClickIntersection);
+ };
+ 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 padding2 = 200 * Math.pow(2, context.map().zoom() - 18);
+ var box2 = pad(deleteLinesLoc, padding2, context);
+ box2.top -= 200;
+ box2.height += 400;
+ reveal(
+ box2,
+ 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() {
+ timeout2(function() {
+ continueTo(deleteLines);
+ }, 500);
+ });
+ }, msec + 100);
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ 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"));
+ timeout2(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 padding2 = 60 * Math.pow(2, context.map().zoom() - 18);
+ var box2 = pad(eleventhAvenueEnd, padding2, context);
+ reveal(
+ box2,
+ 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;
+ timeout2(function() {
+ var node = selectMenuItem(context, "split").node();
+ if (!node)
+ return;
+ continueTo(splitIntersection);
+ }, 50);
+ });
+ context.history().on("change.intro", function() {
+ timeout2(function() {
+ continueTo(deleteLines);
+ }, 300);
+ });
+ }, 600);
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function splitIntersection() {
+ if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(deleteLines);
+ }
+ var node = selectMenuItem(context, "split").node();
+ if (!node) {
+ return continueTo(rightClickIntersection);
+ }
+ 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 node2 = selectMenuItem(context, "split").node();
+ if (!wasChanged && !node2) {
+ return continueTo(rightClickIntersection);
+ }
+ 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;
+ timeout2(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);
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function retrySplit() {
+ context.enter(modeBrowse(context));
+ context.map().centerZoomEase(eleventhAvenueEnd, 18, 500);
+ var advance = function() {
+ continueTo(rightClickIntersection);
+ };
+ 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 padding2 = 60 * Math.pow(2, context.map().zoom() - 18);
+ var box2 = pad(eleventhAvenueEnd, padding2, context);
+ reveal(
+ box2,
+ helpHtml("intro.lines.retry_split"),
+ { duration: 0, buttonText: _t.html("intro.ok"), buttonCallback: advance }
+ );
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ nextStep();
+ }
+ }
+ function didSplit() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
+ }
+ 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 }
+ );
+ timeout2(function() {
+ context.map().centerZoomEase(twelfthAvenue, 18, 500);
+ context.map().on("move.intro drawn.intro", function() {
+ var padding2 = 200 * Math.pow(2, context.map().zoom() - 18);
+ var box2 = pad(twelfthAvenue, padding2, context);
+ box2.width = box2.width / 2;
+ reveal(
+ box2,
+ helpHtml(string, { street1: street, street2: street }),
+ { duration: 0 }
+ );
+ });
+ }, 600);
+ context.on("enter.intro", function() {
+ var ids2 = context.selectedIDs();
+ if (ids2.length === 1 && ids2[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 continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function multiSelect() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
+ }
+ var ids = context.selectedIDs();
+ var hasWashington = ids.indexOf(_washingtonSegmentID) !== -1;
+ var hasTwelfth = ids.indexOf(twelfthAvenueID) !== -1;
+ if (hasWashington && hasTwelfth) {
+ return continueTo(multiRightClick);
+ } else if (!hasWashington && !hasTwelfth) {
+ return continueTo(didSplit);
+ }
+ context.map().centerZoomEase(twelfthAvenue, 18, 500);
+ timeout2(function() {
+ var selected, other, padding, box;
+ 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;
+ }
+ reveal(
+ box,
+ helpHtml(
+ "intro.lines.multi_select",
+ { selected, other1: other }
+ ) + " " + helpHtml(
+ "intro.lines.add_to_selection_" + (context.lastPointerType() === "mouse" ? "click" : "touch"),
+ { 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;
+ }
+ reveal(
+ box,
+ helpHtml(
+ "intro.lines.multi_select",
+ { selected, other1: other }
+ ) + " " + helpHtml(
+ "intro.lines.add_to_selection_" + (context.lastPointerType() === "mouse" ? "click" : "touch"),
+ { 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);
+ }
+ });
+ }, 600);
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ 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 padding2 = 200 * Math.pow(2, context.map().zoom() - 18);
+ var box2 = pad(twelfthAvenue, padding2, context);
+ reveal(box2, rightClickString, { duration: 0 });
+ });
+ context.ui().editMenu().on("toggled.intro", function(open) {
+ if (!open)
+ return;
+ timeout2(function() {
+ var ids = context.selectedIDs();
+ 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);
+ });
+ context.history().on("change.intro", function() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
+ }
+ });
+ 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 multiDelete() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
+ }
+ 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);
+ }
+ });
+ context.history().on("change.intro", function() {
+ if (context.hasEntity(_washingtonSegmentID) || context.hasEntity(twelfthAvenueID)) {
+ continueTo(retryDelete);
+ } else {
+ continueTo(play);
+ }
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("exit.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ 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() {
+ continueTo(multiSelect);
+ }
+ });
+ function continueTo(nextStep) {
+ nextStep();
+ }
+ }
+ function play() {
+ dispatch10.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() {
+ reveal(".ideditor");
+ }
+ }
+ );
+ }
+ chapter.enter = function() {
+ addLine();
+ };
+ chapter.exit = function() {
+ timeouts.forEach(window.clearTimeout);
+ select_default2(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();
+ };
+ return utilRebind(chapter, dispatch10, "on");
+ }
+
+ // modules/ui/intro/building.js
+ function uiIntroBuilding(context, reveal) {
+ var dispatch10 = dispatch_default("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 timeout2(f2, t) {
+ timeouts.push(window.setTimeout(f2, t));
+ }
+ function eventCancel(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ }
+ function revealHouse(center, text2, options2) {
+ var padding = 160 * Math.pow(2, context.map().zoom() - 20);
+ var box = pad(center, padding, context);
+ reveal(box, text2, options2);
+ }
+ function revealTank(center, text2, options2) {
+ var padding = 190 * Math.pow(2, context.map().zoom() - 19.5);
+ var box = pad(center, padding, context);
+ reveal(box, text2, options2);
+ }
+ function addHouse() {
+ context.enter(modeBrowse(context));
+ context.history().reset("initial");
+ _houseID = null;
+ var msec = transitionTime(house, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(house, 19, msec);
+ timeout2(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);
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function startHouse() {
+ if (context.mode().id !== "add-area") {
+ return continueTo(addHouse);
+ }
+ _houseID = null;
+ context.map().zoomEase(20, 500);
+ timeout2(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);
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function continueHouse() {
+ if (context.mode().id !== "draw-area") {
+ return continueTo(addHouse);
+ }
+ _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(n2) {
+ return context.projection(n2.loc);
+ });
+ if (isMostlySquare(points)) {
+ _houseID = way.id;
+ return continueTo(chooseCategoryBuilding);
+ } else {
+ return continueTo(retryHouse);
+ }
+ } else {
+ return chapter.restart();
+ }
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function retryHouse() {
+ var onClick = function() {
+ continueTo(addHouse);
+ };
+ 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 }
+ );
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ nextStep();
+ }
+ }
+ function chooseCategoryBuilding() {
+ if (!_houseID || !context.hasEntity(_houseID)) {
+ return addHouse();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _houseID) {
+ context.enter(modeSelect(context, [_houseID]));
+ }
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ timeout2(function() {
+ 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);
+ context.on("enter.intro", function(mode) {
+ if (!_houseID || !context.hasEntity(_houseID)) {
+ return continueTo(addHouse);
+ }
+ var ids2 = context.selectedIDs();
+ if (mode.id !== "select" || !ids2.length || ids2[0] !== _houseID) {
+ return continueTo(chooseCategoryBuilding);
+ }
+ });
+ 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 choosePresetHouse() {
+ if (!_houseID || !context.hasEntity(_houseID)) {
+ return addHouse();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _houseID) {
+ context.enter(modeSelect(context, [_houseID]));
+ }
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ timeout2(function() {
+ 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);
+ context.on("enter.intro", function(mode) {
+ if (!_houseID || !context.hasEntity(_houseID)) {
+ return continueTo(addHouse);
+ }
+ var ids2 = context.selectedIDs();
+ if (mode.id !== "select" || !ids2.length || ids2[0] !== _houseID) {
+ return continueTo(chooseCategoryBuilding);
+ }
+ });
+ 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();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _houseID) {
+ context.enter(modeSelect(context, [_houseID]));
+ }
+ context.history().checkpoint("hasHouse");
+ context.on("exit.intro", function() {
+ continueTo(rightClickHouse);
+ });
+ timeout2(function() {
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.buildings.close", { button: { html: icon("#iD-icon-close", "inline") } })
+ );
+ }, 500);
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ function rightClickHouse() {
+ if (!_houseID)
+ return chapter.restart();
+ context.enter(modeBrowse(context));
+ context.history().reset("hasHouse");
+ var zoom = context.map().zoom();
+ if (zoom < 20) {
+ zoom = 20;
+ }
+ 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;
+ timeout2(function() {
+ var node = selectMenuItem(context, "orthogonalize").node();
+ if (!node)
+ return;
+ continueTo(clickSquare);
+ }, 50);
+ });
+ 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();
+ }
+ }
+ function clickSquare() {
+ if (!_houseID)
+ return chapter.restart();
+ var entity = context.hasEntity(_houseID);
+ if (!entity)
+ return continueTo(rightClickHouse);
+ var node = selectMenuItem(context, "orthogonalize").node();
+ if (!node) {
+ return continueTo(rightClickHouse);
+ }
+ 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 node2 = selectMenuItem(context, "orthogonalize").node();
+ if (!wasChanged && !node2) {
+ return continueTo(rightClickHouse);
+ }
+ 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);
+ timeout2(function() {
+ if (context.history().undoAnnotation() === _t("operations.orthogonalize.annotation.feature", { n: 1 })) {
+ continueTo(doneSquare);
+ } else {
+ continueTo(retryClickSquare);
+ }
+ }, 500);
+ });
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ context.map().on("move.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function retryClickSquare() {
+ context.enter(modeBrowse(context));
+ revealHouse(house, helpHtml("intro.buildings.retry_square"), {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ continueTo(rightClickHouse);
+ }
+ });
+ function continueTo(nextStep) {
+ nextStep();
+ }
+ }
+ function doneSquare() {
+ context.history().checkpoint("doneSquare");
+ revealHouse(house, helpHtml("intro.buildings.done_square"), {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ continueTo(addTank);
+ }
+ });
+ function continueTo(nextStep) {
+ nextStep();
+ }
+ }
+ function addTank() {
+ context.enter(modeBrowse(context));
+ context.history().reset("doneSquare");
+ _tankID = null;
+ var msec = transitionTime(tank, context.map().center());
+ if (msec) {
+ reveal(null, null, { duration: 0 });
+ }
+ context.map().centerZoomEase(tank, 19.5, msec);
+ timeout2(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 startTank() {
+ if (context.mode().id !== "add-area") {
+ return continueTo(addTank);
+ }
+ _tankID = null;
+ timeout2(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);
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
+ }
+ function continueTank() {
+ if (context.mode().id !== "draw-area") {
+ return continueTo(addTank);
+ }
+ _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);
+ }
+ });
+ 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 ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _tankID) {
+ context.enter(modeSelect(context, [_tankID]));
+ }
+ context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ timeout2(function() {
+ 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);
+ context.on("enter.intro", function(mode) {
+ if (!_tankID || !context.hasEntity(_tankID)) {
+ return continueTo(addTank);
+ }
+ var ids2 = context.selectedIDs();
+ if (mode.id !== "select" || !ids2.length || ids2[0] !== _tankID) {
+ context.enter(modeSelect(context, [_tankID]));
+ context.container().select(".inspector-wrap .panewrap").style("right", "-100%");
+ 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 checkPresetSearch() {
+ var first = context.container().select(".preset-list-item:first-child");
+ 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();
+ }
+ }
+ function closeEditorTank() {
+ if (!_tankID || !context.hasEntity(_tankID)) {
+ return addTank();
+ }
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _tankID) {
+ context.enter(modeSelect(context, [_tankID]));
+ }
+ context.history().checkpoint("hasTank");
+ context.on("exit.intro", function() {
+ continueTo(rightClickTank);
+ });
+ timeout2(function() {
+ reveal(
+ ".entity-editor-pane",
+ helpHtml("intro.buildings.close", { button: { html: icon("#iD-icon-close", "inline") } })
+ );
+ }, 500);
+ function continueTo(nextStep) {
+ context.on("exit.intro", null);
+ nextStep();
+ }
+ }
+ function rightClickTank() {
+ if (!_tankID)
+ return continueTo(addTank);
+ context.enter(modeBrowse(context));
+ context.history().reset("hasTank");
+ context.map().centerEase(tank, 500);
+ timeout2(function() {
+ context.on("enter.intro", function(mode) {
+ if (mode.id !== "select")
+ return;
+ var ids = context.selectedIDs();
+ if (ids.length !== 1 || ids[0] !== _tankID)
+ return;
+ timeout2(function() {
+ var node = selectMenuItem(context, "circularize").node();
+ if (!node)
+ return;
+ continueTo(clickCircle);
+ }, 50);
+ });
+ 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);
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function clickCircle() {
+ if (!_tankID)
+ return chapter.restart();
+ var entity = context.hasEntity(_tankID);
+ if (!entity)
+ return continueTo(rightClickTank);
+ var node = selectMenuItem(context, "circularize").node();
+ 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 node2 = selectMenuItem(context, "circularize").node();
+ if (!wasChanged && !node2) {
+ return continueTo(rightClickTank);
+ }
+ 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);
+ timeout2(function() {
+ if (context.history().undoAnnotation() === _t("operations.circularize.annotation.feature", { n: 1 })) {
+ continueTo(play);
+ } else {
+ continueTo(retryClickCircle);
+ }
+ }, 500);
+ });
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ context.map().on("move.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
+ }
+ }
+ function retryClickCircle() {
+ context.enter(modeBrowse(context));
+ revealTank(tank, helpHtml("intro.buildings.retry_circle"), {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ continueTo(rightClickTank);
+ }
+ });
+ function continueTo(nextStep) {
+ nextStep();
+ }
+ }
+ function play() {
+ dispatch10.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() {
+ reveal(".ideditor");
+ }
+ }
+ );
+ }
+ chapter.enter = function() {
+ addHouse();
+ };
+ 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();
+ };
+ return utilRebind(chapter, dispatch10, "on");
+ }
+
+ // modules/ui/intro/start_editing.js
+ function uiIntroStartEditing(context, reveal) {
+ var dispatch10 = dispatch_default("done", "startEditing");
+ var modalSelection = select_default2(null);
+ var chapter = {
+ title: "intro.startediting.title"
+ };
+ function showHelp() {
+ reveal(
+ ".map-control.help-control",
+ helpHtml("intro.startediting.help"),
+ {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ shortcuts();
+ }
+ }
+ );
+ }
+ function shortcuts() {
+ reveal(
+ ".map-control.help-control",
+ helpHtml("intro.startediting.shortcuts"),
+ {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ showSave();
+ }
+ }
+ );
+ }
+ function showSave() {
+ context.container().selectAll(".shaded").remove();
+ reveal(
+ ".top-toolbar button.save",
+ helpHtml("intro.startediting.save"),
+ {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ showStart();
+ }
+ }
+ );
+ }
+ function showStart() {
+ context.container().selectAll(".shaded").remove();
+ 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").call(_t.append("intro.startediting.start"));
+ dispatch10.call("startEditing");
+ }
+ chapter.enter = function() {
+ showHelp();
+ };
+ chapter.exit = function() {
+ modalSelection.remove();
+ context.container().selectAll(".shaded").remove();
+ };
+ return utilRebind(chapter, dispatch10, "on");
+ }
+
+ // modules/ui/intro/intro.js
+ 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) {
+ const INTRO_IMAGERY = "EsriWorldImageryClarity";
+ let _introGraph = {};
+ let _currChapter;
+ function intro(selection2) {
+ _mainFileFetcher.get("intro_graph").then((dataIntroGraph) => {
+ for (let id2 in dataIntroGraph) {
+ if (!_introGraph[id2]) {
+ _introGraph[id2] = osmEntity(localize(dataIntroGraph[id2]));
+ }
+ }
+ selection2.call(startIntro);
+ }).catch(function() {
+ });
+ }
+ function startIntro(selection2) {
+ context.enter(modeBrowse(context));
+ let osm = context.connection();
+ let history = context.history().toJSON();
+ let hash = window.location.hash;
+ let center = context.map().center();
+ let zoom = context.map().zoom();
+ let background = context.background().baseLayerSource();
+ let overlays = context.background().overlayLayerSources();
+ let opacity = context.container().selectAll(".main-map .layer-background").style("opacity");
+ let caches = osm && osm.caches();
+ let baseEntities = context.history().graph().base().entities;
+ context.ui().sidebar.expand();
+ context.container().selectAll("button.sidebar-toggle").classed("disabled", true);
+ context.inIntro(true);
+ if (osm) {
+ osm.toggle(false).reset();
+ }
+ context.history().reset();
+ context.history().merge(Object.values(coreGraph().load(_introGraph).entities));
+ context.history().checkpoint("initial");
+ let imagery = context.background().findSource(INTRO_IMAGERY);
+ if (imagery) {
+ context.background().baseLayerSource(imagery);
+ } else {
+ context.background().bing();
+ }
+ overlays.forEach((d) => context.background().toggleOverlayLayer(d));
+ let layers = context.layers();
+ layers.all().forEach((item) => {
+ if (typeof item.layer.enabled === "function") {
+ item.layer.enabled(item.id === "osm");
+ }
+ });
+ context.container().selectAll(".main-map .layer-background").style("opacity", 1);
+ let curtain = uiCurtain(context.container().node());
+ selection2.call(curtain);
+ corePreferences("walkthrough_started", "yes");
+ let storedProgress = corePreferences("walkthrough_progress") || "";
+ let progress = storedProgress.split(";").filter(Boolean);
+ let chapters = chapterFlow.map((chapter, i2) => {
+ let s = chapterUi[chapter](context, curtain.reveal).on("done", () => {
+ buttons.filter((d) => d.title === s.title).classed("finished", true);
+ if (i2 < chapterFlow.length - 1) {
+ const next = chapterFlow[i2 + 1];
+ context.container().select(`button.chapter-${next}`).classed("next", true);
+ }
+ progress.push(chapter);
+ corePreferences("walkthrough_progress", utilArrayUniq(progress).join(";"));
+ });
+ return s;
+ });
+ chapters[chapters.length - 1].on("startEditing", () => {
+ progress.push("startEditing");
+ corePreferences("walkthrough_progress", utilArrayUniq(progress).join(";"));
+ let incomplete = utilArrayDifference(chapterFlow, progress);
+ if (!incomplete.length) {
+ corePreferences("walkthrough_completed", "yes");
+ }
+ curtain.remove();
+ navwrap.remove();
+ context.container().selectAll(".main-map .layer-background").style("opacity", opacity);
+ context.container().selectAll("button.sidebar-toggle").classed("disabled", false);
+ if (osm) {
+ osm.toggle(true).reset().caches(caches);
+ }
+ context.history().reset().merge(Object.values(baseEntities));
+ context.background().baseLayerSource(background);
+ overlays.forEach((d) => context.background().toggleOverlayLayer(d));
+ if (history) {
+ context.history().fromJSON(history, false);
+ }
+ context.map().centerZoom(center, zoom);
+ window.location.replace(hash);
+ context.inIntro(false);
+ });
+ let navwrap = selection2.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");
+ let buttonwrap = navwrap.append("div").attr("class", "joined").selectAll("button.chapter");
+ let buttons = buttonwrap.data(chapters).enter().append("button").attr("class", (d, i2) => `chapter chapter-${chapterFlow[i2]}`).on("click", enterChapter);
+ buttons.append("span").html((d) => _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();
+ }
+ context.enter(modeBrowse(context));
+ _currChapter = newChapter;
+ _currChapter.enter();
+ buttons.classed("next", false).classed("active", (d) => d.title === _currChapter.title);
+ }
+ }
+ return intro;
+ }
+
+ // modules/ui/issues_info.js
+ 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(selection2) {
+ var shownItems = [];
+ var liveIssues = context.validator().getIssues({
+ what: corePreferences("validate-what") || "edited",
+ where: corePreferences("validate-where") || "all"
+ });
+ if (liveIssues.length) {
+ warningsItem.count = liveIssues.length;
+ shownItems.push(warningsItem);
+ }
+ if (corePreferences("validate-what") === "all") {
+ var resolvedIssues = context.validator().getResolvedIssues();
+ if (resolvedIssues.length) {
+ resolvedItem.count = resolvedIssues.length;
+ shownItems.push(resolvedItem);
+ }
+ }
+ var chips = selection2.selectAll(".chip").data(shownItems, function(d) {
+ return d.id;
+ });
+ 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_default2(this);
+ var tooltipBehavior = uiTooltip().placement("top").title(() => _t.append(d.descriptionID));
+ chipSelection.call(tooltipBehavior).on("click", function(d3_event) {
+ d3_event.preventDefault();
+ tooltipBehavior.hide(select_default2(this));
+ 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").text(function(d) {
+ return d.count.toString();
+ });
+ }
+ return function(selection2) {
+ update(selection2);
+ context.validator().on("validated.infobox", function() {
+ update(selection2);
+ });
+ };
+ }
+
+ // modules/ui/map_in_map.js
+ function uiMapInMap(context) {
+ function mapInMap(selection2) {
+ var backgroundLayer = rendererTileLayer(context);
+ var overlayLayers = {};
+ var projection2 = geoRawMercator();
+ var dataLayer = svgData(projection2, context).showLabels(false);
+ var debugLayer = svgDebug(projection2, context);
+ var zoom = zoom_default2().scaleExtent([geoZoomToScale(0.5), geoZoomToScale(24)]).on("start", zoomStarted).on("zoom", zoomed).on("end", zoomEnded);
+ var wrap2 = select_default2(null);
+ var tiles = select_default2(null);
+ var viewport = select_default2(null);
+ var _isTransformed = false;
+ var _isHidden = true;
+ var _skipEvents = false;
+ var _gesture = null;
+ var _zDiff = 6;
+ var _dMini;
+ var _cMini;
+ var _tStart;
+ var _tCurr;
+ var _timeoutID;
+ function zoomStarted() {
+ if (_skipEvents)
+ return;
+ _tStart = _tCurr = projection2.transform();
+ _gesture = null;
+ }
+ 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;
+ if (!isZooming && !isPanning) {
+ return;
+ }
+ if (!_gesture) {
+ _gesture = isZooming ? "zoom" : "pan";
+ }
+ var tMini = projection2.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;
+ }
+ utilSetTransform(tiles, tX, tY, scale);
+ utilSetTransform(viewport, 0, 0, scale);
+ _isTransformed = true;
+ _tCurr = identity2.translate(x, y).scale(k);
+ var zMain = geoScaleToZoom(context.projection.scale());
+ var zMini = geoScaleToZoom(k);
+ _zDiff = zMain - zMini;
+ queueRedraw();
+ }
+ function zoomEnded() {
+ if (_skipEvents)
+ return;
+ if (_gesture !== "pan")
+ return;
+ updateProjection();
+ _gesture = null;
+ context.map().center(projection2.invert(_cMini));
+ }
+ 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);
+ projection2.translate([tMain.x, tMain.y]).scale(kMini);
+ var point2 = projection2(loc);
+ var mouse = _gesture === "pan" ? geoVecSubtract([_tCurr.x, _tCurr.y], [_tStart.x, _tStart.y]) : [0, 0];
+ var xMini = _cMini[0] - point2[0] + tMain.x + mouse[0];
+ var yMini = _cMini[1] - point2[1] + tMain.y + mouse[1];
+ projection2.translate([xMini, yMini]).clipExtent([[0, 0], _dMini]);
+ _tCurr = projection2.transform();
+ if (_isTransformed) {
+ utilSetTransform(tiles, 0, 0);
+ utilSetTransform(viewport, 0, 0);
+ _isTransformed = false;
+ }
+ zoom.scaleExtent([geoZoomToScale(0.5), geoZoomToScale(zMain - 3)]);
+ _skipEvents = true;
+ wrap2.call(zoom.transform, _tCurr);
+ _skipEvents = false;
+ }
+ function redraw() {
+ clearTimeout(_timeoutID);
+ if (_isHidden)
+ return;
+ updateProjection();
+ var zMini = geoScaleToZoom(projection2.scale());
+ tiles = wrap2.selectAll(".map-in-map-tiles").data([0]);
+ tiles = tiles.enter().append("div").attr("class", "map-in-map-tiles").merge(tiles);
+ backgroundLayer.source(context.background().baseLayerSource()).projection(projection2).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);
+ var overlaySources = context.background().overlayLayerSources();
+ var activeOverlayLayers = [];
+ for (var i2 = 0; i2 < overlaySources.length; i2++) {
+ if (overlaySources[i2].validZoom(zMini)) {
+ if (!overlayLayers[i2])
+ overlayLayers[i2] = rendererTileLayer(context);
+ activeOverlayLayers.push(overlayLayers[i2].source(overlaySources[i2]).projection(projection2).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_default2(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);
+ if (_gesture !== "pan") {
+ var getPath = path_default(projection2);
+ var bbox2 = { type: "Polygon", coordinates: [context.map().extent().polygon()] };
+ viewport = wrap2.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([bbox2]);
+ path.enter().append("path").attr("class", "map-in-map-bbox").merge(path).attr("d", getPath).classed("thick", function(d) {
+ return getPath.area(d) < 30;
+ });
+ }
+ }
+ function queueRedraw() {
+ clearTimeout(_timeoutID);
+ _timeoutID = setTimeout(function() {
+ redraw();
+ }, 750);
+ }
+ 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 (_isHidden) {
+ wrap2.style("display", "block").style("opacity", "1").transition().duration(200).style("opacity", "0").on("end", function() {
+ selection2.selectAll(".map-in-map").style("display", "none");
+ });
+ } else {
+ wrap2.style("display", "block").style("opacity", "0").transition().duration(200).style("opacity", "1").on("end", function() {
+ redraw();
+ });
+ }
+ }
+ uiMapInMap.toggle = toggle;
+ wrap2 = selection2.selectAll(".map-in-map").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "map-in-map").style("display", _isHidden ? "none" : "block").call(zoom).on("dblclick.zoom", null).merge(wrap2);
+ _dMini = [200, 150];
+ _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);
+ }
+ return mapInMap;
+ }
+
+ // modules/ui/notice.js
+ function uiNotice(context) {
+ return function(selection2) {
+ var div = selection2.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) {
+ var e22 = new WheelEvent(d3_event.type, d3_event);
+ context.surface().node().dispatchEvent(e22);
+ });
+ button.call(svgIcon("#iD-icon-plus", "pre-text")).append("span").attr("class", "label").call(_t.append("zoom_in_edit"));
+ function disableTooHigh() {
+ var canEdit = context.map().zoom() >= context.minEditableZoom();
+ div.style("display", canEdit ? "none" : "block");
+ }
+ context.map().on("move.notice", debounce_default(disableTooHigh, 500));
+ disableTooHigh();
+ };
+ }
+
+ // modules/ui/photoviewer.js
+ function uiPhotoviewer(context) {
+ var dispatch10 = dispatch_default("resize");
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ function photoviewer(selection2) {
+ selection2.append("button").attr("class", "thumb-hide").attr("title", _t("icons.close")).on("click", function() {
+ if (services.streetside) {
+ services.streetside.hideViewer(context);
+ }
+ if (services.mapillary) {
+ services.mapillary.hideViewer(context);
+ }
+ if (services.kartaview) {
+ services.kartaview.hideViewer(context);
+ }
+ }).append("div").call(svgIcon("#iD-icon-close"));
+ function preventDefault(d3_event) {
+ d3_event.preventDefault();
+ }
+ selection2.append("button").attr("class", "resize-handle-xy").on("touchstart touchdown touchend", preventDefault).on(
+ _pointerPrefix + "down",
+ buildResizeListener(selection2, "resize", dispatch10, { resizeOnX: true, resizeOnY: true })
+ );
+ selection2.append("button").attr("class", "resize-handle-x").on("touchstart touchdown touchend", preventDefault).on(
+ _pointerPrefix + "down",
+ buildResizeListener(selection2, "resize", dispatch10, { resizeOnX: true })
+ );
+ selection2.append("button").attr("class", "resize-handle-y").on("touchstart touchdown touchend", preventDefault).on(
+ _pointerPrefix + "down",
+ buildResizeListener(selection2, "resize", dispatch10, { resizeOnY: true })
+ );
+ function buildResizeListener(target, eventName, dispatch11, options2) {
+ var resizeOnX = !!options2.resizeOnX;
+ var resizeOnY = !!options2.resizeOnY;
+ var minHeight = options2.minHeight || 240;
+ var minWidth = options2.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();
+ if (resizeOnX) {
+ var maxWidth = mapSize[0];
+ var newWidth = clamp3(startWidth + d3_event.clientX - startX, minWidth, maxWidth);
+ target.style("width", newWidth + "px");
+ }
+ if (resizeOnY) {
+ var maxHeight = mapSize[1] - 90;
+ var newHeight = clamp3(startHeight + startY - d3_event.clientY, minHeight, maxHeight);
+ target.style("height", newHeight + "px");
+ }
+ dispatch11.call(eventName, target, utilGetDimensions(target, true));
+ }
+ function clamp3(num, min3, max3) {
+ return Math.max(min3, Math.min(num, max3));
+ }
+ function stopResize(d3_event) {
+ if (pointerId !== (d3_event.pointerId || "mouse"))
+ return;
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ select_default2(window).on("." + eventName, null);
+ }
+ 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_default2(window).on(_pointerPrefix + "move." + eventName, startResize, false).on(_pointerPrefix + "up." + eventName, stopResize, false);
+ if (_pointerPrefix === "pointer") {
+ select_default2(window).on("pointercancel." + eventName, stopResize, false);
+ }
+ };
+ }
+ }
+ photoviewer.onMapResize = function() {
+ var photoviewer2 = context.container().select(".photoviewer");
+ var content = context.container().select(".main-content");
+ var mapDimensions = utilGetDimensions(content, true);
+ var photoDimensions = utilGetDimensions(photoviewer2, true);
+ 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)
+ ];
+ photoviewer2.style("width", setPhotoDimensions[0] + "px").style("height", setPhotoDimensions[1] + "px");
+ dispatch10.call("resize", photoviewer2, setPhotoDimensions);
+ }
+ };
+ return utilRebind(photoviewer, dispatch10, "on");
+ }
+
+ // modules/ui/restore.js
+ function uiRestore(context) {
+ return function(selection2) {
+ if (!context.history().hasRestorableChanges())
+ return;
+ let modalSelection = uiModal(selection2, true);
+ modalSelection.select(".modal").attr("class", "modal fillL");
+ let introModal = modalSelection.select(".content");
+ introModal.append("div").attr("class", "modal-section").append("h3").call(_t.append("restore.heading"));
+ introModal.append("div").attr("class", "modal-section").append("p").call(_t.append("restore.description"));
+ let buttonWrap = introModal.append("div").attr("class", "modal-actions");
+ let restore = buttonWrap.append("button").attr("class", "restore").on("click", () => {
+ context.history().restore();
+ modalSelection.remove();
+ });
+ restore.append("svg").attr("class", "logo logo-restore").append("use").attr("xlink:href", "#iD-logo-restore");
+ restore.append("div").call(_t.append("restore.restore"));
+ let reset = buttonWrap.append("button").attr("class", "reset").on("click", () => {
+ context.history().clearSaved();
+ modalSelection.remove();
+ });
+ reset.append("svg").attr("class", "logo logo-reset").append("use").attr("xlink:href", "#iD-logo-reset");
+ reset.append("div").call(_t.append("restore.reset"));
+ restore.node().focus();
+ };
+ }
+
+ // modules/ui/scale.js
+ function uiScale(context) {
+ var projection2 = context.projection, isImperial = !_mainLocalizer.usesMetric(), maxLength = 180, tickHeight = 8;
+ 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, i2, val, dLon;
+ if (isImperial) {
+ buckets = [528e4, 528e3, 52800, 5280, 500, 50, 5, 1];
+ } else {
+ buckets = [5e6, 5e5, 5e4, 5e3, 500, 50, 5, 1];
+ }
+ for (i2 = 0; i2 < buckets.length; i2++) {
+ val = buckets[i2];
+ if (dist >= val) {
+ scale.dist = Math.floor(dist / val) * val;
+ break;
+ } else {
+ scale.dist = +dist.toFixed(2);
+ }
+ }
+ dLon = geoMetersToLon(scale.dist / conversion, lat);
+ scale.px = Math.round(projection2([loc1[0] + dLon, loc1[1]])[0]);
+ scale.text = displayLength(scale.dist / conversion, isImperial);
+ return scale;
+ }
+ function update(selection2) {
+ var dims = context.map().dimensions(), loc1 = projection2.invert([0, dims[1]]), loc2 = projection2.invert([maxLength, dims[1]]), scale = scaleDefs(loc1, loc2);
+ selection2.select(".scale-path").attr("d", "M0.5,0.5v" + tickHeight + "h" + scale.px + "v-" + tickHeight);
+ selection2.select(".scale-text").style(_mainLocalizer.textDirection() === "ltr" ? "left" : "right", scale.px + 16 + "px").text(scale.text);
+ }
+ return function(selection2) {
+ function switchUnits() {
+ isImperial = !isImperial;
+ selection2.call(update);
+ }
+ var scalegroup = selection2.append("svg").attr("class", "scale").on("click", switchUnits).append("g").attr("transform", "translate(10,11)");
+ scalegroup.append("path").attr("class", "scale-path");
+ selection2.append("div").attr("class", "scale-text");
+ selection2.call(update);
+ context.map().on("move.scale", function() {
+ update(selection2);
+ });
+ };
+ }
+
+ // modules/ui/shortcuts.js
+ function uiShortcuts(context) {
+ var detected = utilDetect();
+ var _activeTab = 0;
+ var _modalSelection;
+ var _selection = select_default2(null);
+ var _dataShortcuts;
+ function shortcutsModal(_modalSelection2) {
+ _modalSelection2.select(".modal").classed("modal-shortcuts", true);
+ var content = _modalSelection2.select(".content");
+ content.append("div").attr("class", "modal-section header").append("h2").call(_t.append("shortcuts.title"));
+ _mainFileFetcher.get("shortcuts").then(function(data) {
+ _dataShortcuts = data;
+ content.call(render);
+ }).catch(function() {
+ });
+ }
+ function render(selection2) {
+ if (!_dataShortcuts)
+ return;
+ var wrapper = selection2.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();
+ var i2 = _dataShortcuts.indexOf(d);
+ _activeTab = i2;
+ render(selection2);
+ });
+ tabsEnter.append("span").html(function(d) {
+ return _t.html(d.text);
+ });
+ wrapper.selectAll(".tab").classed("active", function(d, i2) {
+ return i2 === _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 ["\u2318"];
+ } else if (detected.os !== "mac" && d.text === "shortcuts.browsing.display_options.fullscreen") {
+ return [];
+ } else {
+ return d.modifiers;
+ }
+ }).enter().each(function() {
+ var selection3 = select_default2(this);
+ selection3.append("kbd").attr("class", "modifier").text(function(d) {
+ return uiCmd.display(d);
+ });
+ selection3.append("span").text("+");
+ });
+ shortcutKeys.selectAll("kbd.shortcut").data(function(d) {
+ var arr = d.shortcuts;
+ 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"];
+ }
+ 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, i2, nodes) {
+ var selection3 = select_default2(this);
+ var click = d.shortcut.toLowerCase().match(/(.*).click/);
+ if (click && click[1]) {
+ selection3.call(svgIcon("#iD-walkthrough-mouse-" + click[1], "operation"));
+ } else if (d.shortcut.toLowerCase() === "long-press") {
+ selection3.call(svgIcon("#iD-walkthrough-longpress", "longpress operation"));
+ } else if (d.shortcut.toLowerCase() === "tap") {
+ selection3.call(svgIcon("#iD-walkthrough-tap", "tap operation"));
+ } else {
+ selection3.append("kbd").attr("class", "shortcut").text(function(d2) {
+ return d2.shortcut;
+ });
+ }
+ if (i2 < nodes.length - 1) {
+ selection3.append("span").html(d.separator || "\xA0" + _t.html("shortcuts.or") + "\xA0");
+ } else if (i2 === nodes.length - 1 && d.suffix) {
+ selection3.append("span").text(d.suffix);
+ }
+ });
+ shortcutKeys.filter(function(d) {
+ return d.gesture;
+ }).each(function() {
+ var selection3 = select_default2(this);
+ selection3.append("span").text("+");
+ selection3.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";
+ });
+ wrapper.selectAll(".shortcut-tab").style("display", function(d, i2) {
+ return i2 === _activeTab ? "flex" : "none";
+ });
+ }
+ return function(selection2, show) {
+ _selection = selection2;
+ if (show) {
+ _modalSelection = uiModal(selection2);
+ _modalSelection.call(shortcutsModal);
+ } else {
+ context.keybinding().on([_t("shortcuts.toggle.key"), "?"], function() {
+ if (context.container().selectAll(".modal-shortcuts").size()) {
+ if (_modalSelection) {
+ _modalSelection.close();
+ _modalSelection = null;
+ }
+ } else {
+ _modalSelection = uiModal(_selection);
+ _modalSelection.call(shortcutsModal);
+ }
+ });
+ }
+ };
+ }
+
+ // modules/ui/data_header.js
+ function uiDataHeader() {
+ var _datum;
+ function dataHeader(selection2) {
+ var header = selection2.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").call(_t.append("map_data.layers.custom.title"));
+ }
+ dataHeader.datum = function(val) {
+ if (!arguments.length)
+ return _datum;
+ _datum = val;
+ return this;
+ };
+ return dataHeader;
+ }
+
+ // modules/ui/combobox.js
+ var _comboHideTimerID;
+ function uiCombobox(context, klass) {
+ var dispatch10 = dispatch_default("accept", "cancel", "update");
+ 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;
+ var _fetcher = function(val, cb) {
+ cb(_data.filter(function(d) {
+ var terms = d.terms || [];
+ terms.push(d.value);
+ if (d.key) {
+ terms.push(d.key);
+ }
+ return terms.some(function(term) {
+ return term.toString().toLowerCase().indexOf(val.toLowerCase()) !== -1;
+ });
+ }));
+ };
+ var combobox = function(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_default2(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();
+ input.node().focus();
+ mousedown(d3_event);
+ }).on("mouseup.combo-caret", function(d3_event) {
+ d3_event.preventDefault();
+ mouseup(d3_event);
+ });
+ });
+ function mousedown(d3_event) {
+ if (d3_event.button !== 0)
+ return;
+ if (input.classed("disabled"))
+ return;
+ _tDown = +new Date();
+ var start2 = input.property("selectionStart");
+ var end = input.property("selectionEnd");
+ if (start2 !== end) {
+ var val = utilGetSetValue(input);
+ input.node().setSelectionRange(val.length, val.length);
+ return;
+ }
+ input.on("mouseup.combo-input", mouseup);
+ }
+ function mouseup(d3_event) {
+ input.on("mouseup.combo-input", null);
+ if (d3_event.button !== 0)
+ return;
+ if (input.classed("disabled"))
+ return;
+ if (input.node() !== document.activeElement)
+ return;
+ var start2 = input.property("selectionStart");
+ var end = input.property("selectionEnd");
+ if (start2 !== end)
+ return;
+ var combo = container.selectAll(".combobox");
+ if (combo.empty() || combo.datum() !== input.node()) {
+ var tOrig = _tDown;
+ window.setTimeout(function() {
+ if (tOrig !== _tDown)
+ return;
+ fetchComboData("", function() {
+ show();
+ render();
+ });
+ }, 250);
+ } else {
+ hide();
+ }
+ }
+ function focus() {
+ fetchComboData("");
+ }
+ function blur() {
+ _comboHideTimerID = window.setTimeout(hide, 75);
+ }
+ function show() {
+ hide();
+ 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) {
+ d3_event.preventDefault();
+ });
+ container.on("scroll.combo-scroll", render, true);
+ }
+ function hide() {
+ if (_comboHideTimerID) {
+ window.clearTimeout(_comboHideTimerID);
+ _comboHideTimerID = void 0;
+ }
+ container.selectAll(".combobox").remove();
+ container.on("scroll.combo-scroll", null);
+ }
+ function keydown(d3_event) {
+ var shown = !container.selectAll(".combobox").empty();
+ var tagName = input.node() ? input.node().tagName.toLowerCase() : "";
+ switch (d3_event.keyCode) {
+ case 8:
+ case 46:
+ d3_event.stopPropagation();
+ _selected = null;
+ render();
+ input.on("input.combo-input", function() {
+ var start2 = input.property("selectionStart");
+ input.node().setSelectionRange(start2, start2);
+ input.on("input.combo-input", change);
+ change(false);
+ });
+ break;
+ case 9:
+ accept(d3_event);
+ break;
+ case 13:
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ accept(d3_event);
+ break;
+ case 38:
+ if (tagName === "textarea" && !shown)
+ return;
+ d3_event.preventDefault();
+ if (tagName === "input" && !shown) {
+ show();
+ }
+ nav(-1);
+ break;
+ case 40:
+ if (tagName === "textarea" && !shown)
+ return;
+ d3_event.preventDefault();
+ if (tagName === "input" && !shown) {
+ show();
+ }
+ nav(1);
+ break;
+ }
+ }
+ function keyup(d3_event) {
+ switch (d3_event.keyCode) {
+ case 27:
+ cancel();
+ break;
+ }
+ }
+ function change(doAutoComplete) {
+ if (doAutoComplete === void 0)
+ doAutoComplete = true;
+ fetchComboData(value(), function() {
+ _selected = null;
+ var val = input.property("value");
+ if (_suggestions.length) {
+ if (doAutoComplete && input.property("selectionEnd") === val.length) {
+ _selected = tryAutocomplete();
+ }
+ if (!_selected) {
+ _selected = val;
+ }
+ }
+ if (val.length) {
+ var combo = container.selectAll(".combobox");
+ if (combo.empty()) {
+ show();
+ }
+ } else {
+ hide();
+ }
+ render();
+ });
+ }
+ function nav(dir) {
+ if (_suggestions.length) {
+ var index = -1;
+ for (var i2 = 0; i2 < _suggestions.length; i2++) {
+ if (_selected && _suggestions[i2].value === _selected) {
+ index = i2;
+ break;
+ }
+ }
+ index = Math.max(Math.min(index + dir, _suggestions.length - 1), 0);
+ _selected = _suggestions[index].value;
+ utilGetSetValue(input, _selected);
+ dispatch10.call("update");
+ }
+ render();
+ ensureVisible();
+ }
+ function ensureVisible() {
+ var combo = container.selectAll(".combobox");
+ if (combo.empty())
+ return;
+ var containerRect = container.node().getBoundingClientRect();
+ var comboRect = combo.node().getBoundingClientRect();
+ if (comboRect.bottom > containerRect.bottom) {
+ var node = attachTo ? attachTo.node() : input.node();
+ node.scrollIntoView({ behavior: "instant", block: "center" });
+ render();
+ }
+ var selected = combo.selectAll(".combobox-option.selected").node();
+ if (selected) {
+ selected.scrollIntoView({ behavior: "smooth", block: "nearest" });
+ }
+ }
+ function value() {
+ var value2 = input.property("value");
+ var start2 = input.property("selectionStart");
+ var end = input.property("selectionEnd");
+ if (start2 && end) {
+ value2 = value2.substring(0, start2);
+ }
+ return value2;
+ }
+ function fetchComboData(v, cb) {
+ _cancelFetch = false;
+ _fetcher.call(input, v, function(results) {
+ if (_cancelFetch)
+ return;
+ _suggestions = results;
+ results.forEach(function(d) {
+ _fetched[d.value] = d;
+ });
+ if (cb) {
+ cb();
+ }
+ });
+ }
+ function tryAutocomplete() {
+ if (!_canAutocomplete)
+ return;
+ var val = _caseSensitive ? value() : value().toLowerCase();
+ if (!val)
+ return;
+ if (isFinite(val))
+ return;
+ const suggestionValues = [];
+ _suggestions.forEach((s) => {
+ suggestionValues.push(s.value);
+ if (s.key && s.key !== s.value) {
+ suggestionValues.push(s.key);
+ }
+ });
+ var bestIndex = -1;
+ for (var i2 = 0; i2 < suggestionValues.length; i2++) {
+ var suggestion = suggestionValues[i2];
+ var compare = _caseSensitive ? suggestion : suggestion.toLowerCase();
+ if (compare === val) {
+ bestIndex = i2;
+ break;
+ } else if (bestIndex === -1 && compare.indexOf(val) === 0) {
+ bestIndex = i2;
+ }
+ }
+ if (bestIndex !== -1) {
+ var bestVal = suggestionValues[bestIndex];
+ input.property("value", bestVal);
+ input.node().setSelectionRange(val.length, bestVal.length);
+ dispatch10.call("update");
+ return bestVal;
+ }
+ }
+ function render() {
+ if (_suggestions.length < _minItems || document.activeElement !== input.node()) {
+ hide();
+ return;
+ }
+ var shown = !container.selectAll(".combobox").empty();
+ if (!shown)
+ return;
+ var combo = container.selectAll(".combobox");
+ var options2 = combo.selectAll(".combobox-option").data(_suggestions, function(d) {
+ return d.value;
+ });
+ options2.exit().remove();
+ options2.enter().append("a").attr("class", function(d) {
+ return "combobox-option " + (d.klass || "");
+ }).attr("title", function(d) {
+ return d.title;
+ }).each(function(d) {
+ if (d.display) {
+ d.display(select_default2(this));
+ } else {
+ select_default2(this).text(d.value);
+ }
+ }).on("mouseenter", _mouseEnterHandler).on("mouseleave", _mouseLeaveHandler).merge(options2).classed("selected", function(d) {
+ return d.value === _selected || d.key === _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");
+ }
+ function accept(d3_event, d) {
+ _cancelFetch = true;
+ var thiz = input.node();
+ if (d) {
+ utilGetSetValue(input, d.value);
+ utilTriggerEvent(input, "change");
+ }
+ var val = utilGetSetValue(input);
+ thiz.setSelectionRange(val.length, val.length);
+ d = _fetched[val];
+ dispatch10.call("accept", thiz, d, val);
+ hide();
+ }
+ function cancel() {
+ _cancelFetch = true;
+ var thiz = input.node();
+ var val = utilGetSetValue(input);
+ var start2 = input.property("selectionStart");
+ var end = input.property("selectionEnd");
+ val = val.slice(0, start2) + val.slice(end);
+ utilGetSetValue(input, val);
+ thiz.setSelectionRange(val.length, val.length);
+ dispatch10.call("cancel", thiz);
+ hide();
+ }
+ };
+ 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;
+ };
+ combobox.minItems = function(val) {
+ if (!arguments.length)
+ return _minItems;
+ _minItems = val;
+ return combobox;
+ };
+ combobox.itemsMouseEnter = function(val) {
+ if (!arguments.length)
+ return _mouseEnterHandler;
+ _mouseEnterHandler = val;
+ return combobox;
+ };
+ combobox.itemsMouseLeave = function(val) {
+ if (!arguments.length)
+ return _mouseLeaveHandler;
+ _mouseLeaveHandler = val;
+ return combobox;
+ };
+ return utilRebind(combobox, dispatch10, "on");
+ }
+ 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);
+ };
+
+ // modules/ui/disclosure.js
+ function uiDisclosure(context, key, expandedDefault) {
+ var dispatch10 = dispatch_default("toggled");
+ var _expanded;
+ var _label = utilFunctor("");
+ var _updatePreference = true;
+ var _content = function() {
+ };
+ var disclosure = function(selection2) {
+ if (_expanded === void 0 || _expanded === null) {
+ var preference = corePreferences("disclosure." + key + ".expanded");
+ _expanded = preference === null ? !!expandedDefault : preference === "true";
+ }
+ var hideToggle = selection2.selectAll(".hide-toggle-" + key).data([0]);
+ var hideToggleEnter = hideToggle.enter().append("h3").append("a").attr("role", "button").attr("href", "#").attr("class", "hide-toggle hide-toggle-" + key).call(svgIcon("", "pre-text", "hide-toggle-icon"));
+ hideToggleEnter.append("span").attr("class", "hide-toggle-text");
+ hideToggle = hideToggleEnter.merge(hideToggle);
+ hideToggle.on("click", toggle).attr("title", _t(`icons.${_expanded ? "collapse" : "expand"}`)).attr("aria-expanded", _expanded).classed("expanded", _expanded);
+ const label = _label();
+ const labelSelection = hideToggle.selectAll(".hide-toggle-text");
+ if (typeof label !== "function") {
+ labelSelection.text(_label());
+ } else {
+ labelSelection.text("").call(label);
+ }
+ hideToggle.selectAll(".hide-toggle-icon").attr(
+ "xlink:href",
+ _expanded ? "#iD-icon-down" : _mainLocalizer.textDirection() === "rtl" ? "#iD-icon-backward" : "#iD-icon-forward"
+ );
+ var wrap2 = selection2.selectAll(".disclosure-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "disclosure-wrap disclosure-wrap-" + key).merge(wrap2).classed("hide", !_expanded);
+ if (_expanded) {
+ wrap2.call(_content);
+ }
+ function toggle(d3_event) {
+ d3_event.preventDefault();
+ _expanded = !_expanded;
+ if (_updatePreference) {
+ corePreferences("disclosure." + key + ".expanded", _expanded);
+ }
+ hideToggle.classed("expanded", _expanded).attr("aria-expanded", _expanded).attr("title", _t(`icons.${_expanded ? "collapse" : "expand"}`));
+ hideToggle.selectAll(".hide-toggle-icon").attr(
+ "xlink:href",
+ _expanded ? "#iD-icon-down" : _mainLocalizer.textDirection() === "rtl" ? "#iD-icon-backward" : "#iD-icon-forward"
+ );
+ wrap2.call(uiToggle(_expanded));
+ if (_expanded) {
+ wrap2.call(_content);
+ }
+ dispatch10.call("toggled", this, _expanded);
+ }
+ };
+ disclosure.label = function(val) {
+ if (!arguments.length)
+ return _label;
+ _label = utilFunctor(val);
+ return disclosure;
+ };
+ disclosure.expanded = function(val) {
+ if (!arguments.length)
+ return _expanded;
+ _expanded = val;
+ return disclosure;
+ };
+ disclosure.updatePreference = function(val) {
+ if (!arguments.length)
+ return _updatePreference;
+ _updatePreference = val;
+ return disclosure;
+ };
+ disclosure.content = function(val) {
+ if (!arguments.length)
+ return _content;
+ _content = val;
+ return disclosure;
+ };
+ return utilRebind(disclosure, dispatch10, "on");
+ }
+
+ // modules/ui/section.js
+ function uiSection(id2, context) {
+ var _classes = utilFunctor("");
+ var _shouldDisplay;
+ var _content;
+ var _disclosure;
+ var _label;
+ var _expandedByDefault = utilFunctor(true);
+ var _disclosureContent;
+ var _disclosureExpanded;
+ var _containerSelection = select_default2(null);
+ var section = {
+ id: id2
+ };
+ section.classes = function(val) {
+ if (!arguments.length)
+ return _classes;
+ _classes = utilFunctor(val);
+ return section;
+ };
+ section.label = function(val) {
+ if (!arguments.length)
+ return _label;
+ _label = utilFunctor(val);
+ return section;
+ };
+ section.expandedByDefault = function(val) {
+ if (!arguments.length)
+ return _expandedByDefault;
+ _expandedByDefault = utilFunctor(val);
+ return section;
+ };
+ section.shouldDisplay = function(val) {
+ if (!arguments.length)
+ return _shouldDisplay;
+ _shouldDisplay = utilFunctor(val);
+ return section;
+ };
+ section.content = function(val) {
+ if (!arguments.length)
+ return _content;
+ _content = val;
+ return section;
+ };
+ section.disclosureContent = function(val) {
+ if (!arguments.length)
+ return _disclosureContent;
+ _disclosureContent = val;
+ return section;
+ };
+ section.disclosureExpanded = function(val) {
+ if (!arguments.length)
+ return _disclosureExpanded;
+ _disclosureExpanded = val;
+ return section;
+ };
+ section.render = function(selection2) {
+ _containerSelection = selection2.selectAll(".section-" + id2).data([0]);
+ var sectionEnter = _containerSelection.enter().append("div").attr("class", "section section-" + id2 + " " + (_classes && _classes() || ""));
+ _containerSelection = sectionEnter.merge(_containerSelection);
+ _containerSelection.call(renderContent);
+ };
+ section.reRender = function() {
+ _containerSelection.call(renderContent);
+ };
+ section.selection = function() {
+ return _containerSelection;
+ };
+ section.disclosure = function() {
+ return _disclosure;
+ };
+ function renderContent(selection2) {
+ if (_shouldDisplay) {
+ var shouldDisplay = _shouldDisplay();
+ selection2.classed("hide", !shouldDisplay);
+ if (!shouldDisplay) {
+ selection2.html("");
+ return;
+ }
+ }
+ if (_disclosureContent) {
+ if (!_disclosure) {
+ _disclosure = uiDisclosure(context, id2.replace(/-/g, "_"), _expandedByDefault()).label(_label || "").content(_disclosureContent);
+ }
+ if (_disclosureExpanded !== void 0) {
+ _disclosure.expanded(_disclosureExpanded);
+ _disclosureExpanded = void 0;
+ }
+ selection2.call(_disclosure);
+ return;
+ }
+ if (_content) {
+ selection2.call(_content);
+ }
+ }
+ return section;
+ }
+
+ // modules/ui/tag_reference.js
+ function uiTagReference(what) {
+ var wikibase = what.qid ? services.wikidata : services.osmWikibase;
+ var tagReference = {};
+ var _button = select_default2(null);
+ var _body = select_default2(null);
+ var _loaded;
+ var _showing;
+ function load() {
+ if (!wikibase)
+ return;
+ _button.classed("tag-reference-loading", true);
+ wikibase.getDocs(what, gotDocs);
+ }
+ function gotDocs(err, docs) {
+ _body.html("");
+ if (!docs || !docs.title) {
+ _body.append("p").attr("class", "tag-reference-description").call(_t.append("inspector.no_documentation_key"));
+ done();
+ return;
+ }
+ if (docs.imageURL) {
+ _body.append("img").attr("class", "tag-reference-wiki-image").attr("alt", docs.description).attr("src", docs.imageURL).on("load", function() {
+ done();
+ }).on("error", function() {
+ select_default2(this).remove();
+ done();
+ });
+ } else {
+ done();
+ }
+ var tagReferenceDescription = _body.append("p").attr("class", "tag-reference-description").append("span");
+ if (docs.description) {
+ tagReferenceDescription = tagReferenceDescription.attr("class", "localized-text").attr("lang", docs.descriptionLocaleCode || "und").text(docs.description);
+ } else {
+ tagReferenceDescription = tagReferenceDescription.call(_t.append("inspector.no_documentation_key"));
+ }
+ tagReferenceDescription.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").call(_t.append(docs.wiki.text));
+ }
+ 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").call(_t.append("commit.about_changeset_comments"));
+ }
+ }
+ function done() {
+ _loaded = true;
+ _button.classed("tag-reference-loading", false);
+ _body.classed("expanded", true).transition().duration(200).style("max-height", "200px").style("opacity", "1");
+ _showing = true;
+ _button.selectAll("svg.icon use").each(function() {
+ var iconUse = select_default2(this);
+ if (iconUse.attr("href") === "#iD-icon-info") {
+ iconUse.attr("href", "#iD-icon-info-filled");
+ }
+ });
+ }
+ function hide() {
+ _body.transition().duration(200).style("max-height", "0px").style("opacity", "0").on("end", function() {
+ _body.classed("expanded", false);
+ });
+ _showing = false;
+ _button.selectAll("svg.icon use").each(function() {
+ var iconUse = select_default2(this);
+ if (iconUse.attr("href") === "#iD-icon-info-filled") {
+ iconUse.attr("href", "#iD-icon-info");
+ }
+ });
+ }
+ tagReference.button = function(selection2, klass, iconName) {
+ _button = selection2.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);
+ _button.on("click", function(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ this.blur();
+ if (_showing) {
+ hide();
+ } else if (_loaded) {
+ done();
+ } else {
+ load();
+ }
+ });
+ };
+ tagReference.body = function(selection2) {
+ var itemID = what.qid || what.key + "-" + (what.value || "");
+ _body = selection2.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();
+ }
+ };
+ tagReference.showing = function(val) {
+ if (!arguments.length)
+ return _showing;
+ _showing = val;
+ return tagReference;
+ };
+ return tagReference;
+ }
+
+ // modules/ui/field_help.js
+ function uiFieldHelp(context, fieldName) {
+ var fieldHelp = {};
+ var _inspector = select_default2(null);
+ var _wrap = select_default2(null);
+ var _body = select_default2(null);
+ 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: { html: _t.html("restriction.controls.distance") },
+ viaField: { html: _t.html("restriction.controls.via") },
+ fromShadow: { html: icon("#iD-turn-shadow", "inline shadow from") },
+ allowShadow: { html: icon("#iD-turn-shadow", "inline shadow allow") },
+ restrictShadow: { html: icon("#iD-turn-shadow", "inline shadow restrict") },
+ onlyShadow: { html: icon("#iD-turn-shadow", "inline shadow only") },
+ allowTurn: { html: icon("#iD-turn-yes", "inline turn") },
+ restrictTurn: { html: icon("#iD-turn-no", "inline turn") },
+ onlyTurn: { html: icon("#iD-turn-only", "inline turn") }
+ };
+ var docs = fieldHelpKeys[fieldName].map(function(key) {
+ var helpkey = "help.field." + fieldName + "." + key[0];
+ var text2 = key[1].reduce(function(all, part) {
+ var subkey = helpkey + "." + part;
+ var depth = fieldHelpHeadings[subkey];
+ var hhh = depth ? Array(depth + 1).join("#") + " " : "";
+ return all + hhh + _t.html(subkey, replacements) + "\n\n";
+ }, "");
+ return {
+ key: helpkey,
+ title: _t.html(helpkey + ".title"),
+ html: marked(text2.trim())
+ };
+ });
+ function show() {
+ updatePosition();
+ _body.classed("hide", false).style("opacity", "0").transition().duration(200).style("opacity", "1");
+ }
+ function hide() {
+ _body.classed("hide", true).transition().duration(200).style("opacity", "0").on("end", function() {
+ _body.classed("hide", true);
+ });
+ }
+ function clickHelp(index) {
+ var d = docs[index];
+ var tkeys = fieldHelpKeys[fieldName][index][1];
+ _body.selectAll(".field-help-nav-item").classed("active", function(d2, i2) {
+ return i2 === index;
+ });
+ var content = _body.selectAll(".field-help-content").html(d.html);
+ content.selectAll("p").attr("class", function(d2, i2) {
+ return tkeys[i2];
+ });
+ 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"));
+ }
+ }
+ fieldHelp.button = function(selection2) {
+ if (_body.empty())
+ return;
+ var button = selection2.selectAll(".field-help-button").data([0]);
+ 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 (_body.classed("hide")) {
+ show();
+ } else {
+ hide();
+ }
+ });
+ };
+ function updatePosition() {
+ var wrap2 = _wrap.node();
+ var inspector = _inspector.node();
+ var wRect = wrap2.getBoundingClientRect();
+ var iRect = inspector.getBoundingClientRect();
+ _body.style("top", wRect.top + inspector.scrollTop - iRect.top + "px");
+ }
+ fieldHelp.body = function(selection2) {
+ _wrap = selection2.selectAll(".form-field-input-wrap");
+ if (_wrap.empty())
+ return;
+ _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");
+ var titleEnter = enter.append("div").attr("class", "field-help-title cf");
+ titleEnter.append("h2").attr("class", _mainLocalizer.textDirection() === "rtl" ? "fr" : "fl").call(_t.append("help.field." + fieldName + ".title"));
+ titleEnter.append("button").attr("class", "fr close").attr("title", _t("icons.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 fieldHelp;
+ }
+
+ // modules/ui/fields/check.js
+ function uiFieldCheck(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var options2 = field.options;
+ var values = [];
+ var texts = [];
+ var _tags;
+ var input = select_default2(null);
+ var text2 = select_default2(null);
+ var label = select_default2(null);
+ var reverser = select_default2(null);
+ var _impliedYes;
+ var _entityIDs = [];
+ var _value;
+ var stringsField = field.resolveReference("stringsCrossReference");
+ if (!options2 && stringsField.options) {
+ options2 = stringsField.options;
+ }
+ if (options2) {
+ for (var i2 in options2) {
+ var v = options2[i2];
+ values.push(v === "undefined" ? void 0 : v);
+ texts.push(stringsField.t.html("options." + v, { "default": v }));
+ }
+ } else {
+ values = [void 0, "yes"];
+ texts = [_t.html("inspector.unknown"), _t.html("inspector.check.yes")];
+ if (field.type !== "defaultCheck") {
+ values.push("no");
+ texts.push(_t.html("inspector.check.no"));
+ }
+ }
+ function checkImpliedYes() {
+ _impliedYes = field.id === "oneway_yes";
+ if (field.id === "oneway") {
+ var entity = context.entity(_entityIDs[0]);
+ 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 reverserHidden() {
+ if (!context.container().select("div.inspector-hover").empty())
+ return true;
+ return !(_value === "yes" || _impliedYes && !_value);
+ }
+ function reverserSetText(selection2) {
+ var entity = _entityIDs.length && context.hasEntity(_entityIDs[0]);
+ if (reverserHidden() || !entity)
+ return selection2;
+ var first = entity.first();
+ var last = entity.isClosed() ? entity.nodes[entity.nodes.length - 2] : entity.last();
+ var pseudoDirection = first < last;
+ var icon2 = pseudoDirection ? "#iD-icon-forward" : "#iD-icon-backward";
+ selection2.selectAll(".reverser-span").html("").call(_t.append("inspector.check.reverser")).call(svgIcon(icon2, "inline"));
+ return selection2;
+ }
+ var check = function(selection2) {
+ checkImpliedYes();
+ label = selection2.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");
+ if (field.type === "onewayCheck") {
+ enter.append("button").attr("class", "reverser" + (reverserHidden() ? " hide" : "")).append("span").attr("class", "reverser-span");
+ }
+ label = label.merge(enter);
+ input = label.selectAll("input");
+ text2 = label.selectAll("span.value");
+ input.on("click", function(d3_event) {
+ d3_event.stopPropagation();
+ var t = {};
+ 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];
+ }
+ if (t[field.key] === "reversible" || t[field.key] === "alternating") {
+ t[field.key] = values[0];
+ }
+ dispatch10.call("change", this, t);
+ });
+ 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 i3 in _entityIDs) {
+ graph = actionReverse(_entityIDs[i3])(graph);
+ }
+ return graph;
+ },
+ _t("operations.reverse.annotation.line", { n: 1 })
+ );
+ context.validator().validate();
+ select_default2(this).call(reverserSetText);
+ });
+ }
+ };
+ check.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return check;
+ };
+ check.tags = function(tags) {
+ _tags = tags;
+ function isChecked(val) {
+ return val !== "no" && val !== "" && val !== void 0 && val !== null;
+ }
+ function textFor(val) {
+ if (val === "")
+ val = void 0;
+ 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();
+ if (field.type === "onewayCheck" && (_value === "1" || _value === "-1")) {
+ _value = "yes";
+ }
+ input.property("indeterminate", isMixed || field.type !== "defaultCheck" && !_value).property("checked", isChecked(_value));
+ text2.html(isMixed ? _t.html("inspector.multiple_values") : textFor(_value)).classed("mixed", isMixed);
+ label.classed("set", !!_value);
+ if (field.type === "onewayCheck") {
+ reverser.classed("hide", reverserHidden()).call(reverserSetText);
+ }
+ };
+ check.focus = function() {
+ input.node().focus();
+ };
+ return utilRebind(check, dispatch10, "on");
+ }
+
+ // modules/ui/length_indicator.js
+ function uiLengthIndicator(maxChars) {
+ var _wrap = select_default2(null);
+ var _tooltip = uiPopover("tooltip max-length-warning").placement("bottom").hasArrow(true).content(() => (selection2) => {
+ selection2.text("");
+ selection2.call(svgIcon("#iD-icon-alert", "inline"));
+ selection2.call(_t.append("inspector.max_length_reached", { maxChars }));
+ });
+ var _silent = false;
+ var lengthIndicator = function(selection2) {
+ _wrap = selection2.selectAll("span.length-indicator-wrap").data([0]);
+ _wrap = _wrap.enter().append("span").merge(_wrap).classed("length-indicator-wrap", true);
+ selection2.call(_tooltip);
+ };
+ lengthIndicator.update = function(val) {
+ const strLen = utilUnicodeCharsCount(utilCleanOsmString(val, Number.POSITIVE_INFINITY));
+ let indicator = _wrap.selectAll("span.length-indicator").data([strLen]);
+ indicator.enter().append("span").merge(indicator).classed("length-indicator", true).classed("limit-reached", (d) => d > maxChars).style("border-right-width", (d) => `${Math.abs(maxChars - d) * 2}px`).style("margin-right", (d) => d > maxChars ? `${(maxChars - d) * 2}px` : 0).style("opacity", (d) => d > maxChars * 0.8 ? Math.min(1, (d / maxChars - 0.8) / (1 - 0.8)) : 0).style("pointer-events", (d) => d > maxChars * 0.8 ? null : "none");
+ if (_silent)
+ return;
+ if (strLen > maxChars) {
+ _tooltip.show();
+ } else {
+ _tooltip.hide();
+ }
+ };
+ lengthIndicator.silent = function(val) {
+ if (!arguments.length)
+ return _silent;
+ _silent = val;
+ return lengthIndicator;
+ };
+ return lengthIndicator;
+ }
+
+ // modules/ui/fields/combo.js
+ var valueIcons = {
+ "crossing:markings": [
+ "dashes",
+ "dots",
+ "ladder:paired",
+ "ladder:skewed",
+ "ladder",
+ "lines:paired",
+ "lines",
+ "surface",
+ "zebra:bicolour",
+ "zebra:double",
+ "zebra:paired",
+ "zebra"
+ ]
+ };
+ function uiFieldCombo(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var _isMulti = field.type === "multiCombo" || field.type === "manyCombo";
+ var _isNetwork = field.type === "networkCombo";
+ var _isSemi = field.type === "semiCombo";
+ var _showTagInfoSuggestions = field.type !== "manyCombo" && field.autoSuggestions !== false;
+ var _allowCustomValues = field.type !== "manyCombo" && field.customValues !== false;
+ var _snake_case = field.snake_case || field.snake_case === void 0;
+ var _combobox = uiCombobox(context, "combo-" + field.safeid).caseSensitive(field.caseSensitive).minItems(1);
+ var _container = select_default2(null);
+ var _inputWrap = select_default2(null);
+ var _input = select_default2(null);
+ var _lengthIndicator = uiLengthIndicator(context.maxCharsForTagValue());
+ var _comboData = [];
+ var _multiData = [];
+ var _entityIDs = [];
+ var _tags;
+ var _countryCode;
+ var _staticPlaceholder;
+ var _dataDeprecated = [];
+ _mainFileFetcher.get("deprecated").then(function(d) {
+ _dataDeprecated = d;
+ }).catch(function() {
+ });
+ if (_isMulti && field.key && /[^:]$/.test(field.key)) {
+ field.key += ":";
+ }
+ function snake(s) {
+ return s.replace(/\s+/g, "_");
+ }
+ function clean2(s) {
+ return s.split(";").map(function(s2) {
+ return s2.trim();
+ }).join(";");
+ }
+ function tagValue(dval) {
+ dval = clean2(dval || "");
+ var found = getOptions().find(function(o) {
+ return o.key && clean2(o.value) === dval;
+ });
+ if (found)
+ return found.key;
+ if (field.type === "typeCombo" && !dval) {
+ return "yes";
+ }
+ return restrictTagValueSpelling(dval) || void 0;
+ }
+ function restrictTagValueSpelling(dval) {
+ if (_snake_case) {
+ dval = snake(dval);
+ }
+ if (!field.caseSensitive) {
+ dval = dval.toLowerCase();
+ }
+ return dval;
+ }
+ function getLabelId(field2, v) {
+ return field2.hasTextForStringId(`options.${v}.title`) ? `options.${v}.title` : `options.${v}`;
+ }
+ function displayValue(tval) {
+ tval = tval || "";
+ var stringsField = field.resolveReference("stringsCrossReference");
+ const labelId = getLabelId(stringsField, tval);
+ if (stringsField.hasTextForStringId(labelId)) {
+ return stringsField.t(labelId, { default: tval });
+ }
+ if (field.type === "typeCombo" && tval.toLowerCase() === "yes") {
+ return "";
+ }
+ return tval;
+ }
+ function renderValue(tval) {
+ tval = tval || "";
+ var stringsField = field.resolveReference("stringsCrossReference");
+ const labelId = getLabelId(stringsField, tval);
+ if (stringsField.hasTextForStringId(labelId)) {
+ return stringsField.t.append(labelId, { default: tval });
+ }
+ if (field.type === "typeCombo" && tval.toLowerCase() === "yes") {
+ tval = "";
+ }
+ return (selection2) => selection2.text(tval);
+ }
+ function objectDifference(a, b) {
+ return a.filter(function(d1) {
+ return !b.some(function(d2) {
+ return !d2.isMixed && d1.value === d2.value;
+ });
+ });
+ }
+ function initCombo(selection2, attachTo) {
+ if (!_allowCustomValues) {
+ selection2.attr("readonly", "readonly");
+ }
+ if (_showTagInfoSuggestions && services.taginfo) {
+ selection2.call(_combobox.fetcher(setTaginfoValues), attachTo);
+ setTaginfoValues("", setPlaceholder);
+ } else {
+ selection2.call(_combobox, attachTo);
+ setStaticValues(setPlaceholder);
+ }
+ }
+ function getOptions() {
+ var stringsField = field.resolveReference("stringsCrossReference");
+ if (!(field.options || stringsField.options))
+ return [];
+ return (field.options || stringsField.options).map(function(v) {
+ const labelId = getLabelId(stringsField, v);
+ return {
+ key: v,
+ value: stringsField.t(labelId, { default: v }),
+ title: stringsField.t(`options.${v}.description`, { default: v }),
+ display: addComboboxIcons(stringsField.t.append(labelId, { default: v }), v),
+ klass: stringsField.hasTextForStringId(labelId) ? "" : "raw-option"
+ };
+ });
+ }
+ function setStaticValues(callback, filter2) {
+ _comboData = getOptions();
+ if (filter2 !== void 0) {
+ _comboData = _comboData.filter(filter2);
+ }
+ _comboData = objectDifference(_comboData, _multiData);
+ _combobox.data(_comboData);
+ if (callback)
+ callback(_comboData);
+ }
+ function setTaginfoValues(q, callback) {
+ var queryFilter = (d) => d.value.toLowerCase().includes(q.toLowerCase()) || d.key.toLowerCase().includes(q.toLowerCase());
+ setStaticValues(callback, queryFilter);
+ var stringsField = field.resolveReference("stringsCrossReference");
+ var fn = _isMulti ? "multikeys" : "values";
+ var query = (_isMulti ? field.key : "") + q;
+ var hasCountryPrefix = _isNetwork && _countryCode && _countryCode.indexOf(q.toLowerCase()) === 0;
+ if (hasCountryPrefix) {
+ query = _countryCode + ":";
+ }
+ var params = {
+ debounce: q !== "",
+ key: field.key,
+ query
+ };
+ if (_entityIDs.length) {
+ params.geometry = context.graph().geometry(_entityIDs[0]);
+ }
+ services.taginfo[fn](params, function(err, data) {
+ if (err)
+ return;
+ data = data.filter((d) => field.type !== "typeCombo" || d.value !== "yes");
+ data = data.filter((d) => {
+ var value = d.value;
+ if (_isMulti) {
+ value = value.slice(field.key.length);
+ }
+ return value === restrictTagValueSpelling(value);
+ });
+ var deprecatedValues = osmEntity.deprecatedTagValuesByKey(_dataDeprecated)[field.key];
+ if (deprecatedValues) {
+ data = data.filter((d) => !deprecatedValues.includes(d.value));
+ }
+ if (hasCountryPrefix) {
+ data = data.filter((d) => d.value.toLowerCase().indexOf(_countryCode + ":") === 0);
+ }
+ const additionalOptions = (field.options || stringsField.options || []).filter((v) => !data.some((dv) => dv.value === (_isMulti ? field.key + v : v))).map((v) => ({ value: v }));
+ _container.classed("empty-combobox", data.length === 0);
+ _comboData = data.concat(additionalOptions).map(function(d) {
+ var v = d.value;
+ if (_isMulti)
+ v = v.replace(field.key, "");
+ const labelId = getLabelId(stringsField, v);
+ var isLocalizable = stringsField.hasTextForStringId(labelId);
+ var label = stringsField.t(labelId, { default: v });
+ return {
+ key: v,
+ value: label,
+ title: stringsField.t(`options.${v}.description`, { default: isLocalizable ? v : d.title !== label ? d.title : "" }),
+ display: addComboboxIcons(stringsField.t.append(labelId, { default: v }), v),
+ klass: isLocalizable ? "" : "raw-option"
+ };
+ });
+ _comboData = _comboData.filter(queryFilter);
+ _comboData = objectDifference(_comboData, _multiData);
+ if (callback)
+ callback(_comboData);
+ });
+ }
+ function addComboboxIcons(disp, value) {
+ if (valueIcons[field.key]) {
+ return function(selection2) {
+ var span = selection2.insert("span", ":first-child").attr("class", "tag-value-icon");
+ if (valueIcons[field.key].indexOf(value) !== -1) {
+ span.call(svgIcon("#iD-" + field.key.replace(/:/g, "_") + "-" + value.replace(/:/g, "_")));
+ }
+ disp.call(this, selection2);
+ };
+ }
+ return disp;
+ }
+ 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(", ");
+ }
+ if (!/(…|\.\.\.)$/.test(_staticPlaceholder)) {
+ _staticPlaceholder += "\u2026";
+ }
+ var ph;
+ if (!_isMulti && !_isSemi && _tags && Array.isArray(_tags[field.key])) {
+ ph = _t("inspector.multiple_values");
+ } else {
+ ph = _staticPlaceholder;
+ }
+ _container.selectAll("input").attr("placeholder", ph);
+ }
+ function change() {
+ var t = {};
+ var val;
+ if (_isMulti || _isSemi) {
+ var vals;
+ if (_isMulti) {
+ vals = [tagValue(utilGetSetValue(_input))];
+ } else if (_isSemi) {
+ val = tagValue(utilGetSetValue(_input)) || "";
+ val = val.replace(/,/g, ";");
+ vals = val.split(";");
+ }
+ vals = vals.filter(Boolean);
+ if (!vals.length)
+ return;
+ _container.classed("active", false);
+ utilGetSetValue(_input, "");
+ if (_isMulti) {
+ utilArrayUniq(vals).forEach(function(v) {
+ var key = (field.key || "") + v;
+ if (_tags) {
+ var old = _tags[key];
+ if (typeof old === "string" && old.toLowerCase() !== "no")
+ return;
+ }
+ key = context.cleanTagKey(key);
+ field.keys.push(key);
+ t[key] = "yes";
+ });
+ } else if (_isSemi) {
+ var arr = _multiData.map(function(d) {
+ return d.key;
+ });
+ 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);
+ if (!rawValue && Array.isArray(_tags[field.key]))
+ return;
+ val = context.cleanTagValue(tagValue(rawValue));
+ t[field.key] = val || void 0;
+ }
+ dispatch10.call("change", this, t);
+ }
+ function removeMultikey(d3_event, d) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var t = {};
+ if (_isMulti) {
+ t[d.key] = void 0;
+ } 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(";") : void 0;
+ _lengthIndicator.update(t[field.key]);
+ }
+ dispatch10.call("change", this, t);
+ }
+ function combo(selection2) {
+ _container = selection2.selectAll(".form-field-input-wrap").data([0]);
+ var type2 = _isMulti || _isSemi ? "multicombo" : "combo";
+ _container = _container.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + type2).merge(_container);
+ if (_isMulti || _isSemi) {
+ _container = _container.selectAll(".chiplist").data([0]);
+ var listClass = "chiplist";
+ if (field.key === "destination" || field.key === "via") {
+ listClass += " full-line-chips";
+ }
+ _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]);
+ }
+ _input = _input.enter().append("input").attr("type", "text").attr("id", field.domId).call(utilNoAuto).call(initCombo, selection2).merge(_input);
+ if (_isSemi) {
+ _inputWrap.call(_lengthIndicator);
+ } else if (!_isMulti) {
+ _container.call(_lengthIndicator);
+ }
+ if (_isNetwork) {
+ var extent = combinedEntityExtent();
+ var countryCode = extent && iso1A2Code(extent.center());
+ _countryCode = countryCode && countryCode.toLowerCase();
+ }
+ _input.on("change", change).on("blur", change).on("input", function() {
+ let val = utilGetSetValue(_input);
+ updateIcon(val);
+ if (_isSemi && _tags[field.key]) {
+ val += ";" + _tags[field.key];
+ }
+ _lengthIndicator.update(val);
+ });
+ _input.on("keydown.field", function(d3_event) {
+ switch (d3_event.keyCode) {
+ case 13:
+ _input.node().blur();
+ d3_event.stopPropagation();
+ break;
+ }
+ });
+ if (_isMulti || _isSemi) {
+ _combobox.on("accept", function() {
+ _input.node().blur();
+ _input.node().focus();
+ });
+ _input.on("focus", function() {
+ _container.classed("active", true);
+ });
+ }
+ _combobox.on("cancel", function() {
+ _input.node().blur();
+ }).on("update", function() {
+ updateIcon(utilGetSetValue(_input));
+ });
+ }
+ function updateIcon(value) {
+ value = tagValue(value);
+ if (valueIcons[field.key]) {
+ _container.selectAll(".tag-value-icon").remove();
+ if (valueIcons[field.key].indexOf(value) !== -1) {
+ _container.selectAll(".tag-value-icon").data([value]).enter().insert("div", "input").attr("class", "tag-value-icon").call(svgIcon("#iD-" + field.key.replace(/:/g, "_") + "-" + value.replace(/:/g, "_")));
+ }
+ }
+ }
+ combo.tags = function(tags) {
+ _tags = tags;
+ var stringsField = field.resolveReference("stringsCrossReference");
+ if (_isMulti || _isSemi) {
+ _multiData = [];
+ var maxLength;
+ if (_isMulti) {
+ 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.slice(field.key.length) : k;
+ _multiData.push({
+ key: k,
+ value: displayValue(suffix),
+ display: renderValue(suffix),
+ isMixed: Array.isArray(v)
+ });
+ }
+ if (field.key) {
+ field.keys = _multiData.map(function(d) {
+ return d.key;
+ });
+ maxLength = context.maxCharsForTagKey() - utilUnicodeCharsCount(field.key);
+ } else {
+ maxLength = context.maxCharsForTagKey();
+ }
+ } 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);
+ if (!commonValues) {
+ commonValues = thisVals;
+ } else {
+ commonValues = commonValues.filter((value) => thisVals.includes(value));
+ }
+ });
+ allValues = utilArrayUniq(allValues).filter(Boolean);
+ } else {
+ allValues = utilArrayUniq((tags[field.key] || "").split(";")).filter(Boolean);
+ commonValues = allValues;
+ }
+ _multiData = allValues.map(function(v2) {
+ return {
+ key: v2,
+ value: displayValue(v2),
+ display: renderValue(v2),
+ isMixed: !commonValues.includes(v2)
+ };
+ });
+ var currLength = utilUnicodeCharsCount(commonValues.join(";"));
+ maxLength = context.maxCharsForTagValue() - currLength;
+ if (currLength > 0) {
+ maxLength -= 1;
+ }
+ }
+ maxLength = Math.max(0, maxLength);
+ var allowDragAndDrop = _isSemi && !Array.isArray(tags[field.key]);
+ var available = objectDifference(_comboData, _multiData);
+ _combobox.data(available);
+ var hideAdd = !_allowCustomValues && !available.length || maxLength <= 0;
+ _container.selectAll(".chiplist .input-wrap").style("display", hideAdd ? "none" : null);
+ 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 k2 = d.key;
+ if (_isMulti)
+ k2 = k2.replace(field.key, "");
+ return !stringsField.hasTextForStringId("options." + k2);
+ }).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);
+ }
+ chips.select("span").each(function(d) {
+ const selection2 = select_default2(this);
+ if (d.display) {
+ selection2.text("");
+ d.display(selection2);
+ } else {
+ selection2.text(d.value);
+ }
+ });
+ chips.select("a").attr("href", "#").on("click", removeMultikey).attr("class", "remove").text("\xD7");
+ } 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 && !stringsField.hasTextForStringId(`options.${tags[field.key]}`) && !stringsField.hasTextForStringId(`options.${tags[field.key]}.title`);
+ 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" : void 0).attr("title", isMixed ? mixedValues.join("\n") : void 0).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["\u232B"] || d3_event.keyCode === utilKeybinding.keyCodes["\u2326"])) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var t = {};
+ t[field.key] = void 0;
+ dispatch10.call("change", this, t);
+ }
+ });
+ if (!Array.isArray(tags[field.key])) {
+ updateIcon(tags[field.key]);
+ }
+ if (!isMixed) {
+ _lengthIndicator.update(tags[field.key]);
+ }
+ }
+ };
+ function registerDragAndDrop(selection2) {
+ var dragOrigin, targetIndex;
+ selection2.call(
+ drag_default().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_default2(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 = selection2.nodes().indexOf(this);
+ select_default2(this).classed("dragging", true);
+ targetIndex = null;
+ var targetIndexOffsetTop = null;
+ var draggedTagWidth = select_default2(this).node().offsetWidth;
+ if (field.key === "destination" || field.key === "via") {
+ _container.selectAll(".chip").style("transform", function(d2, index2) {
+ var node = select_default2(this).node();
+ 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 "translateY(-100%)";
+ } else if (index2 < index && d3_event.y < node.offsetTop + node.offsetHeight) {
+ if (targetIndex === null || index2 < targetIndex) {
+ targetIndex = index2;
+ }
+ return "translateY(100%)";
+ }
+ return null;
+ });
+ } else {
+ _container.selectAll(".chip").each(function(d2, index2) {
+ var node = select_default2(this).node();
+ 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_default2(this).node();
+ if (index === index2) {
+ return "translate(" + x + "px, " + y + "px)";
+ }
+ if (node.offsetTop === targetIndexOffsetTop) {
+ if (index2 < index && index2 >= targetIndex) {
+ return "translateX(" + draggedTagWidth + "px)";
+ } else if (index2 > index && index2 <= targetIndex) {
+ return "translateX(-" + draggedTagWidth + "px)";
+ }
+ }
+ return null;
+ });
+ }
+ }).on("end", function() {
+ if (!select_default2(this).classed("dragging")) {
+ return;
+ }
+ var index = selection2.nodes().indexOf(this);
+ select_default2(this).classed("dragging", false);
+ _container.selectAll(".chip").style("transform", null);
+ if (typeof targetIndex === "number") {
+ var element = _multiData[index];
+ _multiData.splice(index, 1);
+ _multiData.splice(targetIndex, 0, element);
+ var t = {};
+ if (_multiData.length) {
+ t[field.key] = _multiData.map(function(element2) {
+ return element2.key;
+ }).join(";");
+ } else {
+ t[field.key] = void 0;
+ }
+ dispatch10.call("change", this, t);
+ }
+ dragOrigin = void 0;
+ targetIndex = void 0;
+ })
+ );
+ }
+ combo.focus = function() {
+ _input.node().focus();
+ };
+ combo.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return combo;
+ };
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ return utilRebind(combo, dispatch10, "on");
+ }
+
+ // modules/ui/fields/input.js
+ function uiFieldText(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var input = select_default2(null);
+ var outlinkButton = select_default2(null);
+ var wrap2 = select_default2(null);
+ var _lengthIndicator = uiLengthIndicator(context.maxCharsForTagValue());
+ var _entityIDs = [];
+ var _tags;
+ var _phoneFormats = {};
+ const isDirectionField = field.key.split(":").some((keyPart) => keyPart === "direction");
+ if (field.type === "tel") {
+ _mainFileFetcher.get("phone_formats").then(function(d) {
+ _phoneFormats = d;
+ updatePhonePlaceholder();
+ }).catch(function() {
+ });
+ }
+ function calcLocked() {
+ 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;
+ if (entity.tags.wikidata)
+ return true;
+ var preset = _mainPresetIndex.match(entity, context.graph());
+ var isSuggestion = preset && preset.suggestion;
+ var which = field.id;
+ return isSuggestion && !!entity.tags[which] && !!entity.tags[which + ":wikidata"];
+ });
+ field.locked(isLocked);
+ }
+ function i2(selection2) {
+ calcLocked();
+ var isLocked = field.locked();
+ wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ input = wrap2.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());
+ wrap2.call(_lengthIndicator);
+ 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 = wrap2.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;
+ }).attr("title", function(d) {
+ var which = d > 0 ? "increment" : "decrement";
+ return _t(`inspector.${which}`);
+ }).merge(buttons).on("click", function(d3_event, d) {
+ d3_event.preventDefault();
+ var isMixed = Array.isArray(_tags[field.key]);
+ if (isMixed)
+ return;
+ var raw_vals = input.node().value || "0";
+ var vals = raw_vals.split(";");
+ vals = vals.map(function(v) {
+ var num = Number(v);
+ if (isDirectionField) {
+ const compassDir = cardinal[v.trim().toLowerCase()];
+ if (compassDir !== void 0) {
+ num = compassDir;
+ }
+ }
+ if (!isFinite(num)) {
+ return v.trim();
+ }
+ num += d;
+ if (isDirectionField) {
+ num = (num % 360 + 360) % 360;
+ }
+ const numDecimals = v.includes(".") ? v.split(".")[1].length : 0;
+ return clamped(num).toFixed(numDecimals);
+ });
+ input.node().value = vals.join(";");
+ change()();
+ });
+ } else if (field.type === "identifier" && field.urlFormat && field.pattern) {
+ input.attr("type", "text");
+ outlinkButton = wrap2.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 domain2 = domainResults[1];
+ return _t("icons.view_on", { domain: domain2 });
+ }
+ 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);
+ } else if (field.type === "url") {
+ input.attr("type", "text");
+ outlinkButton = wrap2.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", () => _t("icons.visit_website")).on("click", function(d3_event) {
+ d3_event.preventDefault();
+ const value = validIdentifierValueForLink();
+ if (value)
+ window.open(value, "_blank");
+ }).merge(outlinkButton);
+ } else if (field.type === "colour") {
+ input.attr("type", "text");
+ updateColourPreview();
+ }
+ }
+ function isColourValid(colour) {
+ if (!colour.match(/^(#([0-9a-fA-F]{3}){1,2}|\w+)$/)) {
+ return false;
+ } else if (!CSS.supports("color", colour) || ["unset", "inherit", "initial", "revert"].includes(colour)) {
+ return false;
+ }
+ return true;
+ }
+ function updateColourPreview() {
+ wrap2.selectAll(".colour-preview").remove();
+ const colour = utilGetSetValue(input);
+ if (!isColourValid(colour) && colour !== "")
+ return;
+ var colourSelector = wrap2.selectAll(".colour-selector").data([0]);
+ outlinkButton = wrap2.selectAll(".colour-preview").data([colour]);
+ colourSelector.enter().append("input").attr("type", "color").attr("class", "form-field-button colour-selector").attr("value", colour).on("input", debounce_default(function(d3_event) {
+ d3_event.preventDefault();
+ var colour2 = this.value;
+ if (!isColourValid(colour2))
+ return;
+ utilGetSetValue(input, this.value);
+ change()();
+ updateColourPreview();
+ }, 100));
+ outlinkButton = outlinkButton.enter().append("div").attr("class", "form-field-button colour-preview").append("div").style("background-color", (d) => d).attr("class", "colour-box");
+ if (colour === "") {
+ outlinkButton = outlinkButton.call(svgIcon("#iD-icon-edit"));
+ }
+ outlinkButton.on("click", () => wrap2.select(".colour-selector").node().click()).merge(outlinkButton);
+ }
+ function updatePhonePlaceholder() {
+ if (input.empty() || !Object.keys(_phoneFormats).length)
+ return;
+ var extent = combinedEntityExtent();
+ var countryCode = extent && iso1A2Code(extent.center());
+ var format2 = countryCode && _phoneFormats[countryCode.toLowerCase()];
+ if (format2)
+ input.attr("placeholder", format2);
+ }
+ function validIdentifierValueForLink() {
+ const value = utilGetSetValue(input).trim();
+ if (field.type === "url" && value) {
+ try {
+ return new URL(value).href;
+ } catch (e) {
+ return null;
+ }
+ }
+ if (field.type === "identifier" && field.pattern) {
+ return value && value.match(new RegExp(field.pattern))[0];
+ }
+ return null;
+ }
+ function clamped(num) {
+ if (field.minValue !== void 0) {
+ num = Math.max(num, field.minValue);
+ }
+ if (field.maxValue !== void 0) {
+ num = Math.min(num, field.maxValue);
+ }
+ return num;
+ }
+ function change(onInput) {
+ return function() {
+ var t = {};
+ var val = utilGetSetValue(input);
+ if (!onInput)
+ val = context.cleanTagValue(val);
+ if (!val && Array.isArray(_tags[field.key]))
+ return;
+ if (!onInput) {
+ if (field.type === "number" && val) {
+ var vals = val.split(";");
+ vals = vals.map(function(v) {
+ var num = Number(v);
+ return isFinite(num) ? clamped(num) : v.trim();
+ });
+ val = vals.join(";");
+ }
+ utilGetSetValue(input, val);
+ }
+ t[field.key] = val || void 0;
+ dispatch10.call("change", this, t, onInput);
+ };
+ }
+ i2.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return i2;
+ };
+ i2.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") : void 0).attr("placeholder", isMixed ? _t("inspector.multiple_values") : field.placeholder() || _t("inspector.unknown")).classed("mixed", isMixed);
+ if (field.type === "number") {
+ const buttons = wrap2.selectAll(".increment, .decrement");
+ if (isMixed) {
+ buttons.attr("disabled", "disabled").classed("disabled", true);
+ } else {
+ var raw_vals = tags[field.key] || "0";
+ const canIncDec = raw_vals.split(";").some((val) => isFinite(Number(val)) || isDirectionField && cardinal[val.trim().toLowerCase()]);
+ buttons.attr("disabled", canIncDec ? null : "disabled").classed("disabled", !canIncDec);
+ }
+ }
+ if (field.type === "tel")
+ updatePhonePlaceholder();
+ if (field.key.split(":").includes("colour"))
+ updateColourPreview();
+ if (outlinkButton && !outlinkButton.empty()) {
+ var disabled = !validIdentifierValueForLink();
+ outlinkButton.classed("disabled", disabled);
+ }
+ if (!isMixed) {
+ _lengthIndicator.update(tags[field.key]);
+ }
+ };
+ i2.focus = function() {
+ var node = input.node();
+ if (node)
+ node.focus();
+ };
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ return utilRebind(i2, dispatch10, "on");
+ }
+
+ // modules/ui/fields/access.js
+ function uiFieldAccess(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var items = select_default2(null);
+ var _tags;
+ function access(selection2) {
+ var wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ var list = wrap2.selectAll("ul").data([0]);
+ list = list.enter().append("ul").attr("class", "rows").merge(list);
+ items = list.selectAll("li").data(field.keys);
+ 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_default2(this).call(
+ uiCombobox(context, "access-" + d).data(access.options(d))
+ );
+ });
+ items = items.merge(enter);
+ wrap2.selectAll(".preset-input-access").on("change", change).on("blur", change);
+ }
+ function change(d3_event, d) {
+ var tag = {};
+ var value = context.cleanTagValue(utilGetSetValue(select_default2(this)));
+ if (!value && typeof _tags[d] !== "string")
+ return;
+ tag[d] = value || void 0;
+ dispatch10.call("change", this, tag);
+ }
+ access.options = function(type2) {
+ var options2 = [
+ "yes",
+ "no",
+ "designated",
+ "permissive",
+ "destination",
+ "customers",
+ "private",
+ "permit",
+ "unknown"
+ ];
+ if (type2 === "access") {
+ options2 = options2.filter((v) => v !== "yes" && v !== "designated");
+ }
+ if (type2 === "bicycle") {
+ options2.splice(options2.length - 4, 0, "dismount");
+ }
+ var stringsField = field.resolveReference("stringsCrossReference");
+ return options2.map(function(option) {
+ return {
+ title: stringsField.t("options." + option + ".description"),
+ value: option
+ };
+ });
+ };
+ const placeholdersByTag = {
+ highway: {
+ 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"
+ },
+ construction: {
+ access: "no"
+ }
+ },
+ barrier: {
+ bollard: {
+ access: "no",
+ bicycle: "yes",
+ foot: "yes"
+ },
+ bus_trap: {
+ motor_vehicle: "no",
+ psv: "yes",
+ foot: "yes",
+ bicycle: "yes"
+ },
+ city_wall: {
+ access: "no"
+ },
+ coupure: {
+ access: "yes"
+ },
+ cycle_barrier: {
+ motor_vehicle: "no"
+ },
+ ditch: {
+ access: "no"
+ },
+ entrance: {
+ access: "yes"
+ },
+ fence: {
+ access: "no"
+ },
+ hedge: {
+ access: "no"
+ },
+ jersey_barrier: {
+ access: "no"
+ },
+ motorcycle_barrier: {
+ motor_vehicle: "no"
+ },
+ rail_guard: {
+ access: "no"
+ }
+ }
+ };
+ 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 (d === "bicycle" || d === "motor_vehicle") {
+ if (tags.vehicle && typeof tags.vehicle === "string") {
+ return tags.vehicle;
+ }
+ }
+ if (tags.access && typeof tags.access === "string") {
+ return tags.access;
+ }
+ function getPlaceholdersByTag(key, placeholdersByKey) {
+ if (typeof tags[key] === "string") {
+ if (placeholdersByKey[tags[key]] && placeholdersByKey[tags[key]][d]) {
+ return placeholdersByKey[tags[key]][d];
+ }
+ } else {
+ var impliedAccesses = tags[key].filter(Boolean).map(function(val) {
+ return placeholdersByKey[val] && placeholdersByKey[val][d];
+ }).filter(Boolean);
+ if (impliedAccesses.length === tags[key].length && new Set(impliedAccesses).size === 1) {
+ return impliedAccesses[0];
+ }
+ }
+ }
+ for (const key in placeholdersByTag) {
+ if (tags[key]) {
+ const impliedAccess = getPlaceholdersByTag(key, placeholdersByTag[key]);
+ if (impliedAccess) {
+ return impliedAccess;
+ }
+ }
+ }
+ if (d === "access" && !tags.barrier) {
+ return "yes";
+ }
+ return field.placeholder();
+ });
+ };
+ access.focus = function() {
+ items.selectAll(".preset-input-access").node().focus();
+ };
+ return utilRebind(access, dispatch10, "on");
+ }
+
+ // modules/ui/fields/address.js
+ function uiFieldAddress(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var _selection = select_default2(null);
+ var _wrap = select_default2(null);
+ var addrField = _mainPresetIndex.field("address");
+ var _entityIDs = [];
+ var _tags;
+ var _countryCode;
+ var _addressFormats = [{
+ format: [
+ ["housenumber", "street"],
+ ["city", "postcode"]
+ ]
+ }];
+ _mainFileFetcher.get("address_formats").then(function(d) {
+ _addressFormats = d;
+ if (!_selection.empty()) {
+ _selection.call(address);
+ }
+ }).catch(function() {
+ });
+ 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 isAddressable(d) {
+ return d.tags.highway && d.tags.name && d.type === "way";
+ }
+ }
+ 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");
+ 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;
+ }
+ if (d.tags["addr:city"])
+ return true;
+ return false;
+ }
+ }
+ 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");
+ }
+ function updateForCountryCode() {
+ if (!_countryCode)
+ return;
+ var addressFormat;
+ for (var i2 = 0; i2 < _addressFormats.length; i2++) {
+ var format2 = _addressFormats[i2];
+ if (!format2.countryCodes) {
+ addressFormat = format2;
+ } else if (format2.countryCodes.indexOf(_countryCode) !== -1) {
+ addressFormat = format2;
+ break;
+ }
+ }
+ 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 row(r) {
+ 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();
+ });
+ 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 + "%";
+ });
+ function addDropdown(d) {
+ if (dropdowns.indexOf(d.id) === -1)
+ return;
+ var nearValues = d.id === "street" ? getNearStreets : d.id === "city" ? getNearCities : getNearValues;
+ select_default2(this).call(
+ uiCombobox(context, "address-" + d.id).minItems(1).caseSensitive(true).fetcher(function(value, callback) {
+ callback(nearValues("addr:" + d.id));
+ })
+ );
+ }
+ _wrap.selectAll("input").on("blur", change()).on("change", change());
+ _wrap.selectAll("input:not(.combobox-input)").on("input", change(true));
+ if (_tags)
+ updateTags(_tags);
+ }
+ function address(selection2) {
+ _selection = selection2;
+ _wrap = selection2.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 (extent) {
+ var countryCode;
+ if (context.inIntro()) {
+ countryCode = _t("intro.graph.countrycode");
+ } else {
+ var center = extent.center();
+ countryCode = iso1A2Code(center);
+ }
+ if (countryCode) {
+ _countryCode = countryCode.toLowerCase();
+ updateForCountryCode();
+ }
+ }
+ }
+ function change(onInput) {
+ return function() {
+ var tags = {};
+ _wrap.selectAll("input").each(function(subfield) {
+ var key = field.key + ":" + subfield.id;
+ var value = this.value;
+ if (!onInput)
+ value = context.cleanTagValue(value);
+ if (Array.isArray(_tags[key]) && !value)
+ return;
+ tags[key] = value || void 0;
+ });
+ dispatch10.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");
+ }
+ if (_countryCode) {
+ var localkey = subfield.id + "!" + _countryCode;
+ var tkey = addrField.hasTextForStringId("placeholders." + localkey) ? localkey : subfield.id;
+ return addrField.t("placeholders." + tkey);
+ }
+ });
+ }
+ 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") : void 0;
+ }).classed("mixed", function(subfield) {
+ return Array.isArray(tags[field.key + ":" + subfield.id]);
+ }).call(updatePlaceholder);
+ }
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ 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();
+ if (node)
+ node.focus();
+ };
+ return utilRebind(address, dispatch10, "on");
+ }
+
+ // modules/ui/fields/directional_combo.js
+ function uiFieldDirectionalCombo(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var items = select_default2(null);
+ var wrap2 = select_default2(null);
+ var _tags;
+ var _combos = {};
+ function directionalCombo(selection2) {
+ function stripcolon(s) {
+ return s.replace(":", "");
+ }
+ wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ var div = wrap2.selectAll("ul").data([0]);
+ div = div.enter().append("ul").attr("class", "rows").merge(div);
+ var keys = field.keys.slice(1);
+ items = div.selectAll("li").data(keys);
+ var enter = items.enter().append("li").attr("class", function(d) {
+ return "labeled-input preset-directionalcombo-" + stripcolon(d);
+ });
+ enter.append("span").attr("class", "label preset-label-directionalcombo").attr("for", function(d) {
+ return "preset-input-directionalcombo-" + stripcolon(d);
+ }).html(function(d) {
+ return field.t.html("types." + d);
+ });
+ enter.append("div").attr("class", "preset-input-directionalcombo-wrap form-field-input-wrap").each(function(key) {
+ const subField = {
+ ...field,
+ type: "combo",
+ key
+ };
+ const combo = uiFieldCombo(subField, context);
+ combo.on("change", (t) => change(key, t[key]));
+ _combos[key] = combo;
+ select_default2(this).call(combo);
+ });
+ items = items.merge(enter);
+ wrap2.selectAll(".preset-input-directionalcombo").on("change", change).on("blur", change);
+ }
+ function change(key, newValue) {
+ const commonKey = field.keys[0];
+ const otherKey = key === field.keys[1] ? field.keys[2] : field.keys[1];
+ dispatch10.call("change", this, (tags) => {
+ const otherValue = tags[otherKey] || tags[commonKey];
+ if (newValue === otherValue) {
+ tags[commonKey] = newValue;
+ delete tags[key];
+ delete tags[otherKey];
+ } else {
+ tags[key] = newValue;
+ delete tags[commonKey];
+ tags[otherKey] = otherValue;
+ }
+ return tags;
+ });
+ }
+ directionalCombo.tags = function(tags) {
+ _tags = tags;
+ const commonKey = field.keys[0];
+ for (let key in _combos) {
+ const uniqueValues = [...new Set([].concat(_tags[commonKey]).concat(_tags[key]).filter(Boolean))];
+ _combos[key].tags({ [key]: uniqueValues.length > 1 ? uniqueValues : uniqueValues[0] });
+ }
+ };
+ directionalCombo.focus = function() {
+ var node = wrap2.selectAll("input").node();
+ if (node)
+ node.focus();
+ };
+ return utilRebind(directionalCombo, dispatch10, "on");
+ }
+
+ // modules/ui/fields/lanes.js
+ function uiFieldLanes(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var LANE_WIDTH = 40;
+ var LANE_HEIGHT = 200;
+ var _entityIDs = [];
+ function lanes(selection2) {
+ var lanesData = context.entity(_entityIDs[0]).lanes();
+ if (!context.container().select(".inspector-wrap.inspector-hidden").empty() || !selection2.node().parentNode) {
+ selection2.call(lanes.off);
+ return;
+ }
+ var wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ var surface = wrap2.selectAll(".surface").data([0]);
+ var d = utilGetDimensions(wrap2);
+ 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).text("\u25B2");
+ enter.append("g").attr("class", "bothways").append("text").attr("y", 40).attr("x", 14).text("\u25B2\u25BC");
+ enter.append("g").attr("class", "backward").append("text").attr("y", 40).attr("x", 14).text("\u25BC");
+ lane = lane.merge(enter);
+ lane.attr("transform", function(d2) {
+ return "translate(" + LANE_WIDTH * d2.index * 1.5 + ", 0)";
+ });
+ lane.select(".forward").style("visibility", function(d2) {
+ return d2.direction === "forward" ? "visible" : "hidden";
+ });
+ lane.select(".bothways").style("visibility", function(d2) {
+ return d2.direction === "bothways" ? "visible" : "hidden";
+ });
+ lane.select(".backward").style("visibility", function(d2) {
+ return d2.direction === "backward" ? "visible" : "hidden";
+ });
+ }
+ lanes.entityIDs = function(val) {
+ _entityIDs = val;
+ };
+ lanes.tags = function() {
+ };
+ lanes.focus = function() {
+ };
+ lanes.off = function() {
+ };
+ return utilRebind(lanes, dispatch10, "on");
+ }
+ uiFieldLanes.supportsMultiselection = false;
+
+ // modules/ui/fields/localized.js
+ var _languagesArray = [];
+ function uiFieldLocalized(field, context) {
+ var dispatch10 = dispatch_default("change", "input");
+ var wikipedia = services.wikipedia;
+ var input = select_default2(null);
+ var localizedInputs = select_default2(null);
+ var _lengthIndicator = uiLengthIndicator(context.maxCharsForTagValue());
+ var _countryCode;
+ var _tags;
+ _mainFileFetcher.get("languages").then(loadLanguagesArray).catch(function() {
+ });
+ var _territoryLanguages = {};
+ _mainFileFetcher.get("territory_languages").then(function(d) {
+ _territoryLanguages = d;
+ }).catch(function() {
+ });
+ var langCombo = uiCombobox(context, "localized-lang").fetcher(fetchLanguages).minItems(0);
+ var _selection = select_default2(null);
+ var _multilingual = [];
+ var _buttonTip = uiTooltip().title(() => _t.append("translate.translate")).placement("left");
+ var _wikiTitles;
+ var _entityIDs = [];
+ function loadLanguagesArray(dataLanguages) {
+ if (_languagesArray.length !== 0)
+ return;
+ var replacements = {
+ sr: "sr-Cyrl",
+ // in OSM, `sr` implies Cyrillic
+ "sr-Cyrl": false
+ // `sr-Cyrl` isn't used in OSM
+ };
+ for (var code in dataLanguages) {
+ if (replacements[code] === false)
+ continue;
+ var metaCode = code;
+ if (replacements[code])
+ metaCode = replacements[code];
+ _languagesArray.push({
+ localName: _mainLocalizer.languageName(metaCode, { localOnly: true }),
+ nativeName: dataLanguages[metaCode].nativeName,
+ code,
+ label: _mainLocalizer.languageName(metaCode)
+ });
+ }
+ }
+ function calcLocked() {
+ var isLocked = field.id === "name" && _entityIDs.length && _entityIDs.some(function(entityID) {
+ var entity = context.graph().hasEntity(entityID);
+ if (!entity)
+ return false;
+ if (entity.tags.wikidata)
+ return true;
+ if (entity.tags["name:etymology:wikidata"])
+ return true;
+ var preset = _mainPresetIndex.match(entity, context.graph());
+ 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 false;
+ });
+ field.locked(isLocked);
+ }
+ function calcMultilingual(tags) {
+ var existingLangsOrdered = _multilingual.map(function(item2) {
+ return item2.lang;
+ });
+ var existingLangs = new Set(existingLangsOrdered.filter(Boolean));
+ for (var k in tags) {
+ var m = k.match(/^(.*):([a-z]{2,3}(?:-[A-Z][a-z]{3})?(?:-[A-Z]{2})?)$/);
+ if (m && m[1] === field.key && m[2]) {
+ var item = { lang: m[2], value: tags[k] };
+ if (existingLangs.has(item.lang)) {
+ _multilingual[existingLangsOrdered.indexOf(item.lang)].value = item.value;
+ existingLangs.delete(item.lang);
+ } else {
+ _multilingual.push(item);
+ }
+ }
+ }
+ _multilingual.forEach(function(item2) {
+ if (item2.lang && existingLangs.has(item2.lang)) {
+ item2.value = "";
+ }
+ });
+ }
+ function localized(selection2) {
+ _selection = selection2;
+ calcLocked();
+ var isLocked = field.locked();
+ var wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ input = wrap2.selectAll(".localized-main").data([0]);
+ 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());
+ wrap2.call(_lengthIndicator);
+ var translateButton = wrap2.selectAll(".localized-add").data([0]);
+ translateButton = translateButton.enter().append("button").attr("class", "localized-add form-field-button").attr("aria-label", _t("icons.plus")).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 = selection2.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);
+ selection2.selectAll(".combobox-caret").classed("nope", true);
+ function addNew(d3_event) {
+ d3_event.preventDefault();
+ if (field.locked())
+ return;
+ var defaultLang = _mainLocalizer.languageCode().toLowerCase();
+ var langExists = _multilingual.find(function(datum2) {
+ return datum2.lang === defaultLang;
+ });
+ var isLangEn = defaultLang.indexOf("en") > -1;
+ if (isLangEn || langExists) {
+ defaultLang = "";
+ langExists = _multilingual.find(function(datum2) {
+ return datum2.lang === defaultLang;
+ });
+ }
+ if (!langExists) {
+ _multilingual.unshift({ lang: defaultLang, value: "" });
+ localizedInputs.call(renderMultilingual);
+ }
+ }
+ function change(onInput) {
+ return function(d3_event) {
+ if (field.locked()) {
+ d3_event.preventDefault();
+ return;
+ }
+ var val = utilGetSetValue(select_default2(this));
+ if (!onInput)
+ val = context.cleanTagValue(val);
+ if (!val && Array.isArray(_tags[field.key]))
+ return;
+ var t = {};
+ t[field.key] = val || void 0;
+ dispatch10.call("change", this, t, onInput);
+ };
+ }
+ }
+ function key(lang) {
+ return field.key + ":" + lang;
+ }
+ function changeLang(d3_event, d) {
+ var tags = {};
+ var lang = utilGetSetValue(select_default2(this)).toLowerCase();
+ var language = _languagesArray.find(function(d2) {
+ return d2.label.toLowerCase() === lang || d2.localName && d2.localName.toLowerCase() === lang || d2.nativeName && d2.nativeName.toLowerCase() === lang;
+ });
+ if (language)
+ lang = language.code;
+ if (d.lang && d.lang !== lang) {
+ tags[key(d.lang)] = void 0;
+ }
+ var newKey = lang && context.cleanTagKey(key(lang));
+ var value = utilGetSetValue(select_default2(this.parentNode).selectAll(".localized-value"));
+ if (newKey && value) {
+ tags[newKey] = value;
+ } else if (newKey && _wikiTitles && _wikiTitles[d.lang]) {
+ tags[newKey] = _wikiTitles[d.lang];
+ }
+ d.lang = lang;
+ dispatch10.call("change", this, tags);
+ }
+ function changeValue(d3_event, d) {
+ if (!d.lang)
+ return;
+ var value = context.cleanTagValue(utilGetSetValue(select_default2(this))) || void 0;
+ if (!value && Array.isArray(d.value))
+ return;
+ var t = {};
+ t[key(d.lang)] = value;
+ d.value = value;
+ dispatch10.call("change", this, t);
+ }
+ function fetchLanguages(value, cb) {
+ var v = value.toLowerCase();
+ 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 renderMultilingual(selection2) {
+ var entries = selection2.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 wrap2 = select_default2(this);
+ var domId = utilUniqueDomId(index);
+ var label = wrap2.append("label").attr("class", "field-label").attr("for", domId);
+ var text2 = label.append("span").attr("class", "label-text");
+ text2.append("span").attr("class", "label-textvalue").call(_t.append("translate.localized_translation_label"));
+ text2.append("span").attr("class", "label-textannotation");
+ label.append("button").attr("class", "remove-icon-multilingual").attr("title", _t("icons.remove")).on("click", function(d3_event, d) {
+ if (field.locked())
+ return;
+ d3_event.preventDefault();
+ _multilingual.splice(_multilingual.indexOf(d), 1);
+ var langKey = d.lang && key(d.lang);
+ if (langKey && langKey in _tags) {
+ delete _tags[langKey];
+ var t = {};
+ t[langKey] = void 0;
+ dispatch10.call("change", this, t);
+ return;
+ }
+ renderMultilingual(selection2);
+ }).call(svgIcon("#iD-operation-delete"));
+ wrap2.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);
+ wrap2.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_default2(this).style("max-height", "").style("overflow", "visible");
+ });
+ entries = entries.merge(entriesEnter);
+ entries.order();
+ 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;
+ 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;
+ });
+ }
+ }
+ 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") : void 0).attr("placeholder", isMixed ? _t("inspector.multiple_values") : field.placeholder()).classed("mixed", isMixed);
+ calcMultilingual(tags);
+ _selection.call(localized);
+ if (!isMixed) {
+ _lengthIndicator.update(tags[field.key]);
+ }
+ };
+ 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();
+ }
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ return utilRebind(localized, dispatch10, "on");
+ }
+
+ // modules/ui/fields/roadheight.js
+ function uiFieldRoadheight(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var primaryUnitInput = select_default2(null);
+ var primaryInput = select_default2(null);
+ var secondaryInput = select_default2(null);
+ var secondaryUnitInput = select_default2(null);
+ var _entityIDs = [];
+ var _tags;
+ var _isImperial;
+ var primaryUnits = [
+ {
+ value: "m",
+ title: _t("inspector.roadheight.meter")
+ },
+ {
+ value: "ft",
+ title: _t("inspector.roadheight.foot")
+ }
+ ];
+ var unitCombo = uiCombobox(context, "roadheight-unit").data(primaryUnits);
+ function roadheight(selection2) {
+ var wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ primaryInput = wrap2.selectAll("input.roadheight-number").data([0]);
+ primaryInput = primaryInput.enter().append("input").attr("type", "text").attr("class", "roadheight-number").attr("id", field.domId).call(utilNoAuto).merge(primaryInput);
+ primaryInput.on("change", change).on("blur", change);
+ var loc = combinedEntityExtent().center();
+ _isImperial = roadHeightUnit(loc) === "ft";
+ primaryUnitInput = wrap2.selectAll("input.roadheight-unit").data([0]);
+ primaryUnitInput = primaryUnitInput.enter().append("input").attr("type", "text").attr("class", "roadheight-unit").call(unitCombo).merge(primaryUnitInput);
+ primaryUnitInput.on("blur", changeUnits).on("change", changeUnits);
+ secondaryInput = wrap2.selectAll("input.roadheight-secondary-number").data([0]);
+ secondaryInput = secondaryInput.enter().append("input").attr("type", "text").attr("class", "roadheight-secondary-number").call(utilNoAuto).merge(secondaryInput);
+ secondaryInput.on("change", change).on("blur", change);
+ secondaryUnitInput = wrap2.selectAll("input.roadheight-secondary-unit").data([0]);
+ secondaryUnitInput = secondaryUnitInput.enter().append("input").attr("type", "text").call(utilNoAuto).classed("disabled", true).classed("roadheight-secondary-unit", true).attr("readonly", "readonly").merge(secondaryUnitInput);
+ function changeUnits() {
+ var primaryUnit = utilGetSetValue(primaryUnitInput);
+ if (primaryUnit === "m") {
+ _isImperial = false;
+ } else if (primaryUnit === "ft") {
+ _isImperial = true;
+ }
+ utilGetSetValue(primaryUnitInput, _isImperial ? "ft" : "m");
+ setUnitSuggestions();
+ change();
+ }
+ }
+ function setUnitSuggestions() {
+ utilGetSetValue(primaryUnitInput, _isImperial ? "ft" : "m");
+ }
+ function change() {
+ var tag = {};
+ var primaryValue = utilGetSetValue(primaryInput).trim();
+ var secondaryValue = utilGetSetValue(secondaryInput).trim();
+ if (!primaryValue && !secondaryValue && Array.isArray(_tags[field.key]))
+ return;
+ if (!primaryValue && !secondaryValue) {
+ tag[field.key] = void 0;
+ } else if (isNaN(primaryValue) || isNaN(secondaryValue) || !_isImperial) {
+ tag[field.key] = context.cleanTagValue(primaryValue);
+ } else {
+ if (primaryValue !== "") {
+ primaryValue = context.cleanTagValue(primaryValue + "'");
+ }
+ if (secondaryValue !== "") {
+ secondaryValue = context.cleanTagValue(secondaryValue + '"');
+ }
+ tag[field.key] = primaryValue + secondaryValue;
+ }
+ dispatch10.call("change", this, tag);
+ }
+ roadheight.tags = function(tags) {
+ _tags = tags;
+ var primaryValue = tags[field.key];
+ var secondaryValue;
+ var isMixed = Array.isArray(primaryValue);
+ if (!isMixed) {
+ if (primaryValue && (primaryValue.indexOf("'") >= 0 || primaryValue.indexOf('"') >= 0)) {
+ secondaryValue = primaryValue.match(/(-?[\d.]+)"/);
+ if (secondaryValue !== null) {
+ secondaryValue = secondaryValue[1];
+ }
+ primaryValue = primaryValue.match(/(-?[\d.]+)'/);
+ if (primaryValue !== null) {
+ primaryValue = primaryValue[1];
+ }
+ _isImperial = true;
+ } else if (primaryValue) {
+ _isImperial = false;
+ }
+ }
+ setUnitSuggestions();
+ utilGetSetValue(primaryInput, typeof primaryValue === "string" ? primaryValue : "").attr("title", isMixed ? primaryValue.filter(Boolean).join("\n") : null).attr("placeholder", isMixed ? _t("inspector.multiple_values") : _t("inspector.unknown")).classed("mixed", isMixed);
+ utilGetSetValue(secondaryInput, typeof secondaryValue === "string" ? secondaryValue : "").attr("placeholder", isMixed ? _t("inspector.multiple_values") : _isImperial ? "0" : null).classed("mixed", isMixed).classed("disabled", !_isImperial).attr("readonly", _isImperial ? null : "readonly");
+ secondaryUnitInput.attr("value", _isImperial ? _t("inspector.roadheight.inch") : null);
+ };
+ roadheight.focus = function() {
+ primaryInput.node().focus();
+ };
+ roadheight.entityIDs = function(val) {
+ _entityIDs = val;
+ };
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ return utilRebind(roadheight, dispatch10, "on");
+ }
+
+ // modules/ui/fields/roadspeed.js
+ function uiFieldRoadspeed(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var unitInput = select_default2(null);
+ var input = select_default2(null);
+ var _entityIDs = [];
+ var _tags;
+ var _isImperial;
+ 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];
+ function roadspeed(selection2) {
+ var wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ input = wrap2.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 = wrap2.selectAll("input.roadspeed-unit").data([0]);
+ unitInput = unitInput.enter().append("input").attr("type", "text").attr("class", "roadspeed-unit").attr("aria-label", _t("inspector.speed_unit")).call(unitCombo).merge(unitInput);
+ unitInput.on("blur", changeUnits).on("change", changeUnits);
+ function changeUnits() {
+ var unit2 = utilGetSetValue(unitInput);
+ if (unit2 === "km/h") {
+ _isImperial = false;
+ } else if (unit2 === "mph") {
+ _isImperial = true;
+ }
+ utilGetSetValue(unitInput, _isImperial ? "mph" : "km/h");
+ setUnitSuggestions();
+ change();
+ }
+ }
+ function setUnitSuggestions() {
+ speedCombo.data((_isImperial ? imperialValues : metricValues).map(comboValues));
+ utilGetSetValue(unitInput, _isImperial ? "mph" : "km/h");
+ }
+ function comboValues(d) {
+ return {
+ value: d.toString(),
+ title: d.toString()
+ };
+ }
+ function change() {
+ var tag = {};
+ var value = utilGetSetValue(input).trim();
+ if (!value && Array.isArray(_tags[field.key]))
+ return;
+ if (!value) {
+ tag[field.key] = void 0;
+ } else if (isNaN(value) || !_isImperial) {
+ tag[field.key] = context.cleanTagValue(value);
+ } else {
+ tag[field.key] = context.cleanTagValue(value + " mph");
+ }
+ dispatch10.call("change", this, tag);
+ }
+ roadspeed.tags = function(tags) {
+ _tags = tags;
+ var value = tags[field.key];
+ var isMixed = Array.isArray(value);
+ if (!isMixed) {
+ if (value && value.indexOf("mph") >= 0) {
+ value = parseInt(value, 10).toString();
+ _isImperial = true;
+ } else if (value) {
+ _isImperial = false;
+ }
+ }
+ 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);
+ };
+ roadspeed.focus = function() {
+ input.node().focus();
+ };
+ roadspeed.entityIDs = function(val) {
+ _entityIDs = val;
+ };
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ return utilRebind(roadspeed, dispatch10, "on");
+ }
+
+ // modules/ui/fields/radio.js
+ function uiFieldRadio(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var placeholder = select_default2(null);
+ var wrap2 = select_default2(null);
+ var labels = select_default2(null);
+ var radios = select_default2(null);
+ var radioData = (field.options || field.keys).slice();
+ var typeField;
+ var layerField;
+ var _oldType = {};
+ var _entityIDs = [];
+ function selectedKey() {
+ var node = wrap2.selectAll(".form-field-input-radio label.active input");
+ return !node.empty() && node.datum();
+ }
+ function radio(selection2) {
+ selection2.classed("preset-radio", true);
+ wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ var enter = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-radio");
+ enter.append("span").attr("class", "placeholder");
+ wrap2 = wrap2.merge(enter);
+ placeholder = wrap2.selectAll(".placeholder");
+ labels = wrap2.selectAll("label").data(radioData);
+ enter = labels.enter().append("label");
+ var stringsField = field.resolveReference("stringsCrossReference");
+ enter.append("input").attr("type", "radio").attr("name", field.id).attr("value", function(d) {
+ return stringsField.t("options." + d, { "default": d });
+ }).attr("checked", false);
+ enter.append("span").each(function(d) {
+ stringsField.t.append("options." + d, { "default": d })(select_default2(this));
+ });
+ labels = labels.merge(enter);
+ radios = labels.selectAll("input").on("change", changeRadio);
+ }
+ function structureExtras(selection2, tags) {
+ var selected = selectedKey() || tags.layer !== void 0;
+ var type2 = _mainPresetIndex.field(selected);
+ var layer = _mainPresetIndex.field("layer");
+ var showLayer = selected === "bridge" || selected === "tunnel" || tags.layer !== void 0;
+ var extrasWrap = selection2.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);
+ if (type2) {
+ if (!typeField || typeField.id !== selected) {
+ typeField = uiField(context, type2, _entityIDs, { wrap: false }).on("change", changeType);
+ }
+ typeField.tags(tags);
+ } else {
+ typeField = null;
+ }
+ var typeItem = list.selectAll(".structure-type-item").data(typeField ? [typeField] : [], function(d) {
+ return d.id;
+ });
+ typeItem.exit().remove();
+ 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).call(_t.append("inspector.radio.structure.type"));
+ typeEnter.append("div").attr("class", "structure-input-type-wrap");
+ typeItem = typeItem.merge(typeEnter);
+ if (typeField) {
+ typeItem.selectAll(".structure-input-type-wrap").call(typeField.render);
+ }
+ if (layer && showLayer) {
+ if (!layerField) {
+ layerField = uiField(context, layer, _entityIDs, { wrap: false }).on("change", changeLayer);
+ }
+ layerField.tags(tags);
+ field.keys = utilArrayUnion(field.keys, ["layer"]);
+ } else {
+ layerField = null;
+ field.keys = field.keys.filter(function(k) {
+ return k !== "layer";
+ });
+ }
+ var layerItem = list.selectAll(".structure-layer-item").data(layerField ? [layerField] : []);
+ layerItem.exit().remove();
+ 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").call(_t.append("inspector.radio.structure.layer"));
+ layerEnter.append("div").attr("class", "structure-input-layer-wrap");
+ layerItem = layerItem.merge(layerEnter);
+ if (layerField) {
+ layerItem.selectAll(".structure-input-layer-wrap").call(layerField.render);
+ }
+ }
+ function changeType(t, onInput) {
+ var key = selectedKey();
+ if (!key)
+ return;
+ var val = t[key];
+ if (val !== "no") {
+ _oldType[key] = val;
+ }
+ if (field.type === "structureRadio") {
+ if (val === "no" || key !== "bridge" && key !== "tunnel" || key === "tunnel" && val === "building_passage") {
+ t.layer = void 0;
+ }
+ if (t.layer === void 0) {
+ if (key === "bridge" && val !== "no") {
+ t.layer = "1";
+ }
+ if (key === "tunnel" && val !== "no" && val !== "building_passage") {
+ t.layer = "-1";
+ }
+ }
+ }
+ dispatch10.call("change", this, t, onInput);
+ }
+ function changeLayer(t, onInput) {
+ if (t.layer === "0") {
+ t.layer = void 0;
+ }
+ dispatch10.call("change", this, t, onInput);
+ }
+ function changeRadio() {
+ var t = {};
+ var activeKey;
+ if (field.key) {
+ t[field.key] = void 0;
+ }
+ radios.each(function(d) {
+ var active = select_default2(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 : void 0;
+ }
+ });
+ if (field.type === "structureRadio") {
+ if (activeKey === "bridge") {
+ t.layer = "1";
+ } else if (activeKey === "tunnel" && t.tunnel !== "building_passage") {
+ t.layer = "-1";
+ } else {
+ t.layer = void 0;
+ }
+ }
+ dispatch10.call("change", this, t);
+ }
+ radio.tags = function(tags) {
+ function isOptionChecked(d) {
+ if (field.key) {
+ return tags[field.key] === d;
+ }
+ return !!(typeof tags[d] === "string" && tags[d].toLowerCase() !== "no");
+ }
+ function isMixed(d) {
+ if (field.key) {
+ return Array.isArray(tags[field.key]) && tags[field.key].includes(d);
+ }
+ return Array.isArray(tags[d]);
+ }
+ radios.property("checked", function(d) {
+ return isOptionChecked(d) && (field.key || field.options.filter(isOptionChecked).length === 1);
+ });
+ 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].some((v) => typeof v === "string" && v.toLowerCase() !== "no") || !!(typeof tags[d] === "string" && tags[d].toLowerCase() !== "no");
+ }).classed("mixed", isMixed).attr("title", function(d) {
+ return isMixed(d) ? _t("inspector.unshared_value_tooltip") : null;
+ });
+ var selection2 = radios.filter(function() {
+ return this.checked;
+ });
+ if (selection2.empty()) {
+ placeholder.text("");
+ placeholder.call(_t.append("inspector.none"));
+ } else {
+ placeholder.text(selection2.attr("value"));
+ _oldType[selection2.datum()] = tags[selection2.datum()];
+ }
+ if (field.type === "structureRadio") {
+ if (!!tags.waterway && !_oldType.tunnel) {
+ _oldType.tunnel = "culvert";
+ }
+ wrap2.call(structureExtras, tags);
+ }
+ };
+ radio.focus = function() {
+ radios.node().focus();
+ };
+ radio.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ _oldType = {};
+ return radio;
+ };
+ radio.isAllowed = function() {
+ return _entityIDs.length === 1;
+ };
+ return utilRebind(radio, dispatch10, "on");
+ }
+
+ // modules/ui/fields/restrictions.js
+ function uiFieldRestrictions(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var breathe = behaviorBreathe(context);
+ corePreferences("turn-restriction-via-way", null);
+ var storedViaWay = corePreferences("turn-restriction-via-way0");
+ var storedDistance = corePreferences("turn-restriction-distance");
+ var _maxViaWay = storedViaWay !== null ? +storedViaWay : 0;
+ var _maxDistance = storedDistance ? +storedDistance : 30;
+ var _initialized2 = false;
+ var _parent = select_default2(null);
+ var _container = select_default2(null);
+ var _oldTurns;
+ var _graph;
+ var _vertexID;
+ var _intersection;
+ var _fromWayID;
+ var _lastXPos;
+ function restrictions(selection2) {
+ _parent = selection2;
+ if (_vertexID && (context.graph() !== _graph || !_intersection)) {
+ _graph = context.graph();
+ _intersection = osmIntersection(_graph, _vertexID, _maxDistance);
+ }
+ var isOK = _intersection && _intersection.vertices.length && // has vertices
+ _intersection.vertices.filter(function(vertex) {
+ return vertex.id === _vertexID;
+ }).length && _intersection.ways.length > 2 && // has more than 2 ways
+ _intersection.ways.filter(function(way) {
+ return way.__to;
+ }).length > 1;
+ select_default2(selection2.node().parentNode).classed("hide", !isOK);
+ if (!isOK || !context.container().select(".inspector-wrap.inspector-hidden").empty() || !selection2.node().parentNode || !selection2.node().parentNode.parentNode) {
+ selection2.call(restrictions.off);
+ return;
+ }
+ var wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ var container = wrap2.selectAll(".restriction-container").data([0]);
+ var containerEnter = container.enter().append("div").attr("class", "restriction-container");
+ containerEnter.append("div").attr("class", "restriction-help");
+ _container = containerEnter.merge(container).call(renderViewer);
+ var controls = wrap2.selectAll(".restriction-controls").data([0]);
+ controls.enter().append("div").attr("class", "restriction-controls-container").append("div").attr("class", "restriction-controls").merge(controls).call(renderControls);
+ }
+ function renderControls(selection2) {
+ var distControl = selection2.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").call(_t.append("restriction.controls.distance", { suffix: ":" }));
+ 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");
+ selection2.selectAll(".restriction-distance-input").property("value", _maxDistance).on("input", function() {
+ var val = select_default2(this).property("value");
+ _maxDistance = +val;
+ _intersection = null;
+ _container.selectAll(".layer-osm .layer-turns *").remove();
+ corePreferences("turn-restriction-distance", _maxDistance);
+ _parent.call(restrictions);
+ });
+ selection2.selectAll(".restriction-distance-text").call(displayMaxDistance(_maxDistance));
+ var viaControl = selection2.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").call(_t.append("restriction.controls.via", { suffix: ":" }));
+ 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");
+ selection2.selectAll(".restriction-via-way-input").property("value", _maxViaWay).on("input", function() {
+ var val = select_default2(this).property("value");
+ _maxViaWay = +val;
+ _container.selectAll(".layer-osm .layer-turns *").remove();
+ corePreferences("turn-restriction-via-way0", _maxViaWay);
+ _parent.call(restrictions);
+ });
+ selection2.selectAll(".restriction-via-way-text").call(displayMaxVia(_maxViaWay));
+ }
+ function renderViewer(selection2) {
+ if (!_intersection)
+ return;
+ var vgraph = _intersection.graph;
+ var filter2 = utilFunctor(true);
+ var projection2 = geoRawMercator();
+ var sdims = utilGetDimensions(context.container().select(".sidebar"));
+ var d = [sdims[0] - 50, 370];
+ var c = geoVecScale(d, 0.5);
+ var z = 22;
+ projection2.scale(geoZoomToScale(z));
+ var extent = geoExtent();
+ for (var i2 = 0; i2 < _intersection.vertices.length; i2++) {
+ extent._extend(_intersection.vertices[i2].extent());
+ }
+ var padTop = 35;
+ if (_intersection.vertices.length > 1) {
+ var hPadding = Math.min(160, Math.max(110, d[0] * 0.4));
+ var vPadding = 160;
+ var tl = projection2([extent[0][0], extent[1][1]]);
+ var br = projection2([extent[1][0], extent[0][1]]);
+ var hFactor = (br[0] - tl[0]) / (d[0] - hPadding);
+ var vFactor = (br[1] - tl[1]) / (d[1] - vPadding - padTop);
+ var hZoomDiff = Math.log(Math.abs(hFactor)) / Math.LN2;
+ var vZoomDiff = Math.log(Math.abs(vFactor)) / Math.LN2;
+ z = z - Math.max(hZoomDiff, vZoomDiff);
+ projection2.scale(geoZoomToScale(z));
+ }
+ var extentCenter = projection2(extent.center());
+ extentCenter[1] = extentCenter[1] - padTop / 2;
+ projection2.translate(geoVecSubtract(c, extentCenter)).clipExtent([[0, 0], d]);
+ var drawLayers = svgLayers(projection2, context).only(["osm", "touch"]).dimensions(d);
+ var drawVertices = svgVertices(projection2, context);
+ var drawLines = svgLines(projection2, context);
+ var drawTurns = svgTurns(projection2, context);
+ var firstTime = selection2.selectAll(".surface").empty();
+ selection2.call(drawLayers);
+ var surface = selection2.selectAll(".surface").classed("tr", true);
+ if (firstTime) {
+ _initialized2 = true;
+ surface.call(breathe);
+ }
+ if (_fromWayID && !vgraph.hasEntity(_fromWayID)) {
+ _fromWayID = null;
+ _oldTurns = null;
+ }
+ surface.call(utilSetDimensions, d).call(drawVertices, vgraph, _intersection.vertices, filter2, extent, z).call(drawLines, vgraph, _intersection.ways, filter2).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 (_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);
+ function click(d3_event) {
+ surface.call(breathe.off).call(breathe);
+ var datum2 = d3_event.target.__data__;
+ var entity = datum2 && datum2.properties && datum2.properties.entity;
+ if (entity) {
+ datum2 = entity;
+ }
+ if (datum2 instanceof osmWay && (datum2.__from || datum2.__via)) {
+ _fromWayID = datum2.id;
+ _oldTurns = null;
+ redraw();
+ } else if (datum2 instanceof osmTurn) {
+ var actions, extraActions, turns, i3;
+ var restrictionType = osmInferRestriction(vgraph, datum2, projection2);
+ if (datum2.restrictionID && !datum2.direct) {
+ return;
+ } else if (datum2.restrictionID && !datum2.only) {
+ var seen = {};
+ var datumOnly = JSON.parse(JSON.stringify(datum2));
+ datumOnly.only = true;
+ restrictionType = restrictionType.replace(/^no/, "only");
+ turns = _intersection.turns(_fromWayID, 2);
+ extraActions = [];
+ _oldTurns = [];
+ for (i3 = 0; i3 < turns.length; i3++) {
+ var turn = turns[i3];
+ if (seen[turn.restrictionID])
+ continue;
+ if (turn.direct && turn.path[1] === datum2.path[1]) {
+ seen[turns[i3].restrictionID] = true;
+ turn.restrictionType = osmInferRestriction(vgraph, turn, projection2);
+ _oldTurns.push(turn);
+ extraActions.push(actionUnrestrictTurn(turn));
+ }
+ }
+ actions = _intersection.actions.concat(extraActions, [
+ actionRestrictTurn(datumOnly, restrictionType),
+ _t("operations.restriction.annotation.create")
+ ]);
+ } else if (datum2.restrictionID) {
+ turns = _oldTurns || [];
+ extraActions = [];
+ for (i3 = 0; i3 < turns.length; i3++) {
+ if (turns[i3].key !== datum2.key) {
+ extraActions.push(actionRestrictTurn(turns[i3], turns[i3].restrictionType));
+ }
+ }
+ _oldTurns = null;
+ actions = _intersection.actions.concat(extraActions, [
+ actionUnrestrictTurn(datum2),
+ _t("operations.restriction.annotation.delete")
+ ]);
+ } else {
+ actions = _intersection.actions.concat([
+ actionRestrictTurn(datum2, restrictionType),
+ _t("operations.restriction.annotation.create")
+ ]);
+ }
+ context.perform.apply(context, actions);
+ var s = surface.selectAll("." + datum2.key);
+ datum2 = s.empty() ? null : s.datum();
+ updateHints(datum2);
+ } else {
+ _fromWayID = null;
+ _oldTurns = null;
+ redraw();
+ }
+ }
+ function mouseover(d3_event) {
+ var datum2 = d3_event.target.__data__;
+ updateHints(datum2);
+ }
+ _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);
+ }
+ }
+ }
+ function highlightPathsFrom(wayID) {
+ surface.selectAll(".related").classed("related", false).classed("allow", false).classed("restrict", false).classed("only", false);
+ surface.selectAll("." + wayID).classed("related", true);
+ if (wayID) {
+ var turns = _intersection.turns(wayID, _maxViaWay);
+ for (var i3 = 0; i3 < turns.length; i3++) {
+ var turn = turns[i3];
+ var ids = [turn.to.way];
+ var klass = turn.no ? "restrict" : turn.only ? "only" : "allow";
+ if (turn.only || turns.length === 1) {
+ if (turn.via.ways) {
+ ids = ids.concat(turn.via.ways);
+ }
+ } else if (turn.to.way === wayID) {
+ continue;
+ }
+ surface.selectAll(utilEntitySelector(ids)).classed("related", true).classed("allow", klass === "allow").classed("restrict", klass === "restrict").classed("only", klass === "only");
+ }
+ }
+ }
+ function updateHints(datum2) {
+ var help = _container.selectAll(".restriction-help").html("");
+ var placeholders = {};
+ ["from", "via", "to"].forEach(function(k) {
+ placeholders[k] = { html: '<span class="qualifier">' + _t("restriction.help." + k) + "</span>" };
+ });
+ var entity = datum2 && datum2.properties && datum2.properties.entity;
+ if (entity) {
+ datum2 = entity;
+ }
+ if (_fromWayID) {
+ way = vgraph.entity(_fromWayID);
+ surface.selectAll("." + _fromWayID).classed("selected", true).classed("related", true);
+ }
+ if (datum2 instanceof osmWay && datum2.__from) {
+ way = datum2;
+ highlightPathsFrom(_fromWayID ? null : way.id);
+ surface.selectAll("." + way.id).classed("related", true);
+ var clickSelect = !_fromWayID || _fromWayID !== way.id;
+ help.append("div").html(_t.html("restriction.help." + (clickSelect ? "select_from_name" : "from_name"), {
+ from: placeholders.from,
+ fromName: displayName(way.id, vgraph)
+ }));
+ } else if (datum2 instanceof osmTurn) {
+ var restrictionType = osmInferRestriction(vgraph, datum2, projection2);
+ var turnType = restrictionType.replace(/^(only|no)\_/, "");
+ var indirect = datum2.direct === false ? _t.html("restriction.help.indirect") : "";
+ var klass, turnText, nextText;
+ if (datum2.no) {
+ klass = "restrict";
+ turnText = _t.html("restriction.help.turn.no_" + turnType, { indirect: { html: indirect } });
+ nextText = _t.html("restriction.help.turn.only_" + turnType, { indirect: "" });
+ } else if (datum2.only) {
+ klass = "only";
+ turnText = _t.html("restriction.help.turn.only_" + turnType, { indirect: { html: indirect } });
+ nextText = _t.html("restriction.help.turn.allowed_" + turnType, { indirect: "" });
+ } else {
+ klass = "allow";
+ turnText = _t.html("restriction.help.turn.allowed_" + turnType, { indirect: { html: indirect } });
+ nextText = _t.html("restriction.help.turn.no_" + turnType, { indirect: "" });
+ }
+ help.append("div").attr("class", "qualifier " + klass).html(turnText);
+ help.append("div").html(_t.html("restriction.help.from_name_to_name", {
+ from: placeholders.from,
+ fromName: displayName(datum2.from.way, vgraph),
+ to: placeholders.to,
+ toName: displayName(datum2.to.way, vgraph)
+ }));
+ if (datum2.via.ways && datum2.via.ways.length) {
+ var names = [];
+ for (var i3 = 0; i3 < datum2.via.ways.length; i3++) {
+ var prev = names[names.length - 1];
+ var curr = displayName(datum2.via.ways[i3], vgraph);
+ if (!prev || curr !== prev) {
+ names.push(curr);
+ }
+ }
+ help.append("div").html(_t.html("restriction.help.via_names", {
+ via: placeholders.via,
+ viaNames: names.join(", ")
+ }));
+ }
+ if (!indirect) {
+ help.append("div").html(_t.html("restriction.help.toggle", { turn: { html: nextText.trim() } }));
+ }
+ highlightPathsFrom(null);
+ var alongIDs = datum2.path.slice();
+ surface.selectAll(utilEntitySelector(alongIDs)).classed("related", true).classed("allow", klass === "allow").classed("restrict", klass === "restrict").classed("only", klass === "only");
+ } else {
+ highlightPathsFrom(null);
+ if (_fromWayID) {
+ help.append("div").html(_t.html("restriction.help.from_name", {
+ from: placeholders.from,
+ fromName: displayName(_fromWayID, vgraph)
+ }));
+ } else {
+ help.append("div").html(_t.html("restriction.help.select_from", {
+ from: placeholders.from
+ }));
+ }
+ }
+ }
+ }
+ function displayMaxDistance(maxDist) {
+ return (selection2) => {
+ var isImperial = !_mainLocalizer.usesMetric();
+ var opts;
+ 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 }) };
+ }
+ return selection2.html("").call(_t.append("restriction.controls.distance_up_to", opts));
+ };
+ }
+ function displayMaxVia(maxVia) {
+ return (selection2) => {
+ selection2 = selection2.html("");
+ return maxVia === 0 ? selection2.call(_t.append("restriction.controls.via_node_only")) : maxVia === 1 ? selection2.call(_t.append("restriction.controls.via_up_to_one")) : selection2.call(_t.append("restriction.controls.via_up_to_two"));
+ };
+ }
+ function displayName(entityID, graph) {
+ var entity = graph.entity(entityID);
+ var name = utilDisplayName(entity) || "";
+ var matched = _mainPresetIndex.match(entity, graph);
+ var type2 = matched && matched.name() || utilDisplayType(entity.id);
+ return name || type2;
+ }
+ restrictions.entityIDs = function(val) {
+ _intersection = null;
+ _fromWayID = null;
+ _oldTurns = null;
+ _vertexID = val[0];
+ };
+ restrictions.tags = function() {
+ };
+ restrictions.focus = function() {
+ };
+ restrictions.off = function(selection2) {
+ if (!_initialized2)
+ return;
+ selection2.selectAll(".surface").call(breathe.off).on("click.restrictions", null).on("mouseover.restrictions", null);
+ select_default2(window).on("resize.restrictions", null);
+ };
+ return utilRebind(restrictions, dispatch10, "on");
+ }
+ uiFieldRestrictions.supportsMultiselection = false;
+
+ // modules/ui/fields/textarea.js
+ function uiFieldTextarea(field, context) {
+ var dispatch10 = dispatch_default("change");
+ var input = select_default2(null);
+ var _lengthIndicator = uiLengthIndicator(context.maxCharsForTagValue()).silent(field.usage === "changeset" && field.key === "comment");
+ var _tags;
+ function textarea(selection2) {
+ var wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).style("position", "relative").merge(wrap2);
+ input = wrap2.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);
+ wrap2.call(_lengthIndicator);
+ function change(onInput) {
+ return function() {
+ var val = utilGetSetValue(input);
+ if (!onInput)
+ val = context.cleanTagValue(val);
+ if (!val && Array.isArray(_tags[field.key]))
+ return;
+ var t = {};
+ t[field.key] = val || void 0;
+ dispatch10.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") : void 0).attr("placeholder", isMixed ? _t("inspector.multiple_values") : field.placeholder() || _t("inspector.unknown")).classed("mixed", isMixed);
+ if (!isMixed) {
+ _lengthIndicator.update(tags[field.key]);
+ }
+ };
+ textarea.focus = function() {
+ input.node().focus();
+ };
+ return utilRebind(textarea, dispatch10, "on");
+ }
+
+ // modules/ui/fields/wikidata.js
+ function uiFieldWikidata(field, context) {
+ var wikidata = services.wikidata;
+ var dispatch10 = dispatch_default("change");
+ var _selection = select_default2(null);
+ var _searchInput = select_default2(null);
+ var _qid = null;
+ var _wikidataEntity = null;
+ var _wikiURL = "";
+ var _entityIDs = [];
+ var _wikipediaKey = field.keys && field.keys.find(function(key) {
+ return key.includes("wikipedia");
+ });
+ var _hintKey = field.key === "wikidata" ? "name" : field.key.split(":")[0];
+ var combobox = uiCombobox(context, "combo-" + field.safeid).caseSensitive(true).minItems(1);
+ function wiki(selection2) {
+ _selection = selection2;
+ var wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", "form-field-input-wrap form-field-input-" + field.type).merge(wrap2);
+ var list = wrap2.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_default2(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);
+ 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_default2(this.parentNode).select("input").node().select();
+ document.execCommand("copy");
+ });
+ }
+ function fetchWikidataItems(q, callback) {
+ if (!q && _hintKey) {
+ for (var i2 in _entityIDs) {
+ var entity = context.hasEntity(_entityIDs[i2]);
+ if (entity.tags[_hintKey]) {
+ q = entity.tags[_hintKey];
+ break;
+ }
+ }
+ }
+ wikidata.itemsForSearchQuery(q, function(err, data) {
+ if (err)
+ return;
+ var result = data.map(function(item) {
+ return {
+ id: item.id,
+ value: item.display.label.value + " (" + item.id + ")",
+ display: (selection2) => selection2.append("span").attr("class", "localized-text").attr("lang", item.display.label.language).text(item.display.label.value),
+ title: item.display.description && item.display.description.value,
+ terms: item.aliases
+ };
+ });
+ if (callback)
+ callback(result);
+ });
+ }
+ function change() {
+ var syncTags = {};
+ syncTags[field.key] = _qid;
+ dispatch10.call("change", this, syncTags);
+ var initGraph = context.graph();
+ var initEntityIDs = _entityIDs;
+ wikidata.entityByQID(_qid, function(err, entity) {
+ if (err)
+ return;
+ if (context.graph() !== initGraph)
+ return;
+ if (!entity.sitelinks)
+ return;
+ var langs = wikidata.languagesToQuery();
+ ["labels", "descriptions"].forEach(function(key) {
+ if (!entity[key])
+ return;
+ var valueLangs = Object.keys(entity[key]);
+ if (valueLangs.length === 0)
+ return;
+ var valueLang = valueLangs[0];
+ if (langs.indexOf(valueLang) === -1) {
+ langs.push(valueLang);
+ }
+ });
+ var newWikipediaValue;
+ if (_wikipediaKey) {
+ var foundPreferred;
+ for (var i2 in langs) {
+ var lang = langs[i2];
+ var siteID = lang.replace("-", "_") + "wiki";
+ if (entity.sitelinks[siteID]) {
+ foundPreferred = true;
+ newWikipediaValue = lang + ":" + entity.sitelinks[siteID].title;
+ break;
+ }
+ }
+ if (!foundPreferred) {
+ var wikiSiteKeys = Object.keys(entity.sitelinks).filter(function(site) {
+ return site.endsWith("wiki");
+ });
+ if (wikiSiteKeys.length === 0) {
+ newWikipediaValue = null;
+ } else {
+ var wikiLang = wikiSiteKeys[0].slice(0, -4).replace("_", "-");
+ var wikiTitle = entity.sitelinks[wikiSiteKeys[0]].title;
+ newWikipediaValue = wikiLang + ":" + wikiTitle;
+ }
+ }
+ }
+ if (newWikipediaValue) {
+ newWikipediaValue = context.cleanTagValue(newWikipediaValue);
+ }
+ if (typeof newWikipediaValue === "undefined")
+ return;
+ var actions = initEntityIDs.map(function(entityID) {
+ var entity2 = context.hasEntity(entityID);
+ if (!entity2)
+ return null;
+ var currTags = Object.assign({}, entity2.tags);
+ if (newWikipediaValue === null) {
+ if (!currTags[_wikipediaKey])
+ return null;
+ delete currTags[_wikipediaKey];
+ } else {
+ currTags[_wikipediaKey] = newWikipediaValue;
+ }
+ return actionChangeTags(entityID, currTags);
+ }).filter(Boolean);
+ if (!actions.length)
+ return;
+ context.overwrite(
+ function actionUpdateWikipediaTags(graph) {
+ actions.forEach(function(action) {
+ graph = action(graph);
+ });
+ return graph;
+ },
+ context.history().undoAnnotation()
+ );
+ });
+ }
+ function setLabelForEntity() {
+ var label = "";
+ if (_wikidataEntity) {
+ label = entityPropertyForDisplay(_wikidataEntity, "labels");
+ if (label.length === 0) {
+ label = _wikidataEntity.id.toString();
+ }
+ }
+ utilGetSetValue(_searchInput, label);
+ }
+ wiki.tags = function(tags) {
+ var isMixed = Array.isArray(tags[field.key]);
+ _searchInput.attr("title", isMixed ? tags[field.key].filter(Boolean).join("\n") : null).attr("placeholder", isMixed ? _t("inspector.multiple_values") : "").classed("mixed", isMixed);
+ _qid = typeof tags[field.key] === "string" && tags[field.key] || "";
+ if (!/^Q[0-9]*$/.test(_qid)) {
+ unrecognized();
+ return;
+ }
+ _wikiURL = "https://wikidata.org/wiki/" + _qid;
+ wikidata.entityByQID(_qid, function(err, entity) {
+ if (err) {
+ unrecognized();
+ return;
+ }
+ _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);
+ _selection.select(".preset-wikidata-identifier").style("display", function() {
+ return entity.id ? "flex" : "none";
+ }).select("input").attr("value", entity.id);
+ });
+ function unrecognized() {
+ _wikidataEntity = null;
+ setLabelForEntity();
+ _selection.select(".preset-wikidata-description").style("display", "none");
+ _selection.select(".preset-wikidata-identifier").style("display", "none");
+ _selection.select("button.wiki-link").classed("disabled", true);
+ if (_qid && _qid !== "") {
+ _wikiURL = "https://wikidata.org/wiki/Special:Search?search=" + _qid;
+ } else {
+ _wikiURL = "";
+ }
+ }
+ };
+ function entityPropertyForDisplay(wikidataEntity, propKey) {
+ if (!wikidataEntity[propKey])
+ return "";
+ var propObj = wikidataEntity[propKey];
+ var langKeys = Object.keys(propObj);
+ if (langKeys.length === 0)
+ return "";
+ var langs = wikidata.languagesToQuery();
+ for (var i2 in langs) {
+ var lang = langs[i2];
+ var valueObj = propObj[lang];
+ if (valueObj && valueObj.value && valueObj.value.length > 0)
+ return valueObj.value;
+ }
+ return propObj[langKeys[0]].value;
+ }
+ wiki.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return wiki;
+ };
+ wiki.focus = function() {
+ _searchInput.node().focus();
+ };
+ return utilRebind(wiki, dispatch10, "on");
+ }
+
+ // modules/ui/fields/wikipedia.js
+ function uiFieldWikipedia(field, context) {
+ const dispatch10 = dispatch_default("change");
+ const wikipedia = services.wikipedia;
+ const wikidata = services.wikidata;
+ let _langInput = select_default2(null);
+ let _titleInput = select_default2(null);
+ let _wikiURL = "";
+ let _entityIDs;
+ let _tags;
+ let _dataWikipedia = [];
+ _mainFileFetcher.get("wmf_sitematrix").then((d) => {
+ _dataWikipedia = d;
+ if (_tags)
+ updateForTags(_tags);
+ }).catch(() => {
+ });
+ const langCombo = uiCombobox(context, "wikipedia-lang").fetcher((value, callback) => {
+ const v = value.toLowerCase();
+ callback(
+ _dataWikipedia.filter((d) => {
+ return d[0].toLowerCase().indexOf(v) >= 0 || d[1].toLowerCase().indexOf(v) >= 0 || d[2].toLowerCase().indexOf(v) >= 0;
+ }).map((d) => ({ value: d[1] }))
+ );
+ });
+ const titleCombo = uiCombobox(context, "wikipedia-title").fetcher((value, callback) => {
+ if (!value) {
+ value = "";
+ for (let i2 in _entityIDs) {
+ let entity = context.hasEntity(_entityIDs[i2]);
+ if (entity.tags.name) {
+ value = entity.tags.name;
+ break;
+ }
+ }
+ }
+ const searchfn = value.length > 7 ? wikipedia.search : wikipedia.suggestions;
+ searchfn(language()[2], value, (query, data) => {
+ callback(data.map((d) => ({ value: d })));
+ });
+ });
+ function wiki(selection2) {
+ let wrap2 = selection2.selectAll(".form-field-input-wrap").data([0]);
+ wrap2 = wrap2.enter().append("div").attr("class", `form-field-input-wrap form-field-input-${field.type}`).merge(wrap2);
+ let langContainer = wrap2.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);
+ let titleContainer = wrap2.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);
+ });
+ let link2 = titleContainer.selectAll(".wiki-link").data([0]);
+ link2 = link2.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(link2);
+ link2.on("click", (d3_event) => {
+ d3_event.preventDefault();
+ if (_wikiURL)
+ window.open(_wikiURL, "_blank");
+ });
+ }
+ function defaultLanguageInfo(skipEnglishFallback) {
+ const langCode = _mainLocalizer.languageCode().toLowerCase();
+ for (let i2 in _dataWikipedia) {
+ let d = _dataWikipedia[i2];
+ if (d[2] === langCode)
+ return d;
+ }
+ return skipEnglishFallback ? ["", "", ""] : ["English", "English", "en"];
+ }
+ function language(skipEnglishFallback) {
+ const value = utilGetSetValue(_langInput).toLowerCase();
+ for (let i2 in _dataWikipedia) {
+ let d = _dataWikipedia[i2];
+ if (d[0].toLowerCase() === value || d[1].toLowerCase() === value || d[2] === value)
+ return d;
+ }
+ return defaultLanguageInfo(skipEnglishFallback);
+ }
+ function changeLang() {
+ utilGetSetValue(_langInput, language()[1]);
+ change(true);
+ }
+ function change(skipWikidata) {
+ let value = utilGetSetValue(_titleInput);
+ const m = value.match(/https?:\/\/([-a-z]+)\.wikipedia\.org\/(?:wiki|\1-[-a-z]+)\/([^#]+)(?:#(.+))?/);
+ const langInfo = m && _dataWikipedia.find((d) => m[1] === d[2]);
+ let syncTags = {};
+ if (langInfo) {
+ const nativeLangName = langInfo[1];
+ value = decodeURIComponent(m[2]).replace(/_/g, " ");
+ if (m[3]) {
+ let anchor;
+ anchor = decodeURIComponent(m[3]);
+ value += "#" + anchor.replace(/_/g, " ");
+ }
+ value = value.slice(0, 1).toUpperCase() + value.slice(1);
+ utilGetSetValue(_langInput, nativeLangName);
+ utilGetSetValue(_titleInput, value);
+ }
+ if (value) {
+ syncTags.wikipedia = context.cleanTagValue(language()[2] + ":" + value);
+ } else {
+ syncTags.wikipedia = void 0;
+ }
+ dispatch10.call("change", this, syncTags);
+ if (skipWikidata || !value || !language()[2])
+ return;
+ const initGraph = context.graph();
+ const initEntityIDs = _entityIDs;
+ wikidata.itemsByTitle(language()[2], value, (err, data) => {
+ if (err || !data || !Object.keys(data).length)
+ return;
+ if (context.graph() !== initGraph)
+ return;
+ const qids = Object.keys(data);
+ const value2 = qids && qids.find((id2) => id2.match(/^Q\d+$/));
+ let actions = initEntityIDs.map((entityID) => {
+ let entity = context.entity(entityID).tags;
+ let currTags = Object.assign({}, entity);
+ if (currTags.wikidata !== value2) {
+ currTags.wikidata = value2;
+ return actionChangeTags(entityID, currTags);
+ }
+ return null;
+ }).filter(Boolean);
+ if (!actions.length)
+ return;
+ context.overwrite(
+ function actionUpdateWikidataTags(graph) {
+ actions.forEach(function(action) {
+ graph = action(graph);
+ });
+ return graph;
+ },
+ context.history().undoAnnotation()
+ );
+ });
+ }
+ wiki.tags = (tags) => {
+ _tags = tags;
+ updateForTags(tags);
+ };
+ function updateForTags(tags) {
+ const value = typeof tags[field.key] === "string" ? tags[field.key] : "";
+ const m = value.match(/([^:]+):([^#]+)(?:#(.+))?/);
+ const tagLang = m && m[1];
+ const tagArticleTitle = m && m[2];
+ let anchor = m && m[3];
+ const tagLangInfo = tagLang && _dataWikipedia.find((d) => tagLang === d[2]);
+ if (tagLangInfo) {
+ const nativeLangName = tagLangInfo[1];
+ utilGetSetValue(_langInput, nativeLangName);
+ utilGetSetValue(_titleInput, tagArticleTitle + (anchor ? "#" + anchor : ""));
+ if (anchor) {
+ try {
+ anchor = encodeURIComponent(anchor.replace(/ /g, "_")).replace(/%/g, ".");
+ } catch (e) {
+ anchor = anchor.replace(/ /g, "_");
+ }
+ }
+ _wikiURL = "https://" + tagLang + ".wikipedia.org/wiki/" + tagArticleTitle.replace(/ /g, "_") + (anchor ? "#" + anchor : "");
+ } else {
+ utilGetSetValue(_titleInput, value);
+ if (value && value !== "") {
+ utilGetSetValue(_langInput, "");
+ const defaultLangInfo = defaultLanguageInfo();
+ _wikiURL = `https://${defaultLangInfo[2]}.wikipedia.org/w/index.php?fulltext=1&search=${value}`;
+ } else {
+ const shownOrDefaultLangInfo = language(
+ true
+ /* skipEnglishFallback */
+ );
+ utilGetSetValue(_langInput, shownOrDefaultLangInfo[1]);
+ _wikiURL = "";
+ }
+ }
+ }
+ wiki.entityIDs = (val) => {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return wiki;
+ };
+ wiki.focus = () => {
+ _titleInput.node().focus();
+ };
+ return utilRebind(wiki, dispatch10, "on");
+ }
+ uiFieldWikipedia.supportsMultiselection = false;
+
+ // modules/ui/fields/index.js
+ var uiFields = {
+ access: uiFieldAccess,
+ address: uiFieldAddress,
+ check: uiFieldCheck,
+ colour: uiFieldText,
+ combo: uiFieldCombo,
+ cycleway: uiFieldDirectionalCombo,
+ defaultCheck: uiFieldCheck,
+ directionalCombo: uiFieldDirectionalCombo,
+ email: uiFieldText,
+ identifier: uiFieldText,
+ lanes: uiFieldLanes,
+ localized: uiFieldLocalized,
+ roadheight: uiFieldRoadheight,
+ roadspeed: uiFieldRoadspeed,
+ 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
+ };
+
+ // modules/ui/field.js
+ function uiField(context, presetField2, entityIDs, options2) {
+ options2 = Object.assign({
+ show: true,
+ wrap: true,
+ remove: true,
+ revert: true,
+ info: true
+ }, options2);
+ var dispatch10 = dispatch_default("change", "revert");
+ var field = Object.assign({}, presetField2);
+ field.domId = utilUniqueDomId("form-field-" + field.safeid);
+ var _show = options2.show;
+ var _state = "";
+ var _tags = {};
+ var _entityExtent;
+ if (entityIDs && entityIDs.length) {
+ _entityExtent = entityIDs.reduce(function(extent, entityID) {
+ var entity = context.graph().entity(entityID);
+ return extent.extend(entity.extent(context.graph()));
+ }, geoExtent());
+ }
+ var _locked = false;
+ var _lockedTip = uiTooltip().title(() => _t.append("inspector.lock.suggestion", { label: field.title })).placement("bottom");
+ field.keys = field.keys || [field.key];
+ if (_show && !field.impl) {
+ createField();
+ }
+ function createField() {
+ field.impl = uiFields[field.type](field, context).on("change", function(t, onInput) {
+ dispatch10.call("change", field, t, onInput);
+ });
+ if (entityIDs) {
+ field.entityIDs = entityIDs;
+ if (field.impl.entityIDs) {
+ field.impl.entityIDs(entityIDs);
+ }
+ }
+ }
+ 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 tagsContainFieldKey() {
+ return field.keys.some(function(key) {
+ if (field.type === "multiCombo") {
+ for (var tagKey in _tags) {
+ if (tagKey.indexOf(key) === 0) {
+ return true;
+ }
+ }
+ return false;
+ }
+ return _tags[key] !== void 0;
+ });
+ }
+ function revert(d3_event, d) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ if (!entityIDs || _locked)
+ return;
+ dispatch10.call("revert", d, d.keys);
+ }
+ function remove2(d3_event, d) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ if (_locked)
+ return;
+ var t = {};
+ d.keys.forEach(function(key) {
+ t[key] = void 0;
+ });
+ dispatch10.call("change", d, t);
+ }
+ field.render = function(selection2) {
+ var container = selection2.selectAll(".form-field").data([field]);
+ var enter = container.enter().append("div").attr("class", function(d) {
+ return "form-field form-field-" + d.safeid;
+ }).classed("nowrap", !options2.wrap);
+ if (options2.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").each(function(d) {
+ d.label()(select_default2(this));
+ });
+ textEnter.append("span").attr("class", "label-textannotation");
+ if (options2.remove) {
+ labelEnter.append("button").attr("class", "remove-icon").attr("title", _t("icons.remove")).call(svgIcon("#iD-operation-delete"));
+ }
+ if (options2.revert) {
+ labelEnter.append("button").attr("class", "modified-icon").attr("title", _t("icons.undo")).call(svgIcon(_mainLocalizer.textDirection() === "rtl" ? "#iD-icon-redo" : "#iD-icon-undo"));
+ }
+ }
+ container = container.merge(enter);
+ container.select(".field-label > .remove-icon").on("click", remove2);
+ container.select(".field-label > .modified-icon").on("click", revert);
+ container.each(function(d) {
+ var selection3 = select_default2(this);
+ if (!d.impl) {
+ createField();
+ }
+ var reference, help;
+ if (options2.wrap && field.type === "restrictions") {
+ help = uiFieldHelp(context, "restrictions");
+ }
+ if (options2.wrap && options2.info) {
+ var referenceKey = d.key || "";
+ if (d.type === "multiCombo") {
+ referenceKey = referenceKey.replace(/:$/, "");
+ }
+ reference = uiTagReference(d.reference || { key: referenceKey }, context);
+ if (_state === "hover") {
+ reference.showing(false);
+ }
+ }
+ selection3.call(d.impl);
+ if (help) {
+ selection3.call(help.body).select(".field-label").call(help.button);
+ }
+ if (reference) {
+ selection3.call(reference.body).select(".field-label").call(reference.button);
+ }
+ d.impl.tags(_tags);
+ });
+ container.classed("locked", _locked).classed("modified", isModified()).classed("present", tagsContainFieldKey());
+ var annotation = container.selectAll(".field-label .label-textannotation");
+ var icon2 = annotation.selectAll(".icon").data(_locked ? [0] : []);
+ icon2.exit().remove();
+ icon2.enter().append("svg").attr("class", "icon").append("use").attr("xlink:href", "#fas-lock");
+ container.call(_locked ? _lockedTip : _lockedTip.destroy);
+ };
+ field.state = function(val) {
+ if (!arguments.length)
+ return _state;
+ _state = val;
+ return field;
+ };
+ field.tags = function(val) {
+ if (!arguments.length)
+ return _tags;
+ _tags = val;
+ if (tagsContainFieldKey() && !_show) {
+ _show = true;
+ if (!field.impl) {
+ createField();
+ }
+ }
+ return field;
+ };
+ field.locked = function(val) {
+ if (!arguments.length)
+ return _locked;
+ _locked = val;
+ return field;
+ };
+ field.show = function() {
+ _show = true;
+ if (!field.impl) {
+ createField();
+ }
+ if (field.default && field.key && _tags[field.key] !== field.default) {
+ var t = {};
+ t[field.key] = field.default;
+ dispatch10.call("change", this, t);
+ }
+ };
+ field.isShown = function() {
+ return _show;
+ };
+ 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) {
+ var validHere = _sharedLocationManager.locationSetsAt(_entityExtent.center());
+ if (!validHere[field.locationSetID])
+ return false;
+ }
+ var prerequisiteTag = field.prerequisiteTag;
+ 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;
+ }
+ return true;
+ }))
+ return false;
+ }
+ return true;
+ };
+ field.focus = function() {
+ if (field.impl) {
+ field.impl.focus();
+ }
+ };
+ return utilRebind(field, dispatch10, "on");
+ }
+
+ // modules/ui/form_fields.js
+ function uiFormFields(context) {
+ var moreCombo = uiCombobox(context, "more-fields").minItems(1);
+ var _fieldsArr = [];
+ var _lastPlaceholder = "";
+ var _state = "";
+ var _klass = "";
+ function formFields(selection2) {
+ var allowedFields = _fieldsArr.filter(function(field) {
+ return field.isAllowed();
+ });
+ var shown = allowedFields.filter(function(field) {
+ return field.isShown();
+ });
+ var notShown = allowedFields.filter(function(field) {
+ return !field.isShown();
+ });
+ var container = selection2.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();
+ var enter = fields.enter().append("div").attr("class", function(d) {
+ return "wrap-form-field wrap-form-field-" + d.safeid;
+ });
+ fields = fields.merge(enter);
+ fields.order().each(function(d) {
+ select_default2(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,
+ field,
+ terms
+ };
+ });
+ var placeholder = titles.slice(0, 3).join(", ") + (titles.length > 3 ? "\u2026" : "");
+ var more = selection2.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").call(_t.append("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;
+ var field = d.field;
+ field.show();
+ selection2.call(formFields);
+ field.focus();
+ })
+ );
+ if (_lastPlaceholder !== placeholder) {
+ input.attr("placeholder", placeholder);
+ _lastPlaceholder = placeholder;
+ }
+ }
+ 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;
+ };
+ formFields.klass = function(val) {
+ if (!arguments.length)
+ return _klass;
+ _klass = val;
+ return formFields;
+ };
+ return formFields;
+ }
+
+ // modules/ui/changeset_editor.js
+ function uiChangesetEditor(context) {
+ var dispatch10 = dispatch_default("change");
+ var formFields = uiFormFields(context);
+ var commentCombo = uiCombobox(context, "comment").caseSensitive(true);
+ var _fieldsArr;
+ var _tags;
+ var _changesetID;
+ function changesetEditor(selection2) {
+ render(selection2);
+ }
+ function render(selection2) {
+ 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 })
+ ];
+ _fieldsArr.forEach(function(field) {
+ field.on("change", function(t, onInput) {
+ dispatch10.call("change", field, void 0, t, onInput);
+ });
+ });
+ }
+ _fieldsArr.forEach(function(field) {
+ field.tags(_tags);
+ });
+ selection2.call(formFields.fieldsArr(_fieldsArr));
+ if (initial) {
+ var commentField = selection2.select(".form-field-comment textarea");
+ var commentNode = commentField.node();
+ if (commentNode) {
+ commentNode.focus();
+ commentNode.select();
+ }
+ utilTriggerEvent(commentField, "blur");
+ var osm = context.connection();
+ 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"))
+ );
+ });
+ }
+ }
+ const warnings = [];
+ if (_tags.comment.match(/google/i)) {
+ warnings.push({
+ id: 'contains "google"',
+ msg: _t.append("commit.google_warning"),
+ link: _t("commit.google_warning_link")
+ });
+ }
+ const maxChars = context.maxCharsForTagValue();
+ const strLen = utilUnicodeCharsCount(utilCleanOsmString(_tags.comment, Number.POSITIVE_INFINITY));
+ if (strLen > maxChars || false) {
+ warnings.push({
+ id: "message too long",
+ msg: _t.append("commit.changeset_comment_length_warning", { maxChars })
+ });
+ }
+ var commentWarning = selection2.select(".form-field-comment").selectAll(".comment-warning").data(warnings, (d) => d.id);
+ commentWarning.exit().transition().duration(200).style("opacity", 0).remove();
+ var commentEnter = commentWarning.enter().insert("div", ".comment-warning").attr("class", "comment-warning field-warning").style("opacity", 0);
+ commentEnter.call(svgIcon("#iD-icon-alert", "inline")).append("span");
+ commentEnter.transition().duration(200).style("opacity", 1);
+ commentWarning.merge(commentEnter).selectAll("div > span").text("").each(function(d) {
+ let selection3 = select_default2(this);
+ if (d.link) {
+ selection3 = selection3.append("a").attr("target", "_blank").attr("href", d.link);
+ }
+ selection3.call(d.msg);
+ });
+ }
+ changesetEditor.tags = function(_) {
+ if (!arguments.length)
+ return _tags;
+ _tags = _;
+ return changesetEditor;
+ };
+ changesetEditor.changesetID = function(_) {
+ if (!arguments.length)
+ return _changesetID;
+ if (_changesetID === _)
+ return changesetEditor;
+ _changesetID = _;
+ _fieldsArr = null;
+ return changesetEditor;
+ };
+ return utilRebind(changesetEditor, dispatch10, "on");
+ }
+
+ // modules/ui/commit.js
+ var import_fast_deep_equal9 = __toESM(require_fast_deep_equal());
+
+ // modules/util/jxon.js
+ var JXON = new function() {
+ var sValueProp = "keyValue", sAttributesProp = "keyAttributes", sAttrPref = "@", aCache = [], rIsNull = /^\s*$/, rIsBool = /^(?:true|false)$/i;
+ function parseText(sValue) {
+ if (rIsNull.test(sValue)) {
+ return null;
+ }
+ if (rIsBool.test(sValue)) {
+ return sValue.toLowerCase() === "true";
+ }
+ if (isFinite(sValue)) {
+ return parseFloat(sValue);
+ }
+ if (isFinite(Date.parse(sValue))) {
+ return new Date(sValue);
+ }
+ return sValue;
+ }
+ function EmptyTree() {
+ }
+ EmptyTree.prototype.toString = function() {
+ return "null";
+ };
+ EmptyTree.prototype.valueOf = function() {
+ return null;
+ };
+ function objectify(vValue) {
+ return vValue === null ? new EmptyTree() : vValue instanceof Object ? vValue : new vValue.constructor(vValue);
+ }
+ 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
+ );
+ if (bChildren) {
+ for (var oNode, nItem = 0; nItem < oParentNode.childNodes.length; nItem++) {
+ oNode = oParentNode.childNodes.item(nItem);
+ if (oNode.nodeType === 4) {
+ sCollectedTxt += oNode.nodeValue;
+ } else if (oNode.nodeType === 3) {
+ sCollectedTxt += oNode.nodeValue.trim();
+ } else if (oNode.nodeType === 1 && !oNode.prefix) {
+ aCache.push(oNode);
+ }
+ }
+ }
+ var nLevelEnd = aCache.length, vBuiltVal = parseText(sCollectedTxt);
+ 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 (vResult.hasOwnProperty(sProp)) {
+ if (vResult[sProp].constructor !== Array) {
+ vResult[sProp] = [vResult[sProp]];
+ }
+ vResult[sProp].push(vContent);
+ } else {
+ vResult[sProp] = vContent;
+ nLength++;
+ }
+ }
+ 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 (bNesteAttr) {
+ if (bFreeze) {
+ Object.freeze(oAttrParent);
+ }
+ vResult[sAttributesProp] = oAttrParent;
+ nLength -= nAttrLen - 1;
+ }
+ }
+ 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);
+ }
+ aCache.length = nLevelStart;
+ return vResult;
+ }
+ function loadObjTree(oXMLDoc, oParentEl, oParentObj) {
+ var vValue, oChild;
+ if (oParentObj instanceof String || oParentObj instanceof Number || oParentObj instanceof Boolean) {
+ oParentEl.appendChild(oXMLDoc.createTextNode(oParentObj.toString()));
+ } else if (oParentObj.constructor === Date) {
+ oParentEl.appendChild(oXMLDoc.createTextNode(oParentObj.toGMTString()));
+ }
+ for (var sName in oParentObj) {
+ vValue = oParentObj[sName];
+ if (isFinite(sName) || vValue instanceof Function) {
+ continue;
+ }
+ if (sName === sValueProp) {
+ if (vValue !== null && vValue !== true) {
+ oParentEl.appendChild(oXMLDoc.createTextNode(vValue.constructor === Date ? vValue.toGMTString() : String(vValue)));
+ }
+ } else if (sName === sAttributesProp) {
+ 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 (vValue instanceof Object) {
+ loadObjTree(oXMLDoc, oChild, vValue);
+ } else if (vValue !== null && vValue !== true) {
+ oChild.appendChild(oXMLDoc.createTextNode(vValue.toString()));
+ }
+ oParentEl.appendChild(oChild);
+ }
+ }
+ }
+ this.build = function(oXMLParent, nVerbosity, bFreeze, bNesteAttributes) {
+ 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);
+ };
+ this.unbuild = function(oObjTree) {
+ var oNewDoc = document.implementation.createDocument("", "", null);
+ loadObjTree(oNewDoc, oNewDoc, oObjTree);
+ return oNewDoc;
+ };
+ this.stringify = function(oObjTree) {
+ return new XMLSerializer().serializeToString(JXON.unbuild(oObjTree));
+ };
+ }();
+
+ // modules/ui/sections/changes.js
+ function uiSectionChanges(context) {
+ var _discardTags = {};
+ _mainFileFetcher.get("discarded").then(function(d) {
+ _discardTags = d;
+ }).catch(function() {
+ });
+ var section = uiSection("changes-list", context).label(function() {
+ var history = context.history();
+ var summary = history.difference().summary();
+ return _t.append("inspector.title_count", { title: _t("commit.changes"), count: summary.length });
+ }).disclosureContent(renderDisclosureContent);
+ function renderDisclosureContent(selection2) {
+ var history = context.history();
+ var summary = history.difference().summary();
+ var container = selection2.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_default2(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").text(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").text(function(d) {
+ var name = utilDisplayName(d.entity) || "", string = "";
+ if (name !== "") {
+ string += ":";
+ }
+ return string += " " + name;
+ });
+ items = itemsEnter.merge(items);
+ var changeset = new osmChangeset().update({ id: void 0 });
+ var changes = history.changes(actionDiscardTags(history.difference(), _discardTags));
+ delete changeset.id;
+ 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");
+ linkEnter.attr("href", window.URL.createObjectURL(blob)).attr("download", fileName);
+ linkEnter.call(svgIcon("#iD-icon-load", "inline")).append("span").call(_t.append("commit.download_changes"));
+ 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);
+ }
+ 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;
+ }
+
+ // modules/ui/commit_warnings.js
+ function uiCommitWarnings(context) {
+ function commitWarnings(selection2) {
+ var issuesBySeverity = context.validator().getIssuesBySeverity({ what: "edited", where: "all", includeDisabledRules: true });
+ for (var severity in issuesBySeverity) {
+ var issues = issuesBySeverity[severity];
+ if (severity !== "error") {
+ issues = issues.filter(function(issue) {
+ return issue.type !== "help_request";
+ });
+ }
+ var section = severity + "-section";
+ var issueItem = severity + "-item";
+ var container = selection2.selectAll("." + section).data(issues.length ? [0] : []);
+ container.exit().remove();
+ var containerEnter = container.enter().append("div").attr("class", "modal-section " + section + " fillL2");
+ containerEnter.append("h3").call(severity === "warning" ? _t.append("commit.warnings") : _t.append("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.key;
+ });
+ 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").text("").each(function(d) {
+ return d.message(context)(select_default2(this));
+ });
+ }
+ }
+ return commitWarnings;
+ }
+
+ // modules/ui/commit.js
+ var readOnlyTags = [
+ /^changesets_count$/,
+ /^created_by$/,
+ /^ideditor:/,
+ /^imagery_used$/,
+ /^host$/,
+ /^locale$/,
+ /^warnings:/,
+ /^resolved:/,
+ /^closed:note$/,
+ /^closed:keepright$/,
+ /^closed:improveosm:/,
+ /^closed:osmose:/
+ ];
+ var hashtagRegex = /(#[^\u2000-\u206F\u2E00-\u2E7F\s\\'!"#$%()*,.\/:;<=>?@\[\]^`{|}~]+)/g;
+ function uiCommit(context) {
+ var dispatch10 = dispatch_default("cancel");
+ var _userDetails2;
+ 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(selection2) {
+ _selection = selection2;
+ if (!context.changeset)
+ initChangeset();
+ loadDerivedChangesetTags();
+ selection2.call(render);
+ }
+ function initChangeset() {
+ var commentDate = +corePreferences("commentDate") || 0;
+ var currDate = Date.now();
+ var cutoff = 2 * 86400 * 1e3;
+ if (commentDate > currDate || currDate - commentDate > cutoff) {
+ corePreferences("comment", null);
+ corePreferences("hashtags", null);
+ corePreferences("source", null);
+ }
+ if (context.defaultChangesetComment()) {
+ corePreferences("comment", context.defaultChangesetComment());
+ corePreferences("commentDate", Date.now());
+ }
+ if (context.defaultChangesetSource()) {
+ corePreferences("source", context.defaultChangesetSource());
+ corePreferences("commentDate", Date.now());
+ }
+ if (context.defaultChangesetHashtags()) {
+ corePreferences("hashtags", context.defaultChangesetHashtags());
+ corePreferences("commentDate", Date.now());
+ }
+ 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())
+ };
+ findHashtags(tags, true);
+ var hashtags = corePreferences("hashtags");
+ if (hashtags) {
+ tags.hashtags = hashtags;
+ }
+ var source = corePreferences("source");
+ if (source) {
+ tags.source = source;
+ }
+ var photoOverlaysUsed = context.history().photoOverlaysUsed();
+ if (photoOverlaysUsed.length) {
+ var sources = (tags.source || "").split(";");
+ if (sources.indexOf("streetlevel imagery") === -1) {
+ sources.push("streetlevel imagery");
+ }
+ photoOverlaysUsed.forEach(function(photoOverlay) {
+ if (sources.indexOf(photoOverlay) === -1) {
+ sources.push(photoOverlay);
+ }
+ });
+ tags.source = context.cleanTagValue(sources.join(";"));
+ }
+ context.changeset = new osmChangeset({ tags });
+ }
+ function loadDerivedChangesetTags() {
+ var osm = context.connection();
+ if (!osm)
+ return;
+ var tags = Object.assign({}, context.changeset.tags);
+ var imageryUsed = context.cleanTagValue(context.history().imageryUsed().join(";"));
+ tags.imagery_used = imageryUsed || "None";
+ var osmClosed = osm.getClosedIDs();
+ var itemType;
+ if (osmClosed.length) {
+ tags["closed:note"] = context.cleanTagValue(osmClosed.join(";"));
+ }
+ if (services.keepRight) {
+ var krClosed = services.keepRight.getClosedIDs();
+ if (krClosed.length) {
+ tags["closed:keepright"] = context.cleanTagValue(krClosed.join(";"));
+ }
+ }
+ if (services.improveOSM) {
+ var iOsmClosed = services.improveOSM.getClosedCounts();
+ for (itemType in iOsmClosed) {
+ tags["closed:improveosm:" + itemType] = context.cleanTagValue(iOsmClosed[itemType].toString());
+ }
+ }
+ if (services.osmose) {
+ var osmoseClosed = services.osmose.getClosedCounts();
+ for (itemType in osmoseClosed) {
+ tags["closed:osmose:" + itemType] = context.cleanTagValue(osmoseClosed[itemType].toString());
+ }
+ }
+ for (var key in tags) {
+ if (key.match(/(^warnings:)|(^resolved:)/)) {
+ delete tags[key];
+ }
+ }
+ function addIssueCounts(issues, prefix) {
+ var issuesByType = utilArrayGroupBy(issues, "type");
+ for (var issueType in issuesByType) {
+ var issuesOfType = issuesByType[issueType];
+ if (issuesOfType[0].subtype) {
+ var issuesBySubtype = utilArrayGroupBy(issuesOfType, "subtype");
+ 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());
+ }
+ }
+ }
+ var warnings = context.validator().getIssuesBySeverity({ what: "edited", where: "all", includeIgnored: true, includeDisabledRules: true }).warning.filter(function(issue) {
+ return issue.type !== "help_request";
+ });
+ addIssueCounts(warnings, "warnings");
+ var resolvedIssues = context.validator().getResolvedIssues();
+ addIssueCounts(resolvedIssues, "resolved");
+ context.changeset = context.changeset.update({ tags });
+ }
+ function render(selection2) {
+ var osm = context.connection();
+ if (!osm)
+ return;
+ var header = selection2.selectAll(".header").data([0]);
+ var headerTitle = header.enter().append("div").attr("class", "header fillL");
+ headerTitle.append("div").append("h2").call(_t.append("commit.title"));
+ headerTitle.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", function() {
+ dispatch10.call("cancel", this);
+ }).call(svgIcon("#iD-icon-close"));
+ var body = selection2.selectAll(".body").data([0]);
+ body = body.enter().append("div").attr("class", "body").merge(body);
+ 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)
+ );
+ body.call(commitWarnings);
+ 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()) {
+ _userDetails2 = null;
+ }
+ prose = prose.enter().append("p").attr("class", "commit-info").call(_t.append("commit.upload_explanation")).merge(prose);
+ osm.userDetails(function(err, user) {
+ if (err)
+ return;
+ if (_userDetails2 === user)
+ return;
+ _userDetails2 = user;
+ var userLink = select_default2(document.createElement("div"));
+ 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").text(user.display_name).attr("href", osm.userURL(user.display_name)).attr("target", "_blank");
+ prose.html(_t.html("commit.upload_explanation_with_user", { user: { html: userLink.html() } }));
+ });
+ var requestReview = saveSection.selectAll(".request-review").data([0]);
+ 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.append("commit.request_review_info")).placement("top"));
+ }
+ labelEnter.append("input").attr("type", "checkbox").attr("id", requestReviewDomId);
+ labelEnter.append("span").call(_t.append("commit.request_review"));
+ requestReview = requestReview.merge(requestReviewEnter);
+ var requestReviewInput = requestReview.selectAll("input").property("checked", isReviewRequested(context.changeset.tags)).on("change", toggleRequestReview);
+ var buttonSection = saveSection.selectAll(".buttons").data([0]);
+ 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").call(_t.append("commit.cancel"));
+ var uploadButton = buttonEnter.append("button").attr("class", "action button save-button");
+ uploadButton.append("span").attr("class", "label").call(_t.append("commit.save"));
+ var uploadBlockerTooltipText = getUploadBlockerMessage();
+ buttonSection = buttonSection.merge(buttonEnter);
+ buttonSection.selectAll(".cancel-button").on("click.cancel", function() {
+ dispatch10.call("cancel", this);
+ });
+ buttonSection.selectAll(".save-button").classed("disabled", uploadBlockerTooltipText !== null).on("click.save", function() {
+ if (!select_default2(this).classed("disabled")) {
+ this.blur();
+ for (var key in context.changeset.tags) {
+ if (!key)
+ delete context.changeset.tags[key];
+ }
+ context.uploader().save(context.changeset);
+ }
+ });
+ uiTooltip().destroyAny(buttonSection.selectAll(".save-button"));
+ if (uploadBlockerTooltipText) {
+ buttonSection.selectAll(".save-button").call(uiTooltip().title(() => uploadBlockerTooltipText).placement("top"));
+ }
+ 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)).render
+ );
+ var changesSection = body.selectAll(".commit-changes-section").data([0]);
+ changesSection = changesSection.enter().append("div").attr("class", "modal-section commit-changes-section").merge(changesSection);
+ changesSection.call(commitChanges.render);
+ function toggleRequestReview() {
+ var rr = requestReviewInput.property("checked");
+ updateChangeset({ review_requested: rr ? "yes" : void 0 });
+ tagSection.call(
+ rawTagEditor.tags(Object.assign({}, context.changeset.tags)).render
+ );
+ }
+ }
+ function getUploadBlockerMessage() {
+ var errors = context.validator().getIssuesBySeverity({ what: "edited", where: "all" }).error;
+ if (errors.length) {
+ return _t.append("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.append("commit.comment_needed_message");
+ }
+ }
+ return null;
+ }
+ function changeTags(_, changed, onInput) {
+ if (changed.hasOwnProperty("comment")) {
+ if (changed.comment === void 0) {
+ changed.comment = "";
+ }
+ if (!onInput) {
+ corePreferences("comment", changed.comment);
+ corePreferences("commentDate", Date.now());
+ }
+ }
+ if (changed.hasOwnProperty("source")) {
+ if (changed.source === void 0) {
+ corePreferences("source", null);
+ } else if (!onInput) {
+ corePreferences("source", changed.source);
+ corePreferences("commentDate", Date.now());
+ }
+ }
+ updateChangeset(changed, onInput);
+ if (_selection) {
+ _selection.call(render);
+ }
+ }
+ function findHashtags(tags, commentOnly) {
+ var detectedHashtags = commentHashtags();
+ if (detectedHashtags.length) {
+ corePreferences("hashtags", null);
+ }
+ if (!detectedHashtags.length || !commentOnly) {
+ detectedHashtags = detectedHashtags.concat(hashtagHashtags());
+ }
+ var allLowerCase = /* @__PURE__ */ new Set();
+ return detectedHashtags.filter(function(hashtag) {
+ var lowerCase = hashtag.toLowerCase();
+ if (!allLowerCase.has(lowerCase)) {
+ allLowerCase.add(lowerCase);
+ return true;
+ }
+ return false;
+ });
+ function commentHashtags() {
+ var matches = (tags.comment || "").replace(/http\S*/g, "").match(hashtagRegex);
+ return matches || [];
+ }
+ function hashtagHashtags() {
+ var matches = (tags.hashtags || "").split(/[,;\s]+/).map(function(s) {
+ if (s[0] !== "#") {
+ s = "#" + s;
+ }
+ var matched = s.match(hashtagRegex);
+ return matched && matched[0];
+ }).filter(Boolean);
+ return matches || [];
+ }
+ }
+ function isReviewRequested(tags) {
+ var rr = tags.review_requested;
+ if (rr === void 0)
+ return false;
+ rr = rr.trim().toLowerCase();
+ return !(rr === "" || rr === "no");
+ }
+ function updateChangeset(changed, onInput) {
+ var tags = Object.assign({}, context.changeset.tags);
+ Object.keys(changed).forEach(function(k) {
+ var v = changed[k];
+ k = context.cleanTagKey(k);
+ if (readOnlyTags.indexOf(k) !== -1)
+ return;
+ if (v === void 0) {
+ delete tags[k];
+ } else if (onInput) {
+ tags[k] = v;
+ } else {
+ tags[k] = context.cleanTagValue(v);
+ }
+ });
+ if (!onInput) {
+ 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);
+ }
+ }
+ if (_userDetails2 && _userDetails2.changesets_count !== void 0) {
+ var changesetsCount = parseInt(_userDetails2.changesets_count, 10) + 1;
+ tags.changesets_count = String(changesetsCount);
+ if (changesetsCount <= 100) {
+ var s;
+ s = corePreferences("walkthrough_completed");
+ if (s) {
+ tags["ideditor:walkthrough_completed"] = s;
+ }
+ s = corePreferences("walkthrough_progress");
+ if (s) {
+ tags["ideditor:walkthrough_progress"] = s;
+ }
+ s = corePreferences("walkthrough_started");
+ if (s) {
+ tags["ideditor:walkthrough_started"] = s;
+ }
+ }
+ } else {
+ delete tags.changesets_count;
+ }
+ if (!(0, import_fast_deep_equal9.default)(context.changeset.tags, tags)) {
+ context.changeset = context.changeset.update({ tags });
+ }
+ }
+ commit.reset = function() {
+ context.changeset = null;
+ };
+ return utilRebind(commit, dispatch10, "on");
+ }
+
+ // modules/ui/confirm.js
+ function uiConfirm(selection2) {
+ var modalSelection = uiModal(selection2);
+ 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();
+ }).call(_t.append("confirm.okay")).node().focus();
+ return modalSelection;
+ };
+ return modalSelection;
+ }
+
+ // modules/ui/conflicts.js
+ function uiConflicts(context) {
+ var dispatch10 = dispatch_default("cancel", "save");
+ var keybinding = utilKeybinding("conflicts");
+ var _origChanges;
+ var _conflictList;
+ var _shownConflictIndex;
+ function keybindingOn() {
+ select_default2(document).call(keybinding.on("\u238B", cancel, true));
+ }
+ function keybindingOff() {
+ select_default2(document).call(keybinding.unbind);
+ }
+ function tryAgain() {
+ keybindingOff();
+ dispatch10.call("save");
+ }
+ function cancel() {
+ keybindingOff();
+ dispatch10.call("cancel");
+ }
+ function conflicts(selection2) {
+ keybindingOn();
+ var headerEnter = selection2.selectAll(".header").data([0]).enter().append("div").attr("class", "header fillL");
+ headerEnter.append("button").attr("class", "fr").attr("title", _t("icons.close")).on("click", cancel).call(svgIcon("#iD-icon-close"));
+ headerEnter.append("h2").call(_t.append("save.conflict.header"));
+ var bodyEnter = selection2.selectAll(".body").data([0]).enter().append("div").attr("class", "body fillL");
+ var conflictsHelpEnter = bodyEnter.append("div").attr("class", "conflicts-help").call(_t.append("save.conflict.help"));
+ var changeset = new osmChangeset();
+ delete 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");
+ linkEnter.attr("href", window.URL.createObjectURL(blob)).attr("download", fileName);
+ linkEnter.call(svgIcon("#iD-icon-load", "inline")).append("span").call(_t.append("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").call(_t.append("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").call(_t.append("save.title")).on("click.try_again", tryAgain);
+ buttonsEnter.append("button").attr("class", "secondary-action conflicts-button col6").call(_t.append("confirm.cancel")).on("click.cancel", cancel);
+ }
+ function showConflict(selection2, index) {
+ index = utilWrap(index, _conflictList.length);
+ _shownConflictIndex = index;
+ var parent = select_default2(selection2.node().parentNode);
+ 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);
+ }
+ var conflict = selection2.selectAll(".conflict").data([_conflictList[index]]);
+ conflict.exit().remove();
+ var conflictEnter = conflict.enter().append("div").attr("class", "conflict");
+ conflictEnter.append("h4").attr("class", "conflict-count").call(_t.append("save.conflict.count", { num: index + 1, total: _conflictList.length }));
+ conflictEnter.append("a").attr("class", "conflict-description").attr("href", "#").text(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").attr("class", "conflict-nav-button action col6").attr("disabled", function(d, i2) {
+ return i2 === 0 && index === 0 || i2 === 1 && index === _conflictList.length - 1 || null;
+ }).on("click", function(d3_event, d) {
+ d3_event.preventDefault();
+ var container = parent.selectAll(".conflict-container");
+ var sign2 = d === "previous" ? -1 : 1;
+ container.selectAll(".conflict").remove();
+ container.call(showConflict, index + sign2);
+ }).each(function(d) {
+ _t.append("save.conflict." + d)(select_default2(this));
+ });
+ }
+ function addChoices(selection2) {
+ var choices = selection2.append("ul").attr("class", "layer-list").selectAll("li").data(function(d) {
+ return d.choices || [];
+ });
+ 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").text(function(d) {
+ return d.text;
+ });
+ choicesEnter.merge(choices).each(function(d) {
+ var ul = this.parentNode;
+ if (ul.__data__.chosen === d.id) {
+ choose(null, ul, d);
+ }
+ });
+ }
+ function choose(d3_event, ul, datum2) {
+ if (d3_event)
+ d3_event.preventDefault();
+ select_default2(ul).selectAll("li").classed("active", function(d) {
+ return d === datum2;
+ }).selectAll("input").property("checked", function(d) {
+ return d === datum2;
+ });
+ var extent = geoExtent();
+ var entity;
+ entity = context.graph().hasEntity(datum2.id);
+ if (entity)
+ extent._extend(entity.extent(context.graph()));
+ datum2.action();
+ entity = context.graph().hasEntity(datum2.id);
+ if (entity)
+ extent._extend(entity.extent(context.graph()));
+ zoomToEntity(datum2.id, extent);
+ }
+ function zoomToEntity(id2, extent) {
+ context.surface().selectAll(".hover").classed("hover", false);
+ var entity = context.graph().hasEntity(id2);
+ if (entity) {
+ if (extent) {
+ context.map().trimmedExtent(extent);
+ } else {
+ context.map().zoomToEase(entity);
+ }
+ context.surface().selectAll(utilEntityOrMemberSelector([entity.id], context.graph())).classed("hover", true);
+ }
+ }
+ conflicts.conflictList = function(_) {
+ if (!arguments.length)
+ return _conflictList;
+ _conflictList = _;
+ return conflicts;
+ };
+ conflicts.origChanges = function(_) {
+ if (!arguments.length)
+ return _origChanges;
+ _origChanges = _;
+ return conflicts;
+ };
+ conflicts.shownEntityIds = function() {
+ if (_conflictList && typeof _shownConflictIndex === "number") {
+ return [_conflictList[_shownConflictIndex].id];
+ }
+ return [];
+ };
+ return utilRebind(conflicts, dispatch10, "on");
+ }
+
+ // modules/ui/entity_editor.js
+ var import_fast_deep_equal10 = __toESM(require_fast_deep_equal());
+
+ // modules/ui/sections/entity_issues.js
+ function uiSectionEntityIssues(context) {
+ var preference = corePreferences("entity-issues.reference.expanded");
+ var _expanded = preference === null ? true : preference === "true";
+ var _entityIDs = [];
+ var _issues = [];
+ var _activeIssueID;
+ var section = uiSection("entity-issues", context).shouldDisplay(function() {
+ return _issues.length > 0;
+ }).label(function() {
+ return _t.append("inspector.title_count", { title: _t("issues.list_title"), count: _issues.length });
+ }).disclosureContent(renderDisclosureContent);
+ context.validator().on("validated.entity_issues", function() {
+ reloadIssues();
+ section.reRender();
+ }).on("focusedIssue.entity_issues", function(issue) {
+ makeActiveIssue(issue.id);
+ });
+ function reloadIssues() {
+ _issues = context.validator().getSharedEntityIssues(_entityIDs, { includeDisabledRules: true });
+ }
+ function makeActiveIssue(issueID) {
+ _activeIssueID = issueID;
+ section.selection().selectAll(".issue-container").classed("active", function(d) {
+ return d.id === _activeIssueID;
+ });
+ }
+ function renderDisclosureContent(selection2) {
+ selection2.classed("grouped-items-area", true);
+ _activeIssueID = _issues.length > 0 ? _issues[0].id : null;
+ var containers = selection2.selectAll(".issue-container").data(_issues, function(d) {
+ return d.key;
+ });
+ containers.exit().remove();
+ 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) {
+ 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);
+ var extent = d.extent(context.graph());
+ 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_default2(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();
+ var container = select_default2(this.parentNode.parentNode.parentNode);
+ var info = container.selectAll(".issue-info");
+ var isExpanded = info.classed("expanded");
+ _expanded = !isExpanded;
+ corePreferences("entity-issues.reference.expanded", _expanded);
+ 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);
+ });
+ }
+ });
+ 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_default2(this).call(d.reference);
+ } else {
+ select_default2(this).call(_t.append("inspector.no_documentation_key"));
+ }
+ });
+ containers = containers.merge(containersEnter).classed("active", function(d) {
+ return d.id === _activeIssueID;
+ });
+ containers.selectAll(".issue-message").text("").each(function(d) {
+ return d.message(context)(select_default2(this));
+ });
+ 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) {
+ if (select_default2(this).attr("disabled") || !d.onClick)
+ return;
+ if (d.issue.dateLastRanFix && new Date() - d.issue.dateLastRanFix < 1e3)
+ return;
+ d.issue.dateLastRanFix = new Date();
+ utilHighlightEntities(d.issue.entityIds.concat(d.entityIds), false, context);
+ new Promise(function(resolve, reject) {
+ d.onClick(context, resolve, reject);
+ if (d.onClick.length <= 1) {
+ resolve();
+ }
+ }).then(function() {
+ 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";
+ if (iconName.startsWith("maki")) {
+ iconName += "-15";
+ }
+ select_default2(this).call(svgIcon("#" + iconName, "fix-icon"));
+ });
+ buttons.append("span").attr("class", "fix-message").each(function(d) {
+ return d.title(select_default2(this));
+ });
+ 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;
+ }
+ return null;
+ });
+ }
+ section.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
+ _entityIDs = val;
+ _activeIssueID = null;
+ reloadIssues();
+ }
+ return section;
+ };
+ return section;
+ }
+
+ // modules/ui/preset_icon.js
+ function uiPresetIcon() {
+ let _preset;
+ let _geometry;
+ function presetIcon(selection2) {
+ selection2.each(render);
+ }
+ function getIcon(p, geom) {
+ if (p.isFallback && p.isFallback())
+ return geom === "vertex" ? "" : "iD-icon-" + p.id;
+ if (p.icon)
+ return p.icon;
+ if (geom === "line")
+ return "iD-other-line";
+ if (geom === "vertex")
+ return "temaki-vertex";
+ return "maki-marker-stroked";
+ }
+ function renderPointBorder(container, drawPoint) {
+ let pointBorder = container.selectAll(".preset-icon-point-border").data(drawPoint ? [0] : []);
+ pointBorder.exit().remove();
+ let pointBorderEnter = pointBorder.enter();
+ const w = 40;
+ const h = 40;
+ pointBorderEnter.append("svg").attr("class", "preset-icon-fill preset-icon-point-border").attr("width", w).attr("height", h).attr("viewBox", `0 0 ${w} ${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 renderCategoryBorder(container, category) {
+ let categoryBorder = container.selectAll(".preset-icon-category-border").data(category ? [0] : []);
+ categoryBorder.exit().remove();
+ let categoryBorderEnter = categoryBorder.enter();
+ const d = 60;
+ let svgEnter = categoryBorderEnter.append("svg").attr("class", "preset-icon-fill preset-icon-category-border").attr("width", d).attr("height", d).attr("viewBox", `0 0 ${d} ${d}`);
+ svgEnter.append("path").attr("class", "area").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 (category) {
+ categoryBorder.selectAll("path").attr("class", `area ${category.id}`);
+ }
+ }
+ function renderCircleFill(container, drawVertex) {
+ let vertexFill = container.selectAll(".preset-icon-fill-vertex").data(drawVertex ? [0] : []);
+ vertexFill.exit().remove();
+ let vertexFillEnter = vertexFill.enter();
+ const w = 60;
+ const h = 60;
+ const d = 40;
+ vertexFillEnter.append("svg").attr("class", "preset-icon-fill preset-icon-fill-vertex").attr("width", w).attr("height", h).attr("viewBox", `0 0 ${w} ${h}`).append("circle").attr("cx", w / 2).attr("cy", h / 2).attr("r", d / 2);
+ vertexFill = vertexFillEnter.merge(vertexFill);
+ }
+ function renderSquareFill(container, drawArea, tagClasses) {
+ let fill = container.selectAll(".preset-icon-fill-area").data(drawArea ? [0] : []);
+ fill.exit().remove();
+ let fillEnter = fill.enter();
+ const d = 60;
+ const w = d;
+ const h = d;
+ const l = d * 2 / 3;
+ const c1 = (w - l) / 2;
+ const 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 ${w} ${h}`);
+ ["fill", "stroke"].forEach((klass) => {
+ fillEnter.append("path").attr("d", `M${c1} ${c1} L${c1} ${c2} L${c2} ${c2} L${c2} ${c1} Z`).attr("class", `area ${klass}`);
+ });
+ const rVertex = 2.5;
+ [[c1, c1], [c1, c2], [c2, c2], [c2, c1]].forEach((point2) => {
+ fillEnter.append("circle").attr("class", "vertex").attr("cx", point2[0]).attr("cy", point2[1]).attr("r", rVertex);
+ });
+ const rMidpoint = 1.25;
+ [[c1, w / 2], [c2, w / 2], [h / 2, c1], [h / 2, c2]].forEach((point2) => {
+ fillEnter.append("circle").attr("class", "midpoint").attr("cx", point2[0]).attr("cy", point2[1]).attr("r", rMidpoint);
+ });
+ fill = fillEnter.merge(fill);
+ fill.selectAll("path.stroke").attr("class", `area stroke ${tagClasses}`);
+ fill.selectAll("path.fill").attr("class", `area fill ${tagClasses}`);
+ }
+ function renderLine(container, drawLine, tagClasses) {
+ let line = container.selectAll(".preset-icon-line").data(drawLine ? [0] : []);
+ line.exit().remove();
+ let lineEnter = line.enter();
+ const d = 60;
+ const w = d;
+ const h = d;
+ const y = Math.round(d * 0.72);
+ const l = Math.round(d * 0.6);
+ const r = 2.5;
+ const x12 = (w - l) / 2;
+ const x2 = x12 + l;
+ lineEnter = lineEnter.append("svg").attr("class", "preset-icon-line").attr("width", w).attr("height", h).attr("viewBox", `0 0 ${w} ${h}`);
+ ["casing", "stroke"].forEach((klass) => {
+ lineEnter.append("path").attr("d", `M${x12} ${y} L${x2} ${y}`).attr("class", `line ${klass}`);
+ });
+ [[x12 - 1, y], [x2 + 1, y]].forEach((point2) => {
+ lineEnter.append("circle").attr("class", "vertex").attr("cx", point2[0]).attr("cy", point2[1]).attr("r", r);
+ });
+ line = lineEnter.merge(line);
+ line.selectAll("path.stroke").attr("class", `line stroke ${tagClasses}`);
+ line.selectAll("path.casing").attr("class", `line casing ${tagClasses}`);
+ }
+ function renderRoute(container, drawRoute, p) {
+ let route = container.selectAll(".preset-icon-route").data(drawRoute ? [0] : []);
+ route.exit().remove();
+ let routeEnter = route.enter();
+ const d = 60;
+ const w = d;
+ const h = d;
+ const y12 = Math.round(d * 0.8);
+ const y2 = Math.round(d * 0.68);
+ const l = Math.round(d * 0.6);
+ const r = 2;
+ const x12 = (w - l) / 2;
+ const x2 = x12 + l / 3;
+ const x3 = x2 + l / 3;
+ const x4 = x3 + l / 3;
+ routeEnter = routeEnter.append("svg").attr("class", "preset-icon-route").attr("width", w).attr("height", h).attr("viewBox", `0 0 ${w} ${h}`);
+ ["casing", "stroke"].forEach((klass) => {
+ routeEnter.append("path").attr("d", `M${x12} ${y12} L${x2} ${y2}`).attr("class", `segment0 line ${klass}`);
+ routeEnter.append("path").attr("d", `M${x2} ${y2} L${x3} ${y12}`).attr("class", `segment1 line ${klass}`);
+ routeEnter.append("path").attr("d", `M${x3} ${y12} L${x4} ${y2}`).attr("class", `segment2 line ${klass}`);
+ });
+ [[x12, y12], [x2, y2], [x3, y12], [x4, y2]].forEach((point2) => {
+ routeEnter.append("circle").attr("class", "vertex").attr("cx", point2[0]).attr("cy", point2[1]).attr("r", r);
+ });
+ route = routeEnter.merge(route);
+ if (drawRoute) {
+ let routeType = p.tags.type === "waterway" ? "waterway" : p.tags.route;
+ const segmentPresetIDs = routeSegments[routeType];
+ for (let i2 in segmentPresetIDs) {
+ const segmentPreset = _mainPresetIndex.item(segmentPresetIDs[i2]);
+ const segmentTagClasses = svgTagClasses().getClassesString(segmentPreset.tags, "");
+ route.selectAll(`path.stroke.segment${i2}`).attr("class", `segment${i2} line stroke ${segmentTagClasses}`);
+ route.selectAll(`path.casing.segment${i2}`).attr("class", `segment${i2} line casing ${segmentTagClasses}`);
+ }
+ }
+ }
+ function renderSvgIcon(container, picon, geom, isFramed, category, tagClasses) {
+ const isMaki = picon && /^maki-/.test(picon);
+ const isTemaki = picon && /^temaki-/.test(picon);
+ const isFa = picon && /^fa[srb]-/.test(picon);
+ const isiDIcon = picon && !(isMaki || isTemaki || isFa);
+ let icon2 = container.selectAll(".preset-icon").data(picon ? [0] : []);
+ icon2.exit().remove();
+ icon2 = icon2.enter().append("div").attr("class", "preset-icon").call(svgIcon("")).merge(icon2);
+ icon2.attr("class", "preset-icon " + (geom ? geom + "-geom" : "")).classed("category", category).classed("framed", isFramed).classed("preset-icon-iD", isiDIcon);
+ icon2.selectAll("svg").attr("class", "icon " + picon + " " + (!isiDIcon && geom !== "line" ? "" : tagClasses));
+ icon2.selectAll("use").attr("href", "#" + picon);
+ }
+ function renderImageIcon(container, imageURL) {
+ let imageIcon = container.selectAll("img.image-icon").data(imageURL ? [0] : []);
+ imageIcon.exit().remove();
+ imageIcon = imageIcon.enter().append("img").attr("class", "image-icon").on("load", () => container.classed("showing-img", true)).on("error", () => container.classed("showing-img", false)).merge(imageIcon);
+ imageIcon.attr("src", imageURL);
+ }
+ const 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 render() {
+ let p = _preset.apply(this, arguments);
+ let geom = _geometry ? _geometry.apply(this, arguments) : null;
+ if (geom === "relation" && p.tags && (p.tags.type === "route" && p.tags.route && routeSegments[p.tags.route] || p.tags.type === "waterway")) {
+ geom = "route";
+ }
+ const showThirdPartyIcons = corePreferences("preferences.privacy.thirdpartyicons") || "true";
+ const isFallback = p.isFallback && p.isFallback();
+ const imageURL = showThirdPartyIcons === "true" && p.imageURL;
+ const picon = getIcon(p, geom);
+ const isCategory = !p.setTags;
+ const drawPoint = false;
+ const drawVertex = picon !== null && geom === "vertex";
+ const drawLine = picon && geom === "line" && !isFallback && !isCategory;
+ const drawArea = picon && geom === "area" && !isFallback && !isCategory;
+ const drawRoute = picon && geom === "route";
+ const isFramed = drawVertex || drawArea || drawLine || drawRoute || isCategory;
+ let tags = !isCategory ? p.setTags({}, geom) : {};
+ for (let k in tags) {
+ if (tags[k] === "*") {
+ tags[k] = "yes";
+ }
+ }
+ let tagClasses = svgTagClasses().getClassesString(tags, "");
+ let selection2 = select_default2(this);
+ let container = selection2.selectAll(".preset-icon-container").data([0]);
+ container = container.enter().append("div").attr("class", "preset-icon-container").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);
+ }
+ presetIcon.preset = function(val) {
+ if (!arguments.length)
+ return _preset;
+ _preset = utilFunctor(val);
+ return presetIcon;
+ };
+ presetIcon.geometry = function(val) {
+ if (!arguments.length)
+ return _geometry;
+ _geometry = utilFunctor(val);
+ return presetIcon;
+ };
+ return presetIcon;
+ }
+
+ // modules/ui/sections/feature_type.js
+ function uiSectionFeatureType(context) {
+ var dispatch10 = dispatch_default("choose");
+ var _entityIDs = [];
+ var _presets = [];
+ var _tagReference;
+ var section = uiSection("feature-type", context).label(() => _t.append("inspector.feature_type")).disclosureContent(renderDisclosureContent);
+ function renderDisclosureContent(selection2) {
+ selection2.classed("preset-list-item", true);
+ selection2.classed("mixed-types", _presets.length > 1);
+ var presetButtonWrap = selection2.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.append("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 = selection2.selectAll(".tag-reference-body-wrap").data([0]);
+ tagReferenceBodyWrap = tagReferenceBodyWrap.enter().append("div").attr("class", "tag-reference-body-wrap").merge(tagReferenceBodyWrap);
+ if (_tagReference) {
+ selection2.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);
+ }
+ selection2.selectAll(".preset-reset").on("click", function() {
+ dispatch10.call("choose", this, _presets);
+ }).on("pointerdown pointerup mousedown mouseup", function(d3_event) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ });
+ var geometries = entityGeometries();
+ selection2.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.append("inspector.multiple_types")];
+ var label = selection2.select(".label-inner");
+ var nameparts = label.selectAll(".namepart").data(names, (d) => d.stringId);
+ nameparts.exit().remove();
+ nameparts.enter().append("div").attr("class", "namepart").text("").each(function(d) {
+ d(select_default2(this));
+ });
+ }
+ section.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return section;
+ };
+ section.presets = function(val) {
+ if (!arguments.length)
+ return _presets;
+ if (!utilArrayIdentical(val, _presets)) {
+ _presets = val;
+ if (_presets.length === 1) {
+ _tagReference = uiTagReference(_presets[0].reference(), context).showing(false);
+ }
+ }
+ return section;
+ };
+ function entityGeometries() {
+ var counts = {};
+ for (var i2 in _entityIDs) {
+ var geometry = context.graph().geometry(_entityIDs[i2]);
+ if (!counts[geometry])
+ counts[geometry] = 0;
+ counts[geometry] += 1;
+ }
+ return Object.keys(counts).sort(function(geom1, geom2) {
+ return counts[geom2] - counts[geom1];
+ });
+ }
+ return utilRebind(section, dispatch10, "on");
+ }
+
+ // modules/ui/sections/preset_fields.js
+ function uiSectionPresetFields(context) {
+ var section = uiSection("preset-fields", context).label(() => _t.append("inspector.fields")).disclosureContent(renderDisclosureContent);
+ var dispatch10 = dispatch_default("change", "revert");
+ var formFields = uiFormFields(context);
+ var _state;
+ var _fieldsArr;
+ var _presets = [];
+ var _tags;
+ var _entityIDs;
+ function renderDisclosureContent(selection2) {
+ 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)
+ );
+ }
+ var additionalFields = utilArrayUnion(sharedMoreFields, presetsManager.universal());
+ additionalFields.sort(function(field1, field2) {
+ return field1.title().localeCompare(field2.title(), _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) {
+ dispatch10.call("change", field, _entityIDs, t, onInput);
+ }).on("revert", function(keys) {
+ dispatch10.call("revert", field, keys);
+ });
+ });
+ }
+ _fieldsArr.forEach(function(field) {
+ field.state(_state).tags(_tags);
+ });
+ selection2.call(
+ formFields.fieldsArr(_fieldsArr).state(_state).klass("grouped-items-area")
+ );
+ }
+ section.presets = function(val) {
+ if (!arguments.length)
+ return _presets;
+ if (!_presets || !val || !utilArrayIdentical(_presets, val)) {
+ _presets = val;
+ _fieldsArr = null;
+ }
+ return section;
+ };
+ section.state = function(val) {
+ if (!arguments.length)
+ return _state;
+ _state = val;
+ return section;
+ };
+ section.tags = function(val) {
+ if (!arguments.length)
+ return _tags;
+ _tags = val;
+ return section;
+ };
+ section.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ if (!val || !_entityIDs || !utilArrayIdentical(_entityIDs, val)) {
+ _entityIDs = val;
+ _fieldsArr = null;
+ }
+ return section;
+ };
+ return utilRebind(section, dispatch10, "on");
+ }
+
+ // modules/ui/sections/raw_member_editor.js
+ 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.append("inspector.title_count", { title: _t("inspector.members"), count });
+ }).disclosureContent(renderDisclosureContent);
+ var taginfo = services.taginfo;
+ var _entityIDs;
+ var _maxMembers = 1e3;
+ function downloadMember(d3_event, d) {
+ d3_event.preventDefault();
+ select_default2(this.parentNode).classed("tag-reference-loading", true);
+ context.loadEntity(d.id, function() {
+ section.reRender();
+ });
+ }
+ function zoomToMember(d3_event, d) {
+ d3_event.preventDefault();
+ var entity = context.entity(d.id);
+ context.map().zoomToEase(entity);
+ utilHighlightEntities([d.id], true, context);
+ }
+ function selectMember(d3_event, d) {
+ d3_event.preventDefault();
+ utilHighlightEntities([d.id], false, context);
+ var entity = context.entity(d.id);
+ var mapExtent = context.map().extent();
+ if (!entity.intersects(mapExtent, context.graph())) {
+ context.map().zoomToEase(entity);
+ }
+ context.enter(modeSelect(context, [d.id]));
+ }
+ function changeRole(d3_event, d) {
+ var oldRole = d.role;
+ var newRole = context.cleanRelationRole(select_default2(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();
+ }
+ }
+ function deleteMember(d3_event, d) {
+ 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)) {
+ context.enter(modeBrowse(context));
+ } else {
+ context.validator().validate();
+ }
+ }
+ function renderDisclosureContent(selection2) {
+ var entityID = _entityIDs[0];
+ var memberships = [];
+ var entity = context.entity(entityID);
+ entity.members.slice(0, _maxMembers).forEach(function(member, index) {
+ memberships.push({
+ index,
+ id: member.id,
+ type: member.type,
+ role: member.role,
+ relation: entity,
+ member: context.hasEntity(member.id),
+ domId: utilUniqueDomId(entityID + "-member-" + index)
+ });
+ });
+ var list = selection2.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_default2(this);
+ var label = item.append("label").attr("class", "field-label").attr("for", d.domId);
+ if (d.member) {
+ 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").text(function(d2) {
+ var matched = _mainPresetIndex.match(d2.member, context.graph());
+ return matched && matched.name() || utilDisplayType(d2.member.id);
+ });
+ labelLink.append("span").attr("class", "member-entity-name").text(function(d2) {
+ return utilDisplayName(d2.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").call(_t.append("inspector." + d.type, { id: d.id }));
+ labelText.append("span").attr("class", "member-entity-name").call(_t.append("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);
+ }
+ 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(
+ drag_default().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_default2(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_default2(this).classed("dragging", true);
+ targetIndex = null;
+ selection2.selectAll("li.member-row").style("transform", function(d2, index2) {
+ var node = select_default2(this).node();
+ 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 "translateY(-100%)";
+ } else if (index2 < index && d3_event.y < node.offsetTop + node.offsetHeight) {
+ if (targetIndex === null || index2 < targetIndex) {
+ targetIndex = index2;
+ }
+ return "translateY(100%)";
+ }
+ return null;
+ });
+ }).on("end", function(d3_event, d) {
+ if (!select_default2(this).classed("dragging"))
+ return;
+ var index = items.nodes().indexOf(this);
+ select_default2(this).classed("dragging", false);
+ selection2.selectAll("li.member-row").style("transform", null);
+ if (targetIndex !== null) {
+ context.perform(
+ actionMoveMember(d.relation.id, index, targetIndex),
+ _t("operations.reorder_members.annotation")
+ );
+ context.validator().validate();
+ }
+ })
+ );
+ function bindTypeahead(d) {
+ var row = select_default2(this);
+ var role = row.selectAll("input.member-role");
+ var origValue = role.property("value");
+ function sort(value, data) {
+ var sameletter = [];
+ var other = [];
+ for (var i2 = 0; i2 < data.length; i2++) {
+ if (data[i2].value.substring(0, value.length) === value) {
+ sameletter.push(data[i2]);
+ } else {
+ other.push(data[i2]);
+ }
+ }
+ return sameletter.concat(other);
+ }
+ role.call(
+ uiCombobox(context, "member-role").fetcher(function(role2, callback) {
+ var geometry;
+ 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,
+ query: role2
+ }, function(err, data) {
+ if (!err)
+ callback(sort(role2, data));
+ });
+ }).on("cancel", function() {
+ role.property("value", origValue);
+ })
+ );
+ }
+ function unbind() {
+ var row = select_default2(this);
+ row.selectAll("input.member-role").call(uiCombobox.off, context);
+ }
+ }
+ section.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return section;
+ };
+ return section;
+ }
+
+ // modules/actions/delete_members.js
+ function actionDeleteMembers(relationId, memberIndexes) {
+ return function(graph) {
+ memberIndexes.sort((a, b) => b - a);
+ for (var i2 in memberIndexes) {
+ graph = actionDeleteMember(relationId, memberIndexes[i2])(graph);
+ }
+ return graph;
+ };
+ }
+
+ // modules/ui/sections/raw_membership_editor.js
+ 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.append("inspector.title_count", { title: _t("inspector.relations"), 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 = [];
+ var _showBlank;
+ var _maxMemberships = 1e3;
+ function getSharedParentRelations() {
+ var parents = [];
+ for (var i2 = 0; i2 < _entityIDs.length; i2++) {
+ var entity = context.graph().hasEntity(_entityIDs[i2]);
+ if (!entity)
+ continue;
+ if (i2 === 0) {
+ parents = context.graph().parentRelations(entity);
+ } else {
+ parents = utilArrayIntersection(parents, context.graph().parentRelations(entity));
+ }
+ if (!parents.length)
+ break;
+ }
+ return parents;
+ }
+ function getMemberships() {
+ var memberships = [];
+ var relations = getSharedParentRelations().slice(0, _maxMemberships);
+ var isMultiselect = _entityIDs.length > 1;
+ var i2, relation, membership, index, member, indexedMember;
+ for (i2 = 0; i2 < relations.length; i2++) {
+ relation = relations[i2];
+ membership = {
+ relation,
+ members: [],
+ hash: osmEntity.key(relation)
+ };
+ for (index = 0; index < relation.members.length; index++) {
+ member = relation.members[index];
+ if (_entityIDs.indexOf(member.id) !== -1) {
+ indexedMember = Object.assign({}, member, { index });
+ membership.members.push(indexedMember);
+ membership.hash += "," + index.toString();
+ if (!isMultiselect) {
+ memberships.push(membership);
+ membership = {
+ relation,
+ members: [],
+ hash: osmEntity.key(relation)
+ };
+ }
+ }
+ }
+ if (membership.members.length)
+ memberships.push(membership);
+ }
+ memberships.forEach(function(membership2) {
+ membership2.domId = utilUniqueDomId("membership-" + membership2.relation.id);
+ var roles = [];
+ membership2.members.forEach(function(member2) {
+ if (roles.indexOf(member2.role) === -1)
+ roles.push(member2.role);
+ });
+ membership2.role = roles.length === 1 ? roles[0] : roles;
+ });
+ return memberships;
+ }
+ function selectRelation(d3_event, d) {
+ d3_event.preventDefault();
+ utilHighlightEntities([d.relation.id], false, context);
+ context.enter(modeSelect(context, [d.relation.id]));
+ }
+ function zoomToRelation(d3_event, d) {
+ d3_event.preventDefault();
+ var entity = context.entity(d.relation.id);
+ context.map().zoomToEase(entity);
+ utilHighlightEntities([d.relation.id], true, context);
+ }
+ function changeRole(d3_event, d) {
+ if (d === 0)
+ return;
+ if (_inChange)
+ return;
+ var newRole = context.cleanRelationRole(select_default2(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();
+ }
+ _inChange = false;
+ }
+ function addMembership(d, role) {
+ this.blur();
+ _showBlank = false;
+ function actionAddMembers(relationId, ids, role2) {
+ return function(graph) {
+ for (var i2 in ids) {
+ var member = { id: ids[i2], type: graph.entity(ids[i2]).type, role: role2 };
+ graph = actionAddMember(relationId, member)(graph);
+ }
+ return graph;
+ };
+ }
+ 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")
+ );
+ context.enter(modeSelect(context, [relation.id]).newFeature(true));
+ }
+ }
+ function deleteMembership(d3_event, d) {
+ this.blur();
+ if (d === 0)
+ return;
+ 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();
+ }
+ function fetchNearbyRelations(q, callback) {
+ var newRelation = {
+ relation: null,
+ value: _t("inspector.new_relation"),
+ display: _t.append("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;
+ }
+ var explicitRelation = q && context.hasEntity(q.toLowerCase());
+ if (explicitRelation && explicitRelation.type === "relation" && explicitRelation.id !== entityID) {
+ 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 });
+ });
+ result.sort(function(a, b) {
+ return osmRelation.creationOrder(a.relation, b.relation);
+ });
+ 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;
+ });
+ });
+ }
+ result.forEach(function(obj) {
+ obj.title = obj.value;
+ });
+ result.unshift(newRelation);
+ callback(result);
+ }
+ function renderDisclosureContent(selection2) {
+ var memberships = getMemberships();
+ var list = selection2.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();
+ var itemsEnter = items.enter().append("li").attr("class", "member-row member-row-normal form-field");
+ 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").text(function(d) {
+ var matched = _mainPresetIndex.match(d.relation, context.graph());
+ return matched && matched.name() || _t.html("inspector.relation");
+ });
+ labelLink.append("span").attr("class", "member-entity-name").text(function(d) {
+ return utilDisplayName(d.relation);
+ });
+ labelEnter.append("button").attr("class", "remove member-delete").attr("title", _t("icons.remove")).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);
+ if (taginfo) {
+ wrapEnter.each(bindTypeahead);
+ }
+ var newMembership = list.selectAll(".member-row-new").data(_showBlank ? [0] : []);
+ newMembership.exit().remove();
+ 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").attr("title", _t("icons.remove")).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);
+ newMembership = newMembership.merge(newMembershipEnter);
+ newMembership.selectAll(".member-entity-input").on("blur", cancelEntity).call(
+ nearbyCombo.on("accept", acceptEntity).on("cancel", cancelEntity)
+ );
+ var addRow = selection2.selectAll(".add-row").data([0]);
+ var addRowEnter = addRow.enter().append("div").attr("class", "add-row");
+ var addRelationButton = addRowEnter.append("button").attr("class", "add-relation").attr("aria-label", _t("inspector.add_to_relation"));
+ addRelationButton.call(svgIcon("#iD-icon-plus", "light"));
+ addRelationButton.call(uiTooltip().title(() => _t.append("inspector.add_to_relation")).placement(_mainLocalizer.textDirection() === "ltr" ? "right" : "left"));
+ addRowEnter.append("div").attr("class", "space-value");
+ addRowEnter.append("div").attr("class", "space-buttons");
+ 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;
+ }
+ 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", "");
+ context.surface().selectAll(".highlighted").classed("highlighted", false);
+ }
+ function bindTypeahead(d) {
+ var row = select_default2(this);
+ var role = row.selectAll("input.member-role");
+ var origValue = role.property("value");
+ function sort(value, data) {
+ var sameletter = [];
+ var other = [];
+ for (var i2 = 0; i2 < data.length; i2++) {
+ if (data[i2].value.substring(0, value.length) === value) {
+ sameletter.push(data[i2]);
+ } else {
+ other.push(data[i2]);
+ }
+ }
+ return sameletter.concat(other);
+ }
+ role.call(
+ uiCombobox(context, "member-role").fetcher(function(role2, callback) {
+ var rtype = d.relation.tags.type;
+ taginfo.roles({
+ debounce: true,
+ rtype: rtype || "",
+ geometry: context.graph().geometry(_entityIDs[0]),
+ query: role2
+ }, function(err, data) {
+ if (!err)
+ callback(sort(role2, data));
+ });
+ }).on("cancel", function() {
+ role.property("value", origValue);
+ })
+ );
+ }
+ function unbind() {
+ var row = select_default2(this);
+ row.selectAll("input.member-role").call(uiCombobox.off, context);
+ }
+ }
+ section.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ _showBlank = false;
+ return section;
+ };
+ return section;
+ }
+
+ // modules/ui/sections/selection_list.js
+ function uiSectionSelectionList(context) {
+ var _selectedIDs = [];
+ var section = uiSection("selected-features", context).shouldDisplay(function() {
+ return _selectedIDs.length > 1;
+ }).label(function() {
+ return _t.append("inspector.title_count", { title: _t("inspector.features"), count: _selectedIDs.length });
+ }).disclosureContent(renderDisclosureContent);
+ context.history().on("change.selectionList", function(difference) {
+ if (difference) {
+ section.reRender();
+ }
+ });
+ section.entityIDs = function(val) {
+ if (!arguments.length)
+ return _selectedIDs;
+ _selectedIDs = val;
+ return section;
+ };
+ function selectEntity(d3_event, entity) {
+ context.enter(modeSelect(context, [entity.id]));
+ }
+ function deselectEntity(d3_event, entity) {
+ var selectedIDs = _selectedIDs.slice();
+ var index = selectedIDs.indexOf(entity.id);
+ if (index > -1) {
+ selectedIDs.splice(index, 1);
+ context.enter(modeSelect(context, selectedIDs));
+ }
+ }
+ function renderDisclosureContent(selection2) {
+ var list = selection2.selectAll(".feature-list").data([0]);
+ list = list.enter().append("ul").attr("class", "feature-list").merge(list);
+ var entities = _selectedIDs.map(function(id2) {
+ return context.hasEntity(id2);
+ }).filter(Boolean);
+ var items = list.selectAll(".feature-list-item").data(entities, osmEntity.key);
+ items.exit().remove();
+ var enter = items.enter().append("li").attr("class", "feature-list-item").each(function(d) {
+ select_default2(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"));
+ 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").text(function(entity) {
+ return _mainPresetIndex.match(entity, context.graph()).name();
+ });
+ items.selectAll(".entity-name").text(function(d) {
+ var entity = context.entity(d.id);
+ return utilDisplayName(entity);
+ });
+ }
+ return section;
+ }
+
+ // modules/ui/entity_editor.js
+ function uiEntityEditor(context) {
+ var dispatch10 = dispatch_default("choose");
+ var _state = "select";
+ var _coalesceChanges = false;
+ var _modified = false;
+ var _base;
+ var _entityIDs;
+ var _activePresets = [];
+ var _newFeature;
+ var _sections;
+ function entityEditor(selection2) {
+ var combinedTags = utilCombinedTags(_entityIDs, context.graph());
+ var header = selection2.selectAll(".header").data([0]);
+ var headerEnter = header.enter().append("div").attr("class", "header fillL");
+ var direction = _mainLocalizer.textDirection() === "rtl" ? "forward" : "backward";
+ headerEnter.append("button").attr("class", "preset-reset preset-choose").attr("title", _t("inspector.back_tooltip")).call(svgIcon(`#iD-icon-${direction}`));
+ headerEnter.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", function() {
+ context.enter(modeBrowse(context));
+ }).call(svgIcon(_modified ? "#iD-icon-apply" : "#iD-icon-close"));
+ headerEnter.append("h2");
+ header = header.merge(headerEnter);
+ header.selectAll("h2").text("").call(_entityIDs.length === 1 ? _t.append("inspector.edit") : _t.append("inspector.edit_features"));
+ header.selectAll(".preset-reset").on("click", function() {
+ dispatch10.call("choose", this, _activePresets);
+ });
+ var body = selection2.selectAll(".inspector-body").data([0]);
+ var bodyEnter = body.enter().append("div").attr("class", "entity-editor inspector-body sep-top");
+ body = body.merge(bodyEnter);
+ if (!_sections) {
+ _sections = [
+ uiSectionSelectionList(context),
+ uiSectionFeatureType(context).on("choose", function(presets) {
+ dispatch10.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);
+ }
+ if (section.presets) {
+ section.presets(_activePresets);
+ }
+ if (section.tags) {
+ section.tags(combinedTags);
+ }
+ if (section.state) {
+ section.state(_state);
+ }
+ body.call(section.render);
+ });
+ context.history().on("change.entity-editor", historyChanged);
+ function historyChanged(difference) {
+ if (selection2.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(selection2);
+ if (priorActivePreset && _activePresets.length === 1 && priorActivePreset !== _activePresets[0]) {
+ context.container().selectAll(".entity-editor button.preset-reset .label").style("background-color", "#fff").transition().duration(750).style("background-color", null);
+ }
+ }
+ }
+ function changeTags(entityIDs, changed, onInput) {
+ var actions = [];
+ for (var i2 in entityIDs) {
+ var entityID = entityIDs[i2];
+ var entity = context.entity(entityID);
+ var tags = Object.assign({}, entity.tags);
+ if (typeof changed === "function") {
+ tags = changed(tags);
+ } else {
+ for (var k in changed) {
+ if (!k)
+ continue;
+ var v = changed[k];
+ if (typeof v === "object") {
+ tags[k] = tags[v.oldKey];
+ } else if (v !== void 0 || tags.hasOwnProperty(k)) {
+ tags[k] = v;
+ }
+ }
+ }
+ if (!onInput) {
+ tags = utilCleanTags(tags);
+ }
+ if (!(0, import_fast_deep_equal10.default)(entity.tags, tags)) {
+ actions.push(actionChangeTags(entityID, tags));
+ }
+ }
+ if (actions.length) {
+ var combinedAction = function(graph) {
+ actions.forEach(function(action) {
+ graph = action(graph);
+ });
+ return graph;
+ };
+ var annotation = _t("operations.change_tags.annotation");
+ if (_coalesceChanges) {
+ context.overwrite(combinedAction, annotation);
+ } else {
+ context.perform(combinedAction, annotation);
+ _coalesceChanges = !!onInput;
+ }
+ }
+ if (!onInput) {
+ context.validator().validate();
+ }
+ }
+ function revertTags(keys) {
+ var actions = [];
+ for (var i2 in _entityIDs) {
+ var entityID = _entityIDs[i2];
+ var original = context.graph().base().entities[entityID];
+ var changed = {};
+ for (var j2 in keys) {
+ var key = keys[j2];
+ changed[key] = original ? original.tags[key] : void 0;
+ }
+ var entity = context.entity(entityID);
+ var tags = Object.assign({}, entity.tags);
+ for (var k in changed) {
+ if (!k)
+ continue;
+ var v = changed[k];
+ if (v !== void 0 || tags.hasOwnProperty(k)) {
+ tags[k] = v;
+ }
+ }
+ tags = utilCleanTags(tags);
+ if (!(0, import_fast_deep_equal10.default)(entity.tags, tags)) {
+ actions.push(actionChangeTags(entityID, tags));
+ }
+ }
+ if (actions.length) {
+ var combinedAction = function(graph) {
+ actions.forEach(function(action) {
+ graph = action(graph);
+ });
+ return graph;
+ };
+ var annotation = _t("operations.change_tags.annotation");
+ if (_coalesceChanges) {
+ context.overwrite(combinedAction, annotation);
+ } else {
+ context.perform(combinedAction, annotation);
+ _coalesceChanges = false;
+ }
+ }
+ context.validator().validate();
+ }
+ entityEditor.modified = function(val) {
+ if (!arguments.length)
+ return _modified;
+ _modified = val;
+ return entityEditor;
+ };
+ entityEditor.state = function(val) {
+ if (!arguments.length)
+ return _state;
+ _state = val;
+ return entityEditor;
+ };
+ entityEditor.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _base = context.graph();
+ _coalesceChanges = false;
+ if (val && _entityIDs && utilArrayIdentical(_entityIDs, val))
+ return entityEditor;
+ _entityIDs = val;
+ loadActivePresets(true);
+ return entityEditor.modified(false);
+ };
+ entityEditor.newFeature = function(val) {
+ if (!arguments.length)
+ return _newFeature;
+ _newFeature = val;
+ return entityEditor;
+ };
+ function loadActivePresets(isForNewSelection) {
+ var graph = context.graph();
+ var counts = {};
+ for (var i2 in _entityIDs) {
+ var entity = graph.hasEntity(_entityIDs[i2]);
+ if (!entity)
+ return;
+ var match = _mainPresetIndex.match(entity, graph);
+ if (!counts[match.id])
+ counts[match.id] = 0;
+ counts[match.id] += 1;
+ }
+ var matches = Object.keys(counts).sort(function(p1, p2) {
+ return counts[p2] - counts[p1];
+ }).map(function(pID) {
+ return _mainPresetIndex.item(pID);
+ });
+ if (!isForNewSelection) {
+ var weakPreset = _activePresets.length === 1 && !_activePresets[0].isFallback() && Object.keys(_activePresets[0].addTags || {}).length === 0;
+ if (weakPreset && matches.length === 1 && matches[0].isFallback())
+ return;
+ }
+ entityEditor.presets(matches);
+ }
+ entityEditor.presets = function(val) {
+ if (!arguments.length)
+ return _activePresets;
+ if (!utilArrayIdentical(val, _activePresets)) {
+ _activePresets = val;
+ }
+ return entityEditor;
+ };
+ return utilRebind(entityEditor, dispatch10, "on");
+ }
+
+ // modules/ui/feature_list.js
+ var sexagesimal = __toESM(require_sexagesimal());
+ function uiFeatureList(context) {
+ var _geocodeResults;
+ function featureList(selection2) {
+ var header = selection2.append("div").attr("class", "header fillL");
+ header.append("h2").call(_t.append("inspector.feature_list"));
+ var searchWrap = selection2.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 = selection2.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("\u2318F"), focusSearch);
+ function focusSearch(d3_event) {
+ var mode = context.mode() && context.mode().id;
+ if (mode !== "browse")
+ return;
+ d3_event.preventDefault();
+ search.node().focus();
+ }
+ function keydown(d3_event) {
+ if (d3_event.keyCode === 27) {
+ search.node().blur();
+ }
+ }
+ function keypress(d3_event) {
+ var q = search.property("value"), items = list.selectAll(".feature-list-item");
+ if (d3_event.keyCode === 13 && // ↩ Return
+ q.length && items.size()) {
+ click(d3_event, items.datum());
+ }
+ }
+ function inputevent() {
+ _geocodeResults = void 0;
+ drawList();
+ }
+ 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 = sexagesimal.pair(q.toUpperCase()) || q.match(/^(-?\d+\.?\d*)\s+(-?\d+\.?\d*)$/);
+ if (locationMatch) {
+ var loc = [Number(locationMatch[0]), Number(locationMatch[1])];
+ result.push({
+ id: -1,
+ geometry: "point",
+ type: _t("inspector.location"),
+ name: dmsCoordinatePair([loc[1], loc[0]]),
+ location: loc
+ });
+ }
+ var idMatch = !locationMatch && q.match(/(?:^|\W)(node|way|relation|[nwr])\W?0*([1-9]\d*)(?:\W|$)/i);
+ 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
+ });
+ }
+ var allEntities = graph.entities;
+ var localResults = [];
+ for (var id2 in allEntities) {
+ var entity = allEntities[id2];
+ if (!entity)
+ continue;
+ var name = utilDisplayName(entity) || "";
+ if (name.toLowerCase().indexOf(q) < 0)
+ continue;
+ var matched = _mainPresetIndex.match(entity, graph);
+ var type2 = 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,
+ geometry: entity.geometry(graph),
+ type: type2,
+ name,
+ distance
+ });
+ if (localResults.length > 100)
+ break;
+ }
+ localResults = localResults.sort(function byDistance(a, b) {
+ return a.distance - b.distance;
+ });
+ result = result.concat(localResults);
+ (_geocodeResults || []).forEach(function(d) {
+ if (d.osm_type && d.osm_id) {
+ var id3 = osmEntity.id.fromOSM(d.osm_type, d.osm_id);
+ var tags = {};
+ tags[d.class] = d.type;
+ var attrs = { id: id3, type: d.osm_type, tags };
+ if (d.osm_type === "way") {
+ attrs.nodes = ["a", "a"];
+ }
+ var tempEntity = osmEntity(attrs);
+ var tempGraph = coreGraph([tempEntity]);
+ var matched2 = _mainPresetIndex.match(tempEntity, tempGraph);
+ var type3 = matched2 && matched2.name() || utilDisplayType(id3);
+ result.push({
+ id: tempEntity.id,
+ geometry: tempEntity.geometry(tempGraph),
+ type: type3,
+ name: d.display_name,
+ extent: new geoExtent(
+ [Number(d.boundingbox[3]), Number(d.boundingbox[0])],
+ [Number(d.boundingbox[2]), Number(d.boundingbox[1])]
+ )
+ });
+ }
+ });
+ if (q.match(/^[0-9]+$/)) {
+ 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
+ });
+ }
+ return result;
+ }
+ 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("").call(_t.append("geocoder.no_results_worldwide"));
+ 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").call(_t.append("geocoder.search"));
+ }
+ list.selectAll(".no-results-item").style("display", value.length && !results.length ? "block" : "none");
+ list.selectAll(".geocode-item").style("display", value && _geocodeResults === void 0 ? "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_default2(this).call(svgIcon("#iD-icon-" + d.geometry, "pre-text"));
+ });
+ label.append("span").attr("class", "entity-type").text(function(d) {
+ return d.type;
+ });
+ label.append("span").attr("class", "entity-name").text(function(d) {
+ return d.name;
+ });
+ enter.style("opacity", 0).transition().style("opacity", 1);
+ items.order();
+ items.exit().remove();
+ }
+ function mouseover(d3_event, d) {
+ if (d.id === -1)
+ return;
+ utilHighlightEntities([d.id], true, context);
+ }
+ function mouseout(d3_event, d) {
+ if (d.id === -1)
+ return;
+ utilHighlightEntities([d.id], false, context);
+ }
+ function click(d3_event, d) {
+ d3_event.preventDefault();
+ 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 {
+ context.zoomToEntity(d.id);
+ }
+ }
+ function geocoderSearch() {
+ services.geocoder.search(search.property("value"), function(err, resp) {
+ _geocodeResults = resp || [];
+ drawList();
+ });
+ }
+ }
+ return featureList;
+ }
+
+ // modules/ui/improveOSM_comments.js
+ function uiImproveOsmComments() {
+ let _qaItem;
+ function issueComments(selection2) {
+ let comments = selection2.selectAll(".comments-container").data([0]);
+ comments = comments.enter().append("div").attr("class", "comments-container").merge(comments);
+ services.improveOSM.getComments(_qaItem).then((d) => {
+ if (!d.comments)
+ return;
+ const 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"));
+ const mainEnter = commentEnter.append("div").attr("class", "comment-main");
+ const metadataEnter = mainEnter.append("div").attr("class", "comment-metadata");
+ metadataEnter.append("div").attr("class", "comment-author").each(function(d2) {
+ const osm = services.osm;
+ let selection3 = select_default2(this);
+ if (osm && d2.username) {
+ selection3 = selection3.append("a").attr("class", "comment-author-link").attr("href", osm.userURL(d2.username)).attr("target", "_blank");
+ }
+ selection3.text((d4) => d4.username);
+ });
+ metadataEnter.append("div").attr("class", "comment-date").html((d2) => _t.html("note.status.commented", { when: localeDateString2(d2.timestamp) }));
+ mainEnter.append("div").attr("class", "comment-text").append("p").text((d2) => d2.text);
+ }).catch((err) => {
+ console.log(err);
+ });
+ }
+ function localeDateString2(s) {
+ if (!s)
+ return null;
+ const options2 = { day: "numeric", month: "short", year: "numeric" };
+ const d = new Date(s * 1e3);
+ if (isNaN(d.getTime()))
+ return null;
+ return d.toLocaleDateString(_mainLocalizer.localeCode(), options2);
+ }
+ issueComments.issue = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return issueComments;
+ };
+ return issueComments;
+ }
+
+ // modules/ui/improveOSM_details.js
+ function uiImproveOsmDetails(context) {
+ let _qaItem;
+ function issueDetail(d) {
+ if (d.desc)
+ return d.desc;
+ const issueKey = d.issueKey;
+ d.replacements = d.replacements || {};
+ d.replacements.default = { html: _t.html("inspector.unknown") };
+ return _t.html(`QA.improveOSM.error_types.${issueKey}.description`, d.replacements);
+ }
+ function improveOsmDetails(selection2) {
+ const details = selection2.selectAll(".error-details").data(
+ _qaItem ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ details.exit().remove();
+ const detailsEnter = details.enter().append("div").attr("class", "error-details qa-details-container");
+ const descriptionEnter = detailsEnter.append("div").attr("class", "qa-details-subsection");
+ descriptionEnter.append("h4").call(_t.append("QA.keepRight.detail_description"));
+ descriptionEnter.append("div").attr("class", "qa-details-description-text").html(issueDetail);
+ let relatedEntities = [];
+ descriptionEnter.selectAll(".error_entity_link, .error_object_link").attr("href", "#").each(function() {
+ const link2 = select_default2(this);
+ const isObjectLink = link2.classed("error_object_link");
+ const entityID = isObjectLink ? utilEntityRoot(_qaItem.objectType) + _qaItem.objectId : this.textContent;
+ const entity = context.hasEntity(entityID);
+ relatedEntities.push(entityID);
+ link2.on("mouseenter", () => {
+ utilHighlightEntities([entityID], true, context);
+ }).on("mouseleave", () => {
+ utilHighlightEntities([entityID], false, context);
+ }).on("click", (d3_event) => {
+ d3_event.preventDefault();
+ utilHighlightEntities([entityID], false, context);
+ const osmlayer = context.layers().layer("osm");
+ if (!osmlayer.enabled()) {
+ osmlayer.enabled(true);
+ }
+ context.map().centerZoom(_qaItem.loc, 20);
+ if (entity) {
+ context.enter(modeSelect(context, [entityID]));
+ } else {
+ context.loadEntity(entityID, (err, result) => {
+ if (err)
+ return;
+ const entity2 = result.data.find((e) => e.id === entityID);
+ if (entity2)
+ context.enter(modeSelect(context, [entityID]));
+ });
+ }
+ });
+ if (entity) {
+ let name = utilDisplayName(entity);
+ if (!name && !isObjectLink) {
+ const preset = _mainPresetIndex.match(entity, context.graph());
+ name = preset && !preset.isFallback() && preset.name();
+ }
+ if (name) {
+ this.innerText = name;
+ }
+ }
+ });
+ context.features().forceVisible(relatedEntities);
+ context.map().pan([0, 0]);
+ }
+ improveOsmDetails.issue = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return improveOsmDetails;
+ };
+ return improveOsmDetails;
+ }
+
+ // modules/ui/improveOSM_header.js
+ function uiImproveOsmHeader() {
+ let _qaItem;
+ function issueTitle(d) {
+ const issueKey = d.issueKey;
+ d.replacements = d.replacements || {};
+ d.replacements.default = { html: _t.html("inspector.unknown") };
+ return _t.html(`QA.improveOSM.error_types.${issueKey}.title`, d.replacements);
+ }
+ function improveOsmHeader(selection2) {
+ const header = selection2.selectAll(".qa-header").data(
+ _qaItem ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ header.exit().remove();
+ const headerEnter = header.enter().append("div").attr("class", "qa-header");
+ const svgEnter = headerEnter.append("div").attr("class", "qa-header-icon").classed("new", (d) => d.id < 0).append("svg").attr("width", "20px").attr("height", "30px").attr("viewbox", "0 0 20 30").attr("class", (d) => `preset-icon-28 qaItem ${d.service} itemId-${d.id} itemType-${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", "12px").attr("height", "12px").attr("transform", "translate(4, 5.5)").attr("xlink:href", (d) => d.icon ? "#" + d.icon : "");
+ headerEnter.append("div").attr("class", "qa-header-label").html(issueTitle);
+ }
+ improveOsmHeader.issue = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return improveOsmHeader;
+ };
+ return improveOsmHeader;
+ }
+
+ // modules/ui/improveOSM_editor.js
+ function uiImproveOsmEditor(context) {
+ const dispatch10 = dispatch_default("change");
+ const qaDetails = uiImproveOsmDetails(context);
+ const qaComments = uiImproveOsmComments(context);
+ const qaHeader = uiImproveOsmHeader(context);
+ let _qaItem;
+ function improveOsmEditor(selection2) {
+ const headerEnter = selection2.selectAll(".header").data([0]).enter().append("div").attr("class", "header fillL");
+ headerEnter.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", () => context.enter(modeBrowse(context))).call(svgIcon("#iD-icon-close"));
+ headerEnter.append("h2").call(_t.append("QA.improveOSM.title"));
+ let body = selection2.selectAll(".body").data([0]);
+ body = body.enter().append("div").attr("class", "body").merge(body);
+ const 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 improveOsmSaveSection(selection2) {
+ const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+ const isShown = _qaItem && (isSelected || _qaItem.newComment || _qaItem.comment);
+ let saveSection = selection2.selectAll(".qa-save").data(
+ isShown ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ saveSection.exit().remove();
+ const saveSectionEnter = saveSection.enter().append("div").attr("class", "qa-save save-section cf");
+ saveSectionEnter.append("h4").attr("class", ".qa-save-header").call(_t.append("note.newComment"));
+ saveSectionEnter.append("textarea").attr("class", "new-comment-input").attr("placeholder", _t("QA.keepRight.comment_placeholder")).attr("maxlength", 1e3).property("value", (d) => d.newComment).call(utilNoAuto).on("input", changeInput).on("blur", changeInput);
+ saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
+ function changeInput() {
+ const input = select_default2(this);
+ let val = input.property("value").trim();
+ if (val === "") {
+ val = void 0;
+ }
+ _qaItem = _qaItem.update({ newComment: val });
+ const qaService = services.improveOSM;
+ if (qaService) {
+ qaService.replaceItem(_qaItem);
+ }
+ saveSection.call(qaSaveButtons);
+ }
+ }
+ function qaSaveButtons(selection2) {
+ const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+ let buttonSection = selection2.selectAll(".buttons").data(isSelected ? [_qaItem] : [], (d) => d.status + d.id);
+ buttonSection.exit().remove();
+ const buttonEnter = buttonSection.enter().append("div").attr("class", "buttons");
+ buttonEnter.append("button").attr("class", "button comment-button action").call(_t.append("QA.keepRight.save_comment"));
+ buttonEnter.append("button").attr("class", "button close-button action");
+ buttonEnter.append("button").attr("class", "button ignore-button action");
+ buttonSection = buttonSection.merge(buttonEnter);
+ buttonSection.select(".comment-button").attr("disabled", (d) => d.newComment ? null : true).on("click.comment", function(d3_event, d) {
+ this.blur();
+ const qaService = services.improveOSM;
+ if (qaService) {
+ qaService.postUpdate(d, (err, item) => dispatch10.call("change", item));
+ }
+ });
+ buttonSection.select(".close-button").html((d) => {
+ const andComment = d.newComment ? "_comment" : "";
+ return _t.html(`QA.keepRight.close${andComment}`);
+ }).on("click.close", function(d3_event, d) {
+ this.blur();
+ const qaService = services.improveOSM;
+ if (qaService) {
+ d.newStatus = "SOLVED";
+ qaService.postUpdate(d, (err, item) => dispatch10.call("change", item));
+ }
+ });
+ buttonSection.select(".ignore-button").html((d) => {
+ const andComment = d.newComment ? "_comment" : "";
+ return _t.html(`QA.keepRight.ignore${andComment}`);
+ }).on("click.ignore", function(d3_event, d) {
+ this.blur();
+ const qaService = services.improveOSM;
+ if (qaService) {
+ d.newStatus = "INVALID";
+ qaService.postUpdate(d, (err, item) => dispatch10.call("change", item));
+ }
+ });
+ }
+ improveOsmEditor.error = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return improveOsmEditor;
+ };
+ return utilRebind(improveOsmEditor, dispatch10, "on");
+ }
+
+ // modules/ui/preset_list.js
+ function uiPresetList(context) {
+ var dispatch10 = dispatch_default("cancel", "choose");
+ var _entityIDs;
+ var _currLoc;
+ var _currentPresets;
+ var _autofocus = false;
+ function presetList(selection2) {
+ if (!_entityIDs)
+ return;
+ var presets = _mainPresetIndex.matchAllGeometry(entityGeometries());
+ selection2.html("");
+ var messagewrap = selection2.append("div").attr("class", "header fillL");
+ var message = messagewrap.append("h2").call(_t.append("inspector.choose"));
+ var direction = _mainLocalizer.textDirection() === "rtl" ? "backward" : "forward";
+ messagewrap.append("button").attr("class", "preset-choose").attr("title", _entityIDs.length === 1 ? _t("inspector.edit") : _t("inspector.edit_features")).on("click", function() {
+ dispatch10.call("cancel", this);
+ }).call(svgIcon(`#iD-icon-${direction}`));
+ function initialKeydown(d3_event) {
+ if (search.property("value").length === 0 && (d3_event.keyCode === utilKeybinding.keyCodes["\u232B"] || d3_event.keyCode === utilKeybinding.keyCodes["\u2326"])) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ operationDelete(context, _entityIDs)();
+ } 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) {
+ select_default2(this).on("keydown", keydown);
+ keydown.call(this, d3_event);
+ }
+ }
+ function keydown(d3_event) {
+ if (d3_event.keyCode === utilKeybinding.keyCodes["\u2193"] && // if insertion point is at the end of the string
+ search.node().selectionStart === search.property("value").length) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var buttons = list.selectAll(".preset-list-button");
+ if (!buttons.empty())
+ buttons.nodes()[0].focus();
+ }
+ }
+ function keypress(d3_event) {
+ var value = search.property("value");
+ if (d3_event.keyCode === 13 && // ↩ Return
+ value.length) {
+ list.selectAll(".preset-list-item:first-child").each(function(d) {
+ d.choose.call(this);
+ });
+ }
+ }
+ 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.html("inspector.results", {
+ n: results.collection.length,
+ search: value
+ });
+ } else {
+ var entityPresets2 = _entityIDs.map((entityID) => _mainPresetIndex.match(context.graph().entity(entityID), context.graph()));
+ results = _mainPresetIndex.defaults(entityGeometries()[0], 36, !context.inIntro(), _currLoc, entityPresets2);
+ messageText = _t.html("inspector.choose");
+ }
+ list.call(drawList, results);
+ message.html(messageText);
+ }
+ var searchWrap = selection2.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", debounce_default(inputevent));
+ if (_autofocus) {
+ search.node().focus();
+ setTimeout(function() {
+ search.node().focus();
+ }, 0);
+ }
+ var listWrap = selection2.append("div").attr("class", "inspector-body");
+ var entityPresets = _entityIDs.map((entityID) => _mainPresetIndex.match(context.graph().entity(entityID), context.graph()));
+ var list = listWrap.append("div").attr("class", "preset-list").call(drawList, _mainPresetIndex.defaults(entityGeometries()[0], 36, !context.inIntro(), _currLoc, entityPresets));
+ context.features().on("change.preset-list", updateForFeatureHiddenState);
+ }
+ function drawList(list, presets) {
+ presets = presets.matchAllGeometry(entityGeometries());
+ var collection = presets.collection.reduce(function(collection2, preset) {
+ if (!preset)
+ return collection2;
+ if (preset.members) {
+ if (preset.members.collection.filter(function(preset2) {
+ return preset2.addable();
+ }).length > 1) {
+ collection2.push(CategoryItem(preset));
+ }
+ } else if (preset.addable()) {
+ collection2.push(PresetItem(preset));
+ }
+ return collection2;
+ }, []);
+ 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_default2(this).call(item);
+ }).style("opacity", 0).transition().style("opacity", 1);
+ updateForFeatureHiddenState();
+ }
+ function itemKeydown(d3_event) {
+ var item = select_default2(this.closest(".preset-list-item"));
+ var parentItem = select_default2(item.node().parentNode.closest(".preset-list-item"));
+ if (d3_event.keyCode === utilKeybinding.keyCodes["\u2193"]) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var nextItem = select_default2(item.node().nextElementSibling);
+ if (nextItem.empty()) {
+ if (!parentItem.empty()) {
+ nextItem = select_default2(parentItem.node().nextElementSibling);
+ }
+ } else if (select_default2(this).classed("expanded")) {
+ nextItem = item.select(".subgrid .preset-list-item:first-child");
+ }
+ if (!nextItem.empty()) {
+ nextItem.select(".preset-list-button").node().focus();
+ }
+ } else if (d3_event.keyCode === utilKeybinding.keyCodes["\u2191"]) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var previousItem = select_default2(item.node().previousElementSibling);
+ if (previousItem.empty()) {
+ if (!parentItem.empty()) {
+ previousItem = parentItem;
+ }
+ } else if (previousItem.select(".preset-list-button").classed("expanded")) {
+ previousItem = previousItem.select(".subgrid .preset-list-item:last-child");
+ }
+ if (!previousItem.empty()) {
+ previousItem.select(".preset-list-button").node().focus();
+ } else {
+ var search = select_default2(this.closest(".preset-list-pane")).select(".preset-search-input");
+ search.node().focus();
+ }
+ } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === "rtl" ? "\u2192" : "\u2190"]) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ if (!parentItem.empty()) {
+ parentItem.select(".preset-list-button").node().focus();
+ }
+ } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === "rtl" ? "\u2190" : "\u2192"]) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ item.datum().choose.call(select_default2(this).node());
+ }
+ }
+ function CategoryItem(preset) {
+ var box, sublist, shown = false;
+ function item(selection2) {
+ var wrap2 = selection2.append("div").attr("class", "preset-list-button-wrap category");
+ function click() {
+ var isExpanded = select_default2(this).classed("expanded");
+ var iconName = isExpanded ? _mainLocalizer.textDirection() === "rtl" ? "#iD-icon-backward" : "#iD-icon-forward" : "#iD-icon-down";
+ select_default2(this).classed("expanded", !isExpanded).attr("title", !isExpanded ? _t("icons.collapse") : _t("icons.expand"));
+ select_default2(this).selectAll("div.label-inner svg.icon use").attr("href", iconName);
+ item.choose();
+ }
+ var geometries = entityGeometries();
+ var button = wrap2.append("button").attr("class", "preset-list-button").attr("title", _t("icons.expand")).classed("expanded", false).call(uiPresetIcon().geometry(geometries.length === 1 && geometries[0]).preset(preset)).on("click", click).on("keydown", function(d3_event) {
+ if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === "rtl" ? "\u2190" : "\u2192"]) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ if (!select_default2(this).classed("expanded")) {
+ click.call(this, d3_event);
+ }
+ } else if (d3_event.keyCode === utilKeybinding.keyCodes[_mainLocalizer.textDirection() === "rtl" ? "\u2192" : "\u2190"]) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ if (select_default2(this).classed("expanded")) {
+ 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").call(preset.nameLabel()).append("span").text("\u2026");
+ box = selection2.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");
+ }
+ 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");
+ }
+ };
+ item.preset = preset;
+ return item;
+ }
+ function PresetItem(preset) {
+ function item(selection2) {
+ var wrap2 = selection2.append("div").attr("class", "preset-list-button-wrap");
+ var geometries = entityGeometries();
+ var button = wrap2.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, (d) => d.stringId).enter().append("div").attr("class", "namepart").text("").each(function(d) {
+ d(select_default2(this));
+ });
+ wrap2.call(item.reference.button);
+ selection2.call(item.reference.body);
+ }
+ item.choose = function() {
+ if (select_default2(this).classed("disabled"))
+ return;
+ if (!context.inIntro()) {
+ _mainPresetIndex.setMostRecent(preset, entityGeometries()[0]);
+ }
+ context.perform(
+ function(graph) {
+ for (var i2 in _entityIDs) {
+ var entityID = _entityIDs[i2];
+ var oldPreset = _mainPresetIndex.match(graph.entity(entityID), graph);
+ graph = actionChangePreset(entityID, oldPreset, preset)(graph);
+ }
+ return graph;
+ },
+ _t("operations.change_tags.annotation")
+ );
+ context.validator().validate();
+ dispatch10.call("choose", this, preset);
+ };
+ item.help = function(d3_event) {
+ d3_event.stopPropagation();
+ item.reference.toggle();
+ };
+ item.preset = preset;
+ item.reference = uiTagReference(preset.reference(), context);
+ return item;
+ }
+ function updateForFeatureHiddenState() {
+ if (!_entityIDs.every(context.hasEntity))
+ return;
+ var geometries = entityGeometries();
+ var button = context.container().selectAll(".preset-list .preset-list-button");
+ button.call(uiTooltip().destroyAny);
+ button.each(function(item, index) {
+ var hiddenPresetFeaturesId;
+ for (var i2 in geometries) {
+ hiddenPresetFeaturesId = context.features().isHiddenPreset(item.preset, geometries[i2]);
+ if (hiddenPresetFeaturesId)
+ break;
+ }
+ var isHiddenPreset = !context.inIntro() && !!hiddenPresetFeaturesId && (_currentPresets.length !== 1 || item.preset !== _currentPresets[0]);
+ select_default2(this).classed("disabled", isHiddenPreset);
+ if (isHiddenPreset) {
+ var isAutoHidden = context.features().autoHidden(hiddenPresetFeaturesId);
+ select_default2(this).call(
+ uiTooltip().title(() => _t.append("inspector.hidden_preset." + (isAutoHidden ? "zoom" : "manual"), {
+ features: _t("feature." + hiddenPresetFeaturesId + ".description")
+ })).placement(index < 2 ? "bottom" : "top")
+ );
+ }
+ });
+ }
+ presetList.autofocus = function(val) {
+ if (!arguments.length)
+ return _autofocus;
+ _autofocus = val;
+ return presetList;
+ };
+ presetList.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ _currLoc = null;
+ if (_entityIDs && _entityIDs.length) {
+ const extent = _entityIDs.reduce(function(extent2, entityID) {
+ var entity = context.graph().entity(entityID);
+ return extent2.extend(entity.extent(context.graph()));
+ }, geoExtent());
+ _currLoc = extent.center();
+ var presets = _entityIDs.map(function(entityID) {
+ return _mainPresetIndex.match(context.entity(entityID), context.graph());
+ });
+ presetList.presets(presets);
+ }
+ return presetList;
+ };
+ presetList.presets = function(val) {
+ if (!arguments.length)
+ return _currentPresets;
+ _currentPresets = val;
+ return presetList;
+ };
+ function entityGeometries() {
+ var counts = {};
+ for (var i2 in _entityIDs) {
+ var entityID = _entityIDs[i2];
+ var entity = context.entity(entityID);
+ var geometry = entity.geometry(context.graph());
+ if (geometry === "vertex" && entity.isOnAddressLine(context.graph())) {
+ geometry = "point";
+ }
+ if (!counts[geometry])
+ counts[geometry] = 0;
+ counts[geometry] += 1;
+ }
+ return Object.keys(counts).sort(function(geom1, geom2) {
+ return counts[geom2] - counts[geom1];
+ });
+ }
+ return utilRebind(presetList, dispatch10, "on");
+ }
+
+ // modules/ui/view_on_osm.js
+ function uiViewOnOSM(context) {
+ var _what;
+ function viewOnOSM(selection2) {
+ var url;
+ if (_what instanceof osmEntity) {
+ url = context.connection().entityURL(_what);
+ } else if (_what instanceof osmNote) {
+ url = context.connection().noteURL(_what);
+ }
+ var data = !_what || _what.isNew() ? [] : [_what];
+ var link2 = selection2.selectAll(".view-on-osm").data(data, function(d) {
+ return d.id;
+ });
+ link2.exit().remove();
+ var linkEnter = link2.enter().append("a").attr("class", "view-on-osm").attr("target", "_blank").attr("href", url).call(svgIcon("#iD-icon-out-link", "inline"));
+ linkEnter.append("span").call(_t.append("inspector.view_on_osm"));
+ }
+ viewOnOSM.what = function(_) {
+ if (!arguments.length)
+ return _what;
+ _what = _;
+ return viewOnOSM;
+ };
+ return viewOnOSM;
+ }
+
+ // modules/ui/inspector.js
+ function uiInspector(context) {
+ var presetList = uiPresetList(context);
+ var entityEditor = uiEntityEditor(context);
+ var wrap2 = select_default2(null), presetPane = select_default2(null), editorPane = select_default2(null);
+ var _state = "select";
+ var _entityIDs;
+ var _newFeature = false;
+ function inspector(selection2) {
+ presetList.entityIDs(_entityIDs).autofocus(_newFeature).on("choose", inspector.setPreset).on("cancel", function() {
+ inspector.setPreset();
+ });
+ entityEditor.state(_state).entityIDs(_entityIDs).on("choose", inspector.showList);
+ wrap2 = selection2.selectAll(".panewrap").data([0]);
+ var enter = wrap2.enter().append("div").attr("class", "panewrap");
+ enter.append("div").attr("class", "preset-list-pane pane");
+ enter.append("div").attr("class", "entity-editor-pane pane");
+ wrap2 = wrap2.merge(enter);
+ presetPane = wrap2.selectAll(".preset-list-pane");
+ editorPane = wrap2.selectAll(".entity-editor-pane");
+ function shouldDefaultToPresetList() {
+ if (_state !== "select")
+ return false;
+ if (_entityIDs.length !== 1)
+ return false;
+ var entityID = _entityIDs[0];
+ var entity = context.hasEntity(entityID);
+ if (!entity)
+ return false;
+ if (entity.hasNonGeometryTags())
+ return false;
+ if (_newFeature)
+ return true;
+ if (entity.geometry(context.graph()) !== "vertex")
+ return false;
+ if (context.graph().parentRelations(entity).length)
+ return false;
+ if (context.validator().getEntityIssues(entityID).length)
+ return false;
+ if (entity.isHighwayIntersection(context.graph()))
+ return false;
+ return true;
+ }
+ if (shouldDefaultToPresetList()) {
+ wrap2.style("right", "-100%");
+ editorPane.classed("hide", true);
+ presetPane.classed("hide", false).call(presetList);
+ } else {
+ wrap2.style("right", "0%");
+ presetPane.classed("hide", true);
+ editorPane.classed("hide", false).call(entityEditor);
+ }
+ var footer = selection2.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]))
+ );
+ }
+ inspector.showList = function(presets) {
+ presetPane.classed("hide", false);
+ wrap2.transition().styleTween("right", function() {
+ return value_default("0%", "-100%");
+ }).on("end", function() {
+ editorPane.classed("hide", true);
+ });
+ if (presets) {
+ presetList.presets(presets);
+ }
+ presetPane.call(presetList.autofocus(true));
+ };
+ inspector.setPreset = function(preset) {
+ if (preset && preset.id === "type/multipolygon") {
+ presetPane.call(presetList.autofocus(true));
+ } else {
+ editorPane.classed("hide", false);
+ wrap2.transition().styleTween("right", function() {
+ return value_default("-100%", "0%");
+ }).on("end", function() {
+ presetPane.classed("hide", true);
+ });
+ if (preset) {
+ entityEditor.presets([preset]);
+ }
+ editorPane.call(entityEditor);
+ }
+ };
+ inspector.state = function(val) {
+ if (!arguments.length)
+ return _state;
+ _state = val;
+ entityEditor.state(_state);
+ context.container().selectAll(".field-help-body").remove();
+ return inspector;
+ };
+ inspector.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return inspector;
+ };
+ inspector.newFeature = function(val) {
+ if (!arguments.length)
+ return _newFeature;
+ _newFeature = val;
+ return inspector;
+ };
+ return inspector;
+ }
+
+ // modules/ui/keepRight_details.js
+ function uiKeepRightDetails(context) {
+ let _qaItem;
+ function issueDetail(d) {
+ const { itemType, parentIssueType } = d;
+ const unknown = { html: _t.html("inspector.unknown") };
+ let replacements = d.replacements || {};
+ replacements.default = unknown;
+ if (_mainLocalizer.hasTextForStringId(`QA.keepRight.errorTypes.${itemType}.title`)) {
+ return _t.html(`QA.keepRight.errorTypes.${itemType}.description`, replacements);
+ } else {
+ return _t.html(`QA.keepRight.errorTypes.${parentIssueType}.description`, replacements);
+ }
+ }
+ function keepRightDetails(selection2) {
+ const details = selection2.selectAll(".error-details").data(
+ _qaItem ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ details.exit().remove();
+ const detailsEnter = details.enter().append("div").attr("class", "error-details qa-details-container");
+ const descriptionEnter = detailsEnter.append("div").attr("class", "qa-details-subsection");
+ descriptionEnter.append("h4").call(_t.append("QA.keepRight.detail_description"));
+ descriptionEnter.append("div").attr("class", "qa-details-description-text").html(issueDetail);
+ let relatedEntities = [];
+ descriptionEnter.selectAll(".error_entity_link, .error_object_link").attr("href", "#").each(function() {
+ const link2 = select_default2(this);
+ const isObjectLink = link2.classed("error_object_link");
+ const entityID = isObjectLink ? utilEntityRoot(_qaItem.objectType) + _qaItem.objectId : this.textContent;
+ const entity = context.hasEntity(entityID);
+ relatedEntities.push(entityID);
+ link2.on("mouseenter", () => {
+ utilHighlightEntities([entityID], true, context);
+ }).on("mouseleave", () => {
+ utilHighlightEntities([entityID], false, context);
+ }).on("click", (d3_event) => {
+ d3_event.preventDefault();
+ utilHighlightEntities([entityID], false, context);
+ const osmlayer = context.layers().layer("osm");
+ if (!osmlayer.enabled()) {
+ osmlayer.enabled(true);
+ }
+ context.map().centerZoomEase(_qaItem.loc, 20);
+ if (entity) {
+ context.enter(modeSelect(context, [entityID]));
+ } else {
+ context.loadEntity(entityID, (err, result) => {
+ if (err)
+ return;
+ const entity2 = result.data.find((e) => e.id === entityID);
+ if (entity2)
+ context.enter(modeSelect(context, [entityID]));
+ });
+ }
+ });
+ if (entity) {
+ let name = utilDisplayName(entity);
+ if (!name && !isObjectLink) {
+ const preset = _mainPresetIndex.match(entity, context.graph());
+ name = preset && !preset.isFallback() && preset.name();
+ }
+ if (name) {
+ this.innerText = name;
+ }
+ }
+ });
+ context.features().forceVisible(relatedEntities);
+ context.map().pan([0, 0]);
+ }
+ keepRightDetails.issue = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return keepRightDetails;
+ };
+ return keepRightDetails;
+ }
+
+ // modules/ui/keepRight_header.js
+ function uiKeepRightHeader() {
+ let _qaItem;
+ function issueTitle(d) {
+ const { itemType, parentIssueType } = d;
+ const unknown = _t.html("inspector.unknown");
+ let replacements = d.replacements || {};
+ replacements.default = { html: unknown };
+ if (_mainLocalizer.hasTextForStringId(`QA.keepRight.errorTypes.${itemType}.title`)) {
+ return _t.html(`QA.keepRight.errorTypes.${itemType}.title`, replacements);
+ } else {
+ return _t.html(`QA.keepRight.errorTypes.${parentIssueType}.title`, replacements);
+ }
+ }
+ function keepRightHeader(selection2) {
+ const header = selection2.selectAll(".qa-header").data(
+ _qaItem ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ header.exit().remove();
+ const headerEnter = header.enter().append("div").attr("class", "qa-header");
+ const iconEnter = headerEnter.append("div").attr("class", "qa-header-icon").classed("new", (d) => d.id < 0);
+ iconEnter.append("div").attr("class", (d) => `preset-icon-28 qaItem ${d.service} itemId-${d.id} itemType-${d.parentIssueType}`).call(svgIcon("#iD-icon-bolt", "qaItem-fill"));
+ headerEnter.append("div").attr("class", "qa-header-label").html(issueTitle);
+ }
+ keepRightHeader.issue = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return keepRightHeader;
+ };
+ return keepRightHeader;
+ }
+
+ // modules/ui/view_on_keepRight.js
+ function uiViewOnKeepRight() {
+ let _qaItem;
+ function viewOnKeepRight(selection2) {
+ let url;
+ if (services.keepRight && _qaItem instanceof QAItem) {
+ url = services.keepRight.issueURL(_qaItem);
+ }
+ const link2 = selection2.selectAll(".view-on-keepRight").data(url ? [url] : []);
+ link2.exit().remove();
+ const linkEnter = link2.enter().append("a").attr("class", "view-on-keepRight").attr("target", "_blank").attr("rel", "noopener").attr("href", (d) => d).call(svgIcon("#iD-icon-out-link", "inline"));
+ linkEnter.append("span").call(_t.append("inspector.view_on_keepRight"));
+ }
+ viewOnKeepRight.what = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return viewOnKeepRight;
+ };
+ return viewOnKeepRight;
+ }
+
+ // modules/ui/keepRight_editor.js
+ function uiKeepRightEditor(context) {
+ const dispatch10 = dispatch_default("change");
+ const qaDetails = uiKeepRightDetails(context);
+ const qaHeader = uiKeepRightHeader(context);
+ let _qaItem;
+ function keepRightEditor(selection2) {
+ const headerEnter = selection2.selectAll(".header").data([0]).enter().append("div").attr("class", "header fillL");
+ headerEnter.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", () => context.enter(modeBrowse(context))).call(svgIcon("#iD-icon-close"));
+ headerEnter.append("h2").call(_t.append("QA.keepRight.title"));
+ let body = selection2.selectAll(".body").data([0]);
+ body = body.enter().append("div").attr("class", "body").merge(body);
+ const 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);
+ const footer = selection2.selectAll(".footer").data([0]);
+ footer.enter().append("div").attr("class", "footer").merge(footer).call(uiViewOnKeepRight(context).what(_qaItem));
+ }
+ function keepRightSaveSection(selection2) {
+ const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+ const isShown = _qaItem && (isSelected || _qaItem.newComment || _qaItem.comment);
+ let saveSection = selection2.selectAll(".qa-save").data(
+ isShown ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ saveSection.exit().remove();
+ const saveSectionEnter = saveSection.enter().append("div").attr("class", "qa-save save-section cf");
+ saveSectionEnter.append("h4").attr("class", ".qa-save-header").call(_t.append("QA.keepRight.comment"));
+ saveSectionEnter.append("textarea").attr("class", "new-comment-input").attr("placeholder", _t("QA.keepRight.comment_placeholder")).attr("maxlength", 1e3).property("value", (d) => d.newComment || d.comment).call(utilNoAuto).on("input", changeInput).on("blur", changeInput);
+ saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
+ function changeInput() {
+ const input = select_default2(this);
+ let val = input.property("value").trim();
+ if (val === _qaItem.comment) {
+ val = void 0;
+ }
+ _qaItem = _qaItem.update({ newComment: val });
+ const qaService = services.keepRight;
+ if (qaService) {
+ qaService.replaceItem(_qaItem);
+ }
+ saveSection.call(qaSaveButtons);
+ }
+ }
+ function qaSaveButtons(selection2) {
+ const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+ let buttonSection = selection2.selectAll(".buttons").data(isSelected ? [_qaItem] : [], (d) => d.status + d.id);
+ buttonSection.exit().remove();
+ const buttonEnter = buttonSection.enter().append("div").attr("class", "buttons");
+ buttonEnter.append("button").attr("class", "button comment-button action").call(_t.append("QA.keepRight.save_comment"));
+ buttonEnter.append("button").attr("class", "button close-button action");
+ buttonEnter.append("button").attr("class", "button ignore-button action");
+ buttonSection = buttonSection.merge(buttonEnter);
+ buttonSection.select(".comment-button").attr("disabled", (d) => d.newComment ? null : true).on("click.comment", function(d3_event, d) {
+ this.blur();
+ const qaService = services.keepRight;
+ if (qaService) {
+ qaService.postUpdate(d, (err, item) => dispatch10.call("change", item));
+ }
+ });
+ buttonSection.select(".close-button").html((d) => {
+ const andComment = d.newComment ? "_comment" : "";
+ return _t.html(`QA.keepRight.close${andComment}`);
+ }).on("click.close", function(d3_event, d) {
+ this.blur();
+ const qaService = services.keepRight;
+ if (qaService) {
+ d.newStatus = "ignore_t";
+ qaService.postUpdate(d, (err, item) => dispatch10.call("change", item));
+ }
+ });
+ buttonSection.select(".ignore-button").html((d) => {
+ const andComment = d.newComment ? "_comment" : "";
+ return _t.html(`QA.keepRight.ignore${andComment}`);
+ }).on("click.ignore", function(d3_event, d) {
+ this.blur();
+ const qaService = services.keepRight;
+ if (qaService) {
+ d.newStatus = "ignore";
+ qaService.postUpdate(d, (err, item) => dispatch10.call("change", item));
+ }
+ });
+ }
+ keepRightEditor.error = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return keepRightEditor;
+ };
+ return utilRebind(keepRightEditor, dispatch10, "on");
+ }
+
+ // modules/ui/lasso.js
+ function uiLasso(context) {
+ var group, polygon2;
+ lasso.coordinates = [];
+ function lasso(selection2) {
+ context.container().classed("lasso", true);
+ group = selection2.append("g").attr("class", "lasso hide");
+ polygon2 = group.append("path").attr("class", "lasso-path");
+ group.call(uiToggle(true));
+ }
+ function draw() {
+ if (polygon2) {
+ polygon2.data([lasso.coordinates]).attr("d", function(d) {
+ return "M" + d.join(" L") + " Z";
+ });
+ }
+ }
+ lasso.extent = function() {
+ return lasso.coordinates.reduce(function(extent, point2) {
+ return extent.extend(geoExtent(point2));
+ }, geoExtent());
+ };
+ lasso.p = function(_) {
+ if (!arguments.length)
+ return lasso;
+ lasso.coordinates.push(_);
+ draw();
+ return lasso;
+ };
+ lasso.close = function() {
+ if (group) {
+ group.call(uiToggle(false, function() {
+ select_default2(this).remove();
+ }));
+ }
+ context.container().classed("lasso", false);
+ };
+ return lasso;
+ }
+
+ // modules/ui/note_comments.js
+ function uiNoteComments() {
+ var _note;
+ function noteComments(selection2) {
+ if (_note.isNew())
+ return;
+ var comments = selection2.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 selection3 = select_default2(this);
+ var osm = services.osm;
+ if (osm && d.user) {
+ selection3 = selection3.append("a").attr("class", "comment-author-link").attr("href", osm.userURL(d.user)).attr("target", "_blank");
+ }
+ if (d.user) {
+ selection3.text(d.user);
+ } else {
+ selection3.call(_t.append("note.anonymous"));
+ }
+ });
+ metadataEnter.append("div").attr("class", "comment-date").html(function(d) {
+ return _t.html("note.status." + d.action, { when: localeDateString2(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);
+ }
+ function replaceAvatars(selection2) {
+ var showThirdPartyIcons = corePreferences("preferences.privacy.thirdpartyicons") || "true";
+ var osm = services.osm;
+ if (showThirdPartyIcons !== "true" || !osm)
+ return;
+ var uids = {};
+ _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;
+ selection2.selectAll(".comment-avatar.user-" + uid).html("").append("img").attr("class", "icon comment-avatar-icon").attr("src", user.image_url).attr("alt", user.display_name);
+ });
+ });
+ }
+ function localeDateString2(s) {
+ if (!s)
+ return null;
+ var options2 = { day: "numeric", month: "short", year: "numeric" };
+ s = s.replace(/-/g, "/");
+ var d = new Date(s);
+ if (isNaN(d.getTime()))
+ return null;
+ return d.toLocaleDateString(_mainLocalizer.localeCode(), options2);
+ }
+ noteComments.note = function(val) {
+ if (!arguments.length)
+ return _note;
+ _note = val;
+ return noteComments;
+ };
+ return noteComments;
+ }
+
+ // modules/ui/note_header.js
+ function uiNoteHeader() {
+ var _note;
+ function noteHeader(selection2) {
+ var header = selection2.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 (d.id < 0) {
+ statusIcon = "#iD-icon-plus";
+ } else if (d.status === "open") {
+ statusIcon = "#iD-icon-close";
+ } else {
+ statusIcon = "#iD-icon-apply";
+ }
+ iconEnter.append("div").attr("class", "note-icon-annotation").attr("title", _t("icons.close")).call(svgIcon(statusIcon, "icon-annotation"));
+ });
+ headerEnter.append("div").attr("class", "note-header-label").html(function(d) {
+ if (_note.isNew()) {
+ return _t.html("note.new");
+ }
+ return _t.html("note.note") + " " + d.id + " " + (d.status === "closed" ? _t.html("note.closed") : "");
+ });
+ }
+ noteHeader.note = function(val) {
+ if (!arguments.length)
+ return _note;
+ _note = val;
+ return noteHeader;
+ };
+ return noteHeader;
+ }
+
+ // modules/ui/note_report.js
+ function uiNoteReport() {
+ var _note;
+ function noteReport(selection2) {
+ var url;
+ if (services.osm && _note instanceof osmNote && !_note.isNew()) {
+ url = services.osm.noteReportURL(_note);
+ }
+ var link2 = selection2.selectAll(".note-report").data(url ? [url] : []);
+ link2.exit().remove();
+ var linkEnter = link2.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").call(_t.append("note.report"));
+ }
+ noteReport.note = function(val) {
+ if (!arguments.length)
+ return _note;
+ _note = val;
+ return noteReport;
+ };
+ return noteReport;
+ }
+
+ // modules/ui/note_editor.js
+ function uiNoteEditor(context) {
+ var dispatch10 = dispatch_default("change");
+ var noteComments = uiNoteComments(context);
+ var noteHeader = uiNoteHeader();
+ var _note;
+ var _newNote;
+ function noteEditor(selection2) {
+ var header = selection2.selectAll(".header").data([0]);
+ var headerEnter = header.enter().append("div").attr("class", "header fillL");
+ headerEnter.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", function() {
+ context.enter(modeBrowse(context));
+ }).call(svgIcon("#iD-icon-close"));
+ headerEnter.append("h2").call(_t.append("note.title"));
+ var body = selection2.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 = selection2.selectAll(".footer").data([0]);
+ footer.enter().append("div").attr("class", "footer").merge(footer).call(uiViewOnOSM(context).what(_note)).call(uiNoteReport(context).note(_note));
+ var osm = services.osm;
+ if (osm) {
+ osm.on("change.note-save", function() {
+ selection2.call(noteEditor);
+ });
+ }
+ }
+ function noteSaveSection(selection2) {
+ var isSelected = _note && _note.id === context.selectedNoteID();
+ var noteSave = selection2.selectAll(".note-save").data(isSelected ? [_note] : [], function(d) {
+ return d.status + d.id;
+ });
+ noteSave.exit().remove();
+ var noteSaveEnter = noteSave.enter().append("div").attr("class", "note-save save-section cf");
+ noteSaveEnter.append("h4").attr("class", ".note-save-header").html(function() {
+ return _note.isNew() ? _t.html("note.newDescription") : _t.html("note.newComment");
+ });
+ var commentTextarea = noteSaveEnter.append("textarea").attr("class", "new-comment-input").attr("placeholder", _t("note.inputPlaceholder")).attr("maxlength", 1e3).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) {
+ commentTextarea.node().focus();
+ }
+ noteSave = noteSaveEnter.merge(noteSave).call(userDetails).call(noteSaveButtons);
+ 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_default2(this).on("keydown.note-input", null);
+ window.setTimeout(function() {
+ if (_note.isNew()) {
+ noteSave.selectAll(".save-button").node().focus();
+ clickSave(_note);
+ } else {
+ noteSave.selectAll(".comment-button").node().focus();
+ clickComment(_note);
+ }
+ }, 10);
+ }
+ function changeInput() {
+ var input = select_default2(this);
+ var val = input.property("value").trim() || void 0;
+ _note = _note.update({ newComment: val });
+ var osm = services.osm;
+ if (osm) {
+ osm.replaceNote(_note);
+ }
+ noteSave.call(noteSaveButtons);
+ }
+ }
+ function userDetails(selection2) {
+ var detailSection = selection2.selectAll(".detail-section").data([0]);
+ detailSection = detailSection.enter().append("div").attr("class", "detail-section").merge(detailSection);
+ var osm = services.osm;
+ if (!osm)
+ return;
+ 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").call(_t.append("note.login"));
+ authEnter.append("a").attr("target", "_blank").call(svgIcon("#iD-icon-out-link", "inline")).append("span").call(_t.append("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").call(_t.append("note.upload_explanation")).merge(prose);
+ osm.userDetails(function(err, user) {
+ if (err)
+ return;
+ var userLink = select_default2(document.createElement("div"));
+ 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").text(user.display_name).attr("href", osm.userURL(user.display_name)).attr("target", "_blank");
+ prose.html(_t.html("note.upload_explanation_with_user", { user: { html: userLink.html() } }));
+ });
+ }
+ function noteSaveButtons(selection2) {
+ var osm = services.osm;
+ var hasAuth = osm && osm.authenticated();
+ var isSelected = _note && _note.id === context.selectedNoteID();
+ var buttonSection = selection2.selectAll(".buttons").data(isSelected ? [_note] : [], function(d) {
+ return d.status + d.id;
+ });
+ buttonSection.exit().remove();
+ var buttonEnter = buttonSection.enter().append("div").attr("class", "buttons");
+ if (_note.isNew()) {
+ buttonEnter.append("button").attr("class", "button cancel-button secondary-action").call(_t.append("confirm.cancel"));
+ buttonEnter.append("button").attr("class", "button save-button action").call(_t.append("note.save"));
+ } else {
+ buttonEnter.append("button").attr("class", "button status-button action");
+ buttonEnter.append("button").attr("class", "button comment-button action").call(_t.append("note.comment"));
+ }
+ buttonSection = buttonSection.merge(buttonEnter);
+ buttonSection.select(".cancel-button").on("click.cancel", clickCancel);
+ buttonSection.select(".save-button").attr("disabled", isSaveDisabled).on("click.save", clickSave);
+ buttonSection.select(".status-button").attr("disabled", hasAuth ? null : true).html(function(d) {
+ var action = d.status === "open" ? "close" : "open";
+ var andComment = d.newComment ? "_comment" : "";
+ return _t.html("note." + action + andComment);
+ }).on("click.status", clickStatus);
+ buttonSection.select(".comment-button").attr("disabled", isSaveDisabled).on("click.comment", clickComment);
+ function isSaveDisabled(d) {
+ return hasAuth && d.status === "open" && d.newComment ? null : true;
+ }
+ }
+ function clickCancel(d3_event, d) {
+ this.blur();
+ var osm = services.osm;
+ if (osm) {
+ osm.removeNote(d);
+ }
+ context.enter(modeBrowse(context));
+ dispatch10.call("change");
+ }
+ function clickSave(d3_event, d) {
+ this.blur();
+ var osm = services.osm;
+ if (osm) {
+ osm.postNoteCreate(d, function(err, note) {
+ dispatch10.call("change", note);
+ });
+ }
+ }
+ function clickStatus(d3_event, d) {
+ this.blur();
+ var osm = services.osm;
+ if (osm) {
+ var setStatus = d.status === "open" ? "closed" : "open";
+ osm.postNoteUpdate(d, setStatus, function(err, note) {
+ dispatch10.call("change", note);
+ });
+ }
+ }
+ function clickComment(d3_event, d) {
+ this.blur();
+ var osm = services.osm;
+ if (osm) {
+ osm.postNoteUpdate(d, d.status, function(err, note) {
+ dispatch10.call("change", note);
+ });
+ }
+ }
+ noteEditor.note = function(val) {
+ if (!arguments.length)
+ return _note;
+ _note = val;
+ return noteEditor;
+ };
+ noteEditor.newNote = function(val) {
+ if (!arguments.length)
+ return _newNote;
+ _newNote = val;
+ return noteEditor;
+ };
+ return utilRebind(noteEditor, dispatch10, "on");
+ }
+
+ // modules/ui/source_switch.js
+ function uiSourceSwitch(context) {
+ var keys;
+ 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_default2(this).classed("live");
+ isLive = !isLive;
+ context.enter(modeBrowse(context));
+ context.history().clearSaved();
+ context.flush();
+ select_default2(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]);
+ }
+ var sourceSwitch = function(selection2) {
+ selection2.append("a").attr("href", "#").call(_t.append("source_switch.live")).attr("class", "live chip").on("click", click);
+ };
+ sourceSwitch.keys = function(_) {
+ if (!arguments.length)
+ return keys;
+ keys = _;
+ return sourceSwitch;
+ };
+ return sourceSwitch;
+ }
+
+ // modules/ui/spinner.js
+ function uiSpinner(context) {
+ var osm = context.connection();
+ return function(selection2) {
+ var img = selection2.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);
+ });
+ }
+ };
+ }
+
+ // modules/ui/sections/privacy.js
+ function uiSectionPrivacy(context) {
+ let section = uiSection("preferences-third-party", context).label(() => _t.append("preferences.privacy.title")).disclosureContent(renderDisclosureContent);
+ function renderDisclosureContent(selection2) {
+ selection2.selectAll(".privacy-options-list").data([0]).enter().append("ul").attr("class", "layer-list privacy-options-list");
+ let thirdPartyIconsEnter = selection2.select(".privacy-options-list").selectAll(".privacy-third-party-icons-item").data([corePreferences("preferences.privacy.thirdpartyicons") || "true"]).enter().append("li").attr("class", "privacy-third-party-icons-item").append("label").call(
+ uiTooltip().title(() => _t.append("preferences.privacy.third_party_icons.tooltip")).placement("bottom")
+ );
+ thirdPartyIconsEnter.append("input").attr("type", "checkbox").on("change", (d3_event, d) => {
+ d3_event.preventDefault();
+ corePreferences("preferences.privacy.thirdpartyicons", d === "true" ? "false" : "true");
+ });
+ thirdPartyIconsEnter.append("span").call(_t.append("preferences.privacy.third_party_icons.description"));
+ selection2.selectAll(".privacy-third-party-icons-item").classed("active", (d) => d === "true").select("input").property("checked", (d) => d === "true");
+ selection2.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").call(_t.append("preferences.privacy.privacy_link"));
+ }
+ corePreferences.onChange("preferences.privacy.thirdpartyicons", section.reRender);
+ return section;
+ }
+
+ // modules/ui/splash.js
+ function uiSplash(context) {
+ return (selection2) => {
+ if (context.history().hasRestorableChanges())
+ return;
+ let updateMessage = "";
+ const sawPrivacyVersion = corePreferences("sawPrivacyVersion");
+ let showSplash = !corePreferences("sawSplash");
+ if (sawPrivacyVersion !== context.privacyVersion) {
+ updateMessage = _t("splash.privacy_update");
+ showSplash = true;
+ }
+ if (!showSplash)
+ return;
+ corePreferences("sawSplash", true);
+ corePreferences("sawPrivacyVersion", context.privacyVersion);
+ _mainFileFetcher.get("intro_graph");
+ let modalSelection = uiModal(selection2);
+ modalSelection.select(".modal").attr("class", "modal-splash modal");
+ let introModal = modalSelection.select(".content").append("div").attr("class", "fillL");
+ introModal.append("div").attr("class", "modal-section").append("h3").call(_t.append("splash.welcome"));
+ let modalSection = introModal.append("div").attr("class", "modal-section");
+ modalSection.append("p").html(_t.html("splash.text", {
+ version: context.version,
+ website: { html: '<a target="_blank" href="https://github.com/openstreetmap/iD/blob/develop/CHANGELOG.md#whats-new">' + _t.html("splash.changelog") + "</a>" },
+ github: { html: '<a target="_blank" href="https://github.com/openstreetmap/iD/issues">github.com</a>' }
+ }));
+ modalSection.append("p").html(_t.html("splash.privacy", {
+ updateMessage,
+ privacyLink: { html: '<a target="_blank" href="https://github.com/openstreetmap/iD/blob/release/PRIVACY.md">' + _t("splash.privacy_policy") + "</a>" }
+ }));
+ uiSectionPrivacy(context).label(() => _t.append("splash.privacy_settings")).render(modalSection);
+ let buttonWrap = introModal.append("div").attr("class", "modal-actions");
+ let walkthrough = buttonWrap.append("button").attr("class", "walkthrough").on("click", () => {
+ 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").call(_t.append("splash.walkthrough"));
+ let 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").call(_t.append("splash.start"));
+ modalSelection.select("button.close").attr("class", "hide");
+ };
+ }
+
+ // modules/ui/status.js
+ function uiStatus(context) {
+ var osm = context.connection();
+ return function(selection2) {
+ if (!osm)
+ return;
+ function update(err, apiStatus) {
+ selection2.html("");
+ if (err) {
+ if (apiStatus === "connectionSwitched") {
+ return;
+ } else if (apiStatus === "rateLimited") {
+ selection2.call(_t.append("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").call(_t.append("login")).on("click.login", function(d3_event) {
+ d3_event.preventDefault();
+ osm.authenticate();
+ });
+ } else {
+ var throttledRetry = throttle_default(function() {
+ context.loadTiles(context.projection);
+ osm.reloadApiStatus();
+ }, 2e3);
+ selection2.call(_t.append("osm_api_status.message.error", { suffix: " " })).append("a").attr("href", "#").call(_t.append("osm_api_status.retry")).on("click.retry", function(d3_event) {
+ d3_event.preventDefault();
+ throttledRetry();
+ });
+ }
+ } else if (apiStatus === "readonly") {
+ selection2.call(_t.append("osm_api_status.message.readonly"));
+ } else if (apiStatus === "offline") {
+ selection2.call(_t.append("osm_api_status.message.offline"));
+ }
+ selection2.attr("class", "api-status " + (err ? "error" : apiStatus));
+ }
+ osm.on("apiStatusChange.uiStatus", update);
+ context.history().on("storage_error", () => {
+ selection2.selectAll("span.local-storage-full").remove();
+ selection2.append("span").attr("class", "local-storage-full").call(_t.append("osm_api_status.message.local_storage_full"));
+ selection2.classed("error", true);
+ });
+ window.setInterval(function() {
+ osm.reloadApiStatus();
+ }, 9e4);
+ osm.reloadApiStatus();
+ };
+ }
+
+ // node_modules/osm-community-index/lib/simplify.js
+ var import_diacritics3 = __toESM(require_diacritics(), 1);
+ function simplify2(str2) {
+ if (typeof str2 !== "string")
+ return "";
+ return import_diacritics3.default.remove(
+ str2.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()
+ );
+ }
+
+ // node_modules/osm-community-index/lib/resolve_strings.js
+ function resolveStrings(item, defaults2, localizerFn) {
+ let itemStrings = Object.assign({}, item.strings);
+ let defaultStrings = Object.assign({}, defaults2[item.type]);
+ const anyToken = new RegExp(/(\{\w+\})/, "gi");
+ if (localizerFn) {
+ if (itemStrings.community) {
+ const communityID = simplify2(itemStrings.community);
+ itemStrings.community = localizerFn(`_communities.${communityID}`);
+ }
+ ["name", "description", "extendedDescription"].forEach((prop) => {
+ if (defaultStrings[prop])
+ defaultStrings[prop] = localizerFn(`_defaults.${item.type}.${prop}`);
+ if (itemStrings[prop])
+ itemStrings[prop] = localizerFn(`${item.id}.${prop}`);
+ });
+ }
+ let replacements = {
+ account: item.account,
+ community: itemStrings.community,
+ signupUrl: itemStrings.signupUrl,
+ url: itemStrings.url
+ };
+ if (!replacements.signupUrl) {
+ replacements.signupUrl = resolve(itemStrings.signupUrl || defaultStrings.signupUrl);
+ }
+ if (!replacements.url) {
+ replacements.url = resolve(itemStrings.url || defaultStrings.url);
+ }
+ let 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)
+ };
+ 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 resolve(s, addLinks) {
+ if (!s)
+ return void 0;
+ let result = s;
+ for (let key in replacements) {
+ const token = `{${key}}`;
+ const regex = new RegExp(token, "g");
+ if (regex.test(result)) {
+ let replacement = replacements[key];
+ if (!replacement) {
+ throw new Error(`Cannot resolve token: ${token}`);
+ } else {
+ if (addLinks && (key === "signupUrl" || key === "url")) {
+ replacement = linkify(replacement);
+ }
+ result = result.replace(regex, replacement);
+ }
+ }
+ }
+ const leftovers = result.match(anyToken);
+ if (leftovers) {
+ throw new Error(`Cannot resolve tokens: ${leftovers}`);
+ }
+ if (addLinks && item.type === "reddit") {
+ result = result.replace(/(\/r\/\w+\/*)/i, (match) => linkify(resolved.url, match));
+ }
+ return result;
+ }
+ function linkify(url, text2) {
+ if (!url)
+ return void 0;
+ text2 = text2 || url;
+ return `<a target="_blank" href="${url}">${text2}</a>`;
+ }
+ }
+
+ // modules/ui/success.js
+ var _oci = null;
+ function uiSuccess(context) {
+ const MAXEVENTS = 2;
+ const dispatch10 = dispatch_default("cancel");
+ let _changeset2;
+ let _location;
+ ensureOSMCommunityIndex();
+ function ensureOSMCommunityIndex() {
+ const data = _mainFileFetcher;
+ return Promise.all([
+ data.get("oci_features"),
+ data.get("oci_resources"),
+ data.get("oci_defaults")
+ ]).then((vals) => {
+ if (_oci)
+ return _oci;
+ if (vals[0] && Array.isArray(vals[0].features)) {
+ _sharedLocationManager.mergeCustomGeoJSON(vals[0]);
+ }
+ let ociResources = Object.values(vals[1].resources);
+ if (ociResources.length) {
+ return _sharedLocationManager.mergeLocationSets(ociResources).then(() => {
+ _oci = {
+ resources: ociResources,
+ defaults: vals[2].defaults
+ };
+ return _oci;
+ });
+ } else {
+ _oci = {
+ resources: [],
+ // no resources?
+ defaults: vals[2].defaults
+ };
+ return _oci;
+ }
+ });
+ }
+ function parseEventDate(when) {
+ if (!when)
+ return;
+ let raw = when.trim();
+ if (!raw)
+ return;
+ if (!/Z$/.test(raw)) {
+ raw += "Z";
+ }
+ const parsed = new Date(raw);
+ return new Date(parsed.toUTCString().slice(0, 25));
+ }
+ function success(selection2) {
+ let header = selection2.append("div").attr("class", "header fillL");
+ header.append("h2").call(_t.append("success.just_edited"));
+ header.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", () => dispatch10.call("cancel")).call(svgIcon("#iD-icon-close"));
+ let body = selection2.append("div").attr("class", "body save-success fillL");
+ let summary = body.append("div").attr("class", "save-summary");
+ summary.append("h3").call(_t.append("success.thank_you" + (_location ? "_location" : ""), { where: _location }));
+ summary.append("p").call(_t.append("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").call(_t.append("success.help_link_text"));
+ let osm = context.connection();
+ if (!osm)
+ return;
+ let changesetURL = osm.changesetURL(_changeset2.id);
+ let table = summary.append("table").attr("class", "summary-table");
+ let 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");
+ let 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).call(_t.append("success.view_on_osm"));
+ summaryDetail.append("div").html(_t.html("success.changeset_id", {
+ changeset_id: { html: `<a href="${changesetURL}" target="_blank">${_changeset2.id}</a>` }
+ }));
+ ensureOSMCommunityIndex().then((oci) => {
+ const loc = context.map().center();
+ const validHere = _sharedLocationManager.locationSetsAt(loc);
+ let communities = [];
+ oci.resources.forEach((resource) => {
+ let area = validHere[resource.locationSetID];
+ if (!area)
+ return;
+ const localizer = (stringID) => _t.html(`community.${stringID}`);
+ resource.resolved = resolveStrings(resource, oci.defaults, localizer);
+ communities.push({
+ area,
+ order: resource.order || 0,
+ resource
+ });
+ });
+ communities.sort((a, b) => a.area - b.area || b.order - a.order);
+ body.call(showCommunityLinks, communities.map((c) => c.resource));
+ });
+ }
+ function showCommunityLinks(selection2, resources) {
+ let communityLinks = selection2.append("div").attr("class", "save-communityLinks");
+ communityLinks.append("h3").call(_t.append("success.like_osm"));
+ let table = communityLinks.append("table").attr("class", "community-table");
+ let row = table.selectAll(".community-row").data(resources);
+ let 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", (d) => d.resolved.url).append("svg").attr("class", "logo-small").append("use").attr("xlink:href", (d) => `#community-${d.type}`);
+ let communityDetail = rowEnter.append("td").attr("class", "cell-detail community-detail");
+ communityDetail.each(showCommunityDetails);
+ communityLinks.append("div").attr("class", "community-missing").call(_t.append("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").call(_t.append("success.tell_us"));
+ }
+ function showCommunityDetails(d) {
+ let selection2 = select_default2(this);
+ let communityID = d.id;
+ selection2.append("div").attr("class", "community-name").html(d.resolved.nameHTML);
+ selection2.append("div").attr("class", "community-description").html(d.resolved.descriptionHTML);
+ if (d.resolved.extendedDescriptionHTML || d.languageCodes && d.languageCodes.length) {
+ selection2.append("div").call(
+ uiDisclosure(context, `community-more-${d.id}`, false).expanded(false).updatePreference(false).label(() => _t.append("success.more")).content(showMore)
+ );
+ }
+ let nextEvents = (d.events || []).map((event) => {
+ event.date = parseEventDate(event.when);
+ return event;
+ }).filter((event) => {
+ const t = event.date.getTime();
+ const now3 = new Date().setHours(0, 0, 0, 0);
+ return !isNaN(t) && t >= now3;
+ }).sort((a, b) => {
+ return a.date < b.date ? -1 : a.date > b.date ? 1 : 0;
+ }).slice(0, MAXEVENTS);
+ if (nextEvents.length) {
+ selection2.append("div").call(
+ uiDisclosure(context, `community-events-${d.id}`, false).expanded(false).updatePreference(false).label(_t.html("success.events")).content(showNextEvents)
+ ).select(".hide-toggle").append("span").attr("class", "badge-text").text(nextEvents.length);
+ }
+ function showMore(selection3) {
+ let more = selection3.selectAll(".community-more").data([0]);
+ let 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 (d.languageCodes && d.languageCodes.length) {
+ const languageList = d.languageCodes.map((code) => _mainLocalizer.languageName(code)).join(", ");
+ moreEnter.append("div").attr("class", "community-languages").call(_t.append("success.languages", { languages: languageList }));
+ }
+ }
+ function showNextEvents(selection3) {
+ let events = selection3.append("div").attr("class", "community-events");
+ let item = events.selectAll(".community-event").data(nextEvents);
+ let itemEnter = item.enter().append("div").attr("class", "community-event");
+ itemEnter.append("div").attr("class", "community-event-name").append("a").attr("target", "_blank").attr("href", (d2) => d2.url).text((d2) => {
+ let name = d2.name;
+ if (d2.i18n && d2.id) {
+ name = _t(`community.${communityID}.events.${d2.id}.name`, { default: name });
+ }
+ return name;
+ });
+ itemEnter.append("div").attr("class", "community-event-when").text((d2) => {
+ let options2 = { weekday: "short", day: "numeric", month: "short", year: "numeric" };
+ if (d2.date.getHours() || d2.date.getMinutes()) {
+ options2.hour = "numeric";
+ options2.minute = "numeric";
+ }
+ return d2.date.toLocaleString(_mainLocalizer.localeCode(), options2);
+ });
+ itemEnter.append("div").attr("class", "community-event-where").text((d2) => {
+ let where = d2.where;
+ if (d2.i18n && d2.id) {
+ where = _t(`community.${communityID}.events.${d2.id}.where`, { default: where });
+ }
+ return where;
+ });
+ itemEnter.append("div").attr("class", "community-event-description").text((d2) => {
+ let description = d2.description;
+ if (d2.i18n && d2.id) {
+ description = _t(`community.${communityID}.events.${d2.id}.description`, { default: description });
+ }
+ return description;
+ });
+ }
+ }
+ success.changeset = function(val) {
+ if (!arguments.length)
+ return _changeset2;
+ _changeset2 = val;
+ return success;
+ };
+ success.location = function(val) {
+ if (!arguments.length)
+ return _location;
+ _location = val;
+ return success;
+ };
+ return utilRebind(success, dispatch10, "on");
+ }
+
+ // modules/ui/version.js
+ 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;
+ }
+ return function(selection2) {
+ selection2.append("a").attr("target", "_blank").attr("href", "https://github.com/openstreetmap/iD").text(currVersion);
+ if (isNewVersion && !isNewUser) {
+ selection2.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")).call(
+ uiTooltip().title(() => _t.append("version.whats_new", { version: currVersion })).placement("top").scrollContainer(context.container().select(".main-footer-wrap"))
+ );
+ }
+ };
+ }
+
+ // modules/ui/zoom.js
+ function uiZoom(context) {
+ var zooms = [{
+ id: "zoom-in",
+ icon: "iD-icon-plus",
+ title: _t.append("zoom.in"),
+ action: zoomIn,
+ disabled: function() {
+ return !context.map().canZoomIn();
+ },
+ disabledTitle: _t.append("zoom.disabled.in"),
+ key: "+"
+ }, {
+ id: "zoom-out",
+ icon: "iD-icon-minus",
+ title: _t.append("zoom.out"),
+ action: zoomOut,
+ disabled: function() {
+ return !context.map().canZoomOut();
+ },
+ disabledTitle: _t.append("zoom.disabled.out"),
+ key: "-"
+ }];
+ function zoomIn(d3_event) {
+ if (d3_event.shiftKey)
+ return;
+ d3_event.preventDefault();
+ context.map().zoomIn();
+ }
+ function zoomOut(d3_event) {
+ if (d3_event.shiftKey)
+ return;
+ d3_event.preventDefault();
+ context.map().zoomOut();
+ }
+ function zoomInFurther(d3_event) {
+ if (d3_event.shiftKey)
+ return;
+ d3_event.preventDefault();
+ context.map().zoomInFurther();
+ }
+ function zoomOutFurther(d3_event) {
+ if (d3_event.shiftKey)
+ return;
+ d3_event.preventDefault();
+ context.map().zoomOutFurther();
+ }
+ return function(selection2) {
+ var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === "rtl" ? "right" : "left").title(function(d) {
+ if (d.disabled()) {
+ return d.disabledTitle;
+ }
+ return d.title;
+ }).keys(function(d) {
+ return [d.key];
+ });
+ var lastPointerUpType;
+ var buttons = selection2.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(2e3).iconName("#" + d.icon).iconClass("disabled").label(d.disabledTitle)();
+ }
+ lastPointerUpType = null;
+ }).call(tooltipBehavior);
+ buttons.each(function(d) {
+ select_default2(this).call(svgIcon("#" + d.icon, "light"));
+ });
+ utilKeybinding.plusKeys.forEach(function(key) {
+ context.keybinding().on([key], zoomIn);
+ context.keybinding().on([uiCmd("\u2325" + key)], zoomInFurther);
+ });
+ utilKeybinding.minusKeys.forEach(function(key) {
+ context.keybinding().on([key], zoomOut);
+ context.keybinding().on([uiCmd("\u2325" + key)], zoomOutFurther);
+ });
+ function updateButtonStates() {
+ buttons.classed("disabled", function(d) {
+ return d.disabled();
+ }).each(function() {
+ var selection3 = select_default2(this);
+ if (!selection3.select(".tooltip.in").empty()) {
+ selection3.call(tooltipBehavior.updateContent);
+ }
+ });
+ }
+ updateButtonStates();
+ context.map().on("move.uiZoom", updateButtonStates);
+ };
+ }
+
+ // modules/ui/sections/raw_tag_editor.js
+ function uiSectionRawTagEditor(id2, context) {
+ var section = uiSection(id2, context).classes("raw-tag-editor").label(function() {
+ var count = Object.keys(_tags).filter(function(d) {
+ return d;
+ }).length;
+ return _t.append("inspector.title_count", { title: _t("inspector.tags"), count });
+ }).expandedByDefault(false).disclosureContent(renderDisclosureContent);
+ var taginfo = services.taginfo;
+ var dispatch10 = dispatch_default("change");
+ var availableViews = [
+ { id: "list", icon: "#fas-th-list" },
+ { id: "text", icon: "#fas-i-cursor" }
+ ];
+ var _tagView = corePreferences("raw-tag-editor-view") || "list";
+ var _readOnlyTags = [];
+ var _orderedKeys = [];
+ var _showBlank = false;
+ var _pendingChange = null;
+ var _state;
+ var _presets;
+ var _tags;
+ var _entityIDs;
+ var _didInteract = false;
+ function interacted() {
+ _didInteract = true;
+ }
+ function renderDisclosureContent(wrap2) {
+ _orderedKeys = _orderedKeys.filter(function(key) {
+ return _tags[key] !== void 0;
+ });
+ var all = Object.keys(_tags).sort();
+ var missingKeys = utilArrayDifference(all, _orderedKeys);
+ for (var i2 in missingKeys) {
+ _orderedKeys.push(missingKeys[i2]);
+ }
+ var rowData = _orderedKeys.map(function(key, i3) {
+ return { index: i3, key, value: _tags[key] };
+ });
+ if (!rowData.length || _showBlank) {
+ _showBlank = false;
+ rowData.push({ index: rowData.length, key: "", value: "" });
+ }
+ var options2 = wrap2.selectAll(".raw-tag-options").data([0]);
+ options2.exit().remove();
+ var optionsEnter = options2.enter().insert("div", ":first-child").attr("class", "raw-tag-options").attr("role", "tablist");
+ 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("aria-selected", function(d) {
+ return _tagView === d.id;
+ }).attr("role", "tab").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);
+ wrap2.selectAll(".raw-tag-option").classed("selected", function(datum2) {
+ return datum2 === d;
+ }).attr("aria-selected", function(datum2) {
+ return datum2 === d;
+ });
+ wrap2.selectAll(".tag-text").classed("hide", d.id !== "text").each(setTextareaHeight);
+ wrap2.selectAll(".tag-list, .add-row").classed("hide", d.id !== "list");
+ }).each(function(d) {
+ select_default2(this).call(svgIcon(d.icon));
+ });
+ var textData = rowsToText(rowData);
+ var textarea = wrap2.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);
+ var list = wrap2.selectAll(".tag-list").data([0]);
+ list = list.enter().append("ul").attr("class", "tag-list" + (_tagView !== "list" ? " hide" : "")).merge(list);
+ var addRowEnter = wrap2.selectAll(".add-row").data([0]).enter().append("div").attr("class", "add-row" + (_tagView !== "list" ? " hide" : ""));
+ addRowEnter.append("button").attr("class", "add-tag").attr("aria-label", _t("inspector.add_to_tag")).call(svgIcon("#iD-icon-plus", "light")).call(uiTooltip().title(() => _t.append("inspector.add_to_tag")).placement(_mainLocalizer.textDirection() === "ltr" ? "right" : "left")).on("click", addTag);
+ addRowEnter.append("div").attr("class", "space-value");
+ addRowEnter.append("div").attr("class", "space-buttons");
+ var items = list.selectAll(".tag-row").data(rowData, function(d) {
+ return d.key;
+ });
+ items.exit().each(unbind).remove();
+ 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"));
+ items = items.merge(itemsEnter).sort(function(a, b) {
+ return a.index - b.index;
+ });
+ items.each(function(d) {
+ var row = select_default2(this);
+ var key = row.select("input.key");
+ var value = row.select("input.value");
+ if (_entityIDs && taginfo && _state !== "hover") {
+ bindTypeahead(key, value);
+ }
+ var referenceOptions = { key: d.key };
+ if (typeof d.value === "string") {
+ referenceOptions.value = d.value;
+ }
+ var reference = uiTagReference(referenceOptions, context);
+ if (_state === "hover") {
+ reference.showing(false);
+ }
+ row.select(".inner-wrap").call(reference.button);
+ row.call(reference.body);
+ row.select("button.remove");
+ });
+ 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) || 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);
+ }
+ function isReadOnly(d) {
+ for (var i2 = 0; i2 < _readOnlyTags.length; i2++) {
+ if (d.key.match(_readOnlyTags[i2]) !== null) {
+ return true;
+ }
+ }
+ return false;
+ }
+ function setTextareaHeight() {
+ if (_tagView !== "text")
+ return;
+ var selection2 = select_default2(this);
+ var matches = selection2.node().value.match(/\n/g);
+ var lineCount = 2 + Number(matches && matches.length);
+ var lineHeight = 20;
+ selection2.style("height", lineCount * lineHeight + "px");
+ }
+ function stringify3(s) {
+ return JSON.stringify(s).slice(1, -1);
+ }
+ function unstringify(s) {
+ var leading = "";
+ var trailing = "";
+ if (s.length < 1 || s.charAt(0) !== '"') {
+ leading = '"';
+ }
+ if (s.length < 2 || s.charAt(s.length - 1) !== '"' || s.charAt(s.length - 1) === '"' && s.charAt(s.length - 2) === "\\") {
+ trailing = '"';
+ }
+ return JSON.parse(leading + s + trailing);
+ }
+ function rowsToText(rows) {
+ var str2 = 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 ? stringify3(rawVal) : "";
+ return stringify3(row.key) + "=" + val;
+ }).join("\n");
+ if (_state !== "hover" && str2.length) {
+ return str2 + "\n";
+ }
+ return str2;
+ }
+ function textChanged() {
+ var newText = this.value.trim();
+ var newTags = {};
+ newText.split("\n").forEach(function(row) {
+ var m = row.match(/^\s*([^=]+)=(.*)$/);
+ 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;
+ if (change.newVal === "*" && typeof change.oldVal !== "string")
+ return;
+ if (change.type === "-") {
+ _pendingChange[change.key] = void 0;
+ } else if (change.type === "+") {
+ _pendingChange[change.key] = change.newVal || "";
+ }
+ });
+ if (Object.keys(_pendingChange).length === 0) {
+ _pendingChange = null;
+ return;
+ }
+ scheduleChange();
+ }
+ function pushMore(d3_event) {
+ if (d3_event.keyCode === 9 && !d3_event.shiftKey && section.selection().selectAll(".tag-list li:last-child input.value").node() === this && utilGetSetValue(select_default2(this))) {
+ addTag();
+ }
+ }
+ 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(value2, callback) {
+ var keyString = utilGetSetValue(key);
+ if (!_tags[keyString])
+ return;
+ var data = _tags[keyString].filter(Boolean).map(function(tagValue) {
+ return {
+ value: tagValue,
+ title: tagValue
+ };
+ });
+ callback(data);
+ }));
+ return;
+ }
+ var geometry = context.graph().geometry(_entityIDs[0]);
+ key.call(uiCombobox(context, "tag-key").fetcher(function(value2, callback) {
+ taginfo.keys({
+ debounce: true,
+ geometry,
+ query: value2
+ }, function(err, data) {
+ if (!err) {
+ var filtered = data.filter(function(d) {
+ return _tags[d.value] === void 0;
+ });
+ callback(sort(value2, filtered));
+ }
+ });
+ }));
+ value.call(uiCombobox(context, "tag-value").fetcher(function(value2, callback) {
+ taginfo.values({
+ debounce: true,
+ key: utilGetSetValue(key),
+ geometry,
+ query: value2
+ }, function(err, data) {
+ if (!err)
+ callback(sort(value2, data));
+ });
+ }));
+ function sort(value2, data) {
+ var sameletter = [];
+ var other = [];
+ for (var i2 = 0; i2 < data.length; i2++) {
+ if (data[i2].value.substring(0, value2.length) === value2) {
+ sameletter.push(data[i2]);
+ } else {
+ other.push(data[i2]);
+ }
+ }
+ return sameletter.concat(other);
+ }
+ }
+ function unbind() {
+ var row = select_default2(this);
+ row.selectAll("input.key").call(uiCombobox.off, context);
+ row.selectAll("input.value").call(uiCombobox.off, context);
+ }
+ function keyChange(d3_event, d) {
+ if (select_default2(this).attr("readonly"))
+ return;
+ var kOld = d.key;
+ if (_pendingChange && _pendingChange.hasOwnProperty(kOld) && _pendingChange[kOld] === void 0)
+ return;
+ var kNew = context.cleanTagKey(this.value.trim());
+ if (isReadOnly({ key: kNew })) {
+ this.value = kOld;
+ return;
+ }
+ if (kNew && kNew !== kOld && _tags[kNew] !== void 0) {
+ this.value = kOld;
+ section.selection().selectAll(".tag-list input.value").each(function(d2) {
+ if (d2.key === kNew) {
+ var input = select_default2(this).node();
+ input.focus();
+ input.select();
+ }
+ });
+ return;
+ }
+ _pendingChange = _pendingChange || {};
+ if (kOld) {
+ if (kOld === kNew)
+ return;
+ _pendingChange[kNew] = _pendingChange[kOld] || { oldKey: kOld };
+ _pendingChange[kOld] = void 0;
+ } else {
+ let row = this.parentNode.parentNode;
+ let inputVal = select_default2(row).selectAll("input.value");
+ let vNew = context.cleanTagValue(utilGetSetValue(inputVal));
+ _pendingChange[kNew] = vNew;
+ utilGetSetValue(inputVal, vNew);
+ }
+ var existingKeyIndex = _orderedKeys.indexOf(kOld);
+ if (existingKeyIndex !== -1)
+ _orderedKeys[existingKeyIndex] = kNew;
+ d.key = kNew;
+ this.value = kNew;
+ scheduleChange();
+ }
+ function valueChange(d3_event, d) {
+ if (isReadOnly(d))
+ return;
+ if (typeof d.value !== "string" && !this.value)
+ return;
+ if (_pendingChange && _pendingChange.hasOwnProperty(d.key) && _pendingChange[d.key] === void 0)
+ return;
+ _pendingChange = _pendingChange || {};
+ _pendingChange[d.key] = context.cleanTagValue(this.value);
+ scheduleChange();
+ }
+ function removeTag(d3_event, d) {
+ if (isReadOnly(d))
+ return;
+ if (d.key === "") {
+ _showBlank = false;
+ section.reRender();
+ } else {
+ _orderedKeys = _orderedKeys.filter(function(key) {
+ return key !== d.key;
+ });
+ _pendingChange = _pendingChange || {};
+ _pendingChange[d.key] = void 0;
+ scheduleChange();
+ }
+ }
+ function addTag() {
+ window.setTimeout(function() {
+ _showBlank = true;
+ section.reRender();
+ section.selection().selectAll(".tag-list li:last-child input.key").node().focus();
+ }, 20);
+ }
+ function scheduleChange() {
+ var entityIDs = _entityIDs;
+ window.setTimeout(function() {
+ if (!_pendingChange)
+ return;
+ dispatch10.call("change", this, entityIDs, _pendingChange);
+ _pendingChange = null;
+ }, 10);
+ }
+ section.state = function(val) {
+ if (!arguments.length)
+ return _state;
+ if (_state !== val) {
+ _orderedKeys = [];
+ _state = val;
+ }
+ return section;
+ };
+ section.presets = function(val) {
+ if (!arguments.length)
+ return _presets;
+ _presets = val;
+ if (_presets && _presets.length && _presets[0].isFallback()) {
+ section.disclosureExpanded(true);
+ } else if (!_didInteract) {
+ section.disclosureExpanded(null);
+ }
+ return section;
+ };
+ section.tags = function(val) {
+ if (!arguments.length)
+ return _tags;
+ _tags = val;
+ return section;
+ };
+ section.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
+ _entityIDs = val;
+ _orderedKeys = [];
+ }
+ return section;
+ };
+ section.readOnlyTags = function(val) {
+ if (!arguments.length)
+ return _readOnlyTags;
+ _readOnlyTags = val;
+ return section;
+ };
+ return utilRebind(section, dispatch10, "on");
+ }
+
+ // modules/ui/data_editor.js
+ function uiDataEditor(context) {
+ var dataHeader = uiDataHeader();
+ var rawTagEditor = uiSectionRawTagEditor("custom-data-tag-editor", context).expandedByDefault(true).readOnlyTags([/./]);
+ var _datum;
+ function dataEditor(selection2) {
+ var header = selection2.selectAll(".header").data([0]);
+ var headerEnter = header.enter().append("div").attr("class", "header fillL");
+ headerEnter.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", function() {
+ context.enter(modeBrowse(context));
+ }).call(svgIcon("#iD-icon-close"));
+ headerEnter.append("h2").call(_t.append("map_data.title"));
+ var body = selection2.selectAll(".body").data([0]);
+ body = body.enter().append("div").attr("class", "body").merge(body);
+ var editor = body.selectAll(".data-editor").data([0]);
+ 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]);
+ 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);
+ }
+ dataEditor.datum = function(val) {
+ if (!arguments.length)
+ return _datum;
+ _datum = val;
+ return this;
+ };
+ return dataEditor;
+ }
+
+ // modules/ui/osmose_details.js
+ function uiOsmoseDetails(context) {
+ let _qaItem;
+ function issueString(d, type2) {
+ if (!d)
+ return "";
+ const s = services.osmose.getStrings(d.itemType);
+ return type2 in s ? s[type2] : "";
+ }
+ function osmoseDetails(selection2) {
+ const details = selection2.selectAll(".error-details").data(
+ _qaItem ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ details.exit().remove();
+ const detailsEnter = details.enter().append("div").attr("class", "error-details qa-details-container");
+ if (issueString(_qaItem, "detail")) {
+ const div = detailsEnter.append("div").attr("class", "qa-details-subsection");
+ div.append("h4").call(_t.append("QA.keepRight.detail_description"));
+ div.append("p").attr("class", "qa-details-description-text").html((d) => issueString(d, "detail")).selectAll("a").attr("rel", "noopener").attr("target", "_blank");
+ }
+ const detailsDiv = detailsEnter.append("div").attr("class", "qa-details-subsection");
+ const elemsDiv = detailsEnter.append("div").attr("class", "qa-details-subsection");
+ if (issueString(_qaItem, "fix")) {
+ const div = detailsEnter.append("div").attr("class", "qa-details-subsection");
+ div.append("h4").call(_t.append("QA.osmose.fix_title"));
+ div.append("p").html((d) => issueString(d, "fix")).selectAll("a").attr("rel", "noopener").attr("target", "_blank");
+ }
+ if (issueString(_qaItem, "trap")) {
+ const div = detailsEnter.append("div").attr("class", "qa-details-subsection");
+ div.append("h4").call(_t.append("QA.osmose.trap_title"));
+ div.append("p").html((d) => issueString(d, "trap")).selectAll("a").attr("rel", "noopener").attr("target", "_blank");
+ }
+ const thisItem = _qaItem;
+ services.osmose.loadIssueDetail(_qaItem).then((d) => {
+ if (!d.elems || d.elems.length === 0)
+ return;
+ if (context.selectedErrorID() !== thisItem.id && context.container().selectAll(`.qaItem.osmose.hover.itemId-${thisItem.id}`).empty())
+ return;
+ if (d.detail) {
+ detailsDiv.append("h4").call(_t.append("QA.osmose.detail_title"));
+ detailsDiv.append("p").html((d2) => d2.detail).selectAll("a").attr("rel", "noopener").attr("target", "_blank");
+ }
+ elemsDiv.append("h4").call(_t.append("QA.osmose.elems_title"));
+ elemsDiv.append("ul").selectAll("li").data(d.elems).enter().append("li").append("a").attr("href", "#").attr("class", "error_entity_link").text((d2) => d2).each(function() {
+ const link2 = select_default2(this);
+ const entityID = this.textContent;
+ const entity = context.hasEntity(entityID);
+ link2.on("mouseenter", () => {
+ utilHighlightEntities([entityID], true, context);
+ }).on("mouseleave", () => {
+ utilHighlightEntities([entityID], false, context);
+ }).on("click", (d3_event) => {
+ d3_event.preventDefault();
+ utilHighlightEntities([entityID], false, context);
+ const osmlayer = context.layers().layer("osm");
+ if (!osmlayer.enabled()) {
+ osmlayer.enabled(true);
+ }
+ context.map().centerZoom(d.loc, 20);
+ if (entity) {
+ context.enter(modeSelect(context, [entityID]));
+ } else {
+ context.loadEntity(entityID, (err, result) => {
+ if (err)
+ return;
+ const entity2 = result.data.find((e) => e.id === entityID);
+ if (entity2)
+ context.enter(modeSelect(context, [entityID]));
+ });
+ }
+ });
+ if (entity) {
+ let name = utilDisplayName(entity);
+ if (!name) {
+ const preset = _mainPresetIndex.match(entity, context.graph());
+ name = preset && !preset.isFallback() && preset.name();
+ }
+ if (name) {
+ this.innerText = name;
+ }
+ }
+ });
+ context.features().forceVisible(d.elems);
+ context.map().pan([0, 0]);
+ }).catch((err) => {
+ console.log(err);
+ });
+ }
+ osmoseDetails.issue = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return osmoseDetails;
+ };
+ return osmoseDetails;
+ }
+
+ // modules/ui/osmose_header.js
+ function uiOsmoseHeader() {
+ let _qaItem;
+ function issueTitle(d) {
+ const unknown = _t("inspector.unknown");
+ if (!d)
+ return unknown;
+ const s = services.osmose.getStrings(d.itemType);
+ return "title" in s ? s.title : unknown;
+ }
+ function osmoseHeader(selection2) {
+ const header = selection2.selectAll(".qa-header").data(
+ _qaItem ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ header.exit().remove();
+ const headerEnter = header.enter().append("div").attr("class", "qa-header");
+ const svgEnter = headerEnter.append("div").attr("class", "qa-header-icon").classed("new", (d) => d.id < 0).append("svg").attr("width", "20px").attr("height", "30px").attr("viewbox", "0 0 20 30").attr("class", (d) => `preset-icon-28 qaItem ${d.service} itemId-${d.id} itemType-${d.itemType}`);
+ svgEnter.append("polygon").attr("fill", (d) => 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", "12px").attr("height", "12px").attr("transform", "translate(4, 5.5)").attr("xlink:href", (d) => d.icon ? "#" + d.icon : "");
+ headerEnter.append("div").attr("class", "qa-header-label").text(issueTitle);
+ }
+ osmoseHeader.issue = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return osmoseHeader;
+ };
+ return osmoseHeader;
+ }
+
+ // modules/ui/view_on_osmose.js
+ function uiViewOnOsmose() {
+ let _qaItem;
+ function viewOnOsmose(selection2) {
+ let url;
+ if (services.osmose && _qaItem instanceof QAItem) {
+ url = services.osmose.itemURL(_qaItem);
+ }
+ const link2 = selection2.selectAll(".view-on-osmose").data(url ? [url] : []);
+ link2.exit().remove();
+ const linkEnter = link2.enter().append("a").attr("class", "view-on-osmose").attr("target", "_blank").attr("rel", "noopener").attr("href", (d) => d).call(svgIcon("#iD-icon-out-link", "inline"));
+ linkEnter.append("span").call(_t.append("inspector.view_on_osmose"));
+ }
+ viewOnOsmose.what = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return viewOnOsmose;
+ };
+ return viewOnOsmose;
+ }
+
+ // modules/ui/osmose_editor.js
+ function uiOsmoseEditor(context) {
+ const dispatch10 = dispatch_default("change");
+ const qaDetails = uiOsmoseDetails(context);
+ const qaHeader = uiOsmoseHeader(context);
+ let _qaItem;
+ function osmoseEditor(selection2) {
+ const header = selection2.selectAll(".header").data([0]);
+ const headerEnter = header.enter().append("div").attr("class", "header fillL");
+ headerEnter.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", () => context.enter(modeBrowse(context))).call(svgIcon("#iD-icon-close"));
+ headerEnter.append("h2").call(_t.append("QA.osmose.title"));
+ let body = selection2.selectAll(".body").data([0]);
+ body = body.enter().append("div").attr("class", "body").merge(body);
+ let 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);
+ const footer = selection2.selectAll(".footer").data([0]);
+ footer.enter().append("div").attr("class", "footer").merge(footer).call(uiViewOnOsmose(context).what(_qaItem));
+ }
+ function osmoseSaveSection(selection2) {
+ const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+ const isShown = _qaItem && isSelected;
+ let saveSection = selection2.selectAll(".qa-save").data(
+ isShown ? [_qaItem] : [],
+ (d) => `${d.id}-${d.status || 0}`
+ );
+ saveSection.exit().remove();
+ const saveSectionEnter = saveSection.enter().append("div").attr("class", "qa-save save-section cf");
+ saveSection = saveSectionEnter.merge(saveSection).call(qaSaveButtons);
+ }
+ function qaSaveButtons(selection2) {
+ const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+ let buttonSection = selection2.selectAll(".buttons").data(isSelected ? [_qaItem] : [], (d) => d.status + d.id);
+ buttonSection.exit().remove();
+ const 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");
+ buttonSection = buttonSection.merge(buttonEnter);
+ buttonSection.select(".close-button").call(_t.append("QA.keepRight.close")).on("click.close", function(d3_event, d) {
+ this.blur();
+ const qaService = services.osmose;
+ if (qaService) {
+ d.newStatus = "done";
+ qaService.postUpdate(d, (err, item) => dispatch10.call("change", item));
+ }
+ });
+ buttonSection.select(".ignore-button").call(_t.append("QA.keepRight.ignore")).on("click.ignore", function(d3_event, d) {
+ this.blur();
+ const qaService = services.osmose;
+ if (qaService) {
+ d.newStatus = "false";
+ qaService.postUpdate(d, (err, item) => dispatch10.call("change", item));
+ }
+ });
+ }
+ osmoseEditor.error = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return osmoseEditor;
+ };
+ return utilRebind(osmoseEditor, dispatch10, "on");
+ }
+
+ // modules/ui/sidebar.js
+ 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;
+ var _wasData = false;
+ var _wasNote = false;
+ var _wasQaItem = false;
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ function sidebar(selection2) {
+ var container = context.container();
+ var minWidth = 240;
+ var sidebarWidth;
+ var containerWidth;
+ var dragOffset;
+ selection2.style("min-width", minWidth + "px").style("max-width", "400px").style("width", "33.3333%");
+ var resizer = selection2.append("div").attr("class", "sidebar-resizer").on(_pointerPrefix + "down.sidebar-resizer", pointerdown);
+ var downPointerId, lastClientX, containerLocGetter;
+ 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());
+ dragOffset = utilFastMouse(resizer.node())(d3_event)[0] - 1;
+ sidebarWidth = selection2.node().getBoundingClientRect().width;
+ containerWidth = container.node().getBoundingClientRect().width;
+ var widthPct = sidebarWidth / containerWidth * 100;
+ selection2.style("width", widthPct + "%").style("max-width", "85%");
+ resizer.classed("dragging", true);
+ select_default2(window).on("touchmove.sidebar-resizer", function(d3_event2) {
+ d3_event2.preventDefault();
+ }, { passive: false }).on(_pointerPrefix + "move.sidebar-resizer", pointermove).on(_pointerPrefix + "up.sidebar-resizer pointercancel.sidebar-resizer", pointerup);
+ }
+ 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 = selection2.classed("collapsed");
+ var shouldCollapse = sidebarWidth < minWidth;
+ selection2.classed("collapsed", shouldCollapse);
+ if (shouldCollapse) {
+ if (!isCollapsed) {
+ selection2.style(xMarginProperty, "-400px").style("width", "400px");
+ context.ui().onResize([(sidebarWidth - dx) * scaleX, 0]);
+ }
+ } else {
+ var widthPct = sidebarWidth / containerWidth * 100;
+ selection2.style(xMarginProperty, null).style("width", widthPct + "%");
+ if (isCollapsed) {
+ context.ui().onResize([-sidebarWidth * scaleX, 0]);
+ } else {
+ context.ui().onResize([-dx * scaleX, 0]);
+ }
+ }
+ }
+ function pointerup(d3_event) {
+ if (downPointerId !== (d3_event.pointerId || "mouse"))
+ return;
+ downPointerId = null;
+ resizer.classed("dragging", false);
+ select_default2(window).on("touchmove.sidebar-resizer", null).on(_pointerPrefix + "move.sidebar-resizer", null).on(_pointerPrefix + "up.sidebar-resizer pointercancel.sidebar-resizer", null);
+ }
+ var featureListWrap = selection2.append("div").attr("class", "feature-list-pane").call(uiFeatureList(context));
+ var inspectorWrap = selection2.append("div").attr("class", "inspector-hidden inspector-wrap");
+ var hoverModeSelect = function(targets) {
+ context.container().selectAll(".feature-list-item button").classed("hover", false);
+ 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;
+ });
+ if (!elements.empty()) {
+ elements.classed("hover", true);
+ }
+ }
+ };
+ sidebar.hoverModeSelect = throttle_default(hoverModeSelect, 200);
+ function hover(targets) {
+ var datum2 = targets && targets.length && targets[0];
+ if (datum2 && datum2.__featurehash__) {
+ _wasData = true;
+ sidebar.show(dataEditor.datum(datum2));
+ selection2.selectAll(".sidebar-component").classed("inspector-hover", true);
+ } else if (datum2 instanceof osmNote) {
+ if (context.mode().id === "drag-note")
+ return;
+ _wasNote = true;
+ var osm = services.osm;
+ if (osm) {
+ datum2 = osm.getNote(datum2.id);
+ }
+ sidebar.show(noteEditor.note(datum2));
+ selection2.selectAll(".sidebar-component").classed("inspector-hover", true);
+ } else if (datum2 instanceof QAItem) {
+ _wasQaItem = true;
+ var errService = services[datum2.service];
+ if (errService) {
+ datum2 = errService.getError(datum2.id);
+ }
+ var errEditor;
+ if (datum2.service === "keepRight") {
+ errEditor = keepRightEditor;
+ } else if (datum2.service === "osmose") {
+ errEditor = osmoseEditor;
+ } else {
+ errEditor = improveOsmEditor;
+ }
+ context.container().selectAll(".qaItem." + datum2.service).classed("hover", function(d) {
+ return d.id === datum2.id;
+ });
+ sidebar.show(errEditor.error(datum2));
+ selection2.selectAll(".sidebar-component").classed("inspector-hover", true);
+ } else if (!_current && datum2 instanceof osmEntity) {
+ featureListWrap.classed("inspector-hidden", true);
+ inspectorWrap.classed("inspector-hidden", false).classed("inspector-hover", true);
+ if (!inspector.entityIDs() || !utilArrayIdentical(inspector.entityIDs(), [datum2.id]) || inspector.state() !== "hover") {
+ inspector.state("hover").entityIDs([datum2.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();
+ }
+ }
+ sidebar.hover = throttle_default(hover, 200);
+ sidebar.intersects = function(extent) {
+ var rect = selection2.node().getBoundingClientRect();
+ return extent.intersects([
+ context.projection.invert([0, rect.height]),
+ context.projection.invert([rect.width, 0])
+ ]);
+ };
+ sidebar.select = function(ids, newFeature) {
+ sidebar.hide();
+ if (ids && ids.length) {
+ var entity = ids.length === 1 && context.entity(ids[0]);
+ if (entity && newFeature && selection2.classed("collapsed")) {
+ 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);
+ inspector.state("select").entityIDs(ids).newFeature(newFeature);
+ inspectorWrap.call(inspector);
+ } else {
+ inspector.state("hide");
+ }
+ };
+ sidebar.showPresetList = function() {
+ inspector.showList();
+ };
+ sidebar.show = function(component, element) {
+ featureListWrap.classed("inspector-hidden", true);
+ inspectorWrap.classed("inspector-hidden", true);
+ if (_current)
+ _current.remove();
+ _current = selection2.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;
+ };
+ sidebar.expand = function(moveMap) {
+ if (selection2.classed("collapsed")) {
+ sidebar.toggle(moveMap);
+ }
+ };
+ sidebar.collapse = function(moveMap) {
+ if (!selection2.classed("collapsed")) {
+ sidebar.toggle(moveMap);
+ }
+ };
+ sidebar.toggle = function(moveMap) {
+ if (context.inIntro())
+ return;
+ var isCollapsed = selection2.classed("collapsed");
+ var isCollapsing = !isCollapsed;
+ var isRTL = _mainLocalizer.textDirection() === "rtl";
+ var scaleX = isRTL ? 0 : 1;
+ var xMarginProperty = isRTL ? "margin-right" : "margin-left";
+ sidebarWidth = selection2.node().getBoundingClientRect().width;
+ selection2.style("width", sidebarWidth + "px");
+ var startMargin, endMargin, lastMargin;
+ if (isCollapsing) {
+ startMargin = lastMargin = 0;
+ endMargin = -sidebarWidth;
+ } else {
+ startMargin = lastMargin = -sidebarWidth;
+ endMargin = 0;
+ }
+ if (!isCollapsing) {
+ selection2.classed("collapsed", isCollapsing);
+ }
+ selection2.transition().style(xMarginProperty, endMargin + "px").tween("panner", function() {
+ var i2 = number_default(startMargin, endMargin);
+ return function(t) {
+ var dx = lastMargin - Math.round(i2(t));
+ lastMargin = lastMargin - dx;
+ context.ui().onResize(moveMap ? void 0 : [dx * scaleX, 0]);
+ };
+ }).on("end", function() {
+ if (isCollapsing) {
+ selection2.classed("collapsed", isCollapsing);
+ }
+ if (!isCollapsing) {
+ var containerWidth2 = container.node().getBoundingClientRect().width;
+ var widthPct = sidebarWidth / containerWidth2 * 100;
+ selection2.style(xMarginProperty, null).style("width", widthPct + "%");
+ }
+ });
+ };
+ resizer.on("dblclick", function(d3_event) {
+ d3_event.preventDefault();
+ if (d3_event.sourceEvent) {
+ d3_event.sourceEvent.preventDefault();
+ }
+ sidebar.toggle();
+ });
+ context.map().on("crossEditableZoom.sidebar", function(within) {
+ if (!within && !selection2.select(".inspector-hover").empty()) {
+ hover([]);
+ }
+ });
+ }
+ sidebar.showPresetList = function() {
+ };
+ sidebar.hover = function() {
+ };
+ sidebar.hover.cancel = function() {
+ };
+ sidebar.intersects = function() {
+ };
+ sidebar.select = function() {
+ };
+ sidebar.show = function() {
+ };
+ sidebar.hide = function() {
+ };
+ sidebar.expand = function() {
+ };
+ sidebar.collapse = function() {
+ };
+ sidebar.toggle = function() {
+ };
+ return sidebar;
+ }
+
+ // modules/modes/draw_area.js
+ function modeDrawArea(context, wayID, startGraph, button) {
+ var mode = {
+ button,
+ id: "draw-area"
+ };
+ var behavior = behaviorDrawWay(context, wayID, mode, startGraph).on("rejectedSelfIntersection.modeDrawArea", function() {
+ context.ui().flash.iconName("#iD-icon-no").label(_t.append("self_intersection.error.areas"))();
+ });
+ mode.wayID = wayID;
+ mode.enter = function() {
+ context.install(behavior);
+ };
+ mode.exit = function() {
+ context.uninstall(behavior);
+ };
+ mode.selectedIDs = function() {
+ return [wayID];
+ };
+ mode.activeID = function() {
+ return behavior && behavior.activeID() || [];
+ };
+ return mode;
+ }
+
+ // modules/modes/add_area.js
+ function modeAddArea(context, mode) {
+ mode.id = "add-area";
+ var behavior = behaviorAddWay(context).on("start", start2).on("startFromWay", startFromWay).on("startFromNode", startFromNode);
+ var defaultTags = { area: "yes" };
+ if (mode.preset)
+ defaultTags = mode.preset.setTags(defaultTags, "area");
+ function actionClose(wayId) {
+ return function(graph) {
+ return graph.replace(graph.entity(wayId).close());
+ };
+ }
+ function start2(loc) {
+ var startGraph = context.graph();
+ var node = osmNode({ 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 startFromWay(loc, edge) {
+ var startGraph = context.graph();
+ var node = osmNode({ loc });
+ var way = osmWay({ tags: defaultTags });
+ context.perform(
+ actionAddEntity(node),
+ actionAddEntity(way),
+ actionAddVertex(way.id, node.id),
+ actionClose(way.id),
+ actionAddMidpoint({ loc, edge }, node)
+ );
+ context.enter(modeDrawArea(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),
+ actionClose(way.id)
+ );
+ context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
+ }
+ mode.enter = function() {
+ context.install(behavior);
+ };
+ mode.exit = function() {
+ context.uninstall(behavior);
+ };
+ return mode;
+ }
+
+ // modules/modes/add_line.js
+ function modeAddLine(context, mode) {
+ mode.id = "add-line";
+ var behavior = behaviorAddWay(context).on("start", start2).on("startFromWay", startFromWay).on("startFromNode", startFromNode);
+ var defaultTags = {};
+ if (mode.preset)
+ defaultTags = mode.preset.setTags(defaultTags, "line");
+ function start2(loc) {
+ var startGraph = context.graph();
+ var node = osmNode({ 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));
+ }
+ function startFromWay(loc, edge) {
+ var startGraph = context.graph();
+ var node = osmNode({ loc });
+ var way = osmWay({ tags: defaultTags });
+ context.perform(
+ actionAddEntity(node),
+ actionAddEntity(way),
+ actionAddVertex(way.id, node.id),
+ actionAddMidpoint({ loc, 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));
+ }
+ mode.enter = function() {
+ context.install(behavior);
+ };
+ mode.exit = function() {
+ context.uninstall(behavior);
+ };
+ return mode;
+ }
+
+ // modules/modes/add_point.js
+ 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, tags: defaultTags });
+ context.perform(
+ actionAddEntity(node),
+ _t("operations.add.annotation.point")
+ );
+ enterSelectMode(node);
+ }
+ function addWay(loc, edge) {
+ var node = osmNode({ tags: defaultTags });
+ context.perform(
+ actionAddMidpoint({ loc, edge }, node),
+ _t("operations.add.annotation.vertex")
+ );
+ enterSelectMode(node);
+ }
+ function enterSelectMode(node) {
+ context.enter(
+ modeSelect(context, [node.id]).newFeature(true)
+ );
+ }
+ function addNode(node) {
+ if (Object.keys(defaultTags).length === 0) {
+ enterSelectMode(node);
+ return;
+ }
+ var tags = Object.assign({}, node.tags);
+ for (var key in defaultTags) {
+ tags[key] = defaultTags[key];
+ }
+ 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);
+ };
+ mode.exit = function() {
+ context.uninstall(behavior);
+ };
+ return mode;
+ }
+
+ // modules/modes/select_note.js
+ function modeSelectNote(context, selectedNoteID) {
+ var mode = {
+ id: "select-note",
+ button: "browse"
+ };
+ var _keybinding = utilKeybinding("select-note");
+ var _noteEditor = uiNoteEditor(context).on("change", function() {
+ context.map().pan([0, 0]);
+ var note = checkSelectedID();
+ if (!note)
+ return;
+ context.ui().sidebar.show(_noteEditor.note(note));
+ });
+ var _behaviors = [
+ behaviorBreathe(context),
+ behaviorHover(context),
+ behaviorSelect(context),
+ behaviorLasso(context),
+ modeDragNode(context).behavior,
+ modeDragNote(context).behavior
+ ];
+ var _newFeature = false;
+ function checkSelectedID() {
+ if (!services.osm)
+ return;
+ var note = services.osm.getNote(selectedNoteID);
+ if (!note) {
+ context.enter(modeBrowse(context));
+ }
+ return note;
+ }
+ function selectNote(d3_event, drawn) {
+ if (!checkSelectedID())
+ return;
+ var selection2 = context.surface().selectAll(".layer-notes .note-" + selectedNoteID);
+ if (selection2.empty()) {
+ 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 {
+ selection2.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);
+ if (note) {
+ context.map().centerZoomEase(note.loc, 20);
+ }
+ };
+ mode.newFeature = function(val) {
+ if (!arguments.length)
+ return _newFeature;
+ _newFeature = val;
+ return mode;
+ };
+ mode.enter = function() {
+ var note = checkSelectedID();
+ if (!note)
+ return;
+ _behaviors.forEach(context.install);
+ _keybinding.on(_t("inspector.zoom_to.key"), mode.zoomToSelected).on("\u238B", esc, true);
+ select_default2(document).call(_keybinding);
+ selectNote();
+ var sidebar = context.ui().sidebar;
+ sidebar.show(_noteEditor.note(note).newNote(_newFeature));
+ sidebar.expand(sidebar.intersects(note.extent()));
+ context.map().on("drawn.select", selectNote);
+ };
+ mode.exit = function() {
+ _behaviors.forEach(context.uninstall);
+ select_default2(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);
+ };
+ return mode;
+ }
+
+ // modules/modes/add_note.js
+ function modeAddNote(context) {
+ var mode = {
+ id: "add-note",
+ button: "note",
+ description: _t.append("modes.add_note.description"),
+ key: _t("modes.add_note.key")
+ };
+ var behavior = behaviorDraw(context).on("click", add).on("cancel", cancel).on("finish", cancel);
+ function add(loc) {
+ var osm = services.osm;
+ if (!osm)
+ return;
+ var note = osmNote({ loc, status: "open", comments: [] });
+ osm.replaceNote(note);
+ context.map().pan([0, 0]);
+ context.selectedNoteID(note.id).enter(modeSelectNote(context, note.id).newFeature(true));
+ }
+ function cancel() {
+ context.enter(modeBrowse(context));
+ }
+ mode.enter = function() {
+ context.install(behavior);
+ };
+ mode.exit = function() {
+ context.uninstall(behavior);
+ };
+ return mode;
+ }
+
+ // modules/modes/save.js
+ function modeSave(context) {
+ var mode = { id: "save" };
+ var keybinding = utilKeybinding("modeSave");
+ var commit = uiCommit(context).on("cancel", cancel);
+ var _conflictsUi;
+ var _location;
+ var _success;
+ var uploader = context.uploader().on("saveStarted.modeSave", function() {
+ keybindingOff();
+ }).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 cancel() {
+ context.enter(modeBrowse(context));
+ }
+ function showProgress(num, total) {
+ var modal = context.container().select(".loading-modal .modal-section");
+ var progress = modal.selectAll(".progress").data([0]);
+ progress.enter().append("div").attr("class", "progress").merge(progress).text(_t("save.conflict_progress", { num, total }));
+ }
+ function showConflicts(changeset, conflicts, origChanges) {
+ var selection2 = 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);
+ selection2.remove();
+ keybindingOn();
+ uploader.cancelConflictResolution();
+ }).on("save", function() {
+ context.container().selectAll(".main-content").classed("active", false).classed("inactive", true);
+ selection2.remove();
+ uploader.processResolvedConflicts(changeset);
+ });
+ selection2.call(_conflictsUi);
+ }
+ function showErrors(errors) {
+ keybindingOn();
+ var selection2 = uiConfirm(context.container());
+ selection2.select(".modal-section.header").append("h3").text(_t("save.error"));
+ addErrors(selection2, errors);
+ selection2.okButton();
+ }
+ function addErrors(selection2, data) {
+ var message = selection2.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_default2(this);
+ var detail = select_default2(this.nextElementSibling);
+ var exp2 = error.classed("expanded");
+ detail.style("display", exp2 ? "none" : "block");
+ error.classed("expanded", !exp2);
+ });
+ 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 showSuccess(changeset) {
+ commit.reset();
+ var ui = _success.changeset(changeset).location(_location).on("cancel", function() {
+ context.ui().sidebar.hide();
+ });
+ context.enter(modeBrowse(context).sidebar(ui));
+ }
+ function keybindingOn() {
+ select_default2(document).call(keybinding.on("\u238B", cancel, true));
+ }
+ function keybindingOff() {
+ select_default2(document).call(keybinding.unbind);
+ }
+ 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, separator, region }
+ );
+ });
+ }
+ mode.selectedIDs = function() {
+ return _conflictsUi ? _conflictsUi.shownEntityIds() : [];
+ };
+ mode.enter = function() {
+ context.ui().sidebar.expand();
+ function done() {
+ context.ui().sidebar.show(commit);
+ }
+ keybindingOn();
+ context.container().selectAll(".main-content").classed("active", false).classed("inactive", true);
+ var osm = context.connection();
+ if (!osm) {
+ cancel();
+ return;
+ }
+ if (osm.authenticated()) {
+ done();
+ } else {
+ osm.authenticate(function(err) {
+ if (err) {
+ cancel();
+ } else {
+ done();
+ }
+ });
+ }
+ };
+ mode.exit = function() {
+ keybindingOff();
+ context.container().selectAll(".main-content").classed("active", true).classed("inactive", false);
+ context.ui().sidebar.hide();
+ };
+ return mode;
+ }
+
+ // modules/modes/select_error.js
+ function modeSelectError(context, selectedErrorID, selectedErrorService) {
+ var mode = {
+ id: "select-error",
+ button: "browse"
+ };
+ var keybinding = utilKeybinding("select-error");
+ var errorService = services[selectedErrorService];
+ var errorEditor;
+ switch (selectedErrorService) {
+ case "improveOSM":
+ errorEditor = uiImproveOsmEditor(context).on("change", function() {
+ context.map().pan([0, 0]);
+ var error = checkSelectedID();
+ if (!error)
+ return;
+ context.ui().sidebar.show(errorEditor.error(error));
+ });
+ break;
+ case "keepRight":
+ errorEditor = uiKeepRightEditor(context).on("change", function() {
+ context.map().pan([0, 0]);
+ var error = checkSelectedID();
+ if (!error)
+ return;
+ context.ui().sidebar.show(errorEditor.error(error));
+ });
+ break;
+ case "osmose":
+ errorEditor = uiOsmoseEditor(context).on("change", function() {
+ context.map().pan([0, 0]);
+ var error = checkSelectedID();
+ if (!error)
+ return;
+ context.ui().sidebar.show(errorEditor.error(error));
+ });
+ break;
+ }
+ var behaviors = [
+ behaviorBreathe(context),
+ behaviorHover(context),
+ behaviorSelect(context),
+ behaviorLasso(context),
+ modeDragNode(context).behavior,
+ modeDragNote(context).behavior
+ ];
+ function checkSelectedID() {
+ if (!errorService)
+ return;
+ var error = errorService.getError(selectedErrorID);
+ if (!error) {
+ context.enter(modeBrowse(context));
+ }
+ return error;
+ }
+ mode.zoomToSelected = function() {
+ if (!errorService)
+ return;
+ var error = errorService.getError(selectedErrorID);
+ if (error) {
+ context.map().centerZoomEase(error.loc, 20);
+ }
+ };
+ mode.enter = function() {
+ var error = checkSelectedID();
+ if (!error)
+ return;
+ behaviors.forEach(context.install);
+ keybinding.on(_t("inspector.zoom_to.key"), mode.zoomToSelected).on("\u238B", esc, true);
+ select_default2(document).call(keybinding);
+ selectError();
+ var sidebar = context.ui().sidebar;
+ sidebar.show(errorEditor.error(error));
+ context.map().on("drawn.select-error", selectError);
+ function selectError(d3_event, drawn) {
+ if (!checkSelectedID())
+ return;
+ var selection2 = context.surface().selectAll(".itemId-" + selectedErrorID + "." + selectedErrorService);
+ if (selection2.empty()) {
+ 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 {
+ selection2.classed("selected", true);
+ context.selectedErrorID(selectedErrorID);
+ }
+ }
+ function esc() {
+ if (context.container().select(".combobox").size())
+ return;
+ context.enter(modeBrowse(context));
+ }
+ };
+ mode.exit = function() {
+ behaviors.forEach(context.uninstall);
+ select_default2(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([]);
+ };
+ return mode;
+ }
+
+ // modules/ui/tools/modes.js
+ function uiToolDrawModes(context) {
+ var tool = {
+ id: "old_modes",
+ label: _t.append("toolbar.add_feature")
+ };
+ var modes = [
+ modeAddPoint(context, {
+ title: _t.append("modes.add_point.title"),
+ button: "point",
+ description: _t.append("modes.add_point.description"),
+ preset: _mainPresetIndex.item("point"),
+ key: "1"
+ }),
+ modeAddLine(context, {
+ title: _t.append("modes.add_line.title"),
+ button: "line",
+ description: _t.append("modes.add_line.description"),
+ preset: _mainPresetIndex.item("line"),
+ key: "2"
+ }),
+ modeAddArea(context, {
+ title: _t.append("modes.add_area.title"),
+ button: "area",
+ description: _t.append("modes.add_area.description"),
+ preset: _mainPresetIndex.item("area"),
+ key: "3"
+ })
+ ];
+ function enabled(_mode) {
+ return osmEditable();
+ }
+ function osmEditable() {
+ return context.editable();
+ }
+ modes.forEach(function(mode) {
+ context.keybinding().on(mode.key, function() {
+ if (!enabled(mode))
+ return;
+ if (mode.id === context.mode().id) {
+ context.enter(modeBrowse(context));
+ } else {
+ context.enter(mode);
+ }
+ });
+ });
+ tool.render = function(selection2) {
+ var wrap2 = selection2.append("div").attr("class", "joined").style("display", "flex");
+ var debouncedUpdate = debounce_default(update, 500, { leading: true, trailing: true });
+ context.map().on("move.modes", debouncedUpdate).on("drawn.modes", debouncedUpdate);
+ context.on("enter.modes", update);
+ update();
+ function update() {
+ var buttons = wrap2.selectAll("button.add-button").data(modes, function(d) {
+ return d.id;
+ });
+ buttons.exit().remove();
+ 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(d))
+ return;
+ 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_default2(this).call(svgIcon("#iD-icon-" + d.button));
+ });
+ buttonsEnter.append("span").attr("class", "label").text("").each(function(mode) {
+ mode.title(select_default2(this));
+ });
+ if (buttons.enter().size() || buttons.exit().size()) {
+ context.ui().checkOverflow(".top-toolbar", true);
+ }
+ buttons = buttons.merge(buttonsEnter).attr("aria-disabled", function(d) {
+ return !enabled(d);
+ }).classed("disabled", function(d) {
+ return !enabled(d);
+ }).attr("aria-pressed", function(d) {
+ return context.mode() && context.mode().button === d.button;
+ }).classed("active", function(d) {
+ return context.mode() && context.mode().button === d.button;
+ });
+ }
+ };
+ return tool;
+ }
+
+ // modules/ui/tools/notes.js
+ function uiToolNotes(context) {
+ var tool = {
+ id: "notes",
+ label: _t.append("modes.add_note.label")
+ };
+ var mode = modeAddNote(context);
+ function enabled() {
+ return notesEnabled() && notesEditable();
+ }
+ function notesEnabled() {
+ var noteLayer = context.layers().layer("notes");
+ return noteLayer && noteLayer.enabled();
+ }
+ function notesEditable() {
+ var mode2 = context.mode();
+ return context.map().notesEditable() && mode2 && mode2.id !== "save";
+ }
+ context.keybinding().on(mode.key, function() {
+ if (!enabled())
+ return;
+ if (mode.id === context.mode().id) {
+ context.enter(modeBrowse(context));
+ } else {
+ context.enter(mode);
+ }
+ });
+ tool.render = function(selection2) {
+ var debouncedUpdate = debounce_default(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 = selection2.selectAll("button.add-button").data(data, function(d) {
+ return d.id;
+ });
+ buttons.exit().remove();
+ 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;
+ 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_default2(this).call(svgIcon(d.icon || "#iD-icon-" + d.button));
+ });
+ if (buttons.enter().size() || buttons.exit().size()) {
+ context.ui().checkOverflow(".top-toolbar", true);
+ }
+ buttons = buttons.merge(buttonsEnter).classed("disabled", function() {
+ return !enabled();
+ }).attr("aria-disabled", function() {
+ return !enabled();
+ }).classed("active", function(d) {
+ return context.mode() && context.mode().button === d.button;
+ }).attr("aria-pressed", function(d) {
+ return context.mode() && context.mode().button === d.button;
+ });
+ }
+ };
+ 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);
+ };
+ return tool;
+ }
+
+ // modules/ui/tools/save.js
+ function uiToolSave(context) {
+ var tool = {
+ id: "save",
+ label: _t.append("save.title")
+ };
+ var button = null;
+ var tooltipBehavior = null;
+ var history = context.history();
+ var key = uiCmd("\u2318S");
+ var _numChanges = 0;
+ function isSaving() {
+ var mode = context.mode();
+ return mode && mode.id === "save";
+ }
+ function isDisabled() {
+ return _numChanges === 0 || isSaving();
+ }
+ function save(d3_event) {
+ d3_event.preventDefault();
+ if (!context.inIntro() && !isSaving() && history.hasChanges()) {
+ context.enter(modeSave(context));
+ }
+ }
+ function bgColor(numChanges) {
+ var step;
+ if (numChanges === 0) {
+ return null;
+ } else if (numChanges <= 50) {
+ step = numChanges / 50;
+ return rgb_default("#fff", "#ff8")(step);
+ } else {
+ step = Math.min((numChanges - 50) / 50, 1);
+ return rgb_default("#ff8", "#f88")(step);
+ }
+ }
+ function updateCount() {
+ var val = history.difference().summary().length;
+ if (val === _numChanges)
+ return;
+ _numChanges = val;
+ if (tooltipBehavior) {
+ tooltipBehavior.title(() => _t.append(_numChanges > 0 ? "save.help" : "save.no_changes")).keys([key]);
+ }
+ if (button) {
+ button.classed("disabled", isDisabled()).style("background", bgColor(_numChanges));
+ button.select("span.count").text(_numChanges);
+ }
+ }
+ tool.render = function(selection2) {
+ tooltipBehavior = uiTooltip().placement("bottom").title(() => _t.append("save.no_changes")).keys([key]).scrollContainer(context.container().select(".top-toolbar"));
+ var lastPointerUpType;
+ button = selection2.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);
+ if (_numChanges === 0 && (lastPointerUpType === "touch" || lastPointerUpType === "pen")) {
+ context.ui().flash.duration(2e3).iconName("#iD-icon-save").iconClass("disabled").label(_t.append("save.no_changes"))();
+ }
+ lastPointerUpType = null;
+ }).call(tooltipBehavior);
+ button.call(svgIcon("#iD-icon-save"));
+ button.append("span").attr("class", "count").attr("aria-hidden", "true").text("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 (isSaving()) {
+ button.call(tooltipBehavior.hide);
+ }
+ }
+ });
+ };
+ tool.uninstall = function() {
+ context.keybinding().off(key, true);
+ context.history().on("change.save", null);
+ context.on("enter.save", null);
+ button = null;
+ tooltipBehavior = null;
+ };
+ return tool;
+ }
+
+ // modules/ui/tools/sidebar_toggle.js
+ function uiToolSidebarToggle(context) {
+ var tool = {
+ id: "sidebar_toggle",
+ label: _t.append("toolbar.inspect")
+ };
+ tool.render = function(selection2) {
+ selection2.append("button").attr("class", "bar-button").attr("aria-label", _t("sidebar.tooltip")).on("click", function() {
+ context.ui().sidebar.toggle();
+ }).call(
+ uiTooltip().placement("bottom").title(() => _t.append("sidebar.tooltip")).keys([_t("sidebar.key")]).scrollContainer(context.container().select(".top-toolbar"))
+ ).call(svgIcon("#iD-icon-sidebar-" + (_mainLocalizer.textDirection() === "rtl" ? "right" : "left")));
+ };
+ return tool;
+ }
+
+ // modules/ui/tools/undo_redo.js
+ function uiToolUndoRedo(context) {
+ var tool = {
+ id: "undo_redo",
+ label: _t.append("toolbar.undo_redo")
+ };
+ var commands = [{
+ id: "undo",
+ cmd: uiCmd("\u2318Z"),
+ action: function() {
+ context.undo();
+ },
+ annotation: function() {
+ return context.history().undoAnnotation();
+ },
+ icon: "iD-icon-" + (_mainLocalizer.textDirection() === "rtl" ? "redo" : "undo")
+ }, {
+ id: "redo",
+ cmd: uiCmd("\u2318\u21E7Z"),
+ action: function() {
+ context.redo();
+ },
+ annotation: function() {
+ return context.history().redoAnnotation();
+ },
+ icon: "iD-icon-" + (_mainLocalizer.textDirection() === "rtl" ? "undo" : "redo")
+ }];
+ function editable() {
+ return context.mode() && context.mode().id !== "save" && context.map().editableDataEnabled(
+ true
+ /* ignore min zoom */
+ );
+ }
+ tool.render = function(selection2) {
+ var tooltipBehavior = uiTooltip().placement("bottom").title(function(d) {
+ return d.annotation() ? _t.append(d.id + ".tooltip", { action: d.annotation() }) : _t.append(d.id + ".nothing");
+ }).keys(function(d) {
+ return [d.cmd];
+ }).scrollContainer(context.container().select(".top-toolbar"));
+ var lastPointerUpType;
+ var buttons = selection2.selectAll("button").data(commands).enter().append("button").attr("class", function(d) {
+ return "disabled " + d.id + "-button bar-button";
+ }).on("pointerup", function(d3_event) {
+ lastPointerUpType = d3_event.pointerType;
+ }).on("click", function(d3_event, d) {
+ d3_event.preventDefault();
+ var annotation = d.annotation();
+ if (editable() && annotation) {
+ d.action();
+ }
+ if (editable() && (lastPointerUpType === "touch" || lastPointerUpType === "pen")) {
+ var label = annotation ? _t.append(d.id + ".tooltip", { action: annotation }) : _t.append(d.id + ".nothing");
+ context.ui().flash.duration(2e3).iconName("#" + d.icon).iconClass(annotation ? "" : "disabled").label(label)();
+ }
+ lastPointerUpType = null;
+ }).call(tooltipBehavior);
+ buttons.each(function(d) {
+ select_default2(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_default(update, 500, { leading: true, trailing: true });
+ 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);
+ function update() {
+ buttons.classed("disabled", function(d) {
+ return !editable() || !d.annotation();
+ }).each(function() {
+ var selection3 = select_default2(this);
+ if (!selection3.select(".tooltip.in").empty()) {
+ selection3.call(tooltipBehavior.updateContent);
+ }
+ });
+ }
+ };
+ 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);
+ };
+ return tool;
+ }
+
+ // modules/ui/top_toolbar.js
+ function uiTopToolbar(context) {
+ var sidebarToggle = uiToolSidebarToggle(context), modes = uiToolDrawModes(context), notes = uiToolNotes(context), undoRedo = uiToolUndoRedo(context), save = uiToolSave(context);
+ function notesEnabled() {
+ var noteLayer = context.layers().layer("notes");
+ return noteLayer && noteLayer.enabled();
+ }
+ function topToolbar(bar) {
+ bar.on("wheel.topToolbar", function(d3_event) {
+ if (!d3_event.deltaX) {
+ bar.node().scrollLeft += d3_event.deltaY;
+ }
+ });
+ var debouncedUpdate = debounce_default(update, 500, { leading: true, trailing: true });
+ context.layers().on("change.topToolbar", debouncedUpdate);
+ update();
+ function update() {
+ var tools = [
+ sidebarToggle,
+ "spacer",
+ modes
+ ];
+ tools.push("spacer");
+ if (notesEnabled()) {
+ tools = tools.concat([notes, "spacer"]);
+ }
+ 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_default2(this).call(d.render, bar);
+ });
+ actionableItems.append("div").attr("class", "item-label").each(function(d) {
+ d.label(select_default2(this));
+ });
+ }
+ }
+ return topToolbar;
+ }
+
+ // modules/ui/zoom_to_selection.js
+ function uiZoomToSelection(context) {
+ function isDisabled() {
+ var mode = context.mode();
+ return !mode || !mode.zoomToSelected;
+ }
+ var _lastPointerUpType;
+ function pointerup(d3_event) {
+ _lastPointerUpType = d3_event.pointerType;
+ }
+ function click(d3_event) {
+ d3_event.preventDefault();
+ if (isDisabled()) {
+ if (_lastPointerUpType === "touch" || _lastPointerUpType === "pen") {
+ context.ui().flash.duration(2e3).iconName("#iD-icon-framed-dot").iconClass("disabled").label(_t.append("inspector.zoom_to.no_selection"))();
+ }
+ } else {
+ var mode = context.mode();
+ if (mode && mode.zoomToSelected) {
+ mode.zoomToSelected();
+ }
+ }
+ _lastPointerUpType = null;
+ }
+ return function(selection2) {
+ var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === "rtl" ? "right" : "left").title(function() {
+ if (isDisabled()) {
+ return _t.append("inspector.zoom_to.no_selection");
+ }
+ return _t.append("inspector.zoom_to.title");
+ }).keys([_t("inspector.zoom_to.key")]);
+ var button = selection2.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);
+ }
+ }
+ context.on("enter.uiZoomToSelection", setEnabledState);
+ setEnabledState();
+ };
+ }
+
+ // modules/ui/pane.js
+ function uiPane(id2, context) {
+ var _key;
+ var _label = "";
+ var _description = "";
+ var _iconName = "";
+ var _sections;
+ var _paneSelection = select_default2(null);
+ var _paneTooltip;
+ var pane = {
+ id: id2
+ };
+ pane.label = function(val) {
+ if (!arguments.length)
+ return _label;
+ _label = val;
+ return pane;
+ };
+ pane.key = function(val) {
+ if (!arguments.length)
+ return _key;
+ _key = val;
+ return pane;
+ };
+ pane.description = function(val) {
+ if (!arguments.length)
+ return _description;
+ _description = val;
+ return pane;
+ };
+ pane.iconName = function(val) {
+ if (!arguments.length)
+ return _iconName;
+ _iconName = val;
+ return pane;
+ };
+ pane.sections = function(val) {
+ if (!arguments.length)
+ return _sections;
+ _sections = val;
+ return pane;
+ };
+ pane.selection = function() {
+ return _paneSelection;
+ };
+ function hidePane() {
+ context.ui().togglePanes();
+ }
+ pane.togglePane = function(d3_event) {
+ if (d3_event)
+ d3_event.preventDefault();
+ _paneTooltip.hide();
+ context.ui().togglePanes(!_paneSelection.classed("shown") ? _paneSelection : void 0);
+ };
+ pane.renderToggleButton = function(selection2) {
+ if (!_paneTooltip) {
+ _paneTooltip = uiTooltip().placement(_mainLocalizer.textDirection() === "rtl" ? "right" : "left").title(() => _description).keys([_key]);
+ }
+ selection2.append("button").on("click", pane.togglePane).call(svgIcon("#" + _iconName, "light")).call(_paneTooltip);
+ };
+ pane.renderContent = function(selection2) {
+ if (_sections) {
+ _sections.forEach(function(section) {
+ selection2.call(section.render);
+ });
+ }
+ };
+ pane.renderPane = function(selection2) {
+ _paneSelection = selection2.append("div").attr("class", "fillL map-pane hide " + id2 + "-pane").attr("pane", id2);
+ var heading = _paneSelection.append("div").attr("class", "pane-heading");
+ heading.append("h2").text("").call(_label);
+ heading.append("button").attr("title", _t("icons.close")).on("click", hidePane).call(svgIcon("#iD-icon-close"));
+ _paneSelection.append("div").attr("class", "pane-content").call(pane.renderContent);
+ if (_key) {
+ context.keybinding().on(_key, pane.togglePane);
+ }
+ };
+ return pane;
+ }
+
+ // modules/ui/sections/background_display_options.js
+ function uiSectionBackgroundDisplayOptions(context) {
+ var section = uiSection("background-display-options", context).label(() => _t.append("background.display_options")).disclosureContent(renderDisclosureContent);
+ var _storedOpacity = corePreferences("background-opacity");
+ var _minVal = 0;
+ var _maxVal = 3;
+ var _sliders = ["brightness", "contrast", "saturation", "sharpness"];
+ var _options = {
+ brightness: _storedOpacity !== null ? +_storedOpacity : 1,
+ contrast: 1,
+ saturation: 1,
+ sharpness: 1
+ };
+ function clamp3(x, min3, max3) {
+ return Math.max(min3, Math.min(x, max3));
+ }
+ function updateValue(d, val) {
+ val = clamp3(val, _minVal, _maxVal);
+ _options[d] = val;
+ context.background()[d](val);
+ if (d === "brightness") {
+ corePreferences("background-opacity", val);
+ }
+ section.reRender();
+ }
+ function renderDisclosureContent(selection2) {
+ var container = selection2.selectAll(".display-options-container").data([0]);
+ var containerEnter = container.enter().append("div").attr("class", "display-options-container controls-list");
+ var slidersEnter = containerEnter.selectAll(".display-control").data(_sliders).enter().append("label").attr("class", function(d) {
+ return "display-control display-control-" + d;
+ });
+ slidersEnter.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_default2(this).property("value");
+ if (!val && d3_event && d3_event.target) {
+ val = d3_event.target.value;
+ }
+ updateValue(d, val);
+ });
+ sildersControlEnter.append("button").attr("title", function(d) {
+ return `${_t("background.reset")} ${_t("background." + d)}`;
+ }).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")));
+ containerEnter.append("a").attr("class", "display-option-resetlink").attr("role", "button").attr("href", "#").call(_t.append("background.reset_all")).on("click", function(d3_event) {
+ d3_event.preventDefault();
+ for (var i2 = 0; i2 < _sliders.length; i2++) {
+ updateValue(_sliders[i2], 1);
+ }
+ });
+ container = containerEnter.merge(container);
+ container.selectAll(".display-option-input").property("value", function(d) {
+ return _options[d];
+ });
+ container.selectAll(".display-option-value").text(function(d) {
+ return Math.floor(_options[d] * 100) + "%";
+ });
+ container.selectAll(".display-option-reset").classed("disabled", function(d) {
+ return _options[d] === 1;
+ });
+ if (containerEnter.size() && _options.brightness !== 1) {
+ context.background().brightness(_options.brightness);
+ }
+ }
+ return section;
+ }
+
+ // modules/ui/settings/custom_background.js
+ function uiSettingsCustomBackground() {
+ var dispatch10 = dispatch_default("change");
+ function render(selection2) {
+ 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(selection2).okButton();
+ modal.classed("settings-modal settings-custom-background", true);
+ modal.select(".modal-section.header").append("h3").call(_t.append("settings.custom_background.header"));
+ var textSection = modal.select(".modal-section.message-text");
+ var instructions = `${_t.html("settings.custom_background.instructions.info")}
+
+#### ${_t.html("settings.custom_background.instructions.wms.tokens_label")}
+* ${_t.html("settings.custom_background.instructions.wms.tokens.proj")}
+* ${_t.html("settings.custom_background.instructions.wms.tokens.wkid")}
+* ${_t.html("settings.custom_background.instructions.wms.tokens.dimensions")}
+* ${_t.html("settings.custom_background.instructions.wms.tokens.bbox")}
+
+#### ${_t.html("settings.custom_background.instructions.tms.tokens_label")}
+* ${_t.html("settings.custom_background.instructions.tms.tokens.xyz")}
+* ${_t.html("settings.custom_background.instructions.tms.tokens.flipped_y")}
+* ${_t.html("settings.custom_background.instructions.tms.tokens.switch")}
+* ${_t.html("settings.custom_background.instructions.tms.tokens.quadtile")}
+* ${_t.html("settings.custom_background.instructions.tms.tokens.scale_factor")}
+
+#### ${_t.html("settings.custom_background.instructions.example")}
+\`${example}\``;
+ textSection.append("div").attr("class", "instructions-template").html(marked(instructions));
+ textSection.append("textarea").attr("class", "field-template").attr("placeholder", _t("settings.custom_background.template.placeholder")).call(utilNoAuto).property("value", _currSettings.template);
+ var buttonSection = modal.select(".modal-section.buttons");
+ buttonSection.insert("button", ".ok-button").attr("class", "button cancel-button secondary-action").call(_t.append("confirm.cancel"));
+ buttonSection.select(".cancel-button").on("click.cancel", clickCancel);
+ buttonSection.select(".ok-button").attr("disabled", isSaveDisabled).on("click.save", clickSave);
+ function isSaveDisabled() {
+ return null;
+ }
+ function clickCancel() {
+ textSection.select(".field-template").property("value", _origSettings.template);
+ corePreferences("background-custom-template", _origSettings.template);
+ this.blur();
+ modal.close();
+ }
+ function clickSave() {
+ _currSettings.template = textSection.select(".field-template").property("value");
+ corePreferences("background-custom-template", _currSettings.template);
+ this.blur();
+ modal.close();
+ dispatch10.call("change", this, _currSettings);
+ }
+ }
+ return utilRebind(render, dispatch10, "on");
+ }
+
+ // modules/ui/sections/background_list.js
+ function uiSectionBackgroundList(context) {
+ var _backgroundList = select_default2(null);
+ var _customSource = context.background().findSource("custom");
+ var _settingsCustomBackground = uiSettingsCustomBackground(context).on("change", customChanged);
+ var section = uiSection("background-list", context).label(() => _t.append("background.backgrounds")).disclosureContent(renderDisclosureContent);
+ function previousBackgroundID() {
+ return corePreferences("background-last-used-toggle");
+ }
+ function renderDisclosureContent(selection2) {
+ var container = selection2.selectAll(".layer-background-list").data([0]);
+ _backgroundList = container.enter().append("ul").attr("class", "layer-list layer-background-list").attr("dir", "auto").merge(container);
+ var bgExtrasListEnter = selection2.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.append("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").call(_t.append("background.minimap.description"));
+ var panelLabelEnter = bgExtrasListEnter.append("li").attr("class", "background-panel-toggle-item").append("label").call(
+ uiTooltip().title(() => _t.append("background.panel.tooltip")).keys([uiCmd("\u2318\u21E7" + _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").call(_t.append("background.panel.description"));
+ var locPanelLabelEnter = bgExtrasListEnter.append("li").attr("class", "location-panel-toggle-item").append("label").call(
+ uiTooltip().title(() => _t.append("background.location_panel.tooltip")).keys([uiCmd("\u2318\u21E7" + _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").call(_t.append("background.location_panel.description"));
+ selection2.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").call(_t.append("background.imagery_problem_faq"));
+ _backgroundList.call(drawListItems, "radio", function(d3_event, d) {
+ chooseBackground(d);
+ }, function(d) {
+ return !d.isHidden() && !d.overlay;
+ });
+ }
+ function setTooltips(selection2) {
+ selection2.each(function(d, i2, nodes) {
+ var item = select_default2(this).select("label");
+ var span = item.select("span");
+ var placement = i2 < nodes.length / 2 ? "bottom" : "top";
+ var hasDescription = d.hasDescription();
+ var isOverflowing = span.property("clientWidth") !== span.property("scrollWidth");
+ item.call(uiTooltip().destroyAny);
+ if (d.id === previousBackgroundID()) {
+ item.call(
+ uiTooltip().placement(placement).title(() => _t.append("background.switch")).keys([uiCmd("\u2318" + _t("background.key"))])
+ );
+ } else if (hasDescription || isOverflowing) {
+ item.call(
+ uiTooltip().placement(placement).title(() => hasDescription ? d.description() : d.label())
+ );
+ }
+ });
+ }
+ function drawListItems(layerList, type2, change, filter2) {
+ var sources = context.background().sources(context.map().extent(), context.map().zoom(), true).filter(filter2).sort(function(a, b) {
+ return a.best() && !b.best() ? -1 : b.best() && !a.best() ? 1 : descending(a.area(), b.area()) || ascending(a.name(), b.name()) || 0;
+ });
+ var layerLinks = layerList.selectAll("li").data(sources, function(d, i2) {
+ return d.id + "---" + i2;
+ });
+ 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", type2).attr("name", "background-layer").attr("value", function(d) {
+ return d.id;
+ }).on("change", change);
+ label.append("span").each(function(d) {
+ d.label()(select_default2(this));
+ });
+ enter.filter(function(d) {
+ return d.id === "custom";
+ }).append("button").attr("class", "layer-browse").call(
+ uiTooltip().title(() => _t.append("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.append("background.best_imagery")).placement(_mainLocalizer.textDirection() === "rtl" ? "right" : "left")
+ ).append("span").text("\u2605");
+ layerList.call(updateLayerSelections);
+ }
+ function updateLayerSelections(selection2) {
+ function active(d) {
+ return context.background().showsLayer(d);
+ }
+ selection2.selectAll("li").classed("active", active).classed("switch", function(d) {
+ return d.id === previousBackgroundID();
+ }).call(setTooltips).selectAll("input").property("checked", active);
+ }
+ function chooseBackground(d) {
+ if (d.id === "custom" && !d.template()) {
+ return editCustom();
+ }
+ var previousBackground = context.background().baseLayerSource();
+ corePreferences("background-last-used-toggle", previousBackground.id);
+ corePreferences("background-last-used", d.id);
+ context.background().baseLayerSource(d);
+ }
+ function customChanged(d) {
+ if (d && d.template) {
+ _customSource.template(d.template);
+ chooseBackground(_customSource);
+ } else {
+ _customSource.template("");
+ chooseBackground(context.background().findSource("none"));
+ }
+ }
+ function editCustom() {
+ context.container().call(_settingsCustomBackground);
+ }
+ context.background().on("change.background_list", function() {
+ _backgroundList.call(updateLayerSelections);
+ });
+ context.map().on(
+ "move.background_list",
+ debounce_default(function() {
+ window.requestIdleCallback(section.reRender);
+ }, 1e3)
+ );
+ return section;
+ }
+
+ // modules/ui/sections/background_offset.js
+ function uiSectionBackgroundOffset(context) {
+ var section = uiSection("background-offset", context).label(() => _t.append("background.fix_misalignment")).disclosureContent(renderDisclosureContent).expandedByDefault(false);
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ var _directions = [
+ ["top", [0, -0.5]],
+ ["left", [-0.5, 0]],
+ ["right", [0.5, 0]],
+ ["bottom", [0, 0.5]]
+ ];
+ 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;
+ });
+ }
+ function resetOffset() {
+ context.background().offset([0, 0]);
+ updateValue();
+ }
+ function nudge(d) {
+ context.background().nudge(d, context.map().zoom());
+ updateValue();
+ }
+ function inputOffset() {
+ var input = select_default2(this);
+ var d = input.node().value;
+ if (d === "")
+ return resetOffset();
+ d = d.replace(/;/g, ",").split(",").map(function(n2) {
+ return !isNaN(n2) && n2;
+ });
+ if (d.length !== 2 || !d[0] || !d[1]) {
+ input.classed("error", true);
+ return;
+ }
+ context.background().offset(geoMetersToOffset(d));
+ updateValue();
+ }
+ 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_default2(window).on(_pointerPrefix + "move.drag-bg-offset", pointermove).on(_pointerPrefix + "up.drag-bg-offset", pointerup);
+ if (_pointerPrefix === "pointer") {
+ select_default2(window).on("pointercancel.drag-bg-offset", pointerup);
+ }
+ function pointermove(d3_event2) {
+ if (pointerId !== (d3_event2.pointerId || "mouse"))
+ return;
+ var latest = [d3_event2.clientX, d3_event2.clientY];
+ var d = [
+ -(origin[0] - latest[0]) / 4,
+ -(origin[1] - latest[1]) / 4
+ ];
+ origin = latest;
+ nudge(d);
+ }
+ function pointerup(d3_event2) {
+ if (pointerId !== (d3_event2.pointerId || "mouse"))
+ return;
+ if (d3_event2.button !== 0)
+ return;
+ context.container().selectAll(".nudge-surface").remove();
+ select_default2(window).on(".drag-bg-offset", null);
+ }
+ }
+ function renderDisclosureContent(selection2) {
+ var container = selection2.selectAll(".nudge-container").data([0]);
+ var containerEnter = container.enter().append("div").attr("class", "nudge-container");
+ containerEnter.append("div").attr("class", "nudge-instructions").call(_t.append("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").attr("aria-label", _t("background.offset_label")).on("change", inputOffset);
+ nudgeWrapEnter.append("div").selectAll("button").data(_directions).enter().append("button").attr("title", function(d) {
+ return _t(`background.nudge.${d[0]}`);
+ }).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();
+ }
+ context.background().on("change.backgroundOffset-update", updateValue);
+ return section;
+ }
+
+ // modules/ui/sections/overlay_list.js
+ function uiSectionOverlayList(context) {
+ var section = uiSection("overlay-list", context).label(() => _t.append("background.overlays")).disclosureContent(renderDisclosureContent);
+ var _overlayList = select_default2(null);
+ function setTooltips(selection2) {
+ selection2.each(function(d, i2, nodes) {
+ var item = select_default2(this).select("label");
+ var span = item.select("span");
+ var placement = i2 < nodes.length / 2 ? "bottom" : "top";
+ var description = d.description();
+ var isOverflowing = span.property("clientWidth") !== span.property("scrollWidth");
+ item.call(uiTooltip().destroyAny);
+ if (description || isOverflowing) {
+ item.call(
+ uiTooltip().placement(placement).title(() => description || d.name())
+ );
+ }
+ });
+ }
+ function updateLayerSelections(selection2) {
+ function active(d) {
+ return context.background().showsLayer(d);
+ }
+ selection2.selectAll("li").classed("active", active).call(setTooltips).selectAll("input").property("checked", active);
+ }
+ function chooseOverlay(d3_event, d) {
+ d3_event.preventDefault();
+ context.background().toggleOverlayLayer(d);
+ _overlayList.call(updateLayerSelections);
+ document.activeElement.blur();
+ }
+ function drawListItems(layerList, type2, change, filter2) {
+ var sources = context.background().sources(context.map().extent(), context.map().zoom(), true).filter(filter2);
+ 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", type2).attr("name", "layers").on("change", change);
+ label.append("span").each(function(d) {
+ d.label()(select_default2(this));
+ });
+ layerList.selectAll("li").sort(sortSources);
+ layerList.call(updateLayerSelections);
+ function sortSources(a, b) {
+ return a.best() && !b.best() ? -1 : b.best() && !a.best() ? 1 : descending(a.area(), b.area()) || ascending(a.name(), b.name()) || 0;
+ }
+ }
+ function renderDisclosureContent(selection2) {
+ var container = selection2.selectAll(".layer-overlay-list").data([0]);
+ _overlayList = container.enter().append("ul").attr("class", "layer-list layer-overlay-list").attr("dir", "auto").merge(container);
+ _overlayList.call(drawListItems, "checkbox", chooseOverlay, function(d) {
+ return !d.isHidden() && d.overlay;
+ });
+ }
+ context.map().on(
+ "move.overlay_list",
+ debounce_default(function() {
+ window.requestIdleCallback(section.reRender);
+ }, 1e3)
+ );
+ return section;
+ }
+
+ // modules/ui/panes/background.js
+ function uiPaneBackground(context) {
+ var backgroundPane = uiPane("background", context).key(_t("background.key")).label(_t.append("background.title")).description(_t.append("background.description")).iconName("iD-icon-layers").sections([
+ uiSectionBackgroundList(context),
+ uiSectionOverlayList(context),
+ uiSectionBackgroundDisplayOptions(context),
+ uiSectionBackgroundOffset(context)
+ ]);
+ return backgroundPane;
+ }
+
+ // modules/ui/panes/help.js
+ 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
+ };
+ var docs = docKeys.map(function(key) {
+ var helpkey = "help." + key[0];
+ var helpPaneReplacements = { version: context.version };
+ var text2 = key[1].reduce(function(all, part) {
+ var subkey = helpkey + "." + part;
+ var depth = headings[subkey];
+ var hhh = depth ? Array(depth + 1).join("#") + " " : "";
+ return all + hhh + helpHtml(subkey, helpPaneReplacements) + "\n\n";
+ }, "");
+ return {
+ title: _t.html(helpkey + ".title"),
+ content: marked(text2.trim()).replace(/<code>/g, "<kbd>").replace(/<\/code>/g, "</kbd>")
+ };
+ });
+ var helpPane = uiPane("help", context).key(_t("help.key")).label(_t.append("help.title")).description(_t.append("help.title")).iconName("iD-icon-help");
+ helpPane.renderContent = function(content) {
+ function clickHelp(d, i2) {
+ 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 (rtl) {
+ nav.call(drawNext).call(drawPrevious);
+ } else {
+ nav.call(drawPrevious).call(drawNext);
+ }
+ function drawNext(selection2) {
+ if (i2 < docs.length - 1) {
+ var nextLink = selection2.append("a").attr("href", "#").attr("class", "next").on("click", function(d3_event) {
+ d3_event.preventDefault();
+ clickHelp(docs[i2 + 1], i2 + 1);
+ });
+ nextLink.append("span").html(docs[i2 + 1].title).call(svgIcon(rtl ? "#iD-icon-backward" : "#iD-icon-forward", "inline"));
+ }
+ }
+ function drawPrevious(selection2) {
+ if (i2 > 0) {
+ var prevLink = selection2.append("a").attr("href", "#").attr("class", "previous").on("click", function(d3_event) {
+ d3_event.preventDefault();
+ clickHelp(docs[i2 - 1], i2 - 1);
+ });
+ prevLink.call(svgIcon(rtl ? "#iD-icon-forward" : "#iD-icon-backward", "inline")).append("span").html(docs[i2 - 1].title);
+ }
+ }
+ }
+ function clickWalkthrough(d3_event) {
+ d3_event.preventDefault();
+ if (context.inIntro())
+ return;
+ context.container().call(uiIntro(context));
+ context.ui().togglePanes();
+ }
+ function clickShortcuts(d3_event) {
+ d3_event.preventDefault();
+ context.container().call(context.ui().shortcuts, true);
+ }
+ var toc = content.append("ul").attr("class", "toc");
+ var menuItems = toc.selectAll("li").data(docs).enter().append("li").append("a").attr("role", "button").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.append("shortcuts.tooltip")).keys(["?"]).placement("top")
+ ).append("a").attr("href", "#").on("click", clickShortcuts);
+ shortcuts.append("div").call(_t.append("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").call(_t.append("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);
+ };
+ return helpPane;
+ }
+
+ // modules/ui/sections/validation_issues.js
+ function uiSectionValidationIssues(id2, severity, context) {
+ var _issues = [];
+ var section = uiSection(id2, context).label(function() {
+ if (!_issues)
+ return "";
+ var issueCountText = _issues.length > 1e3 ? "1000+" : String(_issues.length);
+ return _t.append("inspector.title_count", { title: _t("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"
+ };
+ }
+ function reloadIssues() {
+ _issues = context.validator().getIssuesBySeverity(getOptions())[severity];
+ }
+ function renderDisclosureContent(selection2) {
+ var center = context.map().center();
+ var graph = context.graph();
+ 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 });
+ }).sort(function byDistance(a, b) {
+ return a.dist - b.dist;
+ });
+ issues = issues.slice(0, 1e3);
+ selection2.call(drawIssuesList, issues);
+ }
+ function drawIssuesList(selection2, issues) {
+ var list = selection2.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.key;
+ });
+ items.exit().remove();
+ 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_default2(this).call(svgIcon(iconName));
+ });
+ textEnter.append("span").attr("class", "issue-message");
+ items = items.merge(itemsEnter).order();
+ items.selectAll(".issue-message").text("").each(function(d) {
+ return d.message(context)(select_default2(this));
+ });
+ }
+ context.validator().on("validated.uiSectionValidationIssues" + id2, function() {
+ window.requestIdleCallback(function() {
+ reloadIssues();
+ section.reRender();
+ });
+ });
+ context.map().on(
+ "move.uiSectionValidationIssues" + id2,
+ debounce_default(function() {
+ window.requestIdleCallback(function() {
+ if (getOptions().where === "visible") {
+ reloadIssues();
+ }
+ section.reRender();
+ });
+ }, 1e3)
+ );
+ return section;
+ }
+
+ // modules/ui/sections/validation_options.js
+ function uiSectionValidationOptions(context) {
+ var section = uiSection("issues-options", context).content(renderContent);
+ function renderContent(selection2) {
+ var container = selection2.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 options2 = container.selectAll(".issues-option").data(data, function(d) {
+ return d.key;
+ });
+ var optionsEnter = options2.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 getOptions() {
+ return {
+ what: corePreferences("validate-what") || "edited",
+ // 'all', 'edited'
+ where: corePreferences("validate-where") || "all"
+ // 'all', 'visible'
+ };
+ }
+ function updateOptionValue(d3_event, d, val) {
+ if (!val && d3_event && d3_event.target) {
+ val = d3_event.target.value;
+ }
+ corePreferences("validate-" + d, val);
+ context.validator().validate();
+ }
+ return section;
+ }
+
+ // modules/ui/sections/validation_rules.js
+ function uiSectionValidationRules(context) {
+ var MINSQUARE = 0;
+ var MAXSQUARE = 20;
+ var DEFAULTSQUARE = 5;
+ var section = uiSection("issues-rules", context).disclosureContent(renderDisclosureContent).label(() => _t.append("issues.rules.title"));
+ var _ruleKeys = context.validator().getRuleKeys().filter(function(key) {
+ return key !== "maprules";
+ }).sort(function(key1, key2) {
+ return _t("issues." + key1 + ".title") < _t("issues." + key2 + ".title") ? -1 : 1;
+ });
+ function renderDisclosureContent(selection2) {
+ var container = selection2.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("role", "button").attr("href", "#").call(_t.append("issues.disable_all")).on("click", function(d3_event) {
+ d3_event.preventDefault();
+ context.validator().disableRules(_ruleKeys);
+ });
+ ruleLinks.append("a").attr("class", "issue-rules-link").attr("role", "button").attr("href", "#").call(_t.append("issues.enable_all")).on("click", function(d3_event) {
+ d3_event.preventDefault();
+ context.validator().disableRules([]);
+ });
+ container = container.merge(containerEnter);
+ container.selectAll(".issue-rules-list").call(drawListItems, _ruleKeys, "checkbox", "rule", toggleRule, isRuleEnabled);
+ }
+ function drawListItems(selection2, data, type2, name, change, active) {
+ var items = selection2.selectAll("li").data(data);
+ items.exit().remove();
+ var enter = items.enter().append("li");
+ if (name === "rule") {
+ enter.call(
+ uiTooltip().title(function(d) {
+ return _t.append("issues." + d + ".tip");
+ }).placement("top")
+ );
+ }
+ var label = enter.append("label");
+ label.append("input").attr("type", type2).attr("name", name).on("change", change);
+ label.append("span").html(function(d) {
+ var params = {};
+ if (d === "unsquare_way") {
+ params.val = { html: '<span class="square-degrees"></span>' };
+ }
+ return _t.html("issues." + d + ".title", params);
+ });
+ items = items.merge(enter);
+ items.classed("active", active).selectAll("input").property("checked", active).property("indeterminate", false);
+ var degStr = corePreferences("validate-square-degrees");
+ if (degStr === null) {
+ degStr = DEFAULTSQUARE.toString();
+ }
+ var span = items.selectAll(".square-degrees");
+ var input = span.selectAll(".square-degrees-input").data([0]);
+ 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) {
+ this.blur();
+ this.select();
+ }
+ }).on("blur", changeSquare).merge(input).property("value", degStr);
+ }
+ function changeSquare() {
+ var input = select_default2(this);
+ var degStr = utilGetSetValue(input).trim();
+ var degNum = Number(degStr);
+ if (!isFinite(degNum)) {
+ degNum = DEFAULTSQUARE;
+ } else if (degNum > MAXSQUARE) {
+ degNum = MAXSQUARE;
+ } else if (degNum < MINSQUARE) {
+ degNum = MINSQUARE;
+ }
+ degNum = Math.round(degNum * 10) / 10;
+ degStr = degNum.toString();
+ input.property("value", degStr);
+ corePreferences("validate-square-degrees", degStr);
+ context.validator().revalidateUnsquare();
+ }
+ function isRuleEnabled(d) {
+ return context.validator().isRuleEnabled(d);
+ }
+ function toggleRule(d3_event, d) {
+ context.validator().toggleRule(d);
+ }
+ context.validator().on("validated.uiSectionValidationRules", function() {
+ window.requestIdleCallback(section.reRender);
+ });
+ return section;
+ }
+
+ // modules/ui/sections/validation_status.js
+ function uiSectionValidationStatus(context) {
+ var section = uiSection("issues-status", context).content(renderContent).shouldDisplay(function() {
+ var issues = context.validator().getIssues(getOptions());
+ return issues.length === 0;
+ });
+ function getOptions() {
+ return {
+ what: corePreferences("validate-what") || "edited",
+ where: corePreferences("validate-where") || "all"
+ };
+ }
+ function renderContent(selection2) {
+ var box = selection2.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(selection2);
+ setNoIssuesText(selection2);
+ }
+ function renderIgnoredIssuesReset(selection2) {
+ var ignoredIssues = context.validator().getIssues({ what: "all", where: "all", includeDisabledRules: true, includeIgnored: "only" });
+ var resetIgnored = selection2.selectAll(".reset-ignored").data(ignoredIssues.length ? [0] : []);
+ resetIgnored.exit().remove();
+ var resetIgnoredEnter = resetIgnored.enter().append("div").attr("class", "reset-ignored section-footer");
+ resetIgnoredEnter.append("a").attr("href", "#");
+ resetIgnored = resetIgnored.merge(resetIgnoredEnter);
+ resetIgnored.select("a").html(_t.html("inspector.title_count", { title: { html: _t.html("issues.reset_ignored") }, count: ignoredIssues.length }));
+ resetIgnored.on("click", function(d3_event) {
+ d3_event.preventDefault();
+ context.validator().resetIgnoredIssues();
+ });
+ }
+ function setNoIssuesText(selection2) {
+ var opts = getOptions();
+ function checkForHiddenIssues(cases) {
+ for (var type2 in cases) {
+ var hiddenOpts = cases[type2];
+ var hiddenIssues = context.validator().getIssues(hiddenOpts);
+ if (hiddenIssues.length) {
+ selection2.select(".box .details").html("").call(_t.append(
+ "issues.no_issues.hidden_issues." + type2,
+ { count: hiddenIssues.length.toString() }
+ ));
+ return;
+ }
+ }
+ selection2.select(".box .details").html("").call(_t.append("issues.no_issues.hidden_issues.none"));
+ }
+ var messageType;
+ 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 (opts.what === "edited" && context.history().difference().summary().length === 0) {
+ messageType = "no_edits";
+ }
+ selection2.select(".box .message").html("").call(_t.append("issues.no_issues.message." + messageType));
+ }
+ context.validator().on("validated.uiSectionValidationStatus", function() {
+ window.requestIdleCallback(section.reRender);
+ });
+ context.map().on(
+ "move.uiSectionValidationStatus",
+ debounce_default(function() {
+ window.requestIdleCallback(section.reRender);
+ }, 1e3)
+ );
+ return section;
+ }
+
+ // modules/ui/panes/issues.js
+ function uiPaneIssues(context) {
+ var issuesPane = uiPane("issues", context).key(_t("issues.key")).label(_t.append("issues.title")).description(_t.append("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;
+ }
+
+ // modules/ui/settings/custom_data.js
+ function uiSettingsCustomData(context) {
+ var dispatch10 = dispatch_default("change");
+ function render(selection2) {
+ var dataLayer = context.layers().layer("data");
+ 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 modal = uiConfirm(selection2).okButton();
+ modal.classed("settings-modal settings-custom-data", true);
+ modal.select(".modal-section.header").append("h3").call(_t.append("settings.custom_data.header"));
+ var textSection = modal.select(".modal-section.message-text");
+ textSection.append("pre").attr("class", "instructions-file").call(_t.append("settings.custom_data.file.instructions"));
+ textSection.append("input").attr("class", "field-file").attr("type", "file").attr("accept", ".gpx,.kml,.geojson,.json,application/gpx+xml,application/vnd.google-earth.kml+xml,application/geo+json,application/json").property("files", _currSettings.fileList).on("change", function(d3_event) {
+ var files = d3_event.target.files;
+ if (files && files.length) {
+ _currSettings.url = "";
+ textSection.select(".field-url").property("value", "");
+ _currSettings.fileList = files;
+ } else {
+ _currSettings.fileList = null;
+ }
+ });
+ textSection.append("h4").call(_t.append("settings.custom_data.or"));
+ textSection.append("pre").attr("class", "instructions-url").call(_t.append("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);
+ var buttonSection = modal.select(".modal-section.buttons");
+ buttonSection.insert("button", ".ok-button").attr("class", "button cancel-button secondary-action").call(_t.append("confirm.cancel"));
+ buttonSection.select(".cancel-button").on("click.cancel", clickCancel);
+ buttonSection.select(".ok-button").attr("disabled", isSaveDisabled).on("click.save", clickSave);
+ function isSaveDisabled() {
+ return null;
+ }
+ function clickCancel() {
+ textSection.select(".field-url").property("value", _origSettings.url);
+ corePreferences("settings-custom-data-url", _origSettings.url);
+ this.blur();
+ modal.close();
+ }
+ function clickSave() {
+ _currSettings.url = textSection.select(".field-url").property("value").trim();
+ if (_currSettings.url) {
+ _currSettings.fileList = null;
+ }
+ if (_currSettings.fileList) {
+ _currSettings.url = "";
+ }
+ corePreferences("settings-custom-data-url", _currSettings.url);
+ this.blur();
+ modal.close();
+ dispatch10.call("change", this, _currSettings);
+ }
+ }
+ return utilRebind(render, dispatch10, "on");
+ }
+
+ // modules/ui/sections/data_layers.js
+ function uiSectionDataLayers(context) {
+ var settingsCustomData = uiSettingsCustomData(context).on("change", customChanged);
+ var layers = context.layers();
+ var section = uiSection("data-layers", context).label(() => _t.append("map_data.data_layers")).disclosureContent(renderDisclosureContent);
+ function renderDisclosureContent(selection2) {
+ var container = selection2.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).call(drawPanelItems);
+ }
+ function showsLayer(which) {
+ var layer = layers.layer(which);
+ if (layer) {
+ return layer.enabled();
+ }
+ return false;
+ }
+ function setLayer(which, enabled) {
+ var mode = context.mode();
+ if (mode && /^draw/.test(mode.id))
+ return;
+ var layer = layers.layer(which);
+ if (layer) {
+ layer.enabled(enabled);
+ if (!enabled && (which === "osm" || which === "notes")) {
+ context.enter(modeBrowse(context));
+ }
+ }
+ }
+ function toggleLayer(which) {
+ setLayer(which, !showsLayer(which));
+ }
+ function drawOsmItems(selection2) {
+ var osmKeys = ["osm", "notes"];
+ var osmLayers = layers.all().filter(function(obj) {
+ return osmKeys.indexOf(obj.id) !== -1;
+ });
+ var ul = selection2.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_default2(this).call(
+ uiTooltip().title(() => _t.append("map_data.layers." + d.id + ".tooltip")).keys([uiCmd("\u2325" + _t("area_fill.wireframe.key"))]).placement("bottom")
+ );
+ } else {
+ select_default2(this).call(
+ uiTooltip().title(() => _t.append("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");
+ });
+ li.merge(liEnter).classed("active", function(d) {
+ return d.layer.enabled();
+ }).selectAll("input").property("checked", function(d) {
+ return d.layer.enabled();
+ });
+ }
+ function drawQAItems(selection2) {
+ var qaKeys = ["keepRight", "improveOSM", "osmose"];
+ var qaLayers = layers.all().filter(function(obj) {
+ return qaKeys.indexOf(obj.id) !== -1;
+ });
+ var ul = selection2.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_default2(this).call(
+ uiTooltip().title(() => _t.append("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").each(function(d) {
+ _t.append("map_data.layers." + d.id + ".title")(select_default2(this));
+ });
+ li.merge(liEnter).classed("active", function(d) {
+ return d.layer.enabled();
+ }).selectAll("input").property("checked", function(d) {
+ return d.layer.enabled();
+ });
+ }
+ function drawVectorItems(selection2) {
+ 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"
+ }
+ ];
+ var detroit = geoExtent([-83.5, 42.1], [-82.8, 42.5]);
+ var showVectorItems = context.map().zoom() > 9 && detroit.contains(context.map().center());
+ var container = selection2.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").text("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").text("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_default2(this).call(
+ uiTooltip().title(d.tooltip).placement("top")
+ );
+ });
+ labelEnter.append("input").attr("type", "radio").attr("name", "vectortile").on("change", selectVTLayer);
+ labelEnter.append("span").text(function(d) {
+ return d.name;
+ });
+ li.merge(liEnter).classed("active", isVTLayerSelected).selectAll("input").property("checked", isVTLayerSelected);
+ function isVTLayerSelected(d) {
+ return dataLayer && dataLayer.template() === d.template;
+ }
+ function selectVTLayer(d3_event, d) {
+ corePreferences("settings-custom-data-url", d.template);
+ if (dataLayer) {
+ dataLayer.template(d.template, d.src);
+ dataLayer.enabled(true);
+ }
+ }
+ }
+ function drawCustomDataItems(selection2) {
+ var dataLayer = layers.layer("data");
+ var hasData = dataLayer && dataLayer.hasData();
+ var showsData = hasData && dataLayer.enabled();
+ var ul = selection2.selectAll(".layer-list-data").data(dataLayer ? [0] : []);
+ ul.exit().remove();
+ 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.append("map_data.layers.custom.tooltip")).placement("top")
+ );
+ labelEnter.append("input").attr("type", "checkbox").on("change", function() {
+ toggleLayer("data");
+ });
+ labelEnter.append("span").call(_t.append("map_data.layers.custom.title"));
+ liEnter.append("button").attr("class", "open-data-options").call(
+ uiTooltip().title(() => _t.append("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.append("map_data.layers.custom.zoom")).placement(_mainLocalizer.textDirection() === "rtl" ? "right" : "left")
+ ).on("click", function(d3_event) {
+ if (select_default2(this).classed("disabled"))
+ return;
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ dataLayer.fitZoom();
+ }).call(svgIcon("#iD-icon-framed-dot", "monochrome"));
+ 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);
+ }
+ function editCustom() {
+ context.container().call(settingsCustomData);
+ }
+ function customChanged(d) {
+ var dataLayer = layers.layer("data");
+ if (d && d.url) {
+ dataLayer.url(d.url);
+ } else if (d && d.fileList) {
+ dataLayer.fileList(d.fileList);
+ }
+ }
+ function drawPanelItems(selection2) {
+ var panelsListEnter = selection2.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.append("map_data.history_panel.tooltip")).keys([uiCmd("\u2318\u21E7" + _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").call(_t.append("map_data.history_panel.title"));
+ var measurementPanelLabelEnter = panelsListEnter.append("li").attr("class", "measurement-panel-toggle-item").append("label").call(
+ uiTooltip().title(() => _t.append("map_data.measurement_panel.tooltip")).keys([uiCmd("\u2318\u21E7" + _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").call(_t.append("map_data.measurement_panel.title"));
+ }
+ context.layers().on("change.uiSectionDataLayers", section.reRender);
+ context.map().on(
+ "move.uiSectionDataLayers",
+ debounce_default(function() {
+ window.requestIdleCallback(section.reRender);
+ }, 1e3)
+ );
+ return section;
+ }
+
+ // modules/ui/sections/map_features.js
+ function uiSectionMapFeatures(context) {
+ var _features = context.features().keys();
+ var section = uiSection("map-features", context).label(() => _t.append("map_data.map_features")).disclosureContent(renderDisclosureContent).expandedByDefault(false);
+ function renderDisclosureContent(selection2) {
+ var container = selection2.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("role", "button").attr("href", "#").call(_t.append("issues.disable_all")).on("click", function(d3_event) {
+ d3_event.preventDefault();
+ context.features().disableAll();
+ });
+ footer.append("a").attr("class", "feature-list-link").attr("role", "button").attr("href", "#").call(_t.append("issues.enable_all")).on("click", function(d3_event) {
+ d3_event.preventDefault();
+ context.features().enableAll();
+ });
+ container = container.merge(containerEnter);
+ container.selectAll(".layer-feature-list").call(drawListItems, _features, "checkbox", "feature", clickFeature, showsFeature);
+ }
+ function drawListItems(selection2, data, type2, name, change, active) {
+ var items = selection2.selectAll("li").data(data);
+ items.exit().remove();
+ var enter = items.enter().append("li").call(
+ uiTooltip().title(function(d) {
+ var tip = _t.append(name + "." + d + ".tooltip");
+ if (autoHiddenFeature(d)) {
+ var msg = showsLayer("osm") ? _t.append("map_data.autohidden") : _t.append("map_data.osmhidden");
+ return (selection3) => {
+ selection3.call(tip);
+ selection3.append("div").call(msg);
+ };
+ }
+ return tip;
+ }).placement("top")
+ );
+ var label = enter.append("label");
+ label.append("input").attr("type", type2).attr("name", name).on("change", change);
+ label.append("span").html(function(d) {
+ return _t.html(name + "." + d + ".description");
+ });
+ items = items.merge(enter);
+ items.classed("active", active).selectAll("input").property("checked", active).property("indeterminate", autoHiddenFeature);
+ }
+ function autoHiddenFeature(d) {
+ return context.features().autoHidden(d);
+ }
+ function showsFeature(d) {
+ return context.features().enabled(d);
+ }
+ function clickFeature(d3_event, d) {
+ context.features().toggle(d);
+ }
+ function showsLayer(id2) {
+ var layer = context.layers().layer(id2);
+ return layer && layer.enabled();
+ }
+ context.features().on("change.map_features", section.reRender);
+ return section;
+ }
+
+ // modules/ui/sections/map_style_options.js
+ function uiSectionMapStyleOptions(context) {
+ var section = uiSection("fill-area", context).label(() => _t.append("map_data.style_options")).disclosureContent(renderDisclosureContent).expandedByDefault(false);
+ function renderDisclosureContent(selection2) {
+ var container = selection2.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 = selection2.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");
+ });
+ }
+ function drawListItems(selection2, data, type2, name, change, active) {
+ var items = selection2.selectAll("li").data(data);
+ items.exit().remove();
+ var enter = items.enter().append("li").call(
+ uiTooltip().title(function(d) {
+ return _t.append(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", type2).attr("name", name).on("change", change);
+ label.append("span").html(function(d) {
+ return _t.html(name + "." + d + ".description");
+ });
+ items = items.merge(enter);
+ items.classed("active", active).selectAll("input").property("checked", active).property("indeterminate", false);
+ }
+ function isActiveFill(d) {
+ return context.map().activeAreaFill() === d;
+ }
+ function toggleHighlightEdited(d3_event) {
+ d3_event.preventDefault();
+ context.map().toggleHighlightEdited();
+ }
+ function setFill(d3_event, d) {
+ context.map().activeAreaFill(d);
+ }
+ context.map().on("changeHighlighting.ui_style, changeAreaFill.ui_style", section.reRender);
+ return section;
+ }
+
+ // modules/ui/sections/photo_overlays.js
+ function uiSectionPhotoOverlays(context) {
+ var layers = context.layers();
+ var section = uiSection("photo-overlays", context).label(() => _t.append("photo_overlays.title")).disclosureContent(renderDisclosureContent).expandedByDefault(false);
+ function renderDisclosureContent(selection2) {
+ var container = selection2.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 drawPhotoItems(selection2) {
+ 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();
+ }
+ function layerEnabled(d) {
+ return layerSupported(d) && d.layer.enabled();
+ }
+ var ul = selection2.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";
+ }
+ 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 === "kartaview")
+ titleID = "kartaview_images.tooltip";
+ else
+ titleID = d.id.replace(/-/g, "_") + ".tooltip";
+ select_default2(this).call(
+ uiTooltip().title(() => _t.append(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 id2 = d.id;
+ if (id2 === "mapillary-signs")
+ id2 = "photo_overlays.traffic_signs";
+ return _t.html(id2.replace(/-/g, "_") + ".title");
+ });
+ li.merge(liEnter).classed("active", layerEnabled).selectAll("input").property("checked", layerEnabled);
+ }
+ function drawPhotoTypeItems(selection2) {
+ var data = context.photos().allPhotoTypes();
+ function typeEnabled(d) {
+ return context.photos().showsPhotoType(d);
+ }
+ var ul = selection2.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_default2(this).call(
+ uiTooltip().title(() => _t.append("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");
+ });
+ li.merge(liEnter).classed("active", typeEnabled).selectAll("input").property("checked", typeEnabled);
+ }
+ function drawDateFilter(selection2) {
+ var data = context.photos().dateFilters();
+ function filterEnabled(d) {
+ return context.photos().dateFilterValue(d);
+ }
+ var ul = selection2.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_default2(this).call(
+ uiTooltip().title(() => _t.append("photo_overlays.date_filter." + d + ".tooltip")).placement("top")
+ );
+ });
+ labelEnter.append("span").each(function(d) {
+ _t.append("photo_overlays.date_filter." + d + ".title")(select_default2(this));
+ });
+ 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_default2(this), context.photos().dateFilterValue(d) || "");
+ }).on("change", function(d3_event, d) {
+ var value = utilGetSetValue(select_default2(this)).trim();
+ context.photos().setDateFilter(d, value, true);
+ li.selectAll("input").each(function(d2) {
+ utilGetSetValue(select_default2(this), context.photos().dateFilterValue(d2) || "");
+ });
+ });
+ li = li.merge(liEnter).classed("active", filterEnabled);
+ }
+ function drawUsernameFilter(selection2) {
+ function filterEnabled() {
+ return context.photos().usernames();
+ }
+ var ul = selection2.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_default2(this).call(
+ uiTooltip().title(() => _t.append("photo_overlays.username_filter.tooltip")).placement("top")
+ );
+ });
+ labelEnter.append("span").call(_t.append("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_default2(this).property("value");
+ context.photos().setUsernameFilter(value, true);
+ select_default2(this).property("value", usernameValue);
+ });
+ li.merge(liEnter).classed("active", filterEnabled);
+ function usernameValue() {
+ var usernames = context.photos().usernames();
+ if (usernames)
+ return usernames.join("; ");
+ return usernames;
+ }
+ }
+ function toggleLayer(which) {
+ setLayer(which, !showsLayer(which));
+ }
+ function showsLayer(which) {
+ var layer = layers.layer(which);
+ if (layer) {
+ return layer.enabled();
+ }
+ return false;
+ }
+ function setLayer(which, enabled) {
+ var layer = layers.layer(which);
+ if (layer) {
+ layer.enabled(enabled);
+ }
+ }
+ context.layers().on("change.uiSectionPhotoOverlays", section.reRender);
+ context.photos().on("change.uiSectionPhotoOverlays", section.reRender);
+ return section;
+ }
+
+ // modules/ui/panes/map_data.js
+ function uiPaneMapData(context) {
+ var mapDataPane = uiPane("map-data", context).key(_t("map_data.key")).label(_t.append("map_data.title")).description(_t.append("map_data.description")).iconName("iD-icon-data").sections([
+ uiSectionDataLayers(context),
+ uiSectionPhotoOverlays(context),
+ uiSectionMapStyleOptions(context),
+ uiSectionMapFeatures(context)
+ ]);
+ return mapDataPane;
+ }
+
+ // modules/ui/panes/preferences.js
+ function uiPanePreferences(context) {
+ let preferencesPane = uiPane("preferences", context).key(_t("preferences.key")).label(_t.append("preferences.title")).description(_t.append("preferences.description")).iconName("fas-user-cog").sections([
+ uiSectionPrivacy(context)
+ ]);
+ return preferencesPane;
+ }
+
+ // modules/ui/init.js
+ function uiInit(context) {
+ var _initCounter = 0;
+ var _needWidth = {};
+ var _lastPointerType;
+ function render(container) {
+ container.on("click.ui", function(d3_event) {
+ if (d3_event.button !== 0)
+ return;
+ if (!d3_event.composedPath)
+ return;
+ var isOkayTarget = d3_event.composedPath().some(function(node) {
+ 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;
+ d3_event.preventDefault();
+ });
+ var detected = utilDetect();
+ 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) {
+ container.on("gesturestart.ui gesturechange.ui gestureend.ui", function(d3_event) {
+ d3_event.preventDefault();
+ });
+ }
+ if ("PointerEvent" in window) {
+ select_default2(window).on("pointerdown.ui pointerup.ui", function(d3_event) {
+ var pointerType = d3_event.pointerType || "mouse";
+ 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());
+ container.call(uiFullScreen(context));
+ var map2 = context.map();
+ map2.redrawEnable(false);
+ map2.on("hitMinZoom.ui", function() {
+ ui.flash.iconName("#iD-icon-no").label(_t.append("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");
+ 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(map2);
+ var overMap = content.append("div").attr("class", "over-map");
+ 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));
+ var controlsWrap = overMap.append("div").attr("class", "map-controls-wrap");
+ var controls = controlsWrap.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));
+ controlsWrap.on("wheel.mapControls", function(d3_event) {
+ if (!d3_event.deltaX) {
+ controlsWrap.node().scrollTop += d3_event.deltaY;
+ }
+ });
+ 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).classed("hide", true).call(ui.photoviewer);
+ overMap.append("div").attr("class", "attribution-wrap").attr("dir", "ltr").call(uiAttribution(context));
+ 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.connection().apiConnections();
+ if (apiConnections && apiConnections.length > 1) {
+ aboutList.append("li").attr("class", "source-switch").call(
+ uiSourceSwitch(context).keys(apiConnections)
+ );
+ }
+ 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").attr("aria-label", _t("report_a_bug")).call(svgIcon("#iD-icon-bug", "light")).call(uiTooltip().title(() => _t.append("report_a_bug")).placement("top"));
+ issueLinks.append("a").attr("target", "_blank").attr("href", "https://github.com/openstreetmap/iD/blob/develop/CONTRIBUTING.md#translating").attr("aria-label", _t("help_translate")).call(svgIcon("#iD-icon-translate", "light")).call(uiTooltip().title(() => _t.append("help_translate")).placement("top"));
+ aboutList.append("li").attr("class", "version").call(uiVersion(context));
+ if (!context.embed()) {
+ aboutList.call(uiAccount(context));
+ }
+ ui.onResize();
+ map2.redrawEnable(true);
+ ui.hash = behaviorHash(context);
+ ui.hash();
+ if (!ui.hash.hadLocation) {
+ map2.centerZoom([0, 0], 2);
+ }
+ window.onbeforeunload = function() {
+ return context.save();
+ };
+ window.onunload = function() {
+ context.history().unlock();
+ };
+ select_default2(window).on("resize.editor", function() {
+ ui.onResize();
+ });
+ var panPixels = 80;
+ context.keybinding().on("\u232B", function(d3_event) {
+ d3_event.preventDefault();
+ }).on([_t("sidebar.key"), "`", "\xB2", "@"], ui.sidebar.toggle).on("\u2190", pan([panPixels, 0])).on("\u2191", pan([0, panPixels])).on("\u2192", pan([-panPixels, 0])).on("\u2193", pan([0, -panPixels])).on(uiCmd("\u2325\u2190"), pan([map2.dimensions()[0], 0])).on(uiCmd("\u2325\u2191"), pan([0, map2.dimensions()[1]])).on(uiCmd("\u2325\u2192"), pan([-map2.dimensions()[0], 0])).on(uiCmd("\u2325\u2193"), pan([0, -map2.dimensions()[1]])).on(uiCmd("\u2318" + _t("background.key")), function quickSwitch(d3_event) {
+ if (d3_event) {
+ d3_event.stopImmediatePropagation();
+ d3_event.preventDefault();
+ }
+ var previousBackground = context.background().findSource(corePreferences("background-last-used-toggle"));
+ 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("\u2325" + _t("area_fill.wireframe.key")), function toggleOsmData(d3_event) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var mode = context.mode();
+ if (mode && /^draw/.test(mode.id))
+ return;
+ var layer = context.layers().layer("osm");
+ if (layer) {
+ layer.enabled(!layer.enabled());
+ 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);
+ }
+ var osm = context.connection();
+ var auth = uiLoading(context).message(_t.html("loading_auth")).blocking(true);
+ if (osm && auth) {
+ osm.on("authLoading.ui", function() {
+ context.container().call(auth);
+ }).on("authDone.ui", function() {
+ auth.close();
+ });
+ }
+ _initCounter++;
+ if (ui.hash.startWalkthrough) {
+ ui.hash.startWalkthrough = false;
+ context.container().call(uiIntro(context));
+ }
+ 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);
+ };
+ }
+ }
+ let ui = {};
+ let _loadPromise;
+ ui.ensureLoaded = () => {
+ if (_loadPromise)
+ return _loadPromise;
+ return _loadPromise = Promise.all([
+ // must have strings and presets before loading the UI
+ _mainLocalizer.ensureLoaded(),
+ _mainPresetIndex.ensureLoaded()
+ ]).then(() => {
+ if (!context.container().empty())
+ render(context.container());
+ }).catch((err) => console.error(err));
+ };
+ ui.restart = function() {
+ context.keybinding().clear();
+ _loadPromise = null;
+ context.container().selectAll("*").remove();
+ ui.ensureLoaded();
+ };
+ ui.lastPointerType = function() {
+ return _lastPointerType;
+ };
+ ui.svgDefs = svgDefs(context);
+ ui.flash = uiFlash(context);
+ ui.sidebar = uiSidebar(context);
+ ui.photoviewer = uiPhotoviewer(context);
+ ui.shortcuts = uiShortcuts(context);
+ ui.onResize = function(withPan) {
+ var map2 = context.map();
+ var mapDimensions = utilGetDimensions(context.container().select(".main-content"), true);
+ utilGetDimensions(context.container().select(".sidebar"), true);
+ if (withPan !== void 0) {
+ map2.redrawEnable(false);
+ map2.pan(withPan);
+ map2.redrawEnable(true);
+ }
+ map2.dimensions(mapDimensions);
+ ui.photoviewer.onMapResize();
+ ui.checkOverflow(".top-toolbar");
+ ui.checkOverflow(".map-footer-bar");
+ var resizeWindowEvent = document.createEvent("Event");
+ resizeWindowEvent.initEvent("resizeWindow", true, true);
+ document.dispatchEvent(resizeWindowEvent);
+ };
+ ui.checkOverflow = function(selector, reset) {
+ if (reset) {
+ delete _needWidth[selector];
+ }
+ var selection2 = context.container().select(selector);
+ if (selection2.empty())
+ return;
+ var scrollWidth = selection2.property("scrollWidth");
+ var clientWidth = selection2.property("clientWidth");
+ var needed = _needWidth[selector] || scrollWidth;
+ if (scrollWidth > clientWidth) {
+ selection2.classed("narrow", true);
+ if (!_needWidth[selector]) {
+ _needWidth[selector] = scrollWidth;
+ }
+ } else if (scrollWidth >= needed) {
+ selection2.classed("narrow", false);
+ }
+ };
+ 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);
+ 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);
+ 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_default2(this).classed("shown", false).classed("hide", true);
+ });
+ }
+ };
+ var _editMenu = uiEditMenu(context);
+ ui.editMenu = function() {
+ return _editMenu;
+ };
+ ui.showEditMenu = function(anchorPoint, triggerType, operations) {
+ ui.closeEditMenu();
+ if (!operations && context.mode().operations)
+ operations = context.mode().operations();
+ if (!operations || !operations.length)
+ return;
+ if (!context.map().editableDataEnabled())
+ return;
+ var surfaceNode = context.surface().node();
+ if (surfaceNode.focus) {
+ surfaceNode.focus();
+ }
+ operations.forEach(function(operation) {
+ if (operation.point)
+ operation.point(anchorPoint);
+ });
+ _editMenu.anchorLoc(anchorPoint).triggerType(triggerType).operations(operations);
+ context.map().supersurface.call(_editMenu);
+ };
+ ui.closeEditMenu = function() {
+ context.map().supersurface.select(".edit-menu").remove();
+ };
+ var _saveLoading = select_default2(null);
+ context.uploader().on("saveStarted.ui", function() {
+ _saveLoading = uiLoading(context).message(_t.html("save.uploading")).blocking(true);
+ context.container().call(_saveLoading);
+ }).on("saveEnded.ui", function() {
+ _saveLoading.close();
+ _saveLoading = select_default2(null);
+ });
+ return ui;
+ }
+
+ // modules/core/context.js
+ function coreContext() {
+ const dispatch10 = dispatch_default("enter", "exit", "change");
+ let context = utilRebind({}, dispatch10, "on");
+ let _deferred2 = /* @__PURE__ */ new Set();
+ context.version = package_default.version;
+ context.privacyVersion = "20201202";
+ context.initialHashParams = window.location.hash ? utilStringQs(window.location.hash) : {};
+ context.changeset = null;
+ let _defaultChangesetComment = context.initialHashParams.comment;
+ let _defaultChangesetSource = context.initialHashParams.source;
+ let _defaultChangesetHashtags = context.initialHashParams.hashtags;
+ context.defaultChangesetComment = function(val) {
+ if (!arguments.length)
+ return _defaultChangesetComment;
+ _defaultChangesetComment = val;
+ return context;
+ };
+ context.defaultChangesetSource = function(val) {
+ if (!arguments.length)
+ return _defaultChangesetSource;
+ _defaultChangesetSource = val;
+ return context;
+ };
+ context.defaultChangesetHashtags = function(val) {
+ if (!arguments.length)
+ return _defaultChangesetHashtags;
+ _defaultChangesetHashtags = val;
+ return context;
+ };
+ let _setsDocumentTitle = true;
+ context.setsDocumentTitle = function(val) {
+ if (!arguments.length)
+ return _setsDocumentTitle;
+ _setsDocumentTitle = val;
+ return context;
+ };
+ let _documentTitleBase = document.title;
+ context.documentTitleBase = function(val) {
+ if (!arguments.length)
+ return _documentTitleBase;
+ _documentTitleBase = val;
+ return context;
+ };
+ let _ui;
+ context.ui = () => _ui;
+ context.lastPointerType = () => _ui.lastPointerType();
+ let _keybinding = utilKeybinding("context");
+ context.keybinding = () => _keybinding;
+ select_default2(document).call(_keybinding);
+ let _connection = services.osm;
+ let _history;
+ let _validator;
+ let _uploader;
+ context.connection = () => _connection;
+ context.history = () => _history;
+ context.validator = () => _validator;
+ context.uploader = () => _uploader;
+ context.preauth = (options2) => {
+ if (_connection) {
+ _connection.switch(options2);
+ }
+ return context;
+ };
+ context.locale = function(locale2) {
+ if (!arguments.length)
+ return _mainLocalizer.localeCode();
+ _mainLocalizer.preferredLocaleCodes(locale2);
+ return context;
+ };
+ function afterLoad(cid, callback) {
+ return (err, result) => {
+ if (err) {
+ if (err.status === 400 || err.status === 401 || err.status === 403) {
+ if (_connection) {
+ _connection.logout();
+ }
+ }
+ if (typeof callback === "function") {
+ callback(err);
+ }
+ return;
+ } else if (_connection && _connection.getConnectionId() !== cid) {
+ if (typeof callback === "function") {
+ callback({ message: "Connection Switched", status: -1 });
+ }
+ return;
+ } else {
+ _history.merge(result.data, result.extent);
+ if (typeof callback === "function") {
+ callback(err, result);
+ }
+ return;
+ }
+ };
+ }
+ context.loadTiles = (projection2, callback) => {
+ const handle = window.requestIdleCallback(() => {
+ _deferred2.delete(handle);
+ if (_connection && context.editableDataEnabled()) {
+ const cid = _connection.getConnectionId();
+ _connection.loadTiles(projection2, afterLoad(cid, callback));
+ }
+ });
+ _deferred2.add(handle);
+ };
+ context.loadTileAtLoc = (loc, callback) => {
+ const handle = window.requestIdleCallback(() => {
+ _deferred2.delete(handle);
+ if (_connection && context.editableDataEnabled()) {
+ const cid = _connection.getConnectionId();
+ _connection.loadTileAtLoc(loc, afterLoad(cid, callback));
+ }
+ });
+ _deferred2.add(handle);
+ };
+ context.loadEntity = (entityID, callback) => {
+ if (_connection) {
+ const cid = _connection.getConnectionId();
+ _connection.loadEntity(entityID, afterLoad(cid, callback));
+ _connection.loadEntityRelations(entityID, afterLoad(cid, callback));
+ }
+ };
+ context.zoomToEntity = (entityID, zoomTo) => {
+ context.loadEntity(entityID, (err, result) => {
+ if (err)
+ return;
+ if (zoomTo !== false) {
+ const entity = result.data.find((e) => e.id === entityID);
+ if (entity) {
+ _map.zoomTo(entity);
+ }
+ }
+ });
+ _map.on("drawn.zoomToEntity", () => {
+ if (!context.hasEntity(entityID))
+ return;
+ _map.on("drawn.zoomToEntity", null);
+ context.on("enter.zoomToEntity", null);
+ context.enter(modeSelect(context, [entityID]));
+ });
+ context.on("enter.zoomToEntity", () => {
+ if (_mode.id !== "browse") {
+ _map.on("drawn.zoomToEntity", null);
+ context.on("enter.zoomToEntity", null);
+ }
+ });
+ };
+ let _minEditableZoom = 16;
+ context.minEditableZoom = function(val) {
+ if (!arguments.length)
+ return _minEditableZoom;
+ _minEditableZoom = val;
+ if (_connection) {
+ _connection.tileZoom(val);
+ }
+ return context;
+ };
+ context.maxCharsForTagKey = () => 255;
+ context.maxCharsForTagValue = () => 255;
+ context.maxCharsForRelationRole = () => 255;
+ context.cleanTagKey = (val) => utilCleanOsmString(val, context.maxCharsForTagKey());
+ context.cleanTagValue = (val) => utilCleanOsmString(val, context.maxCharsForTagValue());
+ context.cleanRelationRole = (val) => utilCleanOsmString(val, context.maxCharsForRelationRole());
+ let _inIntro = false;
+ context.inIntro = function(val) {
+ if (!arguments.length)
+ return _inIntro;
+ _inIntro = val;
+ return context;
+ };
+ context.save = () => {
+ if (_inIntro || context.container().select(".modal").size())
+ return;
+ let canSave;
+ if (_mode && _mode.id === "save") {
+ canSave = false;
+ if (services.osm && services.osm.isChangesetInflight()) {
+ _history.clearSaved();
+ return;
+ }
+ } else {
+ canSave = context.selectedIDs().every((id2) => {
+ const entity = context.hasEntity(id2);
+ return entity && !entity.isDegenerate();
+ });
+ }
+ if (canSave) {
+ _history.save();
+ }
+ if (_history.hasChanges()) {
+ return _t("save.unsaved_changes");
+ }
+ };
+ context.debouncedSave = debounce_default(context.save, 350);
+ function withDebouncedSave(fn) {
+ return function() {
+ const result = fn.apply(_history, arguments);
+ context.debouncedSave();
+ return result;
+ };
+ }
+ context.hasEntity = (id2) => _history.graph().hasEntity(id2);
+ context.entity = (id2) => _history.graph().entity(id2);
+ let _mode;
+ context.mode = () => _mode;
+ context.enter = (newMode) => {
+ if (_mode) {
+ _mode.exit();
+ dispatch10.call("exit", this, _mode);
+ }
+ _mode = newMode;
+ _mode.enter();
+ dispatch10.call("enter", this, _mode);
+ };
+ context.selectedIDs = () => _mode && _mode.selectedIDs && _mode.selectedIDs() || [];
+ context.activeID = () => _mode && _mode.activeID && _mode.activeID();
+ let _selectedNoteID;
+ context.selectedNoteID = function(noteID) {
+ if (!arguments.length)
+ return _selectedNoteID;
+ _selectedNoteID = noteID;
+ return context;
+ };
+ let _selectedErrorID;
+ context.selectedErrorID = function(errorID) {
+ if (!arguments.length)
+ return _selectedErrorID;
+ _selectedErrorID = errorID;
+ return context;
+ };
+ context.install = (behavior) => context.surface().call(behavior);
+ context.uninstall = (behavior) => context.surface().call(behavior.off);
+ let _copyGraph;
+ context.copyGraph = () => _copyGraph;
+ let _copyIDs = [];
+ context.copyIDs = function(val) {
+ if (!arguments.length)
+ return _copyIDs;
+ _copyIDs = val;
+ _copyGraph = _history.graph();
+ return context;
+ };
+ let _copyLonLat;
+ context.copyLonLat = function(val) {
+ if (!arguments.length)
+ return _copyLonLat;
+ _copyLonLat = val;
+ return context;
+ };
+ let _background;
+ context.background = () => _background;
+ let _features;
+ context.features = () => _features;
+ context.hasHiddenConnections = (id2) => {
+ const graph = _history.graph();
+ const entity = graph.entity(id2);
+ return _features.hasHiddenConnections(entity, graph);
+ };
+ let _photos;
+ context.photos = () => _photos;
+ let _map;
+ context.map = () => _map;
+ context.layers = () => _map.layers();
+ context.surface = () => _map.surface;
+ context.editableDataEnabled = () => _map.editableDataEnabled();
+ context.surfaceRect = () => _map.surface.node().getBoundingClientRect();
+ context.editable = () => {
+ const mode = context.mode();
+ if (!mode || mode.id === "save")
+ return false;
+ return _map.editableDataEnabled();
+ };
+ let _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
+ };
+ context.debugFlags = () => _debugFlags;
+ context.getDebug = (flag) => flag && _debugFlags[flag];
+ context.setDebug = function(flag, val) {
+ if (arguments.length === 1)
+ val = true;
+ _debugFlags[flag] = val;
+ dispatch10.call("change");
+ return context;
+ };
+ let _container = select_default2(null);
+ context.container = function(val) {
+ if (!arguments.length)
+ return _container;
+ _container = val;
+ _container.classed("ideditor", true);
+ return context;
+ };
+ context.containerNode = function(val) {
+ if (!arguments.length)
+ return context.container().node();
+ context.container(select_default2(val));
+ return context;
+ };
+ let _embed;
+ context.embed = function(val) {
+ if (!arguments.length)
+ return _embed;
+ _embed = val;
+ return context;
+ };
+ let _assetPath = "";
+ context.assetPath = function(val) {
+ if (!arguments.length)
+ return _assetPath;
+ _assetPath = val;
+ _mainFileFetcher.assetPath(val);
+ return context;
+ };
+ let _assetMap = {};
+ context.assetMap = function(val) {
+ if (!arguments.length)
+ return _assetMap;
+ _assetMap = val;
+ _mainFileFetcher.assetMap(val);
+ return context;
+ };
+ context.asset = (val) => {
+ if (/^http(s)?:\/\//i.test(val))
+ return val;
+ const filename = _assetPath + val;
+ return _assetMap[filename] || filename;
+ };
+ context.imagePath = (val) => context.asset(`img/${val}`);
+ context.reset = context.flush = () => {
+ context.debouncedSave.cancel();
+ Array.from(_deferred2).forEach((handle) => {
+ window.cancelIdleCallback(handle);
+ _deferred2.delete(handle);
+ });
+ Object.values(services).forEach((service) => {
+ if (service && typeof service.reset === "function") {
+ service.reset(context);
+ }
+ });
+ context.changeset = null;
+ _validator.reset();
+ _features.reset();
+ _history.reset();
+ _uploader.reset();
+ context.container().select(".inspector-wrap *").remove();
+ return context;
+ };
+ context.projection = geoRawMercator();
+ context.curtainProjection = geoRawMercator();
+ context.init = () => {
+ instantiateInternal();
+ initializeDependents();
+ return context;
+ 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);
+ }
+ function initializeDependents() {
+ if (context.initialHashParams.presets) {
+ _mainPresetIndex.addablePresetIDs(new Set(context.initialHashParams.presets.split(",")));
+ }
+ if (context.initialHashParams.locale) {
+ _mainLocalizer.preferredLocaleCodes(context.initialHashParams.locale);
+ }
+ _mainLocalizer.ensureLoaded();
+ _mainPresetIndex.ensureLoaded();
+ _background.ensureLoaded();
+ Object.values(services).forEach((service) => {
+ if (service && typeof service.init === "function") {
+ service.init();
+ }
+ });
+ _map.init();
+ _validator.init();
+ _features.init();
+ if (services.maprules && context.initialHashParams.maprules) {
+ json_default(context.initialHashParams.maprules).then((mapcss) => {
+ services.maprules.init();
+ mapcss.forEach((mapcssSelector) => services.maprules.addRule(mapcssSelector));
+ }).catch(() => {
+ });
+ }
+ if (!context.container().empty()) {
+ _ui.ensureLoaded().then(() => {
+ _background.init();
+ _photos.init();
+ });
+ }
+ }
+ };
+ return context;
+ }
+
+ // modules/services/nsi.js
+ var _nsiStatus = "loading";
+ var _nsi = {};
+ var buildingPreset = {
+ "building/commercial": true,
+ "building/government": true,
+ "building/hotel": true,
+ "building/retail": true,
+ "building/office": true,
+ "building/supermarket": true,
+ "building/yes": true
+ };
+ var notNames = /:(colou?r|type|forward|backward|left|right|etymology|pronunciation|wikipedia)$/i;
+ var notBranches = /(coop|express|wireless|factory|outlet)/i;
+ function setNsiSources() {
+ const nsiVersion = package_default.dependencies["name-suggestion-index"] || package_default.devDependencies["name-suggestion-index"];
+ const v = (0, import_vparse2.default)(nsiVersion);
+ const vMinor = `${v.major}.${v.minor}`;
+ const cdn = nsiCdnUrl.replace("{version}", vMinor);
+ const sources = {
+ "nsi_data": cdn + "dist/nsi.min.json",
+ "nsi_dissolved": cdn + "dist/dissolved.min.json",
+ "nsi_features": cdn + "dist/featureCollection.min.json",
+ "nsi_generics": cdn + "dist/genericWords.min.json",
+ "nsi_presets": cdn + "dist/presets/nsi-id-presets.min.json",
+ "nsi_replacements": cdn + "dist/replacements.min.json",
+ "nsi_trees": cdn + "dist/trees.min.json"
+ };
+ let fileMap = _mainFileFetcher.fileMap();
+ for (const k in sources) {
+ if (!fileMap[k])
+ fileMap[k] = sources[k];
+ }
+ }
+ function loadNsiPresets() {
+ return Promise.all([
+ _mainFileFetcher.get("nsi_presets"),
+ _mainFileFetcher.get("nsi_features")
+ ]).then((vals) => {
+ Object.values(vals[0].presets).forEach((preset) => preset.suggestion = true);
+ _mainPresetIndex.merge({
+ presets: vals[0].presets,
+ featureCollection: vals[1]
+ });
+ });
+ }
+ function loadNsiData() {
+ return Promise.all([
+ _mainFileFetcher.get("nsi_data"),
+ _mainFileFetcher.get("nsi_dissolved"),
+ _mainFileFetcher.get("nsi_replacements"),
+ _mainFileFetcher.get("nsi_trees")
+ ]).then((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: /* @__PURE__ */ new Map(),
+ // Map (k -> Map (v -> t) )
+ qids: /* @__PURE__ */ new Map(),
+ // Map (wd/wp tag values -> qids)
+ ids: /* @__PURE__ */ new Map()
+ // Map (id -> NSI item)
+ };
+ const matcher = _nsi.matcher = new Matcher();
+ matcher.buildMatchIndex(_nsi.data);
+ matcher.itemLocation = /* @__PURE__ */ new Map();
+ matcher.locationSets = /* @__PURE__ */ new Map();
+ Object.keys(_nsi.data).forEach((tkv) => {
+ const items = _nsi.data[tkv].items;
+ if (!Array.isArray(items) || !items.length)
+ return;
+ items.forEach((item) => {
+ if (matcher.itemLocation.has(item.id))
+ return;
+ const locationSetID = _sharedLocationManager.locationSetID(item.locationSet);
+ matcher.itemLocation.set(item.id, locationSetID);
+ if (matcher.locationSets.has(locationSetID))
+ return;
+ const fakeFeature = { id: locationSetID, properties: { id: locationSetID, area: 1 } };
+ matcher.locationSets.set(locationSetID, fakeFeature);
+ });
+ });
+ matcher.locationIndex = (bbox2) => {
+ const validHere = _sharedLocationManager.locationSetsAt([bbox2[0], bbox2[1]]);
+ const results = [];
+ for (const [locationSetID, area] of Object.entries(validHere)) {
+ const fakeFeature = matcher.locationSets.get(locationSetID);
+ if (fakeFeature) {
+ fakeFeature.properties.area = area;
+ results.push(fakeFeature);
+ }
+ }
+ return results;
+ };
+ Object.keys(_nsi.data).forEach((tkv) => {
+ const category = _nsi.data[tkv];
+ const parts = tkv.split("/", 3);
+ const t = parts[0];
+ const k = parts[1];
+ const v = parts[2];
+ let vmap = _nsi.kvt.get(k);
+ if (!vmap) {
+ vmap = /* @__PURE__ */ new Map();
+ _nsi.kvt.set(k, vmap);
+ }
+ vmap.set(v, t);
+ const tree = _nsi.trees[t];
+ const mainTag = tree.mainTag;
+ const items = category.items || [];
+ items.forEach((item) => {
+ item.tkv = tkv;
+ item.mainTag = mainTag;
+ _nsi.ids.set(item.id, item);
+ const wd = item.tags[mainTag];
+ const wp = item.tags[mainTag.replace("wikidata", "wikipedia")];
+ if (wd)
+ _nsi.qids.set(wd, wd);
+ if (wp && wd)
+ _nsi.qids.set(wp, wd);
+ });
+ });
+ });
+ }
+ function gatherKVs(tags) {
+ let primary = /* @__PURE__ */ new Set();
+ let alternate = /* @__PURE__ */ new Set();
+ Object.keys(tags).forEach((osmkey) => {
+ const osmvalue = tags[osmkey];
+ if (!osmvalue)
+ return;
+ if (osmkey === "route_master")
+ osmkey = "route";
+ const vmap = _nsi.kvt.get(osmkey);
+ if (!vmap)
+ return;
+ if (vmap.get(osmvalue)) {
+ primary.add(`${osmkey}/${osmvalue}`);
+ } else if (osmvalue === "yes") {
+ alternate.add(`${osmkey}/${osmvalue}`);
+ }
+ });
+ const preset = _mainPresetIndex.matchTags(tags, "area");
+ if (buildingPreset[preset.id]) {
+ alternate.add("building/yes");
+ }
+ return { primary, alternate };
+ }
+ function identifyTree(tags) {
+ let unknown;
+ let t;
+ Object.keys(tags).forEach((osmkey) => {
+ if (t)
+ return;
+ const osmvalue = tags[osmkey];
+ if (!osmvalue)
+ return;
+ if (osmkey === "route_master")
+ osmkey = "route";
+ const vmap = _nsi.kvt.get(osmkey);
+ if (!vmap)
+ return;
+ if (osmvalue === "yes") {
+ unknown = "unknown";
+ } else {
+ t = vmap.get(osmvalue);
+ }
+ });
+ return t || unknown || null;
+ }
+ function gatherNames(tags) {
+ const empty2 = { primary: /* @__PURE__ */ new Set(), alternate: /* @__PURE__ */ new Set() };
+ let primary = /* @__PURE__ */ new Set();
+ let alternate = /* @__PURE__ */ new Set();
+ let foundSemi = false;
+ let testNameFragments = false;
+ let patterns2;
+ let t = identifyTree(tags);
+ if (!t)
+ return empty2;
+ if (t === "transit") {
+ patterns2 = {
+ primary: /^network$/i,
+ alternate: /^(operator|operator:\w+|network:\w+|\w+_name|\w+_name:\w+)$/i
+ };
+ } else if (t === "flags") {
+ patterns2 = {
+ 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 (t === "brands") {
+ testNameFragments = true;
+ patterns2 = {
+ primary: /^(name|name:\w+)$/i,
+ alternate: /^(brand|brand:\w+|operator|operator:\w+|\w+_name|\w+_name:\w+)/i
+ };
+ } else if (t === "operators") {
+ testNameFragments = true;
+ patterns2 = {
+ primary: /^(name|name:\w+|operator|operator:\w+)$/i,
+ alternate: /^(brand|brand:\w+|\w+_name|\w+_name:\w+)/i
+ };
+ } else {
+ testNameFragments = true;
+ patterns2 = {
+ primary: /^(name|name:\w+)$/i,
+ alternate: /^(brand|brand:\w+|network|network:\w+|operator|operator:\w+|\w+_name|\w+_name:\w+)/i
+ };
+ }
+ if (tags.name && testNameFragments) {
+ const nameParts = tags.name.split(/[\s\-\/,.]/);
+ for (let split = nameParts.length; split > 0; split--) {
+ const name = nameParts.slice(0, split).join(" ");
+ primary.add(name);
+ }
+ }
+ Object.keys(tags).forEach((osmkey) => {
+ const osmvalue = tags[osmkey];
+ if (!osmvalue)
+ return;
+ 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);
+ }
+ }
+ });
+ if (tags.man_made === "flagpole" && !primary.size && !alternate.size && !!tags.country) {
+ const osmvalue = tags.country;
+ if (/;/.test(osmvalue)) {
+ foundSemi = true;
+ } else {
+ alternate.add(osmvalue);
+ }
+ }
+ if (foundSemi) {
+ return empty2;
+ } else {
+ return { primary, alternate };
+ }
+ function isNamelike(osmkey, which) {
+ if (osmkey === "old_name")
+ return false;
+ return patterns2[which].test(osmkey) && !notNames.test(osmkey);
+ }
+ }
+ function gatherTuples(tryKVs, tryNames) {
+ let tuples = [];
+ ["primary", "alternate"].forEach((whichName) => {
+ const arr = Array.from(tryNames[whichName]).sort((a, b) => b.length - a.length);
+ arr.forEach((n2) => {
+ ["primary", "alternate"].forEach((whichKV) => {
+ tryKVs[whichKV].forEach((kv) => {
+ const parts = kv.split("/", 2);
+ const k = parts[0];
+ const v = parts[1];
+ tuples.push({ k, v, n: n2 });
+ });
+ });
+ });
+ });
+ return tuples;
+ }
+ function _upgradeTags(tags, loc) {
+ let newTags = Object.assign({}, tags);
+ let changed = false;
+ Object.keys(newTags).forEach((osmkey) => {
+ const matchTag = osmkey.match(/^(\w+:)?wikidata$/);
+ if (matchTag) {
+ const prefix = matchTag[1] || "";
+ const wd = newTags[osmkey];
+ const replace = _nsi.replacements[wd];
+ if (replace && replace.wikidata !== void 0) {
+ changed = true;
+ if (replace.wikidata) {
+ newTags[osmkey] = replace.wikidata;
+ } else {
+ delete newTags[osmkey];
+ }
+ }
+ if (replace && replace.wikipedia !== void 0) {
+ changed = true;
+ const wpkey = `${prefix}wikipedia`;
+ if (replace.wikipedia) {
+ newTags[wpkey] = replace.wikipedia;
+ } else {
+ delete newTags[wpkey];
+ }
+ }
+ }
+ });
+ const isRouteMaster = tags.type === "route_master";
+ const tryKVs = gatherKVs(tags);
+ if (!tryKVs.primary.size && !tryKVs.alternate.size) {
+ return changed ? { newTags, matched: null } : null;
+ }
+ const tryNames = gatherNames(tags);
+ const foundQID = _nsi.qids.get(tags.wikidata) || _nsi.qids.get(tags.wikipedia);
+ if (foundQID)
+ tryNames.primary.add(foundQID);
+ if (!tryNames.primary.size && !tryNames.alternate.size) {
+ return changed ? { newTags, matched: null } : null;
+ }
+ const tuples = gatherTuples(tryKVs, tryNames);
+ for (let i2 = 0; i2 < tuples.length; i2++) {
+ const tuple = tuples[i2];
+ const hits = _nsi.matcher.match(tuple.k, tuple.v, tuple.n, loc);
+ if (!hits || !hits.length)
+ continue;
+ if (hits[0].match !== "primary" && hits[0].match !== "alternate")
+ break;
+ let itemID, item;
+ for (let j2 = 0; j2 < hits.length; j2++) {
+ const hit = hits[j2];
+ itemID = hit.itemID;
+ if (_nsi.dissolved[itemID])
+ continue;
+ item = _nsi.ids.get(itemID);
+ if (!item)
+ continue;
+ const mainTag = item.mainTag;
+ const itemQID = item.tags[mainTag];
+ const notQID = newTags[`not:${mainTag}`];
+ if (
+ // Exceptions, skip this hit
+ !itemQID || itemQID === notQID || // No `*:wikidata` or matched a `not:*:wikidata`
+ newTags.office && !item.tags.office
+ ) {
+ item = null;
+ continue;
+ } else {
+ break;
+ }
+ }
+ if (!item)
+ continue;
+ item = JSON.parse(JSON.stringify(item));
+ const tkv = item.tkv;
+ const parts = tkv.split("/", 3);
+ const k = parts[1];
+ const v = parts[2];
+ const category = _nsi.data[tkv];
+ const properties = category.properties || {};
+ let preserveTags = item.preserveTags || properties.preserveTags || [];
+ ["building", "emergency", "internet_access", "takeaway"].forEach((osmkey) => {
+ if (k !== osmkey)
+ preserveTags.push(`^${osmkey}$`);
+ });
+ const regexes = preserveTags.map((s) => new RegExp(s, "i"));
+ let keepTags = {};
+ Object.keys(newTags).forEach((osmkey) => {
+ if (regexes.some((regex) => regex.test(osmkey))) {
+ keepTags[osmkey] = newTags[osmkey];
+ }
+ });
+ _nsi.kvt.forEach((vmap, k2) => {
+ if (newTags[k2] === "yes")
+ delete newTags[k2];
+ });
+ if (foundQID) {
+ delete newTags.wikipedia;
+ delete newTags.wikidata;
+ }
+ Object.assign(newTags, item.tags, keepTags);
+ if (isRouteMaster) {
+ newTags.route_master = newTags.route;
+ delete newTags.route;
+ }
+ const origName = tags.name;
+ const newName = newTags.name;
+ if (newName && origName && newName !== origName && !newTags.branch) {
+ const newNames = gatherNames(newTags);
+ const newSet = /* @__PURE__ */ new Set([...newNames.primary, ...newNames.alternate]);
+ const isMoved = newSet.has(origName);
+ if (!isMoved) {
+ const nameParts = origName.split(/[\s\-\/,.]/);
+ for (let split = nameParts.length; split > 0; split--) {
+ const name = nameParts.slice(0, split).join(" ");
+ const branch = nameParts.slice(split).join(" ");
+ const nameHits = _nsi.matcher.match(k, v, name, loc);
+ if (!nameHits || !nameHits.length)
+ continue;
+ if (nameHits.some((hit) => hit.itemID === itemID)) {
+ if (branch) {
+ if (notBranches.test(branch)) {
+ newTags.name = origName;
+ } else {
+ const branchHits = _nsi.matcher.match(k, v, branch, loc);
+ if (branchHits && branchHits.length) {
+ if (branchHits[0].match === "primary" || branchHits[0].match === "alternate") {
+ return null;
+ }
+ } else {
+ newTags.branch = branch;
+ }
+ }
+ }
+ break;
+ }
+ }
+ }
+ }
+ return { newTags, matched: item };
+ }
+ return changed ? { newTags, matched: null } : null;
+ }
+ function _isGenericName(tags) {
+ const n2 = tags.name;
+ if (!n2)
+ return false;
+ const tryNames = { primary: /* @__PURE__ */ new Set([n2]), alternate: /* @__PURE__ */ new Set() };
+ const tryKVs = gatherKVs(tags);
+ if (!tryKVs.primary.size && !tryKVs.alternate.size)
+ return false;
+ const tuples = gatherTuples(tryKVs, tryNames);
+ for (let i2 = 0; i2 < tuples.length; i2++) {
+ const tuple = tuples[i2];
+ const hits = _nsi.matcher.match(tuple.k, tuple.v, tuple.n);
+ if (hits && hits.length && hits[0].match === "excludeGeneric")
+ return true;
+ }
+ return false;
+ }
+ var nsi_default = {
+ // `init()`
+ // On init, start preparing the name-suggestion-index
+ //
+ init: () => {
+ setNsiSources();
+ _mainPresetIndex.ensureLoaded().then(() => loadNsiPresets()).then(() => loadNsiData()).then(() => _nsiStatus = "ok").catch(() => _nsiStatus = "failed");
+ },
+ // `reset()`
+ // Reset is called when user saves data to OSM (does nothing here)
+ //
+ reset: () => {
+ },
+ // `status()`
+ // To let other code know how it's going...
+ //
+ // Returns
+ // `String`: 'loading', 'ok', 'failed'
+ //
+ status: () => _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: (tags) => _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` containing the result, or `null` if no changes needed:
+ // {
+ // 'newTags': `Object` - The tags the the feature should have
+ // 'matched': `Object` - The matched item
+ // }
+ //
+ upgradeTags: (tags, loc) => _upgradeTags(tags, loc),
+ // `cache()`
+ // Direct access to the NSI cache, useful for testing or breaking things
+ //
+ // Returns
+ // `Object`: the internal NSI cache
+ //
+ cache: () => _nsi
+ };
+
+ // modules/services/kartaview.js
+ var import_rbush8 = __toESM(require_rbush_min());
+ var apibase2 = "https://kartaview.org";
+ var maxResults = 1e3;
+ var tileZoom = 14;
+ var tiler4 = utilTiler().zoomExtent([tileZoom, tileZoom]).skipNullIsland(true);
+ var dispatch6 = dispatch_default("loadedImages");
+ var imgZoom = zoom_default2().extent([[0, 0], [320, 240]]).translateExtent([[0, 0], [320, 240]]).scaleExtent([1, 15]);
+ var _oscCache;
+ var _oscSelectedImage;
+ var _loadViewerPromise2;
+ function abortRequest4(controller) {
+ controller.abort();
+ }
+ 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;
+ }
+ function loadTiles2(which, url, projection2) {
+ var currZoom = Math.floor(geoScaleToZoom(projection2.scale()));
+ var tiles = tiler4.getTiles(projection2);
+ var cache = _oscCache[which];
+ Object.keys(cache.inflight).forEach(function(k) {
+ var wanted = tiles.find(function(tile) {
+ return k.indexOf(tile.id + ",") === 0;
+ });
+ if (!wanted) {
+ abortRequest4(cache.inflight[k]);
+ delete cache.inflight[k];
+ }
+ });
+ tiles.forEach(function(tile) {
+ loadNextTilePage(which, currZoom, url, tile);
+ });
+ }
+ function loadNextTilePage(which, currZoom, url, tile) {
+ var cache = _oscCache[which];
+ var bbox2 = tile.extent.bbox();
+ var maxPages = maxPageAtZoom(currZoom);
+ var nextPage = cache.nextPage[tile.id] || 1;
+ var params = utilQsString({
+ ipp: maxResults,
+ page: nextPage,
+ // client_id: clientId,
+ bbTopLeft: [bbox2.maxY, bbox2.minX].join(","),
+ bbBottomRight: [bbox2.minY, bbox2.maxX].join(",")
+ }, true);
+ if (nextPage > maxPages)
+ return;
+ var id2 = tile.id + "," + String(nextPage);
+ if (cache.loaded[id2] || cache.inflight[id2])
+ return;
+ var controller = new AbortController();
+ cache.inflight[id2] = controller;
+ var options2 = {
+ method: "POST",
+ signal: controller.signal,
+ body: params,
+ headers: { "Content-Type": "application/x-www-form-urlencoded" }
+ };
+ json_default(url, options2).then(function(data) {
+ cache.loaded[id2] = true;
+ delete cache.inflight[id2];
+ if (!data || !data.currentPageItems || !data.currentPageItems.length) {
+ throw new Error("No Data");
+ }
+ var features = data.currentPageItems.map(function(item) {
+ var loc = [+item.lng, +item.lat];
+ var d;
+ if (which === "images") {
+ d = {
+ 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
+ };
+ var seq = _oscCache.sequences[d.sequence_id];
+ if (!seq) {
+ seq = { rotation: 0, images: [] };
+ _oscCache.sequences[d.sequence_id] = seq;
+ }
+ seq.images[d.sequence_index] = d;
+ _oscCache.images.forImageKey[d.key] = d;
+ }
+ return {
+ minX: loc[0],
+ minY: loc[1],
+ maxX: loc[0],
+ maxY: loc[1],
+ data: d
+ };
+ });
+ cache.rtree.load(features);
+ if (data.currentPageItems.length === maxResults) {
+ cache.nextPage[tile.id] = nextPage + 1;
+ loadNextTilePage(which, currZoom, url, tile);
+ } else {
+ cache.nextPage[tile.id] = Infinity;
+ }
+ if (which === "images") {
+ dispatch6.call("loadedImages");
+ }
+ }).catch(function() {
+ cache.loaded[id2] = true;
+ delete cache.inflight[id2];
+ });
+ }
+ function partitionViewport2(projection2) {
+ var z = geoScaleToZoom(projection2.scale());
+ var z2 = Math.ceil(z * 2) / 2 + 2.5;
+ var tiler8 = utilTiler().zoomExtent([z2, z2]);
+ return tiler8.getTiles(projection2).map(function(tile) {
+ return tile.extent;
+ });
+ }
+ function searchLimited2(limit, projection2, rtree) {
+ limit = limit || 5;
+ return partitionViewport2(projection2).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 kartaview_default = {
+ init: function() {
+ if (!_oscCache) {
+ this.reset();
+ }
+ this.event = utilRebind(this, dispatch6, "on");
+ },
+ reset: function() {
+ if (_oscCache) {
+ Object.values(_oscCache.images.inflight).forEach(abortRequest4);
+ }
+ _oscCache = {
+ images: { inflight: {}, loaded: {}, nextPage: {}, rtree: new import_rbush8.default(), forImageKey: {} },
+ sequences: {}
+ };
+ _oscSelectedImage = null;
+ },
+ images: function(projection2) {
+ var limit = 5;
+ return searchLimited2(limit, projection2, _oscCache.images.rtree);
+ },
+ sequences: function(projection2) {
+ var viewport = projection2.clipExtent();
+ var min3 = [viewport[0][0], viewport[1][1]];
+ var max3 = [viewport[1][0], viewport[0][1]];
+ var bbox2 = geoExtent(projection2.invert(min3), projection2.invert(max3)).bbox();
+ var sequenceKeys = {};
+ _oscCache.images.rtree.search(bbox2).forEach(function(d) {
+ sequenceKeys[d.data.sequence_id] = true;
+ });
+ 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(imageKey) {
+ return _oscCache.images.forImageKey[imageKey];
+ },
+ loadImages: function(projection2) {
+ var url = apibase2 + "/1.0/list/nearby-photos/";
+ loadTiles2("images", url, projection2);
+ },
+ ensureViewerLoaded: function(context) {
+ if (_loadViewerPromise2)
+ return _loadViewerPromise2;
+ var wrap2 = context.container().select(".photoviewer").selectAll(".kartaview-wrapper").data([0]);
+ var that = this;
+ var wrapEnter = wrap2.enter().append("div").attr("class", "photo-wrapper kartaview-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)).text("\u25C4");
+ controlsEnter.append("button").on("click.rotate-ccw", rotate(-90)).text("\u293F");
+ controlsEnter.append("button").on("click.rotate-cw", rotate(90)).text("\u293E");
+ controlsEnter.append("button").on("click.forward", step(1)).text("\u25BA");
+ wrapEnter.append("div").attr("class", "kartaview-image-wrap");
+ context.ui().photoviewer.on("resize.kartaview", function(dimensions) {
+ imgZoom = zoom_default2().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 .kartaview-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 wrap3 = context.container().select(".photoviewer .kartaview-wrapper");
+ wrap3.transition().duration(100).call(imgZoom.transform, identity2);
+ wrap3.selectAll(".kartaview-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);
+ };
+ }
+ _loadViewerPromise2 = Promise.resolve();
+ return _loadViewerPromise2;
+ },
+ showViewer: function(context) {
+ var viewer = context.container().select(".photoviewer").classed("hide", false);
+ var isHidden = viewer.selectAll(".photo-wrapper.kartaview-wrapper.hide").size();
+ if (isHidden) {
+ viewer.selectAll(".photo-wrapper:not(.kartaview-wrapper)").classed("hide", true);
+ viewer.selectAll(".photo-wrapper.kartaview-wrapper").classed("hide", false);
+ }
+ return this;
+ },
+ hideViewer: function(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(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 wrap2 = context.container().select(".photoviewer .kartaview-wrapper");
+ var imageWrap = wrap2.selectAll(".kartaview-image-wrap");
+ var attribution = wrap2.selectAll(".photo-attribution").text("");
+ wrap2.transition().duration(100).call(imgZoom.transform, identity2);
+ imageWrap.selectAll(".kartaview-image").remove();
+ if (d) {
+ var sequence = _oscCache.sequences[d.sequence_id];
+ var r = sequence && sequence.rotation || 0;
+ imageWrap.append("img").attr("class", "kartaview-image").attr("src", apibase2 + "/" + d.imagePath).style("transform", "rotate(" + r + "deg)");
+ if (d.captured_by) {
+ attribution.append("a").attr("class", "captured_by").attr("target", "_blank").attr("href", "https://kartaview.org/user/" + encodeURIComponent(d.captured_by)).text("@" + d.captured_by);
+ attribution.append("span").text("|");
+ }
+ if (d.captured_at) {
+ attribution.append("span").attr("class", "captured_at").text(localeDateString2(d.captured_at));
+ attribution.append("span").text("|");
+ }
+ attribution.append("a").attr("class", "image-link").attr("target", "_blank").attr("href", "https://kartaview.org/details/" + d.sequence_id + "/" + d.sequence_index).text("kartaview.org");
+ }
+ return this;
+ function localeDateString2(s) {
+ if (!s)
+ return null;
+ var options2 = { day: "numeric", month: "short", year: "numeric" };
+ var d2 = new Date(s);
+ if (isNaN(d2.getTime()))
+ return null;
+ return d2.toLocaleDateString(_mainLocalizer.localeCode(), options2);
+ }
+ },
+ getSelectedImage: function() {
+ return _oscSelectedImage;
+ },
+ getSequenceKeyForImage: function(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(context, hovered, reset) {
+ if (reset) {
+ 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() ? void 0 : 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;
+ }) || [];
+ var highlightedImageKeys = utilArrayUnion(hoveredImageKeys, selectedImageKeys);
+ context.container().selectAll(".layer-kartaview .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-kartaview .sequence").classed("highlighted", function(d) {
+ return d.properties.key === hoveredSequenceKey;
+ }).classed("currentView", function(d) {
+ return d.properties.key === selectedSequenceKey;
+ });
+ context.container().selectAll(".layer-kartaview .viewfield-group .viewfield").attr("d", viewfieldPath);
+ function viewfieldPath() {
+ var d = this.parentNode.__data__;
+ 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";
+ }
+ }
+ return this;
+ },
+ updateUrlImage: function(imageKey) {
+ if (!window.mocha) {
+ var hash = utilStringQs(window.location.hash);
+ if (imageKey) {
+ hash.photo = "kartaview/" + imageKey;
+ } else {
+ delete hash.photo;
+ }
+ window.location.replace("#" + utilQsString(hash, true));
+ }
+ },
+ cache: function() {
+ return _oscCache;
+ }
+ };
+
+ // node_modules/osm-auth/src/osm-auth.mjs
+ var import_store = __toESM(require_store_legacy(), 1);
+ function osmAuth(o) {
+ var oauth2 = {};
+ oauth2.authenticated = function() {
+ return !!token("oauth2_access_token");
+ };
+ oauth2.logout = function() {
+ token("oauth2_access_token", "");
+ token("oauth_token", "");
+ token("oauth_token_secret", "");
+ token("oauth_request_token_secret", "");
+ return oauth2;
+ };
+ oauth2.authenticate = function(callback) {
+ if (oauth2.authenticated()) {
+ callback(null, oauth2);
+ return;
+ }
+ oauth2.logout();
+ var url = o.url + "/oauth2/authorize?" + utilQsString2({
+ client_id: o.client_id,
+ redirect_uri: o.redirect_uri,
+ response_type: "code",
+ scope: o.scope
+ });
+ if (!o.singlepage) {
+ var w = 600;
+ var h = 550;
+ var settings = [
+ ["width", w],
+ ["height", h],
+ ["left", screen.width / 2 - w / 2],
+ ["top", screen.height / 2 - h / 2]
+ ].map(function(x) {
+ return x.join("=");
+ }).join(",");
+ var popup = window.open("about:blank", "oauth_window", settings);
+ oauth2.popupWindow = popup;
+ popup.location = url;
+ if (!popup) {
+ var error = new Error("Popup was blocked");
+ error.status = "popup-blocked";
+ throw error;
+ }
+ }
+ window.authComplete = function(url2) {
+ var params = utilStringQs2(url2.split("?")[1]);
+ getAccessToken(params.code);
+ delete window.authComplete;
+ };
+ function getAccessToken(auth_code) {
+ var url2 = o.url + "/oauth2/token?" + utilQsString2({
+ client_id: o.client_id,
+ grant_type: "authorization_code",
+ code: auth_code,
+ redirect_uri: o.redirect_uri,
+ client_secret: o.client_secret
+ });
+ oauth2.rawxhr("POST", url2, null, null, null, accessTokenDone);
+ o.loading();
+ }
+ function accessTokenDone(err, xhr) {
+ o.done();
+ if (err) {
+ callback(err);
+ return;
+ }
+ var access_token = JSON.parse(xhr.response);
+ token("oauth2_access_token", access_token.access_token);
+ callback(null, oauth2);
+ }
+ };
+ oauth2.bringPopupWindowToFront = function() {
+ var broughtPopupToFront = false;
+ try {
+ if (oauth2.popupWindow && !oauth2.popupWindow.closed) {
+ oauth2.popupWindow.focus();
+ broughtPopupToFront = true;
+ }
+ } catch (err) {
+ }
+ return broughtPopupToFront;
+ };
+ oauth2.bootstrapToken = function(auth_code, callback) {
+ function getAccessToken(auth_code2) {
+ var url = o.url + "/oauth2/token?" + utilQsString2({
+ client_id: o.client_id,
+ grant_type: "authorization_code",
+ code: auth_code2,
+ redirect_uri: o.redirect_uri,
+ client_secret: o.client_secret
+ });
+ oauth2.rawxhr("POST", url, null, null, null, accessTokenDone);
+ o.loading();
+ }
+ function accessTokenDone(err, xhr) {
+ o.done();
+ if (err) {
+ callback(err);
+ return;
+ }
+ var access_token = JSON.parse(xhr.response);
+ token("oauth2_access_token", access_token.access_token);
+ callback(null, oauth2);
+ }
+ getAccessToken(auth_code);
+ };
+ oauth2.xhr = function(options2, callback) {
+ if (oauth2.authenticated()) {
+ return run();
+ } else {
+ if (o.auto) {
+ oauth2.authenticate(run);
+ return;
+ } else {
+ callback("not authenticated", null);
+ return;
+ }
+ }
+ function run() {
+ var url = options2.prefix !== false ? o.url + options2.path : options2.path;
+ return oauth2.rawxhr(
+ options2.method,
+ url,
+ token("oauth2_access_token"),
+ options2.content,
+ options2.headers,
+ done
+ );
+ }
+ function done(err, xhr) {
+ if (err) {
+ callback(err);
+ } else if (xhr.responseXML) {
+ callback(err, xhr.responseXML);
+ } else {
+ callback(err, xhr.response);
+ }
+ }
+ };
+ oauth2.rawxhr = function(method, url, access_token, data, headers, callback) {
+ headers = headers || { "Content-Type": "application/x-www-form-urlencoded" };
+ if (access_token) {
+ headers.Authorization = "Bearer " + access_token;
+ }
+ var xhr = new XMLHttpRequest();
+ xhr.onreadystatechange = function() {
+ if (4 === xhr.readyState && 0 !== xhr.status) {
+ if (/^20\d$/.test(xhr.status)) {
+ callback(null, xhr);
+ } else {
+ callback(xhr, null);
+ }
+ }
+ };
+ xhr.onerror = function(e) {
+ callback(e, null);
+ };
+ xhr.open(method, url, true);
+ for (var h in headers)
+ xhr.setRequestHeader(h, headers[h]);
+ xhr.send(data);
+ return xhr;
+ };
+ oauth2.preauth = function(val) {
+ if (val && val.access_token) {
+ token("oauth2_access_token", val.access_token);
+ }
+ return oauth2;
+ };
+ oauth2.options = function(val) {
+ if (!arguments.length)
+ return o;
+ o = val;
+ o.url = o.url || "https://www.openstreetmap.org";
+ o.auto = o.auto || false;
+ o.singlepage = o.singlepage || false;
+ o.loading = o.loading || function() {
+ };
+ o.done = o.done || function() {
+ };
+ return oauth2.preauth(o);
+ };
+ var token;
+ if (import_store.default.enabled) {
+ token = function(x, y) {
+ if (arguments.length === 1)
+ return import_store.default.get(o.url + x);
+ else if (arguments.length === 2)
+ return import_store.default.set(o.url + x, y);
+ };
+ } else {
+ var storage = {};
+ token = function(x, y) {
+ if (arguments.length === 1)
+ return storage[o.url + x];
+ else if (arguments.length === 2)
+ return storage[o.url + x] = y;
+ };
+ }
+ oauth2.options(o);
+ return oauth2;
+ }
+ function utilQsString2(obj) {
+ return Object.keys(obj).sort().map(function(key) {
+ return encodeURIComponent(key) + "=" + encodeURIComponent(obj[key]);
+ }).join("&");
+ }
+ function utilStringQs2(str2) {
+ var i2 = 0;
+ while (i2 < str2.length && (str2[i2] === "?" || str2[i2] === "#"))
+ i2++;
+ str2 = str2.slice(i2);
+ return str2.split("&").reduce(function(obj, pair2) {
+ var parts = pair2.split("=");
+ if (parts.length === 2) {
+ obj[parts[0]] = decodeURIComponent(parts[1]);
+ }
+ return obj;
+ }, {});
+ }
+
+ // modules/services/osm.js
+ var import_rbush9 = __toESM(require_rbush_min());
+ var tiler5 = utilTiler();
+ var dispatch7 = dispatch_default("apiStatusChange", "authLoading", "authDone", "change", "loading", "loaded", "loadedNotes");
+ var urlroot = osmApiConnections[0].url;
+ var redirectPath = window.location.origin + window.location.pathname;
+ var oauth = osmAuth({
+ url: urlroot,
+ client_id: osmApiConnections[0].client_id,
+ client_secret: osmApiConnections[0].client_secret,
+ scope: "read_prefs write_prefs write_api read_gpx write_notes",
+ redirect_uri: redirectPath + "land.html",
+ loading: authLoading,
+ done: authDone
+ });
+ var _apiConnections = osmApiConnections;
+ var _imageryBlocklists = [/.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/];
+ var _tileCache = { toLoad: {}, loaded: {}, inflight: {}, seen: {}, rtree: new import_rbush9.default() };
+ var _noteCache = { toLoad: {}, loaded: {}, inflight: {}, inflightPost: {}, note: {}, closed: {}, rtree: new import_rbush9.default() };
+ var _userCache = { toLoad: {}, user: {} };
+ var _cachedApiStatus;
+ var _changeset = {};
+ var _deferred = /* @__PURE__ */ new Set();
+ var _connectionID = 1;
+ var _tileZoom4 = 16;
+ var _noteZoom = 12;
+ var _rateLimitError;
+ var _userChangesets;
+ var _userDetails;
+ var _off;
+ var _maxWayNodes = 2e3;
+ function authLoading() {
+ dispatch7.call("authLoading");
+ }
+ function authDone() {
+ dispatch7.call("authDone");
+ }
+ function abortRequest5(controllerOrXHR) {
+ if (controllerOrXHR) {
+ controllerOrXHR.abort();
+ }
+ }
+ function hasInflightRequests(cache) {
+ return Object.keys(cache.inflight).length;
+ }
+ function abortUnwantedRequests4(cache, visibleTiles) {
+ Object.keys(cache.inflight).forEach(function(k) {
+ if (cache.toLoad[k])
+ return;
+ if (visibleTiles.find(function(tile) {
+ return k === tile.id;
+ }))
+ return;
+ abortRequest5(cache.inflight[k]);
+ delete cache.inflight[k];
+ });
+ }
+ function getLoc(attrs) {
+ var lon = attrs.lon && attrs.lon.value;
+ var lat = attrs.lat && attrs.lat.value;
+ return [Number(lon), Number(lat)];
+ }
+ function getNodes(obj) {
+ var elems = obj.getElementsByTagName("nd");
+ var nodes = new Array(elems.length);
+ for (var i2 = 0, l = elems.length; i2 < l; i2++) {
+ nodes[i2] = "n" + elems[i2].attributes.ref.value;
+ }
+ return nodes;
+ }
+ function getNodesJSON(obj) {
+ var elems = obj.nodes;
+ var nodes = new Array(elems.length);
+ for (var i2 = 0, l = elems.length; i2 < l; i2++) {
+ nodes[i2] = "n" + elems[i2];
+ }
+ return nodes;
+ }
+ function getTags(obj) {
+ var elems = obj.getElementsByTagName("tag");
+ var tags = {};
+ for (var i2 = 0, l = elems.length; i2 < l; i2++) {
+ var attrs = elems[i2].attributes;
+ tags[attrs.k.value] = attrs.v.value;
+ }
+ return tags;
+ }
+ function getMembers(obj) {
+ var elems = obj.getElementsByTagName("member");
+ var members = new Array(elems.length);
+ for (var i2 = 0, l = elems.length; i2 < l; i2++) {
+ var attrs = elems[i2].attributes;
+ members[i2] = {
+ id: attrs.type.value[0] + attrs.ref.value,
+ type: attrs.type.value,
+ role: attrs.role.value
+ };
+ }
+ return members;
+ }
+ function getMembersJSON(obj) {
+ var elems = obj.members;
+ var members = new Array(elems.length);
+ for (var i2 = 0, l = elems.length; i2 < l; i2++) {
+ var attrs = elems[i2];
+ members[i2] = {
+ id: attrs.type[0] + attrs.ref,
+ type: attrs.type,
+ role: attrs.role
+ };
+ }
+ return members;
+ }
+ function getVisible(attrs) {
+ return !attrs.visible || attrs.visible.value !== "false";
+ }
+ function parseComments(comments) {
+ var parsedComments = [];
+ for (var i2 = 0; i2 < comments.length; i2++) {
+ var comment = comments[i2];
+ if (comment.nodeName === "comment") {
+ var childNodes = comment.childNodes;
+ var parsedComment = {};
+ for (var j2 = 0; j2 < childNodes.length; j2++) {
+ var node = childNodes[j2];
+ var nodeName = node.nodeName;
+ if (nodeName === "#text")
+ continue;
+ parsedComment[nodeName] = node.textContent;
+ if (nodeName === "uid") {
+ var uid = node.textContent;
+ if (uid && !_userCache.user[uid]) {
+ _userCache.toLoad[uid] = true;
+ }
+ }
+ }
+ if (parsedComment) {
+ parsedComments.push(parsedComment);
+ }
+ }
+ }
+ return parsedComments;
+ }
+ function encodeNoteRtree(note) {
+ return {
+ minX: note.loc[0],
+ minY: note.loc[1],
+ maxX: note.loc[0],
+ maxY: note.loc[1],
+ data: note
+ };
+ }
+ 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: [Number(obj.lon), Number(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"
+ };
+ }
+ };
+ function parseJSON(payload, callback, options2) {
+ options2 = Object.assign({ skipSeen: true }, options2);
+ if (!payload) {
+ return callback({ message: "No JSON", status: -1 });
+ }
+ var json = payload;
+ if (typeof json !== "object")
+ json = JSON.parse(payload);
+ if (!json.elements)
+ return callback({ message: "No JSON", status: -1 });
+ var children2 = json.elements;
+ var handle = window.requestIdleCallback(function() {
+ _deferred.delete(handle);
+ var results = [];
+ var result;
+ for (var i2 = 0; i2 < children2.length; i2++) {
+ result = parseChild(children2[i2]);
+ if (result)
+ results.push(result);
+ }
+ callback(null, results);
+ });
+ _deferred.add(handle);
+ function parseChild(child) {
+ var parser3 = jsonparsers[child.type];
+ if (!parser3)
+ return null;
+ var uid;
+ uid = osmEntity.id.fromOSM(child.type, child.id);
+ if (options2.skipSeen) {
+ if (_tileCache.seen[uid])
+ return null;
+ _tileCache.seen[uid] = true;
+ }
+ return parser3(child, uid);
+ }
+ }
+ function parseUserJSON(payload, callback, options2) {
+ options2 = Object.assign({ skipSeen: true }, options2);
+ if (!payload) {
+ return callback({ message: "No JSON", status: -1 });
+ }
+ 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 results = [];
+ var result;
+ for (var i2 = 0; i2 < objs.length; i2++) {
+ result = parseObj(objs[i2]);
+ if (result)
+ results.push(result);
+ }
+ callback(null, results);
+ });
+ _deferred.add(handle);
+ function parseObj(obj) {
+ var uid = obj.user.id && obj.user.id.toString();
+ if (options2.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;
+ }
+ }
+ var parsers = {
+ node: function nodeData2(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 wayData2(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 relationData2(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);
+ var coincident = false;
+ var epsilon3 = 1e-5;
+ do {
+ if (coincident) {
+ props.loc = geoVecAdd(props.loc, [epsilon3, epsilon3]);
+ }
+ var bbox2 = geoExtent(props.loc).bbox();
+ coincident = _noteCache.rtree.search(bbox2).length;
+ } while (coincident);
+ for (var i2 = 0; i2 < childNodes.length; i2++) {
+ var node = childNodes[i2];
+ var nodeName = node.nodeName;
+ if (nodeName === "#text")
+ continue;
+ if (nodeName === "comments") {
+ props[nodeName] = parseComments(node.childNodes);
+ } else {
+ props[nodeName] = node.textContent;
+ }
+ }
+ var note = new osmNote(props);
+ var item = encodeNoteRtree(note);
+ _noteCache.note[note.id] = note;
+ _noteCache.rtree.insert(item);
+ return note;
+ },
+ user: function parseUser2(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");
+ if (img && img[0] && img[0].getAttribute("href")) {
+ user.image_url = img[0].getAttribute("href");
+ }
+ var changesets = obj.getElementsByTagName("changesets");
+ if (changesets && changesets[0] && changesets[0].getAttribute("count")) {
+ user.changesets_count = changesets[0].getAttribute("count");
+ }
+ var blocks = obj.getElementsByTagName("blocks");
+ if (blocks && blocks[0]) {
+ var received = blocks[0].getElementsByTagName("received");
+ if (received && received[0] && received[0].getAttribute("active")) {
+ user.active_blocks = received[0].getAttribute("active");
+ }
+ }
+ _userCache.user[uid] = user;
+ delete _userCache.toLoad[uid];
+ return user;
+ }
+ };
+ function parseXML(xml, callback, options2) {
+ options2 = Object.assign({ skipSeen: true }, options2);
+ if (!xml || !xml.childNodes) {
+ return callback({ message: "No XML", status: -1 });
+ }
+ var root3 = xml.childNodes[0];
+ var children2 = root3.childNodes;
+ var handle = window.requestIdleCallback(function() {
+ _deferred.delete(handle);
+ var results = [];
+ var result;
+ for (var i2 = 0; i2 < children2.length; i2++) {
+ result = parseChild(children2[i2]);
+ if (result)
+ results.push(result);
+ }
+ callback(null, results);
+ });
+ _deferred.add(handle);
+ function parseChild(child) {
+ var parser3 = parsers[child.nodeName];
+ if (!parser3)
+ return null;
+ var uid;
+ if (child.nodeName === "user") {
+ uid = child.attributes.id.value;
+ if (options2.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);
+ if (options2.skipSeen) {
+ if (_tileCache.seen[uid])
+ return null;
+ _tileCache.seen[uid] = true;
+ }
+ }
+ return parser3(child, uid);
+ }
+ }
+ function updateRtree4(item, replace) {
+ _noteCache.rtree.remove(item, function isEql(a, b) {
+ return a.data.id === b.data.id;
+ });
+ if (replace) {
+ _noteCache.rtree.insert(item);
+ }
+ }
+ function wrapcb(thisArg, callback, cid) {
+ return function(err, result) {
+ if (err) {
+ if (err.status === 400 || err.status === 401 || err.status === 403) {
+ thisArg.logout();
+ }
+ 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);
+ }
+ };
+ }
+ var osm_default = {
+ init: function() {
+ utilRebind(this, dispatch7, "on");
+ },
+ reset: function() {
+ Array.from(_deferred).forEach(function(handle) {
+ window.cancelIdleCallback(handle);
+ _deferred.delete(handle);
+ });
+ _connectionID++;
+ _userChangesets = void 0;
+ _userDetails = void 0;
+ _rateLimitError = void 0;
+ Object.values(_tileCache.inflight).forEach(abortRequest5);
+ Object.values(_noteCache.inflight).forEach(abortRequest5);
+ Object.values(_noteCache.inflightPost).forEach(abortRequest5);
+ if (_changeset.inflight)
+ abortRequest5(_changeset.inflight);
+ _tileCache = { toLoad: {}, loaded: {}, inflight: {}, seen: {}, rtree: new import_rbush9.default() };
+ _noteCache = { toLoad: {}, loaded: {}, inflight: {}, inflightPost: {}, note: {}, closed: {}, rtree: new import_rbush9.default() };
+ _userCache = { toLoad: {}, user: {} };
+ _cachedApiStatus = void 0;
+ _changeset = {};
+ return this;
+ },
+ getConnectionId: function() {
+ return _connectionID;
+ },
+ getUrlRoot: function() {
+ return urlroot;
+ },
+ changesetURL: function(changesetID) {
+ return urlroot + "/changeset/" + changesetID;
+ },
+ changesetsURL: function(center, zoom) {
+ var precision2 = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2));
+ return urlroot + "/history#map=" + Math.floor(zoom) + "/" + center[1].toFixed(precision2) + "/" + center[0].toFixed(precision2);
+ },
+ entityURL: function(entity) {
+ return urlroot + "/" + entity.type + "/" + entity.osmId();
+ },
+ historyURL: function(entity) {
+ return urlroot + "/" + entity.type + "/" + entity.osmId() + "/history";
+ },
+ userURL: function(username) {
+ return urlroot + "/user/" + encodeURIComponent(username);
+ },
+ noteURL: function(note) {
+ return urlroot + "/note/" + note.id;
+ },
+ noteReportURL: function(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(path, callback, options2) {
+ options2 = Object.assign({ skipSeen: true }, options2);
+ var that = this;
+ var cid = _connectionID;
+ function done(err, payload) {
+ if (that.getConnectionId() !== cid) {
+ if (callback)
+ callback({ message: "Connection Switched", status: -1 });
+ return;
+ }
+ var isAuthenticated = that.authenticated();
+ if (isAuthenticated && err && err.status && (err.status === 400 || err.status === 401 || err.status === 403)) {
+ that.logout();
+ that.loadFromAPI(path, callback, options2);
+ } else {
+ if (!isAuthenticated && !_rateLimitError && err && err.status && (err.status === 509 || err.status === 429)) {
+ _rateLimitError = err;
+ dispatch7.call("change");
+ that.reloadApiStatus();
+ } else if (err && _cachedApiStatus === "online" || !err && _cachedApiStatus !== "online") {
+ that.reloadApiStatus();
+ }
+ if (callback) {
+ if (err) {
+ return callback(err);
+ } else {
+ if (path.indexOf(".json") !== -1) {
+ return parseJSON(payload, callback, options2);
+ } else {
+ return parseXML(payload, callback, options2);
+ }
+ }
+ }
+ }
+ }
+ if (this.authenticated()) {
+ return oauth.xhr({ method: "GET", path }, done);
+ } else {
+ var url = urlroot + path;
+ var controller = new AbortController();
+ var fn;
+ if (path.indexOf(".json") !== -1) {
+ fn = json_default;
+ } else {
+ fn = xml_default;
+ }
+ fn(url, { signal: controller.signal }).then(function(data) {
+ done(null, data);
+ }).catch(function(err) {
+ if (err.name === "AbortError")
+ return;
+ var match = err.message.match(/^\d{3}/);
+ 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 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(id2, callback) {
+ var type2 = osmEntity.id.type(id2);
+ var osmID = osmEntity.id.toOSM(id2);
+ var options2 = { skipSeen: false };
+ this.loadFromAPI(
+ "/api/0.6/" + type2 + "/" + osmID + (type2 !== "node" ? "/full" : "") + ".json",
+ function(err, entities) {
+ if (callback)
+ callback(err, { data: entities });
+ },
+ options2
+ );
+ },
+ // Load a single entity with a specific version
+ // GET /api/0.6/[node|way|relation]/#id/#version
+ loadEntityVersion: function(id2, version, callback) {
+ var type2 = osmEntity.id.type(id2);
+ var osmID = osmEntity.id.toOSM(id2);
+ var options2 = { skipSeen: false };
+ this.loadFromAPI(
+ "/api/0.6/" + type2 + "/" + osmID + "/" + version + ".json",
+ function(err, entities) {
+ if (callback)
+ callback(err, { data: entities });
+ },
+ options2
+ );
+ },
+ // Load the relations of a single entity with the given.
+ // GET /api/0.6/[node|way|relation]/#id/relations
+ loadEntityRelations: function(id2, callback) {
+ var type2 = osmEntity.id.type(id2);
+ var osmID = osmEntity.id.toOSM(id2);
+ var options2 = { skipSeen: false };
+ this.loadFromAPI(
+ "/api/0.6/" + type2 + "/" + osmID + "/relations.json",
+ function(err, entities) {
+ if (callback)
+ callback(err, { data: entities });
+ },
+ options2
+ );
+ },
+ // 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(ids, callback) {
+ var that = this;
+ var groups = utilArrayGroupBy(utilArrayUniq(ids), osmEntity.id.type);
+ Object.keys(groups).forEach(function(k) {
+ var type2 = k + "s";
+ var osmIDs = groups[k].map(function(id2) {
+ return osmEntity.id.toOSM(id2);
+ });
+ var options2 = { skipSeen: false };
+ utilArrayChunk(osmIDs, 150).forEach(function(arr) {
+ that.loadFromAPI(
+ "/api/0.6/" + type2 + ".json?" + type2 + "=" + arr.join(),
+ function(err, entities) {
+ if (callback)
+ callback(err, { data: entities });
+ },
+ options2
+ );
+ });
+ });
+ },
+ // 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(changeset, changes, callback) {
+ var cid = _connectionID;
+ if (_changeset.inflight) {
+ return callback({ message: "Changeset already inflight", status: -2 }, changeset);
+ } else if (_changeset.open) {
+ return createdChangeset.call(this, null, _changeset.open);
+ } else {
+ var options2 = {
+ method: "PUT",
+ path: "/api/0.6/changeset/create",
+ headers: { "Content-Type": "text/xml" },
+ content: JXON.stringify(changeset.asJXON())
+ };
+ _changeset.inflight = oauth.xhr(
+ options2,
+ wrapcb(this, createdChangeset, cid)
+ );
+ }
+ function createdChangeset(err, changesetID) {
+ _changeset.inflight = null;
+ if (err) {
+ return callback(err, changeset);
+ }
+ _changeset.open = changesetID;
+ changeset = changeset.update({ id: changesetID });
+ var options3 = {
+ method: "POST",
+ path: "/api/0.6/changeset/" + changesetID + "/upload",
+ headers: { "Content-Type": "text/xml" },
+ content: JXON.stringify(changeset.osmChangeJXON(changes))
+ };
+ _changeset.inflight = oauth.xhr(
+ options3,
+ wrapcb(this, uploadedChangeset, cid)
+ );
+ }
+ function uploadedChangeset(err) {
+ _changeset.inflight = null;
+ if (err)
+ return callback(err, changeset);
+ window.setTimeout(function() {
+ callback(null, changeset);
+ }, 2500);
+ _changeset.open = null;
+ if (this.getConnectionId() === cid) {
+ oauth.xhr({
+ method: "PUT",
+ path: "/api/0.6/changeset/" + changeset.id + "/close",
+ headers: { "Content-Type": "text/xml" }
+ }, 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(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(void 0, cached);
+ if (!this.authenticated())
+ return;
+ }
+ 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));
+ function done(err, payload) {
+ if (err)
+ return callback(err);
+ var options2 = { skipSeen: true };
+ return parseUserJSON(payload, function(err2, results) {
+ if (err2)
+ return callback(err2);
+ return callback(void 0, results);
+ }, options2);
+ }
+ },
+ // Load a given user by id
+ // GET /api/0.6/user/#id
+ loadUser: function(uid, callback) {
+ if (_userCache.user[uid] || !this.authenticated()) {
+ delete _userCache.toLoad[uid];
+ return callback(void 0, _userCache.user[uid]);
+ }
+ oauth.xhr(
+ { method: "GET", path: "/api/0.6/user/" + uid + ".json" },
+ wrapcb(this, done, _connectionID)
+ );
+ function done(err, payload) {
+ if (err)
+ return callback(err);
+ var options2 = { skipSeen: true };
+ return parseUserJSON(payload, function(err2, results) {
+ if (err2)
+ return callback(err2);
+ return callback(void 0, results[0]);
+ }, options2);
+ }
+ },
+ // Load the details of the logged-in user
+ // GET /api/0.6/user/details
+ userDetails: function(callback) {
+ if (_userDetails) {
+ return callback(void 0, _userDetails);
+ }
+ oauth.xhr(
+ { method: "GET", path: "/api/0.6/user/details.json" },
+ wrapcb(this, done, _connectionID)
+ );
+ function done(err, payload) {
+ if (err)
+ return callback(err);
+ var options2 = { skipSeen: false };
+ return parseUserJSON(payload, function(err2, results) {
+ if (err2)
+ return callback(err2);
+ _userDetails = results[0];
+ return callback(void 0, _userDetails);
+ }, options2);
+ }
+ },
+ // Load previous changesets for the logged in user
+ // GET /api/0.6/changesets?user=#id
+ userChangesets: function(callback) {
+ if (_userChangesets) {
+ return callback(void 0, _userChangesets);
+ }
+ this.userDetails(
+ wrapcb(this, gotDetails, _connectionID)
+ );
+ function gotDetails(err, user) {
+ if (err) {
+ return callback(err);
+ }
+ oauth.xhr(
+ { method: "GET", path: "/api/0.6/changesets?user=" + user.id },
+ wrapcb(this, done, _connectionID)
+ );
+ }
+ function done(err, xml) {
+ if (err) {
+ return callback(err);
+ }
+ _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(void 0, _userChangesets);
+ }
+ },
+ // Fetch the status of the OSM API
+ // GET /api/capabilities
+ status: function(callback) {
+ var url = urlroot + "/api/capabilities";
+ var errback = wrapcb(this, done, _connectionID);
+ xml_default(url).then(function(data) {
+ errback(null, data);
+ }).catch(function(err) {
+ errback(err.message);
+ });
+ function done(err, xml) {
+ if (err) {
+ return callback(err, null);
+ }
+ var elements = xml.getElementsByTagName("blacklist");
+ var regexes = [];
+ for (var i2 = 0; i2 < elements.length; i2++) {
+ var regexString = elements[i2].getAttribute("regex");
+ if (regexString) {
+ try {
+ var regex = new RegExp(regexString);
+ regexes.push(regex);
+ } catch (e) {
+ }
+ }
+ }
+ if (regexes.length) {
+ _imageryBlocklists = regexes;
+ }
+ 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(void 0, val);
+ }
+ }
+ },
+ // Calls `status` and dispatches an `apiStatusChange` event if the returned
+ // status differs from the cached status.
+ reloadApiStatus: function() {
+ if (!this.throttledReloadApiStatus) {
+ var that = this;
+ this.throttledReloadApiStatus = throttle_default(function() {
+ that.status(function(err, status) {
+ if (status !== _cachedApiStatus) {
+ _cachedApiStatus = status;
+ dispatch7.call("apiStatusChange", that, err, status);
+ }
+ });
+ }, 500);
+ }
+ this.throttledReloadApiStatus();
+ },
+ // Returns the maximum number of nodes a single way can have
+ maxWayNodes: function() {
+ return _maxWayNodes;
+ },
+ // Load data (entities) from the API in tiles
+ // GET /api/0.6/map?bbox=
+ loadTiles: function(projection2, callback) {
+ if (_off)
+ return;
+ var tiles = tiler5.zoomExtent([_tileZoom4, _tileZoom4]).getTiles(projection2);
+ var hadRequests = hasInflightRequests(_tileCache);
+ abortUnwantedRequests4(_tileCache, tiles);
+ if (hadRequests && !hasInflightRequests(_tileCache)) {
+ dispatch7.call("loaded");
+ }
+ tiles.forEach(function(tile) {
+ this.loadTile(tile, callback);
+ }, this);
+ },
+ // Load a single data tile
+ // GET /api/0.6/map?bbox=
+ loadTile: function(tile, callback) {
+ if (_off)
+ return;
+ if (_tileCache.loaded[tile.id] || _tileCache.inflight[tile.id])
+ return;
+ if (!hasInflightRequests(_tileCache)) {
+ dispatch7.call("loading");
+ }
+ var path = "/api/0.6/map.json?bbox=";
+ var options2 = { skipSeen: true };
+ _tileCache.inflight[tile.id] = this.loadFromAPI(
+ path + tile.extent.toParam(),
+ tileCallback,
+ options2
+ );
+ function tileCallback(err, parsed) {
+ delete _tileCache.inflight[tile.id];
+ if (!err) {
+ delete _tileCache.toLoad[tile.id];
+ _tileCache.loaded[tile.id] = true;
+ var bbox2 = tile.extent.bbox();
+ bbox2.id = tile.id;
+ _tileCache.rtree.insert(bbox2);
+ }
+ if (callback) {
+ callback(err, Object.assign({ data: parsed }, tile));
+ }
+ if (!hasInflightRequests(_tileCache)) {
+ dispatch7.call("loaded");
+ }
+ }
+ },
+ isDataLoaded: function(loc) {
+ var bbox2 = { minX: loc[0], minY: loc[1], maxX: loc[0], maxY: loc[1] };
+ return _tileCache.rtree.collides(bbox2);
+ },
+ // load the tile that covers the given `loc`
+ loadTileAtLoc: function(loc, callback) {
+ if (Object.keys(_tileCache.toLoad).length > 50)
+ return;
+ var k = geoZoomToScale(_tileZoom4 + 1);
+ var offset = geoRawMercator().scale(k)(loc);
+ var projection2 = geoRawMercator().transform({ k, x: -offset[0], y: -offset[1] });
+ var tiles = tiler5.zoomExtent([_tileZoom4, _tileZoom4]).getTiles(projection2);
+ 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(projection2, noteOptions) {
+ noteOptions = Object.assign({ limit: 1e4, closed: 7 }, noteOptions);
+ if (_off)
+ return;
+ var that = this;
+ var path = "/api/0.6/notes?limit=" + noteOptions.limit + "&closed=" + noteOptions.closed + "&bbox=";
+ var throttleLoadUsers = throttle_default(function() {
+ var uids = Object.keys(_userCache.toLoad);
+ if (!uids.length)
+ return;
+ that.loadUsers(uids, function() {
+ });
+ }, 750);
+ var tiles = tiler5.zoomExtent([_noteZoom, _noteZoom]).getTiles(projection2);
+ abortUnwantedRequests4(_noteCache, tiles);
+ tiles.forEach(function(tile) {
+ if (_noteCache.loaded[tile.id] || _noteCache.inflight[tile.id])
+ return;
+ var options2 = { skipSeen: false };
+ _noteCache.inflight[tile.id] = that.loadFromAPI(
+ path + tile.extent.toParam(),
+ function(err) {
+ delete _noteCache.inflight[tile.id];
+ if (!err) {
+ _noteCache.loaded[tile.id] = true;
+ }
+ throttleLoadUsers();
+ dispatch7.call("loadedNotes");
+ },
+ options2
+ );
+ });
+ },
+ // Create a note
+ // POST /api/0.6/notes?params
+ postNoteCreate: function(note, callback) {
+ if (!this.authenticated()) {
+ return callback({ message: "Not Authenticated", status: -3 }, note);
+ }
+ if (_noteCache.inflightPost[note.id]) {
+ return callback({ message: "Note update already inflight", status: -2 }, note);
+ }
+ if (!note.loc[0] || !note.loc[1] || !note.newComment)
+ return;
+ var comment = note.newComment;
+ if (note.newCategory && note.newCategory !== "None") {
+ comment += " #" + note.newCategory;
+ }
+ 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 },
+ wrapcb(this, done, _connectionID)
+ );
+ function done(err, xml) {
+ delete _noteCache.inflightPost[note.id];
+ if (err) {
+ return callback(err);
+ }
+ this.removeNote(note);
+ var options2 = { skipSeen: false };
+ return parseXML(xml, function(err2, results) {
+ if (err2) {
+ return callback(err2);
+ } else {
+ return callback(void 0, results[0]);
+ }
+ }, options2);
+ }
+ },
+ // 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(note, newStatus, callback) {
+ if (!this.authenticated()) {
+ return callback({ message: "Not Authenticated", status: -3 }, note);
+ }
+ if (_noteCache.inflightPost[note.id]) {
+ return callback({ message: "Note update already inflight", status: -2 }, note);
+ }
+ var action;
+ if (note.status !== "closed" && newStatus === "closed") {
+ action = "close";
+ } else if (note.status !== "open" && newStatus === "open") {
+ action = "reopen";
+ } else {
+ action = "comment";
+ if (!note.newComment)
+ return;
+ }
+ var path = "/api/0.6/notes/" + note.id + "/" + action;
+ if (note.newComment) {
+ path += "?" + utilQsString({ text: note.newComment });
+ }
+ _noteCache.inflightPost[note.id] = oauth.xhr(
+ { method: "POST", path },
+ wrapcb(this, done, _connectionID)
+ );
+ function done(err, xml) {
+ delete _noteCache.inflightPost[note.id];
+ if (err) {
+ return callback(err);
+ }
+ this.removeNote(note);
+ if (action === "close") {
+ _noteCache.closed[note.id] = true;
+ } else if (action === "reopen") {
+ delete _noteCache.closed[note.id];
+ }
+ var options2 = { skipSeen: false };
+ return parseXML(xml, function(err2, results) {
+ if (err2) {
+ return callback(err2);
+ } else {
+ return callback(void 0, results[0]);
+ }
+ }, options2);
+ }
+ },
+ /* connection options for source switcher (optional) */
+ apiConnections: function(val) {
+ if (!arguments.length)
+ return _apiConnections;
+ _apiConnections = val;
+ return this;
+ },
+ switch: function(newOptions) {
+ urlroot = newOptions.url;
+ var oldOptions = utilObjectOmit(oauth.options(), "access_token");
+ oauth.options(Object.assign(oldOptions, newOptions));
+ this.reset();
+ this.userChangesets(function() {
+ });
+ dispatch7.call("change");
+ return this;
+ },
+ toggle: function(val) {
+ _off = !val;
+ return this;
+ },
+ isChangesetInflight: function() {
+ 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(obj) {
+ function cloneCache(source) {
+ var target = {};
+ Object.keys(source).forEach(function(k) {
+ if (k === "rtree") {
+ target.rtree = new import_rbush9.default().fromJSON(source.rtree.toJSON());
+ } else if (k === "note") {
+ target.note = {};
+ Object.keys(source.note).forEach(function(id2) {
+ target.note[id2] = osmNote(source.note[id2]);
+ });
+ } else {
+ target[k] = JSON.parse(JSON.stringify(source[k]));
+ }
+ });
+ return target;
+ }
+ if (!arguments.length) {
+ return {
+ tile: cloneCache(_tileCache),
+ note: cloneCache(_noteCache),
+ user: cloneCache(_userCache)
+ };
+ }
+ if (obj === "get") {
+ return {
+ tile: _tileCache,
+ note: _noteCache,
+ user: _userCache
+ };
+ }
+ if (obj.tile) {
+ _tileCache = obj.tile;
+ _tileCache.inflight = {};
+ }
+ if (obj.note) {
+ _noteCache = obj.note;
+ _noteCache.inflight = {};
+ _noteCache.inflightPost = {};
+ }
+ if (obj.user) {
+ _userCache = obj.user;
+ }
+ return this;
+ },
+ logout: function() {
+ _userChangesets = void 0;
+ _userDetails = void 0;
+ oauth.logout();
+ dispatch7.call("change");
+ return this;
+ },
+ authenticated: function() {
+ return oauth.authenticated();
+ },
+ authenticate: function(callback) {
+ var that = this;
+ var cid = _connectionID;
+ _userChangesets = void 0;
+ _userDetails = void 0;
+ function done(err, res) {
+ if (err) {
+ if (callback)
+ callback(err);
+ return;
+ }
+ if (that.getConnectionId() !== cid) {
+ if (callback)
+ callback({ message: "Connection Switched", status: -1 });
+ return;
+ }
+ _rateLimitError = void 0;
+ dispatch7.call("change");
+ if (callback)
+ callback(err, res);
+ that.userChangesets(function() {
+ });
+ }
+ oauth.authenticate(done);
+ },
+ imageryBlocklists: function() {
+ return _imageryBlocklists;
+ },
+ tileZoom: function(val) {
+ if (!arguments.length)
+ return _tileZoom4;
+ _tileZoom4 = val;
+ return this;
+ },
+ // get all cached notes covering the viewport
+ notes: function(projection2) {
+ var viewport = projection2.clipExtent();
+ var min3 = [viewport[0][0], viewport[1][1]];
+ var max3 = [viewport[1][0], viewport[0][1]];
+ var bbox2 = geoExtent(projection2.invert(min3), projection2.invert(max3)).bbox();
+ return _noteCache.rtree.search(bbox2).map(function(d) {
+ return d.data;
+ });
+ },
+ // get a single note from the cache
+ getNote: function(id2) {
+ return _noteCache.note[id2];
+ },
+ // remove a single note from the cache
+ removeNote: function(note) {
+ if (!(note instanceof osmNote) || !note.id)
+ return;
+ delete _noteCache.note[note.id];
+ updateRtree4(encodeNoteRtree(note), false);
+ },
+ // replace a single note in the cache
+ replaceNote: function(note) {
+ if (!(note instanceof osmNote) || !note.id)
+ return;
+ _noteCache.note[note.id] = note;
+ updateRtree4(encodeNoteRtree(note), true);
+ return note;
+ },
+ // Get an array of note IDs closed during this session.
+ // Used to populate `closed:note` changeset tag
+ getClosedIDs: function() {
+ return Object.keys(_noteCache.closed).sort();
+ }
+ };
+
+ // modules/services/osm_wikibase.js
+ var apibase3 = "https://wiki.openstreetmap.org/w/api.php";
+ var _inflight2 = {};
+ var _wikibaseCache = {};
+ var _localeIDs = { en: false };
+ var debouncedRequest = debounce_default(request, 500, { leading: false });
+ function request(url, callback) {
+ if (_inflight2[url])
+ return;
+ var controller = new AbortController();
+ _inflight2[url] = controller;
+ json_default(url, { signal: controller.signal }).then(function(result) {
+ delete _inflight2[url];
+ if (callback)
+ callback(null, result);
+ }).catch(function(err) {
+ delete _inflight2[url];
+ if (err.name === "AbortError")
+ return;
+ if (callback)
+ callback(err.message);
+ });
+ }
+ var osm_wikibase_default = {
+ init: function() {
+ _inflight2 = {};
+ _wikibaseCache = {};
+ _localeIDs = {};
+ },
+ reset: function() {
+ Object.values(_inflight2).forEach(function(controller) {
+ controller.abort();
+ });
+ _inflight2 = {};
+ },
+ /**
+ * 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(entity, property, langCode) {
+ if (!entity.claims[property])
+ return void 0;
+ var locale2 = _localeIDs[langCode];
+ var preferredPick, localePick;
+ entity.claims[property].forEach(function(stmt) {
+ if (!preferredPick && stmt.rank === "preferred") {
+ preferredPick = stmt;
+ }
+ if (locale2 && stmt.qualifiers && stmt.qualifiers.P26 && stmt.qualifiers.P26[0].datavalue.value.id === locale2) {
+ localePick = stmt;
+ }
+ });
+ var result = localePick || preferredPick;
+ if (result) {
+ var datavalue = result.mainsnak.datavalue;
+ return datavalue.type === "wikibase-entityid" ? datavalue.value.id : datavalue.value;
+ } else {
+ return void 0;
+ }
+ },
+ /**
+ * 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(entity, property) {
+ if (!entity || !entity.claims[property])
+ return void 0;
+ return entity.claims[property].reduce(function(acc, obj) {
+ var value = obj.mainsnak.datavalue.value;
+ acc[value.language] = value.text;
+ return acc;
+ }, {});
+ },
+ toSitelink: function(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(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;
+ if (params.langCodes) {
+ params.langCodes.forEach(function(langCode) {
+ if (_localeIDs[langCode] === void 0) {
+ localeSitelink = ("Locale:" + langCode).replace(/_/g, " ").trim();
+ titles.push(localeSitelink);
+ }
+ });
+ }
+ if (rtypeSitelink) {
+ if (_wikibaseCache[rtypeSitelink]) {
+ result.rtype = _wikibaseCache[rtypeSitelink];
+ } else {
+ titles.push(rtypeSitelink);
+ }
+ }
+ if (keySitelink) {
+ if (_wikibaseCache[keySitelink]) {
+ result.key = _wikibaseCache[keySitelink];
+ } else {
+ titles.push(keySitelink);
+ }
+ }
+ if (tagSitelink) {
+ if (_wikibaseCache[tagSitelink]) {
+ result.tag = _wikibaseCache[tagSitelink];
+ } else {
+ titles.push(tagSitelink);
+ }
+ }
+ if (!titles.length) {
+ return callback(null, result);
+ }
+ 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,
+ };
+ var url = apibase3 + "?" + 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;
+ 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);
+ }
+ }
+ });
+ if (localeSitelink) {
+ that.addLocale(params.langCodes[0], localeID);
+ }
+ 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(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;
+ }
+ var entity = data.rtype || data.tag || data.key;
+ if (!entity) {
+ callback("No entity");
+ return;
+ }
+ var i2;
+ var description;
+ for (i2 in langCodes) {
+ let code2 = langCodes[i2];
+ if (entity.descriptions[code2] && entity.descriptions[code2].language === code2) {
+ description = entity.descriptions[code2];
+ break;
+ }
+ }
+ if (!description && Object.values(entity.descriptions).length)
+ description = Object.values(entity.descriptions)[0];
+ var result = {
+ title: entity.title,
+ description: description ? description.value : "",
+ descriptionLocaleCode: description ? description.language : "",
+ editURL: "https://wiki.openstreetmap.org/wiki/" + entity.title
+ };
+ if (entity.claims) {
+ var imageroot;
+ var image = that.claimToValue(entity, "P4", langCodes[0]);
+ 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";
+ }
+ }
+ if (imageroot && image) {
+ result.imageURL = imageroot + "?" + utilQsString({
+ title: "Special:Redirect/file/" + image,
+ width: 400
+ });
+ }
+ }
+ 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];
+ for (i2 in wikis) {
+ var wiki = wikis[i2];
+ for (var j2 in langCodes) {
+ var code = langCodes[j2];
+ var referenceId = langCodes[0].split("-")[0] !== "en" && code.split("-")[0] === "en" ? "inspector.wiki_en_reference" : "inspector.wiki_reference";
+ var info = getWikiInfo(wiki, code, referenceId);
+ if (info) {
+ result.wiki = info;
+ break;
+ }
+ }
+ if (result.wiki)
+ break;
+ }
+ callback(null, result);
+ function getWikiInfo(wiki2, langCode, tKey) {
+ if (wiki2 && wiki2[langCode]) {
+ return {
+ title: wiki2[langCode],
+ text: tKey,
+ url: "https://wiki.openstreetmap.org/wiki/" + wiki2[langCode]
+ };
+ }
+ }
+ });
+ },
+ addLocale: function(langCode, qid) {
+ _localeIDs[langCode] = qid;
+ },
+ apibase: function(val) {
+ if (!arguments.length)
+ return apibase3;
+ apibase3 = val;
+ return this;
+ }
+ };
+
+ // modules/services/streetside.js
+ var import_rbush10 = __toESM(require_rbush_min());
+
+ // modules/util/jsonp_request.js
+ var jsonpCache = {};
+ window.jsonpCache = jsonpCache;
+ function jsonpRequest(url, callback) {
+ var request3 = {
+ abort: function() {
+ }
+ };
+ 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);
+ request3.abort = function() {
+ window.clearTimeout(t);
+ };
+ }
+ return request3;
+ }
+ function rand() {
+ var chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz";
+ var c = "";
+ var i2 = -1;
+ while (++i2 < 15)
+ c += chars.charAt(Math.floor(Math.random() * 52));
+ return c;
+ }
+ function create2(url2) {
+ var e = url2.match(/callback=(\w+)/);
+ var c = e ? e[1] : rand();
+ jsonpCache[c] = function(data) {
+ if (jsonpCache[c]) {
+ callback(data);
+ }
+ finalize();
+ };
+ function finalize() {
+ delete jsonpCache[c];
+ script.remove();
+ }
+ request3.abort = finalize;
+ return "jsonpCache." + c;
+ }
+ var cb = create2(url);
+ var script = select_default2("head").append("script").attr("type", "text/javascript").attr("src", url.replace(/(\{|%7B)callback(\}|%7D)/, cb));
+ return request3;
+ }
+
+ // modules/services/streetside.js
+ 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 maxResults2 = 2e3;
+ var tileZoom2 = 16.5;
+ var tiler6 = utilTiler().zoomExtent([tileZoom2, tileZoom2]).skipNullIsland(true);
+ var dispatch8 = dispatch_default("loadedImages", "viewerChanged");
+ var minHfov = 10;
+ var maxHfov = 90;
+ var defaultHfov = 45;
+ var _hires = false;
+ var _resolution = 512;
+ var _currScene = 0;
+ var _ssCache;
+ var _pannellumViewer;
+ var _sceneOptions = {
+ showFullscreenCtrl: false,
+ autoLoad: true,
+ compass: true,
+ yaw: 0,
+ minHfov,
+ maxHfov,
+ hfov: defaultHfov,
+ type: "cubemap",
+ cubeMap: []
+ };
+ var _loadViewerPromise3;
+ function abortRequest6(i2) {
+ i2.abort();
+ }
+ function localeTimestamp(s) {
+ if (!s)
+ return null;
+ const options2 = { day: "numeric", month: "short", year: "numeric" };
+ const d = new Date(s);
+ if (isNaN(d.getTime()))
+ return null;
+ return d.toLocaleString(_mainLocalizer.localeCode(), options2);
+ }
+ function loadTiles3(which, url, projection2, margin) {
+ const tiles = tiler6.margin(margin).getTiles(projection2);
+ const cache = _ssCache[which];
+ Object.keys(cache.inflight).forEach((k) => {
+ const wanted = tiles.find((tile) => k.indexOf(tile.id + ",") === 0);
+ if (!wanted) {
+ abortRequest6(cache.inflight[k]);
+ delete cache.inflight[k];
+ }
+ });
+ tiles.forEach((tile) => loadNextTilePage2(which, url, tile));
+ }
+ function loadNextTilePage2(which, url, tile) {
+ const cache = _ssCache[which];
+ const nextPage = cache.nextPage[tile.id] || 0;
+ const id2 = tile.id + "," + String(nextPage);
+ if (cache.loaded[id2] || cache.inflight[id2])
+ return;
+ cache.inflight[id2] = getBubbles(url, tile, (bubbles) => {
+ cache.loaded[id2] = true;
+ delete cache.inflight[id2];
+ if (!bubbles)
+ return;
+ bubbles.shift();
+ const features = bubbles.map((bubble) => {
+ if (cache.points[bubble.id])
+ return null;
+ const loc = [bubble.lo, bubble.la];
+ const d = {
+ 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;
+ if (bubble.pr === void 0) {
+ cache.leaders.push(bubble.id);
+ }
+ return {
+ minX: loc[0],
+ minY: loc[1],
+ maxX: loc[0],
+ maxY: loc[1],
+ data: d
+ };
+ }).filter(Boolean);
+ cache.rtree.load(features);
+ connectSequences();
+ if (which === "bubbles") {
+ dispatch8.call("loadedImages");
+ }
+ });
+ }
+ function connectSequences() {
+ let cache = _ssCache.bubbles;
+ let keepLeaders = [];
+ for (let i2 = 0; i2 < cache.leaders.length; i2++) {
+ let bubble = cache.points[cache.leaders[i2]];
+ let seen = {};
+ let sequence = { key: bubble.key, bubbles: [] };
+ let complete = false;
+ do {
+ sequence.bubbles.push(bubble);
+ seen[bubble.key] = true;
+ if (bubble.ne === void 0) {
+ complete = true;
+ } else {
+ bubble = cache.points[bubble.ne];
+ }
+ } while (bubble && !seen[bubble.key] && !complete);
+ if (complete) {
+ _ssCache.sequences[sequence.key] = sequence;
+ for (let j2 = 0; j2 < sequence.bubbles.length; j2++) {
+ sequence.bubbles[j2].sequenceKey = sequence.key;
+ }
+ 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((d) => d.loc)
+ };
+ } else {
+ keepLeaders.push(cache.leaders[i2]);
+ }
+ }
+ cache.leaders = keepLeaders;
+ }
+ function getBubbles(url, tile, callback) {
+ let rect = tile.extent.rectangle();
+ let urlForRequest = url + utilQsString({
+ n: rect[3],
+ s: rect[1],
+ e: rect[2],
+ w: rect[0],
+ c: maxResults2,
+ appkey: bubbleAppKey,
+ jsCallback: "{callback}"
+ });
+ return jsonpRequest(urlForRequest, (data) => {
+ if (!data || data.error) {
+ callback(null);
+ } else {
+ callback(data);
+ }
+ });
+ }
+ function partitionViewport3(projection2) {
+ let z = geoScaleToZoom(projection2.scale());
+ let z2 = Math.ceil(z * 2) / 2 + 2.5;
+ let tiler8 = utilTiler().zoomExtent([z2, z2]);
+ return tiler8.getTiles(projection2).map((tile) => tile.extent);
+ }
+ function searchLimited3(limit, projection2, rtree) {
+ limit = limit || 5;
+ return partitionViewport3(projection2).reduce((result, extent) => {
+ let found = rtree.search(extent.bbox()).slice(0, limit).map((d) => d.data);
+ return found.length ? result.concat(found) : result;
+ }, []);
+ }
+ function loadImage(imgInfo) {
+ return new Promise((resolve) => {
+ let img = new Image();
+ img.onload = () => {
+ let canvas = document.getElementById("ideditor-canvas" + imgInfo.face);
+ let ctx = canvas.getContext("2d");
+ ctx.drawImage(img, imgInfo.x, imgInfo.y);
+ resolve({ imgInfo, status: "ok" });
+ };
+ img.onerror = () => {
+ resolve({ data: imgInfo, status: "error" });
+ };
+ img.setAttribute("crossorigin", "");
+ img.src = imgInfo.url;
+ });
+ }
+ function loadCanvas(imageGroup) {
+ return Promise.all(imageGroup.map(loadImage)).then((data) => {
+ let canvas = document.getElementById("ideditor-canvas" + data[0].imgInfo.face);
+ const which = { "01": 0, "02": 1, "03": 2, "10": 3, "11": 4, "12": 5 };
+ let face = data[0].imgInfo.face;
+ _sceneOptions.cubeMap[which[face]] = canvas.toDataURL("image/jpeg", 1);
+ return { status: "loadCanvas for face " + data[0].imgInfo.face + "ok" };
+ });
+ }
+ function loadFaces(faceGroup) {
+ return Promise.all(faceGroup.map(loadCanvas)).then(() => {
+ return { status: "loadFaces done" };
+ });
+ }
+ function setupCanvas(selection2, reset) {
+ if (reset) {
+ selection2.selectAll("#ideditor-stitcher-canvases").remove();
+ }
+ selection2.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", (d) => "ideditor-" + d).attr("width", _resolution).attr("height", _resolution);
+ }
+ function qkToXY(qk) {
+ let x = 0;
+ let y = 0;
+ let scale = 256;
+ for (let i2 = qk.length; i2 > 0; i2--) {
+ const key = qk[i2 - 1];
+ x += +(key === "1" || key === "3") * scale;
+ y += +(key === "2" || key === "3") * scale;
+ scale *= 2;
+ }
+ return [x, y];
+ }
+ function getQuadKeys() {
+ let dim = _resolution / 256;
+ let 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 {
+ quadKeys = [
+ "0",
+ "1",
+ "2",
+ "3"
+ ];
+ }
+ return quadKeys;
+ }
+ var streetside_default = {
+ /**
+ * init() initialize streetside.
+ */
+ init: function() {
+ if (!_ssCache) {
+ this.reset();
+ }
+ this.event = utilRebind(this, dispatch8, "on");
+ },
+ /**
+ * reset() reset the cache.
+ */
+ reset: function() {
+ if (_ssCache) {
+ Object.values(_ssCache.bubbles.inflight).forEach(abortRequest6);
+ }
+ _ssCache = {
+ bubbles: { inflight: {}, loaded: {}, nextPage: {}, rtree: new import_rbush10.default(), points: {}, leaders: [] },
+ sequences: {}
+ };
+ },
+ /**
+ * bubbles()
+ */
+ bubbles: function(projection2) {
+ const limit = 5;
+ return searchLimited3(limit, projection2, _ssCache.bubbles.rtree);
+ },
+ cachedImage: function(imageKey) {
+ return _ssCache.bubbles.points[imageKey];
+ },
+ sequences: function(projection2) {
+ const viewport = projection2.clipExtent();
+ const min3 = [viewport[0][0], viewport[1][1]];
+ const max3 = [viewport[1][0], viewport[0][1]];
+ const bbox2 = geoExtent(projection2.invert(min3), projection2.invert(max3)).bbox();
+ let seen = {};
+ let results = [];
+ _ssCache.bubbles.rtree.search(bbox2).forEach((d) => {
+ const key = d.data.sequenceKey;
+ if (key && !seen[key]) {
+ seen[key] = true;
+ results.push(_ssCache.sequences[key].geojson);
+ }
+ });
+ return results;
+ },
+ /**
+ * loadBubbles()
+ */
+ loadBubbles: function(projection2, margin) {
+ if (margin === void 0)
+ margin = 2;
+ loadTiles3("bubbles", bubbleApi, projection2, margin);
+ },
+ viewer: function() {
+ return _pannellumViewer;
+ },
+ initViewer: function() {
+ if (!window.pannellum)
+ return;
+ if (_pannellumViewer)
+ return;
+ _currScene += 1;
+ const sceneID = _currScene.toString();
+ const options2 = {
+ "default": { firstScene: sceneID },
+ scenes: {}
+ };
+ options2.scenes[sceneID] = _sceneOptions;
+ _pannellumViewer = window.pannellum.viewer("ideditor-viewer-streetside", options2);
+ },
+ ensureViewerLoaded: function(context) {
+ if (_loadViewerPromise3)
+ return _loadViewerPromise3;
+ let wrap2 = context.container().select(".photoviewer").selectAll(".ms-wrapper").data([0]);
+ let wrapEnter = wrap2.enter().append("div").attr("class", "photo-wrapper ms-wrapper").classed("hide", true);
+ let that = this;
+ let pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ wrapEnter.append("div").attr("id", "ideditor-viewer-streetside").on(pointerPrefix + "down.streetside", () => {
+ select_default2(window).on(pointerPrefix + "move.streetside", () => {
+ dispatch8.call("viewerChanged");
+ }, true);
+ }).on(pointerPrefix + "up.streetside pointercancel.streetside", () => {
+ select_default2(window).on(pointerPrefix + "move.streetside", null);
+ let t = timer((elapsed) => {
+ dispatch8.call("viewerChanged");
+ if (elapsed > 2e3) {
+ t.stop();
+ }
+ });
+ }).append("div").attr("class", "photo-attribution fillD");
+ let controlsEnter = wrapEnter.append("div").attr("class", "photo-controls-wrap").append("div").attr("class", "photo-controls");
+ controlsEnter.append("button").on("click.back", step(-1)).text("\u25C4");
+ controlsEnter.append("button").on("click.forward", step(1)).text("\u25BA");
+ wrap2 = wrap2.merge(wrapEnter).call(setupCanvas, true);
+ context.ui().photoviewer.on("resize.streetside", () => {
+ if (_pannellumViewer) {
+ _pannellumViewer.resize();
+ }
+ });
+ _loadViewerPromise3 = new Promise((resolve, reject) => {
+ let loadedCount = 0;
+ function loaded() {
+ loadedCount += 1;
+ if (loadedCount === 2)
+ resolve();
+ }
+ const head = select_default2("head");
+ 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();
+ });
+ 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() {
+ _loadViewerPromise3 = null;
+ });
+ return _loadViewerPromise3;
+ function step(stepBy) {
+ return () => {
+ let viewer = context.container().select(".photoviewer");
+ let selected = viewer.empty() ? void 0 : viewer.datum();
+ if (!selected)
+ return;
+ let nextID = stepBy === 1 ? selected.ne : selected.pr;
+ let yaw = _pannellumViewer.getYaw();
+ let ca = selected.ca + yaw;
+ let origin = selected.loc;
+ const meters = 35;
+ let p1 = [
+ origin[0] + geoMetersToLon(meters / 5, origin[1]),
+ origin[1]
+ ];
+ let p2 = [
+ origin[0] + geoMetersToLon(meters / 2, origin[1]),
+ origin[1] + geoMetersToLat(meters)
+ ];
+ let p3 = [
+ origin[0] - geoMetersToLon(meters / 2, origin[1]),
+ origin[1] + geoMetersToLat(meters)
+ ];
+ let p4 = [
+ origin[0] - geoMetersToLon(meters / 5, origin[1]),
+ origin[1]
+ ];
+ let poly = [p1, p2, p3, p4, p1];
+ let angle2 = (stepBy === 1 ? ca : ca + 180) * (Math.PI / 180);
+ poly = geoRotate(poly, -angle2, origin);
+ let extent = poly.reduce((extent2, point2) => {
+ return extent2.extend(geoExtent(point2));
+ }, geoExtent());
+ let minDist = Infinity;
+ _ssCache.bubbles.rtree.search(extent.bbox()).forEach((d) => {
+ if (d.data.key === selected.key)
+ return;
+ if (!geoPointInPolygon(d.data.loc, poly))
+ return;
+ let dist = geoVecLength(d.data.loc, selected.loc);
+ let theta = selected.ca - d.data.ca;
+ let minTheta = Math.min(Math.abs(theta), 360 - Math.abs(theta));
+ if (minTheta > 20) {
+ dist += 5;
+ }
+ if (dist < minDist) {
+ nextID = d.data.key;
+ minDist = dist;
+ }
+ });
+ let nextBubble = nextID && that.cachedImage(nextID);
+ if (!nextBubble)
+ return;
+ context.map().centerEase(nextBubble.loc);
+ that.selectImage(context, nextBubble.key).yaw(yaw).showViewer(context);
+ };
+ }
+ },
+ yaw: function(yaw) {
+ if (typeof yaw !== "number")
+ return yaw;
+ _sceneOptions.yaw = yaw;
+ return this;
+ },
+ /**
+ * showViewer()
+ */
+ showViewer: function(context) {
+ let wrap2 = context.container().select(".photoviewer").classed("hide", false);
+ let isHidden = wrap2.selectAll(".photo-wrapper.ms-wrapper.hide").size();
+ if (isHidden) {
+ wrap2.selectAll(".photo-wrapper:not(.ms-wrapper)").classed("hide", true);
+ wrap2.selectAll(".photo-wrapper.ms-wrapper").classed("hide", false);
+ }
+ return this;
+ },
+ /**
+ * hideViewer()
+ */
+ hideViewer: function(context) {
+ let 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);
+ },
+ /**
+ * selectImage().
+ */
+ selectImage: function(context, key) {
+ let that = this;
+ let d = this.cachedImage(key);
+ let viewer = context.container().select(".photoviewer");
+ if (!viewer.empty())
+ viewer.datum(d);
+ this.setStyles(context, null, true);
+ let wrap2 = context.container().select(".photoviewer .ms-wrapper");
+ let attribution = wrap2.selectAll(".photo-attribution").html("");
+ wrap2.selectAll(".pnlm-load-box").style("display", "block");
+ if (!d)
+ return this;
+ this.updateUrlImage(key);
+ _sceneOptions.northOffset = d.ca;
+ let line1 = attribution.append("div").attr("class", "attribution-row");
+ const hiresDomId = utilUniqueDomId("streetside-hires");
+ let 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", (d3_event) => {
+ d3_event.stopPropagation();
+ _hires = !_hires;
+ _resolution = _hires ? 1024 : 512;
+ wrap2.call(setupCanvas, true);
+ let 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").call(_t.append("streetside.hires"));
+ let captureInfo = line1.append("div").attr("class", "attribution-capture-info");
+ if (d.captured_by) {
+ const yyyy = new Date().getFullYear();
+ captureInfo.append("a").attr("class", "captured_by").attr("target", "_blank").attr("href", "https://www.microsoft.com/en-us/maps/streetside").text("\xA9" + yyyy + " Microsoft");
+ captureInfo.append("span").text("|");
+ }
+ if (d.captured_at) {
+ captureInfo.append("span").attr("class", "captured_at").text(localeTimestamp(d.captured_at));
+ }
+ let 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").call(_t.append("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").call(_t.append("streetside.report"));
+ let bubbleIdQuadKey = d.key.toString(4);
+ const paddingNeeded = 16 - bubbleIdQuadKey.length;
+ for (let i2 = 0; i2 < paddingNeeded; i2++) {
+ bubbleIdQuadKey = "0" + bubbleIdQuadKey;
+ }
+ const imgUrlPrefix = streetsideImagesApi + "hs" + bubbleIdQuadKey;
+ const imgUrlSuffix = ".jpg?g=6338&n=z";
+ const faceKeys = ["01", "02", "03", "10", "11", "12"];
+ let quadKeys = getQuadKeys();
+ let faces = faceKeys.map((faceKey) => {
+ return quadKeys.map((quadKey) => {
+ const 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 {
+ _currScene += 1;
+ let sceneID = _currScene.toString();
+ _pannellumViewer.addScene(sceneID, _sceneOptions).loadScene(sceneID);
+ if (_currScene > 2) {
+ sceneID = (_currScene - 1).toString();
+ _pannellumViewer.removeScene(sceneID);
+ }
+ }
+ });
+ return this;
+ },
+ getSequenceKeyForBubble: function(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(context, hovered, reset) {
+ if (reset) {
+ context.container().selectAll(".viewfield-group").classed("highlighted", false).classed("hovered", false).classed("currentView", false);
+ context.container().selectAll(".sequence").classed("highlighted", false).classed("currentView", false);
+ }
+ let hoveredBubbleKey = hovered && hovered.key;
+ let hoveredSequenceKey = this.getSequenceKeyForBubble(hovered);
+ let hoveredSequence = hoveredSequenceKey && _ssCache.sequences[hoveredSequenceKey];
+ let hoveredBubbleKeys = hoveredSequence && hoveredSequence.bubbles.map((d) => d.key) || [];
+ let viewer = context.container().select(".photoviewer");
+ let selected = viewer.empty() ? void 0 : viewer.datum();
+ let selectedBubbleKey = selected && selected.key;
+ let selectedSequenceKey = this.getSequenceKeyForBubble(selected);
+ let selectedSequence = selectedSequenceKey && _ssCache.sequences[selectedSequenceKey];
+ let selectedBubbleKeys = selectedSequence && selectedSequence.bubbles.map((d) => d.key) || [];
+ let highlightedBubbleKeys = utilArrayUnion(hoveredBubbleKeys, selectedBubbleKeys);
+ context.container().selectAll(".layer-streetside-images .viewfield-group").classed("highlighted", (d) => highlightedBubbleKeys.indexOf(d.key) !== -1).classed("hovered", (d) => d.key === hoveredBubbleKey).classed("currentView", (d) => d.key === selectedBubbleKey);
+ context.container().selectAll(".layer-streetside-images .sequence").classed("highlighted", (d) => d.properties.key === hoveredSequenceKey).classed("currentView", (d) => d.properties.key === selectedSequenceKey);
+ context.container().selectAll(".layer-streetside-images .viewfield-group .viewfield").attr("d", viewfieldPath);
+ function viewfieldPath() {
+ let d = this.parentNode.__data__;
+ 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 this;
+ },
+ updateUrlImage: function(imageKey) {
+ if (!window.mocha) {
+ var hash = utilStringQs(window.location.hash);
+ if (imageKey) {
+ hash.photo = "streetside/" + imageKey;
+ } else {
+ delete hash.photo;
+ }
+ window.location.replace("#" + utilQsString(hash, true));
+ }
+ },
+ /**
+ * cache().
+ */
+ cache: function() {
+ return _ssCache;
+ }
+ };
+
+ // modules/services/taginfo.js
+ var apibase4 = taginfoApiUrl;
+ var _inflight3 = {};
+ 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"
+ };
+ function sets(params, n2, o) {
+ if (params.geometry && o[params.geometry]) {
+ params[n2] = o[params.geometry];
+ }
+ return params;
+ }
+ function setFilter(params) {
+ return sets(params, "filter", tag_filters);
+ }
+ function setSort(params) {
+ return sets(params, "sortname", tag_sorts);
+ }
+ function setSortMembers(params) {
+ return sets(params, "sortname", tag_sort_members);
+ }
+ function clean(params) {
+ return utilObjectOmit(params, ["geometry", "debounce"]);
+ }
+ function filterKeys(type2) {
+ var count_type = type2 ? "count_" + type2 : "count_all";
+ return function(d) {
+ return Number(d[count_type]) > 2500 || d.in_wiki;
+ };
+ }
+ function filterMultikeys(prefix) {
+ return function(d) {
+ var re2 = new RegExp("^" + prefix + "(.*)$", "i");
+ var matches = d.key.match(re2) || [];
+ return matches.length === 2 && matches[1].indexOf(":") === -1;
+ };
+ }
+ function filterValues(allowUpperCase) {
+ return function(d) {
+ if (d.value.match(/[;,]/) !== null)
+ return false;
+ if (!allowUpperCase && d.value.match(/[A-Z*]/) !== null)
+ return false;
+ return d.count > 100 || d.in_wiki;
+ };
+ }
+ function filterRoles(geometry) {
+ return function(d) {
+ if (d.role === "")
+ return false;
+ if (d.role.match(/[A-Z*;,]/) !== null)
+ return false;
+ return Number(d[tag_members_fractions[geometry]]) > 0;
+ };
+ }
+ function valKey(d) {
+ return {
+ value: d.key,
+ title: d.key
+ };
+ }
+ function valKeyDescription(d) {
+ var obj = {
+ value: d.value,
+ title: d.description || d.value
+ };
+ return obj;
+ }
+ function roleKey(d) {
+ return {
+ value: d.role,
+ title: d.role
+ };
+ }
+ function sortKeys(a, b) {
+ return a.key.indexOf(":") === -1 && b.key.indexOf(":") !== -1 ? -1 : a.key.indexOf(":") !== -1 && b.key.indexOf(":") === -1 ? 1 : 0;
+ }
+ var debouncedRequest2 = debounce_default(request2, 300, { leading: false });
+ function request2(url, params, exactMatch, callback, loaded) {
+ if (_inflight3[url])
+ return;
+ if (checkCache(url, params, exactMatch, callback))
+ return;
+ var controller = new AbortController();
+ _inflight3[url] = controller;
+ json_default(url, { signal: controller.signal }).then(function(result) {
+ delete _inflight3[url];
+ if (loaded)
+ loaded(null, result);
+ }).catch(function(err) {
+ delete _inflight3[url];
+ if (err.name === "AbortError")
+ return;
+ if (loaded)
+ loaded(err.message);
+ });
+ }
+ function checkCache(url, params, exactMatch, callback) {
+ var rp = params.rp || 25;
+ var testQuery = params.query || "";
+ var testUrl = url;
+ do {
+ var hit = _taginfoCache[testUrl];
+ if (hit && (url === testUrl || hit.length < rp)) {
+ callback(null, hit);
+ return true;
+ }
+ if (exactMatch || !testQuery.length)
+ return false;
+ testQuery = testQuery.slice(0, -1);
+ testUrl = url.replace(/&query=(.*?)&/, "&query=" + testQuery + "&");
+ } while (testQuery.length >= 0);
+ return false;
+ }
+ var taginfo_default = {
+ init: function() {
+ _inflight3 = {};
+ _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,
+ via: true,
+ "bridge:name": true
+ };
+ 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;
+ _popularKeys[d.value] = true;
+ });
+ });
+ },
+ reset: function() {
+ Object.values(_inflight3).forEach(function(controller) {
+ controller.abort();
+ });
+ _inflight3 = {};
+ },
+ keys: function(params, callback) {
+ var doRequest = params.debounce ? debouncedRequest2 : request2;
+ params = clean(setSort(params));
+ params = Object.assign({
+ rp: 10,
+ sortname: "count_all",
+ sortorder: "desc",
+ page: 1,
+ lang: _mainLocalizer.languageCode()
+ }, params);
+ var url = apibase4 + "keys/all?" + utilQsString(params);
+ doRequest(url, params, false, callback, function(err, d) {
+ if (err) {
+ callback(err);
+ } else {
+ var f2 = filterKeys(params.filter);
+ var result = d.data.filter(f2).sort(sortKeys).map(valKey);
+ _taginfoCache[url] = result;
+ callback(null, result);
+ }
+ });
+ },
+ multikeys: function(params, callback) {
+ var doRequest = params.debounce ? debouncedRequest2 : request2;
+ 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 = apibase4 + "keys/all?" + utilQsString(params);
+ doRequest(url, params, true, callback, function(err, d) {
+ if (err) {
+ callback(err);
+ } else {
+ var f2 = filterMultikeys(prefix);
+ var result = d.data.filter(f2).map(valKey);
+ _taginfoCache[url] = result;
+ callback(null, result);
+ }
+ });
+ },
+ values: function(params, callback) {
+ var key = params.key;
+ if (key && _popularKeys[key]) {
+ callback(null, []);
+ return;
+ }
+ var doRequest = params.debounce ? debouncedRequest2 : request2;
+ params = clean(setSort(setFilter(params)));
+ params = Object.assign({
+ rp: 25,
+ sortname: "count_all",
+ sortorder: "desc",
+ page: 1,
+ lang: _mainLocalizer.languageCode()
+ }, params);
+ var url = apibase4 + "key/values?" + utilQsString(params);
+ doRequest(url, params, false, callback, function(err, d) {
+ if (err) {
+ callback(err);
+ } else {
+ var re2 = /network|taxon|genus|species|brand|grape_variety|royal_cypher|listed_status|booth|rating|stars|:output|_hours|_times|_ref|manufacturer|country|target|brewery|cai_scale/;
+ var allowUpperCase = re2.test(params.key);
+ var f2 = filterValues(allowUpperCase);
+ var result = d.data.filter(f2).map(valKeyDescription);
+ _taginfoCache[url] = result;
+ callback(null, result);
+ }
+ });
+ },
+ roles: function(params, callback) {
+ var doRequest = params.debounce ? debouncedRequest2 : request2;
+ 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 = apibase4 + "relation/roles?" + utilQsString(params);
+ doRequest(url, params, true, callback, function(err, d) {
+ if (err) {
+ callback(err);
+ } else {
+ var f2 = filterRoles(geometry);
+ var result = d.data.filter(f2).map(roleKey);
+ _taginfoCache[url] = result;
+ callback(null, result);
+ }
+ });
+ },
+ docs: function(params, callback) {
+ var doRequest = params.debounce ? debouncedRequest2 : request2;
+ 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 = apibase4 + 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(_) {
+ if (!arguments.length)
+ return apibase4;
+ apibase4 = _;
+ return this;
+ }
+ };
+
+ // modules/services/vector_tile.js
+ var import_fast_deep_equal11 = __toESM(require_fast_deep_equal());
+ var import_fast_json_stable_stringify2 = __toESM(require_fast_json_stable_stringify());
+ var import_polygon_clipping2 = __toESM(require_polygon_clipping_umd());
+ var import_pbf2 = __toESM(require_pbf());
+ var import_vector_tile2 = __toESM(require_vector_tile());
+ var tiler7 = utilTiler().tileSize(512).margin(1);
+ var dispatch9 = dispatch_default("loadedData");
+ var _vtCache;
+ function abortRequest7(controller) {
+ controller.abort();
+ }
+ function vtToGeoJSON(data, tile, mergeCache) {
+ var vectorTile = new import_vector_tile2.default.VectorTile(new import_pbf2.default(data));
+ var layers = Object.keys(vectorTile.layers);
+ if (!Array.isArray(layers)) {
+ layers = [layers];
+ }
+ var features = [];
+ layers.forEach(function(layerID) {
+ var layer = vectorTile.layers[layerID];
+ if (layer) {
+ for (var i2 = 0; i2 < layer.length; i2++) {
+ var feature3 = layer.feature(i2).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+ var geometry = feature3.geometry;
+ if (geometry.type === "Polygon") {
+ geometry.type = "MultiPolygon";
+ geometry.coordinates = [geometry.coordinates];
+ }
+ var isClipped = false;
+ if (geometry.type === "MultiPolygon") {
+ var featureClip = bboxClip(feature3, tile.extent.rectangle());
+ if (!(0, import_fast_deep_equal11.default)(feature3.geometry, featureClip.geometry)) {
+ isClipped = true;
+ }
+ if (!feature3.geometry.coordinates.length)
+ continue;
+ if (!feature3.geometry.coordinates[0].length)
+ continue;
+ }
+ var featurehash = utilHashcode((0, import_fast_json_stable_stringify2.default)(feature3));
+ var propertyhash = utilHashcode((0, import_fast_json_stable_stringify2.default)(feature3.properties || {}));
+ feature3.__layerID__ = layerID.replace(/[^_a-zA-Z0-9\-]/g, "_");
+ feature3.__featurehash__ = featurehash;
+ feature3.__propertyhash__ = propertyhash;
+ features.push(feature3);
+ if (isClipped && geometry.type === "MultiPolygon") {
+ var merged = mergeCache[propertyhash];
+ if (merged && merged.length) {
+ var other = merged[0];
+ var coords = import_polygon_clipping2.default.union(
+ feature3.geometry.coordinates,
+ other.geometry.coordinates
+ );
+ if (!coords || !coords.length) {
+ continue;
+ }
+ merged.push(feature3);
+ for (var j2 = 0; j2 < merged.length; j2++) {
+ merged[j2].geometry.coordinates = coords;
+ merged[j2].__featurehash__ = featurehash;
+ }
+ } else {
+ mergeCache[propertyhash] = [feature3];
+ }
+ }
+ }
+ }
+ });
+ return features;
+ }
+ function loadTile2(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]).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);
+ }
+ source.loaded[tile.id] = [];
+ delete source.inflight[tile.id];
+ return response.arrayBuffer();
+ }).then(function(data) {
+ if (!data) {
+ throw new Error("No Data");
+ }
+ var z = tile.xyz[2];
+ if (!source.canMerge[z]) {
+ source.canMerge[z] = {};
+ }
+ source.loaded[tile.id] = vtToGeoJSON(data, tile, source.canMerge[z]);
+ dispatch9.call("loadedData");
+ }).catch(function() {
+ source.loaded[tile.id] = [];
+ delete source.inflight[tile.id];
+ });
+ }
+ var vector_tile_default = {
+ init: function() {
+ if (!_vtCache) {
+ this.reset();
+ }
+ this.event = utilRebind(this, dispatch9, "on");
+ },
+ reset: function() {
+ for (var sourceID in _vtCache) {
+ var source = _vtCache[sourceID];
+ if (source && source.inflight) {
+ Object.values(source.inflight).forEach(abortRequest7);
+ }
+ }
+ _vtCache = {};
+ },
+ addSource: function(sourceID, template) {
+ _vtCache[sourceID] = { template, inflight: {}, loaded: {}, canMerge: {} };
+ return _vtCache[sourceID];
+ },
+ data: function(sourceID, projection2) {
+ var source = _vtCache[sourceID];
+ if (!source)
+ return [];
+ var tiles = tiler7.getTiles(projection2);
+ var seen = {};
+ var results = [];
+ for (var i2 = 0; i2 < tiles.length; i2++) {
+ var features = source.loaded[tiles[i2].id];
+ if (!features || !features.length)
+ continue;
+ for (var j2 = 0; j2 < features.length; j2++) {
+ var feature3 = features[j2];
+ var hash = feature3.__featurehash__;
+ if (seen[hash])
+ continue;
+ seen[hash] = true;
+ results.push(Object.assign({}, feature3));
+ }
+ }
+ return results;
+ },
+ loadTiles: function(sourceID, template, projection2) {
+ var source = _vtCache[sourceID];
+ if (!source) {
+ source = this.addSource(sourceID, template);
+ }
+ var tiles = tiler7.getTiles(projection2);
+ Object.keys(source.inflight).forEach(function(k) {
+ var wanted = tiles.find(function(tile) {
+ return k === tile.id;
+ });
+ if (!wanted) {
+ abortRequest7(source.inflight[k]);
+ delete source.inflight[k];
+ }
+ });
+ tiles.forEach(function(tile) {
+ loadTile2(source, tile);
+ });
+ },
+ cache: function() {
+ return _vtCache;
+ }
+ };
+
+ // modules/services/wikidata.js
+ var apibase5 = "https://www.wikidata.org/w/api.php?";
+ var _wikidataCache = {};
+ var wikidata_default = {
+ init: function() {
+ },
+ reset: function() {
+ _wikidataCache = {};
+ },
+ // Search for Wikidata items matching the query
+ itemsForSearchQuery: function(query, callback) {
+ if (!query) {
+ if (callback)
+ callback("No query", {});
+ return;
+ }
+ var lang = this.languagesToQuery()[0];
+ var url = apibase5 + 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: "*"
+ });
+ json_default(url).then(function(result) {
+ if (result && result.error) {
+ throw new Error(result.error);
+ }
+ 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(lang, title, callback) {
+ if (!title) {
+ if (callback)
+ callback("No title", {});
+ return;
+ }
+ lang = lang || "en";
+ var url = apibase5 + utilQsString({
+ action: "wbgetentities",
+ format: "json",
+ formatversion: 2,
+ sites: lang.replace(/-/g, "_") + "wiki",
+ titles: title,
+ languages: "en",
+ // shrink response by filtering to one language
+ origin: "*"
+ });
+ json_default(url).then(function(result) {
+ if (result && result.error) {
+ throw new Error(result.error);
+ }
+ if (callback)
+ callback(null, result.entities || {});
+ }).catch(function(err) {
+ if (callback)
+ callback(err.message, {});
+ });
+ },
+ languagesToQuery: function() {
+ return _mainLocalizer.localeCodes().map(function(code) {
+ return code.toLowerCase();
+ }).filter(function(code) {
+ return code !== "en-us";
+ });
+ },
+ entityByQID: function(qid, callback) {
+ if (!qid) {
+ callback("No qid", {});
+ return;
+ }
+ if (_wikidataCache[qid]) {
+ if (callback)
+ callback(null, _wikidataCache[qid]);
+ return;
+ }
+ var langs = this.languagesToQuery();
+ var url = apibase5 + 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: "*"
+ });
+ json_default(url).then(function(result) {
+ if (result && result.error) {
+ throw new Error(result.error);
+ }
+ 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(params, callback) {
+ var langs = this.languagesToQuery();
+ this.entityByQID(params.qid, function(err, entity) {
+ if (err || !entity) {
+ callback(err || "No entity");
+ return;
+ }
+ var i2;
+ var description;
+ for (i2 in langs) {
+ let code = langs[i2];
+ if (entity.descriptions[code] && entity.descriptions[code].language === code) {
+ description = entity.descriptions[code];
+ break;
+ }
+ }
+ if (!description && Object.values(entity.descriptions).length)
+ description = Object.values(entity.descriptions)[0];
+ var result = {
+ title: entity.id,
+ description: description ? description.value : "",
+ descriptionLocaleCode: description ? description.language : "",
+ editURL: "https://www.wikidata.org/wiki/" + entity.id
+ };
+ if (entity.claims) {
+ var imageroot = "https://commons.wikimedia.org/w/index.php";
+ var props = ["P154", "P18"];
+ var prop, image;
+ for (i2 = 0; i2 < props.length; i2++) {
+ prop = entity.claims[props[i2]];
+ if (prop && Object.keys(prop).length > 0) {
+ image = prop[Object.keys(prop)[0]].mainsnak.datavalue.value;
+ if (image) {
+ result.imageURL = imageroot + "?" + utilQsString({
+ title: "Special:Redirect/file/" + image,
+ width: 400
+ });
+ break;
+ }
+ }
+ }
+ }
+ if (entity.sitelinks) {
+ var englishLocale = _mainLocalizer.languageCode().toLowerCase() === "en";
+ for (i2 = 0; i2 < langs.length; i2++) {
+ var w = langs[i2] + "wiki";
+ if (entity.sitelinks[w]) {
+ var title = entity.sitelinks[w].title;
+ var tKey = "inspector.wiki_reference";
+ if (!englishLocale && langs[i2] === "en") {
+ tKey = "inspector.wiki_en_reference";
+ }
+ result.wiki = {
+ title,
+ text: tKey,
+ url: "https://" + langs[i2] + ".wikipedia.org/wiki/" + title.replace(/ /g, "_")
+ };
+ break;
+ }
+ }
+ }
+ callback(null, result);
+ });
+ }
+ };
+
+ // modules/services/wikipedia.js
+ var endpoint = "https://en.wikipedia.org/w/api.php?";
+ var wikipedia_default = {
+ init: function() {
+ },
+ reset: function() {
+ },
+ search: function(lang, query, callback) {
+ if (!query) {
+ if (callback)
+ callback("No Query", []);
+ return;
+ }
+ lang = lang || "en";
+ var url = endpoint.replace("en", lang) + utilQsString({
+ action: "query",
+ list: "search",
+ srlimit: "10",
+ srinfo: "suggestion",
+ format: "json",
+ origin: "*",
+ srsearch: query
+ });
+ json_default(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");
+ }
+ 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(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
+ });
+ json_default(url).then(function(result) {
+ if (result && result.error) {
+ throw new Error(result.error);
+ } else if (!result || result.length < 2) {
+ throw new Error("No Results");
+ }
+ if (callback)
+ callback(null, result[1] || []);
+ }).catch(function(err) {
+ if (callback)
+ callback(err.message, []);
+ });
+ },
+ translations: function(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
+ });
+ json_default(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 (callback) {
+ var list = result.query.pages[Object.keys(result.query.pages)[0]];
+ var translations = {};
+ if (list && list.langlinks) {
+ list.langlinks.forEach(function(d) {
+ translations[d.lang] = d["*"];
+ });
+ }
+ callback(null, translations);
+ }
+ }).catch(function(err) {
+ if (callback)
+ callback(err.message);
+ });
+ }
+ };
+
+ // modules/services/index.js
+ var services = {
+ geocoder: nominatim_default,
+ keepRight: keepRight_default,
+ improveOSM: improveOSM_default,
+ osmose: osmose_default,
+ mapillary: mapillary_default,
+ nsi: nsi_default,
+ kartaview: kartaview_default,
+ osm: osm_default,
+ osmWikibase: osm_wikibase_default,
+ maprules: maprules_default,
+ streetside: streetside_default,
+ taginfo: taginfo_default,
+ vectorTile: vector_tile_default,
+ wikidata: wikidata_default,
+ wikipedia: wikipedia_default
+ };
+
+ // modules/modes/drag_note.js
+ function modeDragNote(context) {
+ var mode = {
+ id: "drag-note",
+ button: "browse"
+ };
+ var edit2 = behaviorEdit(context);
+ var _nudgeInterval;
+ var _lastLoc;
+ var _note;
+ function startNudge(d3_event, nudge) {
+ if (_nudgeInterval)
+ window.clearInterval(_nudgeInterval);
+ _nudgeInterval = window.setInterval(function() {
+ context.map().pan(nudge);
+ doMove(d3_event, nudge);
+ }, 50);
+ }
+ function stopNudge() {
+ if (_nudgeInterval) {
+ window.clearInterval(_nudgeInterval);
+ _nudgeInterval = null;
+ }
+ }
+ function origin(note) {
+ return context.projection(note.loc);
+ }
+ function start2(d3_event, note) {
+ _note = note;
+ var osm = services.osm;
+ if (osm) {
+ _note = osm.getNote(_note.id);
+ }
+ context.surface().selectAll(".note-" + _note.id).classed("active", true);
+ context.perform(actionNoop());
+ context.enter(mode);
+ context.selectedNoteID(_note.id);
+ }
+ function move(d3_event, entity, point2) {
+ d3_event.stopPropagation();
+ _lastLoc = context.projection.invert(point2);
+ doMove(d3_event);
+ var nudge = geoViewportEdge(point2, context.map().dimensions());
+ if (nudge) {
+ startNudge(d3_event, nudge);
+ } else {
+ stopNudge();
+ }
+ }
+ 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;
+ if (osm) {
+ osm.replaceNote(_note);
+ }
+ context.replace(actionNoop());
+ }
+ function end() {
+ context.replace(actionNoop());
+ 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", start2).on("move", move).on("end", end);
+ mode.enter = function() {
+ context.install(edit2);
+ };
+ mode.exit = function() {
+ context.ui().sidebar.hover.cancel();
+ context.uninstall(edit2);
+ context.surface().selectAll(".active").classed("active", false);
+ stopNudge();
+ };
+ mode.behavior = drag;
+ return mode;
+ }
+
+ // modules/modes/select_data.js
+ function modeSelectData(context, selectedDatum) {
+ var mode = {
+ id: "select-data",
+ button: "browse"
+ };
+ var keybinding = utilKeybinding("select-data");
+ var dataEditor = uiDataEditor(context);
+ var behaviors = [
+ behaviorBreathe(context),
+ behaviorHover(context),
+ behaviorSelect(context),
+ behaviorLasso(context),
+ modeDragNode(context).behavior,
+ modeDragNote(context).behavior
+ ];
+ function selectData(d3_event, drawn) {
+ var selection2 = context.surface().selectAll(".layer-mapdata .data" + selectedDatum.__featurehash__);
+ if (selection2.empty()) {
+ 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 {
+ selection2.classed("selected", true);
+ }
+ }
+ function esc() {
+ if (context.container().select(".combobox").size())
+ return;
+ context.enter(modeBrowse(context));
+ }
+ mode.zoomToSelected = function() {
+ var extent = geoExtent(bounds_default(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("\u238B", esc, true);
+ select_default2(document).call(keybinding);
+ selectData();
+ var sidebar = context.ui().sidebar;
+ sidebar.show(dataEditor.datum(selectedDatum));
+ var extent = geoExtent(bounds_default(selectedDatum));
+ sidebar.expand(sidebar.intersects(extent));
+ context.map().on("drawn.select-data", selectData);
+ };
+ mode.exit = function() {
+ behaviors.forEach(context.uninstall);
+ select_default2(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;
+ }
+
+ // modules/behavior/select.js
+ function behaviorSelect(context) {
+ var _tolerancePx = 4;
+ var _lastMouseEvent = null;
+ var _showMenu = false;
+ var _downPointers = {};
+ var _longPressTimeout = null;
+ var _lastInteractionType = null;
+ var _multiselectionPointerId = null;
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ function keydown(d3_event) {
+ if (d3_event.keyCode === 32) {
+ var activeNode = document.activeElement;
+ if (activeNode && (/* @__PURE__ */ new Set(["INPUT", "TEXTAREA"])).has(activeNode.nodeName))
+ return;
+ }
+ if (d3_event.keyCode === 93 || // context menu key
+ d3_event.keyCode === 32) {
+ d3_event.preventDefault();
+ }
+ if (d3_event.repeat)
+ return;
+ cancelLongPress();
+ if (d3_event.shiftKey) {
+ context.surface().classed("behavior-multiselect", true);
+ }
+ if (d3_event.keyCode === 32) {
+ if (!_downPointers.spacebar && _lastMouseEvent) {
+ cancelLongPress();
+ _longPressTimeout = window.setTimeout(didLongPress, 500, "spacebar", "spacebar");
+ _downPointers.spacebar = {
+ firstEvent: _lastMouseEvent,
+ lastEvent: _lastMouseEvent
+ };
+ }
+ }
+ }
+ function keyup(d3_event) {
+ cancelLongPress();
+ if (!d3_event.shiftKey) {
+ context.surface().classed("behavior-multiselect", false);
+ }
+ if (d3_event.keyCode === 93) {
+ d3_event.preventDefault();
+ _lastInteractionType = "menukey";
+ contextmenu(d3_event);
+ } else if (d3_event.keyCode === 32) {
+ var pointer = _downPointers.spacebar;
+ if (pointer) {
+ delete _downPointers.spacebar;
+ if (pointer.done)
+ return;
+ d3_event.preventDefault();
+ _lastInteractionType = "spacebar";
+ click(pointer.firstEvent, pointer.lastEvent, "spacebar");
+ }
+ }
+ }
+ function pointerdown(d3_event) {
+ var id2 = (d3_event.pointerId || "mouse").toString();
+ cancelLongPress();
+ if (d3_event.buttons && d3_event.buttons !== 1)
+ return;
+ context.ui().closeEditMenu();
+ if (d3_event.pointerType !== "mouse") {
+ _longPressTimeout = window.setTimeout(didLongPress, 500, id2, "longdown-" + (d3_event.pointerType || "mouse"));
+ }
+ _downPointers[id2] = {
+ firstEvent: d3_event,
+ lastEvent: d3_event
+ };
+ }
+ function didLongPress(id2, interactionType) {
+ var pointer = _downPointers[id2];
+ if (!pointer)
+ return;
+ for (var i2 in _downPointers) {
+ _downPointers[i2].done = true;
+ }
+ _longPressTimeout = null;
+ _lastInteractionType = interactionType;
+ _showMenu = true;
+ click(pointer.firstEvent, pointer.lastEvent, id2);
+ }
+ function pointermove(d3_event) {
+ var id2 = (d3_event.pointerId || "mouse").toString();
+ if (_downPointers[id2]) {
+ _downPointers[id2].lastEvent = d3_event;
+ }
+ if (!d3_event.pointerType || d3_event.pointerType === "mouse") {
+ _lastMouseEvent = d3_event;
+ if (_downPointers.spacebar) {
+ _downPointers.spacebar.lastEvent = d3_event;
+ }
+ }
+ }
+ function pointerup(d3_event) {
+ var id2 = (d3_event.pointerId || "mouse").toString();
+ var pointer = _downPointers[id2];
+ if (!pointer)
+ return;
+ delete _downPointers[id2];
+ if (_multiselectionPointerId === id2) {
+ _multiselectionPointerId = null;
+ }
+ if (pointer.done)
+ return;
+ click(pointer.firstEvent, d3_event, id2);
+ }
+ function pointercancel(d3_event) {
+ var id2 = (d3_event.pointerId || "mouse").toString();
+ if (!_downPointers[id2])
+ return;
+ delete _downPointers[id2];
+ if (_multiselectionPointerId === id2) {
+ _multiselectionPointerId = null;
+ }
+ }
+ function contextmenu(d3_event) {
+ d3_event.preventDefault();
+ if (!+d3_event.clientX && !+d3_event.clientY) {
+ if (_lastMouseEvent) {
+ d3_event = _lastMouseEvent;
+ } else {
+ return;
+ }
+ } else {
+ _lastMouseEvent = d3_event;
+ _lastInteractionType = "rightclick";
+ }
+ _showMenu = true;
+ click(d3_event, d3_event);
+ }
+ function click(firstEvent, lastEvent, pointerId) {
+ cancelLongPress();
+ var mapNode = context.container().select(".main-map").node();
+ 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) {
+ var selectPointerInfo = pointerDownOnSelection(pointerId);
+ if (selectPointerInfo) {
+ _multiselectionPointerId = selectPointerInfo.pointerId;
+ multiselectEntityId = !selectPointerInfo.selected && selectPointerInfo.entityId;
+ _downPointers[selectPointerInfo.pointerId].done = true;
+ }
+ }
+ 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 mapContains(event) {
+ var rect = mapNode.getBoundingClientRect();
+ return event.clientX >= rect.left && event.clientX <= rect.right && event.clientY >= rect.top && event.clientY <= rect.bottom;
+ }
+ function pointerDownOnSelection(skipPointerId) {
+ var mode = context.mode();
+ var selectedIDs = mode.id === "select" ? mode.selectedIDs() : [];
+ for (var pointerId2 in _downPointers) {
+ if (pointerId2 === "spacebar" || pointerId2 === skipPointerId)
+ continue;
+ var pointerInfo = _downPointers[pointerId2];
+ var p12 = pointGetter(pointerInfo.firstEvent);
+ var p22 = pointGetter(pointerInfo.lastEvent);
+ if (geoVecLength(p12, p22) > _tolerancePx)
+ continue;
+ var datum2 = pointerInfo.firstEvent.target.__data__;
+ var entity = datum2 && datum2.properties && datum2.properties.entity || datum2;
+ if (context.graph().hasEntity(entity.id)) {
+ return {
+ pointerId: pointerId2,
+ entityId: entity.id,
+ selected: selectedIDs.indexOf(entity.id) !== -1
+ };
+ }
+ }
+ return null;
+ }
+ }
+ function processClick(datum2, isMultiselect, point2, alsoSelectId) {
+ var mode = context.mode();
+ var showMenu = _showMenu;
+ var interactionType = _lastInteractionType;
+ var entity = datum2 && datum2.properties && datum2.properties.entity;
+ if (entity)
+ datum2 = entity;
+ if (datum2 && datum2.type === "midpoint") {
+ datum2 = datum2.parents[0];
+ }
+ var newMode;
+ if (datum2 instanceof osmEntity) {
+ var selectedIDs = context.selectedIDs();
+ context.selectedNoteID(null);
+ context.selectedErrorID(null);
+ if (!isMultiselect) {
+ if (!showMenu || selectedIDs.length <= 1 || selectedIDs.indexOf(datum2.id) === -1) {
+ if (alsoSelectId === datum2.id)
+ alsoSelectId = null;
+ selectedIDs = (alsoSelectId ? [alsoSelectId] : []).concat([datum2.id]);
+ newMode = mode.id === "select" ? mode.selectedIDs(selectedIDs) : modeSelect(context, selectedIDs).selectBehavior(behavior);
+ context.enter(newMode);
+ }
+ } else {
+ if (selectedIDs.indexOf(datum2.id) !== -1) {
+ if (!showMenu) {
+ selectedIDs = selectedIDs.filter(function(id2) {
+ return id2 !== datum2.id;
+ });
+ newMode = selectedIDs.length ? mode.selectedIDs(selectedIDs) : modeBrowse(context).selectBehavior(behavior);
+ context.enter(newMode);
+ }
+ } else {
+ selectedIDs = selectedIDs.concat([datum2.id]);
+ newMode = mode.selectedIDs(selectedIDs);
+ context.enter(newMode);
+ }
+ }
+ } else if (datum2 && datum2.__featurehash__ && !isMultiselect) {
+ context.selectedNoteID(null).enter(modeSelectData(context, datum2));
+ } else if (datum2 instanceof osmNote && !isMultiselect) {
+ context.selectedNoteID(datum2.id).enter(modeSelectNote(context, datum2.id));
+ } else if (datum2 instanceof QAItem && !isMultiselect) {
+ context.selectedErrorID(datum2.id).enter(modeSelectError(context, datum2.id, datum2.service));
+ } else {
+ context.selectedNoteID(null);
+ context.selectedErrorID(null);
+ if (!isMultiselect && mode.id !== "browse") {
+ context.enter(modeBrowse(context));
+ }
+ }
+ context.ui().closeEditMenu();
+ if (showMenu)
+ context.ui().showEditMenu(point2, interactionType);
+ resetProperties();
+ }
+ function cancelLongPress() {
+ if (_longPressTimeout)
+ window.clearTimeout(_longPressTimeout);
+ _longPressTimeout = null;
+ }
+ function resetProperties() {
+ cancelLongPress();
+ _showMenu = false;
+ _lastInteractionType = null;
+ }
+ function behavior(selection2) {
+ resetProperties();
+ _lastMouseEvent = context.map().lastPointerEvent();
+ select_default2(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) {
+ var e = d3_event;
+ if (+e.clientX === 0 && +e.clientY === 0) {
+ d3_event.preventDefault();
+ }
+ });
+ selection2.on(_pointerPrefix + "down.select", pointerdown).on("contextmenu.select", contextmenu);
+ }
+ behavior.off = function(selection2) {
+ cancelLongPress();
+ select_default2(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);
+ selection2.on(_pointerPrefix + "down.select", null).on("contextmenu.select", null);
+ context.surface().classed("behavior-multiselect", false);
+ };
+ return behavior;
+ }
+
+ // modules/operations/index.js
+ var operations_exports = {};
+ __export(operations_exports, {
+ operationCircularize: () => operationCircularize,
+ operationContinue: () => operationContinue,
+ operationCopy: () => operationCopy,
+ operationDelete: () => operationDelete,
+ operationDisconnect: () => operationDisconnect,
+ operationDowngrade: () => operationDowngrade,
+ operationExtract: () => operationExtract,
+ operationMerge: () => operationMerge,
+ operationMove: () => operationMove,
+ operationOrthogonalize: () => operationOrthogonalize,
+ operationPaste: () => operationPaste,
+ operationReflectLong: () => operationReflectLong,
+ operationReflectShort: () => operationReflectShort,
+ operationReverse: () => operationReverse,
+ operationRotate: () => operationRotate,
+ operationSplit: () => operationSplit,
+ operationStraighten: () => operationStraighten
+ });
+
+ // modules/operations/continue.js
+ function operationContinue(context, selectedIDs) {
+ var _entities = selectedIDs.map(function(id2) {
+ return context.graph().entity(id2);
+ });
+ var _geometries = Object.assign(
+ { line: [], vertex: [] },
+ utilArrayGroupBy(_entities, function(entity) {
+ return entity.geometry(context.graph());
+ })
+ );
+ var _vertex = _geometries.vertex.length && _geometries.vertex[0];
+ 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);
+ }) : [];
+ }
+ var _candidates = candidateWays();
+ var operation = function() {
+ 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] : [];
+ };
+ operation.available = function() {
+ return _geometries.vertex.length === 1 && _geometries.line.length <= 1 && !context.features().hasHiddenConnections(_vertex, context.graph());
+ };
+ operation.disabled = function() {
+ if (_candidates.length === 0) {
+ return "not_eligible";
+ } else if (_candidates.length > 1) {
+ return "multiple";
+ }
+ return false;
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.continue." + disable) : _t.append("operations.continue.description");
+ };
+ operation.annotation = function() {
+ return _t("operations.continue.annotation.line");
+ };
+ operation.id = "continue";
+ operation.keys = [_t("operations.continue.key")];
+ operation.title = _t.append("operations.continue.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/copy.js
+ function operationCopy(context, selectedIDs) {
+ function getFilteredIdsToCopy() {
+ return selectedIDs.filter(function(selectedID) {
+ var entity = context.graph().hasEntity(selectedID);
+ return entity.hasInterestingTags() || entity.geometry(context.graph()) !== "vertex";
+ });
+ }
+ var operation = function() {
+ var graph = context.graph();
+ var selected = groupEntities(getFilteredIdsToCopy(), graph);
+ var canCopy = [];
+ var skip = {};
+ var entity;
+ var i2;
+ for (i2 = 0; i2 < selected.relation.length; i2++) {
+ entity = selected.relation[i2];
+ if (!skip[entity.id] && entity.isComplete(graph)) {
+ canCopy.push(entity.id);
+ skip = getDescendants(entity.id, graph, skip);
+ }
+ }
+ for (i2 = 0; i2 < selected.way.length; i2++) {
+ entity = selected.way[i2];
+ if (!skip[entity.id]) {
+ canCopy.push(entity.id);
+ skip = getDescendants(entity.id, graph, skip);
+ }
+ }
+ for (i2 = 0; i2 < selected.node.length; i2++) {
+ entity = selected.node[i2];
+ if (!skip[entity.id]) {
+ canCopy.push(entity.id);
+ }
+ }
+ context.copyIDs(canCopy);
+ if (_point && (canCopy.length !== 1 || graph.entity(canCopy[0]).type !== "node")) {
+ context.copyLonLat(context.projection.invert(_point));
+ } else {
+ context.copyLonLat(null);
+ }
+ };
+ function groupEntities(ids, graph) {
+ var entities = ids.map(function(id2) {
+ return graph.entity(id2);
+ });
+ return Object.assign(
+ { relation: [], way: [], node: [] },
+ utilArrayGroupBy(entities, "type")
+ );
+ }
+ function getDescendants(id2, graph, descendants) {
+ var entity = graph.entity(id2);
+ var children2;
+ descendants = descendants || {};
+ if (entity.type === "relation") {
+ children2 = entity.members.map(function(m) {
+ return m.id;
+ });
+ } else if (entity.type === "way") {
+ children2 = entity.nodes;
+ } else {
+ children2 = [];
+ }
+ for (var i2 = 0; i2 < children2.length; i2++) {
+ if (!descendants[children2[i2]]) {
+ descendants[children2[i2]] = true;
+ descendants = getDescendants(children2[i2], graph, descendants);
+ }
+ }
+ return descendants;
+ }
+ operation.available = function() {
+ return getFilteredIdsToCopy().length > 0;
+ };
+ operation.disabled = function() {
+ var extent = utilTotalExtent(getFilteredIdsToCopy(), context.graph());
+ if (extent.percentContainedIn(context.map().extent()) < 0.8) {
+ return "too_large";
+ }
+ return false;
+ };
+ operation.availableForKeypress = function() {
+ var selection2 = window.getSelection && window.getSelection();
+ return !selection2 || !selection2.toString();
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.copy." + disable, { n: selectedIDs.length }) : _t.append("operations.copy.description", { n: selectedIDs.length });
+ };
+ operation.annotation = function() {
+ return _t("operations.copy.annotation", { n: selectedIDs.length });
+ };
+ var _point;
+ operation.point = function(val) {
+ _point = val;
+ return operation;
+ };
+ operation.id = "copy";
+ operation.keys = [uiCmd("\u2318C")];
+ operation.title = _t.append("operations.copy.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/disconnect.js
+ function operationDisconnect(context, selectedIDs) {
+ var _vertexIDs = [];
+ var _wayIDs = [];
+ var _otherIDs = [];
+ var _actions = [];
+ selectedIDs.forEach(function(id2) {
+ var entity = context.entity(id2);
+ if (entity.type === "way") {
+ _wayIDs.push(id2);
+ } else if (entity.geometry(context.graph()) === "vertex") {
+ _vertexIDs.push(id2);
+ } else {
+ _otherIDs.push(id2);
+ }
+ });
+ var _coords, _descriptionID = "", _annotationID = "features";
+ var _disconnectingVertexIds = [];
+ var _disconnectingWayIds = [];
+ if (_vertexIDs.length > 0) {
+ _disconnectingVertexIds = _vertexIDs;
+ _vertexIDs.forEach(function(vertexID) {
+ var action = actionDisconnect(vertexID);
+ if (_wayIDs.length > 0) {
+ var waysIDsForVertex = _wayIDs.filter(function(wayID) {
+ var way = context.entity(wayID);
+ return way.nodes.indexOf(vertexID) !== -1;
+ });
+ action.limitWays(waysIDsForVertex);
+ }
+ _actions.push(action);
+ _disconnectingWayIds = _disconnectingWayIds.concat(context.graph().parentWays(context.graph().entity(vertexID)).map((d) => d.id));
+ });
+ _disconnectingWayIds = utilArrayUniq(_disconnectingWayIds).filter(function(id2) {
+ return _wayIDs.indexOf(id2) === -1;
+ });
+ _descriptionID += _actions.length === 1 ? "single_point." : "multiple_points.";
+ 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) {
+ var ways = _wayIDs.map(function(id2) {
+ return context.entity(id2);
+ });
+ var nodes = utilGetAllNodes(_wayIDs, context.graph());
+ _coords = nodes.map(function(n2) {
+ return n2.loc;
+ });
+ var sharedActions = [];
+ var sharedNodes = [];
+ 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 i2 in ways) {
+ var way = ways[i2];
+ if (way.nodes.indexOf(node.id) !== -1) {
+ count += 1;
+ }
+ if (count > 1)
+ break;
+ }
+ 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) {
+ _actions = sharedActions;
+ _disconnectingVertexIds = sharedNodes.map((node) => node.id);
+ _descriptionID += "conjoined";
+ _annotationID = "from_each_other";
+ } else {
+ _actions = unsharedActions;
+ _disconnectingVertexIds = unsharedNodes.map((node) => node.id);
+ if (_wayIDs.length === 1) {
+ _descriptionID += context.graph().geometry(_wayIDs[0]);
+ } else {
+ _descriptionID += "separate";
+ }
+ }
+ }
+ var _extent = utilTotalExtent(_disconnectingVertexIds, context.graph());
+ var operation = function() {
+ context.perform(function(graph) {
+ return _actions.reduce(function(graph2, action) {
+ return action(graph2);
+ }, 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 (_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";
+ }
+ return false;
+ function someMissing() {
+ if (context.inIntro())
+ return false;
+ var osm = context.connection();
+ if (osm) {
+ var missing = _coords.filter(function(loc) {
+ return !osm.isDataLoaded(loc);
+ });
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
+ }
+ }
+ return false;
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.disconnect." + disable) : _t.append("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.append("operations.disconnect.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/downgrade.js
+ function operationDowngrade(context, selectedIDs) {
+ var _affectedFeatureCount = 0;
+ var _downgradeType = downgradeTypeForEntityIDs(selectedIDs);
+ var _multi = _affectedFeatureCount === 1 ? "single" : "multiple";
+ function downgradeTypeForEntityIDs(entityIds) {
+ var downgradeType;
+ _affectedFeatureCount = 0;
+ for (var i2 in entityIds) {
+ var entityID = entityIds[i2];
+ var type2 = downgradeTypeForEntityID(entityID);
+ if (type2) {
+ _affectedFeatureCount += 1;
+ if (downgradeType && type2 !== downgradeType) {
+ if (downgradeType !== "generic" && type2 !== "generic") {
+ downgradeType = "building_address";
+ } else {
+ downgradeType = "generic";
+ }
+ } else {
+ downgradeType = type2;
+ }
+ }
+ }
+ return downgradeType;
+ }
+ function downgradeTypeForEntityID(entityID) {
+ var graph = context.graph();
+ var entity = graph.entity(entityID);
+ var preset = _mainPresetIndex.match(entity, graph);
+ if (!preset || preset.isFallback())
+ return null;
+ if (entity.type === "node" && preset.id !== "address" && Object.keys(entity.tags).some(function(key) {
+ return key.match(/^addr:.{1,}/);
+ })) {
+ return "address";
+ }
+ var geometry = entity.geometry(graph);
+ if (geometry === "area" && entity.tags.building && !preset.tags.building) {
+ return "building";
+ }
+ if (geometry === "vertex" && Object.keys(entity.tags).length) {
+ return "generic";
+ }
+ return null;
+ }
+ var buildingKeysToKeep = ["architect", "building", "height", "layer", "source", "type", "wheelchair"];
+ var addressKeysToKeep = ["source"];
+ var operation = function() {
+ context.perform(function(graph) {
+ for (var i2 in selectedIDs) {
+ var entityID = selectedIDs[i2];
+ var type2 = downgradeTypeForEntityID(entityID);
+ if (!type2)
+ continue;
+ var tags = Object.assign({}, graph.entity(entityID).tags);
+ for (var key in tags) {
+ if (type2 === "address" && addressKeysToKeep.indexOf(key) !== -1)
+ continue;
+ if (type2 === "building") {
+ if (buildingKeysToKeep.indexOf(key) !== -1 || key.match(/^building:.{1,}/) || key.match(/^roof:.{1,}/))
+ continue;
+ }
+ if (type2 !== "generic") {
+ if (key.match(/^addr:.{1,}/) || key.match(/^source:.{1,}/))
+ continue;
+ }
+ delete tags[key];
+ }
+ graph = actionChangeTags(entityID, tags)(graph);
+ }
+ return graph;
+ }, operation.annotation());
+ context.validator().validate();
+ context.enter(modeSelect(context, selectedIDs));
+ };
+ operation.available = function() {
+ return _downgradeType;
+ };
+ operation.disabled = function() {
+ if (selectedIDs.some(hasWikidataTag)) {
+ return "has_wikidata_tag";
+ }
+ return false;
+ function hasWikidataTag(id2) {
+ var entity = context.entity(id2);
+ return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.downgrade." + disable + "." + _multi) : _t.append("operations.downgrade.description." + _downgradeType);
+ };
+ operation.annotation = function() {
+ var suffix;
+ if (_downgradeType === "building_address") {
+ suffix = "generic";
+ } else {
+ suffix = _downgradeType;
+ }
+ return _t("operations.downgrade.annotation." + suffix, { n: _affectedFeatureCount });
+ };
+ operation.id = "downgrade";
+ operation.keys = [uiCmd("\u232B")];
+ operation.title = _t.append("operations.downgrade.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/extract.js
+ function operationExtract(context, selectedIDs) {
+ var _amount = selectedIDs.length === 1 ? "single" : "multiple";
+ var _geometries = utilArrayUniq(selectedIDs.map(function(entityID) {
+ return context.graph().hasEntity(entityID) && context.graph().geometry(entityID);
+ }).filter(Boolean));
+ var _geometryID = _geometries.length === 1 ? _geometries[0] : "feature";
+ var _extent;
+ 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);
+ if (preset.geometry.indexOf("point") === -1)
+ return null;
+ }
+ _extent = _extent ? _extent.extend(entity.extent(graph)) : entity.extent(graph);
+ return actionExtract(entityID, context.projection);
+ }).filter(Boolean);
+ var operation = function() {
+ var combinedAction = function(graph) {
+ _actions.forEach(function(action) {
+ graph = action(graph);
+ });
+ return graph;
+ };
+ context.perform(combinedAction, operation.annotation());
+ var extractedNodeIDs = _actions.map(function(action) {
+ return action.getExtractedNodeID();
+ });
+ context.enter(modeSelect(context, extractedNodeIDs));
+ };
+ operation.available = function() {
+ return _actions.length && selectedIDs.length === _actions.length;
+ };
+ 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";
+ }
+ return false;
+ };
+ operation.tooltip = function() {
+ var disableReason = operation.disabled();
+ if (disableReason) {
+ return _t.append("operations.extract." + disableReason + "." + _amount);
+ } else {
+ return _t.append("operations.extract.description." + _geometryID + "." + _amount);
+ }
+ };
+ operation.annotation = function() {
+ return _t("operations.extract.annotation", { n: selectedIDs.length });
+ };
+ operation.id = "extract";
+ operation.keys = [_t("operations.extract.key")];
+ operation.title = _t.append("operations.extract.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/merge.js
+ function operationMerge(context, selectedIDs) {
+ var _action = getAction();
+ function getAction() {
+ var join = actionJoin(selectedIDs);
+ if (!join.disabled(context.graph()))
+ return join;
+ var merge3 = actionMerge(selectedIDs);
+ if (!merge3.disabled(context.graph()))
+ return merge3;
+ var mergePolygon = actionMergePolygon(selectedIDs);
+ if (!mergePolygon.disabled(context.graph()))
+ return mergePolygon;
+ var mergeNodes = actionMergeNodes(selectedIDs);
+ if (!mergeNodes.disabled(context.graph()))
+ return mergeNodes;
+ if (join.disabled(context.graph()) !== "not_eligible")
+ return join;
+ if (merge3.disabled(context.graph()) !== "not_eligible")
+ return merge3;
+ if (mergePolygon.disabled(context.graph()) !== "not_eligible")
+ return mergePolygon;
+ return mergeNodes;
+ }
+ var operation = function() {
+ if (operation.disabled())
+ return;
+ context.perform(_action, operation.annotation());
+ context.validator().validate();
+ var resultIDs = selectedIDs.filter(context.hasEntity);
+ if (resultIDs.length > 1) {
+ var interestingIDs = resultIDs.filter(function(id2) {
+ return context.entity(id2).hasInterestingTags();
+ });
+ if (interestingIDs.length)
+ resultIDs = interestingIDs;
+ }
+ context.enter(modeSelect(context, resultIDs));
+ };
+ operation.available = function() {
+ return selectedIDs.length >= 2;
+ };
+ operation.disabled = function() {
+ var actionDisabled = _action.disabled(context.graph());
+ if (actionDisabled)
+ return actionDisabled;
+ var osm = context.connection();
+ if (osm && _action.resultingWayNodesLength && _action.resultingWayNodesLength(context.graph()) > osm.maxWayNodes()) {
+ return "too_many_vertices";
+ }
+ return false;
+ };
+ operation.tooltip = function() {
+ var disabled = operation.disabled();
+ if (disabled) {
+ if (disabled === "conflicting_relations") {
+ return _t.append("operations.merge.conflicting_relations");
+ }
+ if (disabled === "restriction" || disabled === "connectivity") {
+ return _t.append(
+ "operations.merge.damage_relation",
+ { relation: _mainPresetIndex.item("type/" + disabled).name() }
+ );
+ }
+ return _t.append("operations.merge." + disabled);
+ }
+ return _t.append("operations.merge.description");
+ };
+ operation.annotation = function() {
+ return _t("operations.merge.annotation", { n: selectedIDs.length });
+ };
+ operation.id = "merge";
+ operation.keys = [_t("operations.merge.key")];
+ operation.title = _t.append("operations.merge.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/paste.js
+ function operationPaste(context) {
+ var _pastePoint;
+ var operation = function() {
+ if (!_pastePoint)
+ return;
+ var oldIDs = context.copyIDs();
+ if (!oldIDs.length)
+ return;
+ var projection2 = 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 = /* @__PURE__ */ new Set();
+ Object.values(copies).forEach(function(entity) {
+ originals.add(entity.id);
+ });
+ for (var id2 in copies) {
+ var oldEntity = oldGraph.entity(id2);
+ var newEntity = copies[id2];
+ extent._extend(oldEntity.extent(oldGraph));
+ var parents = context.graph().parentWays(newEntity);
+ var parentCopied = parents.some(function(parent) {
+ return originals.has(parent.id);
+ });
+ if (!parentCopied) {
+ newIDs.push(newEntity.id);
+ }
+ }
+ var copyPoint = context.copyLonLat() && projection2(context.copyLonLat()) || projection2(extent.center());
+ var delta = geoVecSubtract(_pastePoint, copyPoint);
+ context.replace(actionMove(newIDs, delta, projection2), operation.annotation());
+ context.enter(modeSelect(context, newIDs));
+ };
+ operation.point = function(val) {
+ _pastePoint = val;
+ return operation;
+ };
+ operation.available = function() {
+ return context.mode().id === "browse";
+ };
+ operation.disabled = function() {
+ return !context.copyIDs().length;
+ };
+ operation.tooltip = function() {
+ var oldGraph = context.copyGraph();
+ var ids = context.copyIDs();
+ if (!ids.length) {
+ return _t.append("operations.paste.nothing_copied");
+ }
+ return _t.append("operations.paste.description", { feature: utilDisplayLabel(oldGraph.entity(ids[0]), oldGraph), n: ids.length });
+ };
+ operation.annotation = function() {
+ var ids = context.copyIDs();
+ return _t("operations.paste.annotation", { n: ids.length });
+ };
+ operation.id = "paste";
+ operation.keys = [uiCmd("\u2318V")];
+ operation.title = _t.append("operations.paste.title");
+ return operation;
+ }
+
+ // modules/operations/reverse.js
+ function operationReverse(context, selectedIDs) {
+ var operation = function() {
+ context.perform(function combinedReverseAction(graph) {
+ actions().forEach(function(action) {
+ graph = action(graph);
+ });
+ return graph;
+ }, operation.annotation());
+ context.validator().validate();
+ };
+ function actions(situation) {
+ return selectedIDs.map(function(entityID) {
+ var entity = context.hasEntity(entityID);
+ if (!entity)
+ return null;
+ if (situation === "toolbar") {
+ if (entity.type === "way" && (!entity.isOneWay() && !entity.isSided()))
+ return null;
+ }
+ 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 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";
+ }
+ operation.available = function(situation) {
+ return actions(situation).length > 0;
+ };
+ operation.disabled = function() {
+ return false;
+ };
+ operation.tooltip = function() {
+ return _t.append("operations.reverse.description." + reverseTypeID());
+ };
+ operation.annotation = function() {
+ var acts = actions();
+ return _t("operations.reverse.annotation." + reverseTypeID(), { n: acts.length });
+ };
+ operation.id = "reverse";
+ operation.keys = [_t("operations.reverse.key")];
+ operation.title = _t.append("operations.reverse.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/split.js
+ function operationSplit(context, selectedIDs) {
+ var _vertexIds = selectedIDs.filter(function(id2) {
+ return context.graph().geometry(id2) === "vertex";
+ });
+ var _selectedWayIds = selectedIDs.filter(function(id2) {
+ var entity = context.graph().hasEntity(id2);
+ return entity && entity.type === "way";
+ });
+ var _isAvailable = _vertexIds.length > 0 && _vertexIds.length + _selectedWayIds.length === selectedIDs.length;
+ var _action = actionSplit(_vertexIds);
+ var _ways = [];
+ var _geometry = "feature";
+ var _waysAmount = "single";
+ var _nodesAmount = _vertexIds.length === 1 ? "single" : "multiple";
+ if (_isAvailable) {
+ if (_selectedWayIds.length)
+ _action.limitWays(_selectedWayIds);
+ _ways = _action.ways(context.graph());
+ var geometries = {};
+ _ways.forEach(function(way) {
+ geometries[way.geometry(context.graph())] = true;
+ });
+ if (Object.keys(geometries).length === 1) {
+ _geometry = Object.keys(geometries)[0];
+ }
+ _waysAmount = _ways.length === 1 ? "single" : "multiple";
+ }
+ var operation = function() {
+ var difference = context.perform(_action, operation.annotation());
+ var idsToSelect = _vertexIds.concat(difference.extantIDs().filter(function(id2) {
+ return context.entity(id2).type === "way";
+ }));
+ context.enter(modeSelect(context, idsToSelect));
+ };
+ operation.relatedEntityIds = function() {
+ return _selectedWayIds.length ? [] : _ways.map((way) => way.id);
+ };
+ operation.available = function() {
+ return _isAvailable;
+ };
+ operation.disabled = function() {
+ var reason = _action.disabled(context.graph());
+ if (reason) {
+ return reason;
+ } else if (selectedIDs.some(context.hasHiddenConnections)) {
+ return "connected_to_hidden";
+ }
+ return false;
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.split." + disable) : _t.append("operations.split.description." + _geometry + "." + _waysAmount + "." + _nodesAmount + "_node");
+ };
+ operation.annotation = function() {
+ return _t("operations.split.annotation." + _geometry, { n: _ways.length });
+ };
+ operation.icon = function() {
+ if (_waysAmount === "multiple") {
+ return "#iD-operation-split-multiple";
+ } else {
+ return "#iD-operation-split";
+ }
+ };
+ operation.id = "split";
+ operation.keys = [_t("operations.split.key")];
+ operation.title = _t.append("operations.split.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/operations/straighten.js
+ function operationStraighten(context, selectedIDs) {
+ var _wayIDs = selectedIDs.filter(function(id2) {
+ return id2.charAt(0) === "w";
+ });
+ var _nodeIDs = selectedIDs.filter(function(id2) {
+ return id2.charAt(0) === "n";
+ });
+ var _amount = (_wayIDs.length ? _wayIDs : _nodeIDs).length === 1 ? "single" : "multiple";
+ var _nodes = utilGetAllNodes(selectedIDs, context.graph());
+ var _coords = _nodes.map(function(n2) {
+ return n2.loc;
+ });
+ var _extent = utilTotalExtent(selectedIDs, context.graph());
+ var _action = chooseAction();
+ var _geometry;
+ function chooseAction() {
+ if (_wayIDs.length === 0 && _nodeIDs.length > 2) {
+ _geometry = "point";
+ return actionStraightenNodes(_nodeIDs, context.projection);
+ } else if (_wayIDs.length > 0 && (_nodeIDs.length === 0 || _nodeIDs.length === 2)) {
+ var startNodeIDs = [];
+ var endNodeIDs = [];
+ for (var i2 = 0; i2 < selectedIDs.length; i2++) {
+ var entity = context.entity(selectedIDs[i2]);
+ if (entity.type === "node") {
+ continue;
+ } else if (entity.type !== "way" || entity.isClosed()) {
+ return null;
+ }
+ startNodeIDs.push(entity.first());
+ endNodeIDs.push(entity.last());
+ }
+ startNodeIDs = startNodeIDs.filter(function(n2) {
+ return startNodeIDs.indexOf(n2) === startNodeIDs.lastIndexOf(n2);
+ });
+ endNodeIDs = endNodeIDs.filter(function(n2) {
+ return endNodeIDs.indexOf(n2) === endNodeIDs.lastIndexOf(n2);
+ });
+ if (utilArrayDifference(startNodeIDs, endNodeIDs).length + utilArrayDifference(endNodeIDs, startNodeIDs).length !== 2)
+ return null;
+ var wayNodeIDs = utilGetAllNodes(_wayIDs, context.graph()).map(function(node) {
+ return node.id;
+ });
+ if (wayNodeIDs.length <= 2)
+ return null;
+ if (_nodeIDs.length === 2 && (wayNodeIDs.indexOf(_nodeIDs[0]) === -1 || wayNodeIDs.indexOf(_nodeIDs[1]) === -1))
+ return null;
+ if (_nodeIDs.length) {
+ _extent = utilTotalExtent(_nodeIDs, context.graph());
+ }
+ _geometry = "line";
+ return actionStraightenWay(selectedIDs, context.projection);
+ }
+ return null;
+ }
+ function operation() {
+ if (!_action)
+ return;
+ context.perform(_action, operation.annotation());
+ window.setTimeout(function() {
+ context.validator().validate();
+ }, 300);
+ }
+ operation.available = function() {
+ return Boolean(_action);
+ };
+ operation.disabled = function() {
+ var reason = _action.disabled(context.graph());
+ 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;
+ function someMissing() {
+ if (context.inIntro())
+ return false;
+ var osm = context.connection();
+ if (osm) {
+ var missing = _coords.filter(function(loc) {
+ return !osm.isDataLoaded(loc);
+ });
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
+ }
+ }
+ return false;
+ }
+ };
+ operation.tooltip = function() {
+ var disable = operation.disabled();
+ return disable ? _t.append("operations.straighten." + disable + "." + _amount) : _t.append("operations.straighten.description." + _geometry + (_wayIDs.length === 1 ? "" : "s"));
+ };
+ operation.annotation = function() {
+ return _t("operations.straighten.annotation." + _geometry, { n: _wayIDs.length ? _wayIDs.length : _nodeIDs.length });
+ };
+ operation.id = "straighten";
+ operation.keys = [_t("operations.straighten.key")];
+ operation.title = _t.append("operations.straighten.title");
+ operation.behavior = behaviorOperation(context).which(operation);
+ return operation;
+ }
+
+ // modules/modes/select.js
+ function modeSelect(context, selectedIDs) {
+ var mode = {
+ id: "select",
+ button: "browse"
+ };
+ var keybinding = utilKeybinding("select");
+ var _breatheBehavior = behaviorBreathe(context);
+ var _modeDragNode = modeDragNode(context);
+ var _selectBehavior;
+ var _behaviors = [];
+ var _operations = [];
+ var _newFeature = false;
+ var _follow = false;
+ var _focusedParentWayId;
+ var _focusedVertexIds;
+ function singular() {
+ if (selectedIDs && selectedIDs.length === 1) {
+ return context.hasEntity(selectedIDs[0]);
+ }
+ }
+ function selectedEntities() {
+ return selectedIDs.map(function(id2) {
+ return context.hasEntity(id2);
+ }).filter(Boolean);
+ }
+ function checkSelectedIDs() {
+ var ids = [];
+ if (Array.isArray(selectedIDs)) {
+ ids = selectedIDs.filter(function(id2) {
+ return context.hasEntity(id2);
+ });
+ }
+ if (!ids.length) {
+ context.enter(modeBrowse(context));
+ return false;
+ } else if (selectedIDs.length > 1 && ids.length === 1 || selectedIDs.length === 1 && ids.length > 1) {
+ context.enter(modeSelect(context, ids));
+ return false;
+ }
+ selectedIDs = ids;
+ return true;
+ }
+ function parentWaysIdsOfSelection(onlyCommonParents) {
+ var graph = context.graph();
+ var parents = [];
+ for (var i2 = 0; i2 < selectedIDs.length; i2++) {
+ var entity = context.hasEntity(selectedIDs[i2]);
+ if (!entity || entity.geometry(graph) !== "vertex") {
+ return [];
+ }
+ var currParents = graph.parentWays(entity).map(function(w) {
+ return w.id;
+ });
+ if (!parents.length) {
+ parents = currParents;
+ continue;
+ }
+ parents = (onlyCommonParents ? utilArrayIntersection : utilArrayUnion)(parents, currParents);
+ if (!parents.length) {
+ return [];
+ }
+ }
+ return parents;
+ }
+ function childNodeIdsOfSelection(onlyCommon) {
+ var graph = context.graph();
+ var childs = [];
+ for (var i2 = 0; i2 < selectedIDs.length; i2++) {
+ var entity = context.hasEntity(selectedIDs[i2]);
+ if (!entity || !["area", "line"].includes(entity.geometry(graph))) {
+ return [];
+ }
+ var currChilds = graph.childNodes(entity).map(function(node) {
+ return node.id;
+ });
+ if (!childs.length) {
+ childs = currChilds;
+ continue;
+ }
+ childs = (onlyCommon ? utilArrayIntersection : utilArrayUnion)(childs, currChilds);
+ if (!childs.length) {
+ return [];
+ }
+ }
+ return childs;
+ }
+ function checkFocusedParent() {
+ if (_focusedParentWayId) {
+ var parents = parentWaysIdsOfSelection(true);
+ if (parents.indexOf(_focusedParentWayId) === -1)
+ _focusedParentWayId = null;
+ }
+ }
+ function parentWayIdForVertexNavigation() {
+ var parentIds = parentWaysIdsOfSelection(true);
+ if (_focusedParentWayId && parentIds.indexOf(_focusedParentWayId) !== -1) {
+ return _focusedParentWayId;
+ }
+ return parentIds.length ? parentIds[0] : null;
+ }
+ mode.selectedIDs = function(val) {
+ if (!arguments.length)
+ return selectedIDs;
+ selectedIDs = val;
+ return mode;
+ };
+ mode.zoomToSelected = function() {
+ context.map().zoomToEase(selectedEntities());
+ };
+ mode.newFeature = function(val) {
+ if (!arguments.length)
+ return _newFeature;
+ _newFeature = val;
+ return mode;
+ };
+ mode.selectBehavior = function(val) {
+ if (!arguments.length)
+ return _selectBehavior;
+ _selectBehavior = val;
+ return mode;
+ };
+ mode.follow = function(val) {
+ if (!arguments.length)
+ return _follow;
+ _follow = val;
+ return mode;
+ };
+ function loadOperations() {
+ _operations.forEach(function(operation) {
+ if (operation.behavior) {
+ context.uninstall(operation.behavior);
+ }
+ });
+ _operations = Object.values(operations_exports).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();
+ });
+ _operations.forEach(function(operation) {
+ if (operation.behavior) {
+ context.install(operation.behavior);
+ }
+ });
+ context.ui().closeEditMenu();
+ }
+ mode.operations = function() {
+ return _operations;
+ };
+ mode.enter = function() {
+ if (!checkSelectedIDs())
+ return;
+ context.features().forceVisible(selectedIDs);
+ _modeDragNode.restoreSelectedIDs(selectedIDs);
+ loadOperations();
+ 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
+ ];
+ }
+ _behaviors.forEach(context.install);
+ keybinding.on(_t("inspector.zoom_to.key"), mode.zoomToSelected).on(["[", "pgup"], previousVertex).on(["]", "pgdown"], nextVertex).on(["{", uiCmd("\u2318["), "home"], firstVertex).on(["}", uiCmd("\u2318]"), "end"], lastVertex).on(uiCmd("\u21E7\u2190"), nudgeSelection([-10, 0])).on(uiCmd("\u21E7\u2191"), nudgeSelection([0, -10])).on(uiCmd("\u21E7\u2192"), nudgeSelection([10, 0])).on(uiCmd("\u21E7\u2193"), nudgeSelection([0, 10])).on(uiCmd("\u21E7\u2325\u2190"), nudgeSelection([-100, 0])).on(uiCmd("\u21E7\u2325\u2191"), nudgeSelection([0, -100])).on(uiCmd("\u21E7\u2325\u2192"), nudgeSelection([100, 0])).on(uiCmd("\u21E7\u2325\u2193"), nudgeSelection([0, 100])).on(utilKeybinding.plusKeys.map((key) => uiCmd("\u21E7" + key)), scaleSelection(1.05)).on(utilKeybinding.plusKeys.map((key) => uiCmd("\u21E7\u2325" + key)), scaleSelection(Math.pow(1.05, 5))).on(utilKeybinding.minusKeys.map((key) => uiCmd("\u21E7" + key)), scaleSelection(1 / 1.05)).on(utilKeybinding.minusKeys.map((key) => uiCmd("\u21E7\u2325" + key)), scaleSelection(1 / Math.pow(1.05, 5))).on(["\\", "pause"], focusNextParent).on(uiCmd("\u2318\u2191"), selectParent).on(uiCmd("\u2318\u2193"), selectChild).on("\u238B", esc, true);
+ select_default2(document).call(keybinding);
+ context.ui().sidebar.select(selectedIDs, _newFeature);
+ context.history().on("change.select", function() {
+ loadOperations();
+ selectElements();
+ }).on("undone.select", checkSelectedIDs).on("redone.select", checkSelectedIDs);
+ context.map().on("drawn.select", selectElements).on("crossEditableZoom.select", function() {
+ selectElements();
+ _breatheBehavior.restartIfNeeded(context.surface());
+ });
+ context.map().doubleUpHandler().on("doubleUp.modeSelect", didDoubleUp);
+ selectElements();
+ if (_follow) {
+ var extent = geoExtent();
+ var graph = context.graph();
+ selectedIDs.forEach(function(id2) {
+ var entity = context.entity(id2);
+ extent._extend(entity.extent(graph));
+ });
+ var loc = extent.center();
+ context.map().centerEase(loc);
+ _follow = false;
+ }
+ function nudgeSelection(delta) {
+ return function() {
+ if (!context.map().withinEditableZoom())
+ return;
+ var moveOp = operationMove(context, selectedIDs);
+ if (moveOp.disabled()) {
+ context.ui().flash.duration(4e3).iconName("#iD-operation-" + moveOp.id).iconClass("operation disabled").label(moveOp.tooltip())();
+ } else {
+ context.perform(actionMove(selectedIDs, delta, context.projection), moveOp.annotation());
+ context.validator().validate();
+ }
+ };
+ }
+ function scaleSelection(factor) {
+ return function() {
+ if (!context.map().withinEditableZoom())
+ return;
+ let nodes = utilGetAllNodes(selectedIDs, context.graph());
+ let isUp = factor > 1;
+ if (nodes.length <= 1)
+ return;
+ let extent2 = utilTotalExtent(selectedIDs, context.graph());
+ function scalingDisabled() {
+ if (tooSmall()) {
+ return "too_small";
+ } else if (extent2.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 false;
+ function tooSmall() {
+ if (isUp)
+ return false;
+ let dLon = Math.abs(extent2[1][0] - extent2[0][0]);
+ let dLat = Math.abs(extent2[1][1] - extent2[0][1]);
+ return dLon < geoMetersToLon(1, extent2[1][1]) && dLat < geoMetersToLat(1);
+ }
+ function someMissing() {
+ if (context.inIntro())
+ return false;
+ let osm = context.connection();
+ if (osm) {
+ let missing = nodes.filter(function(n2) {
+ return !osm.isDataLoaded(n2.loc);
+ });
+ if (missing.length) {
+ missing.forEach(function(loc2) {
+ context.loadTileAtLoc(loc2);
+ });
+ return true;
+ }
+ }
+ return false;
+ }
+ function incompleteRelation(id2) {
+ let entity = context.entity(id2);
+ return entity.type === "relation" && !entity.isComplete(context.graph());
+ }
+ }
+ const disabled = scalingDisabled();
+ if (disabled) {
+ let multi = selectedIDs.length === 1 ? "single" : "multiple";
+ context.ui().flash.duration(4e3).iconName("#iD-icon-no").iconClass("operation disabled").label(_t.append("operations.scale." + disabled + "." + multi))();
+ } else {
+ const pivot = context.projection(extent2.center());
+ const 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 didDoubleUp(d3_event, loc2) {
+ if (!context.map().withinEditableZoom())
+ return;
+ var target = select_default2(d3_event.target);
+ var datum2 = target.datum();
+ var entity = datum2 && datum2.properties && datum2.properties.entity;
+ if (!entity)
+ return;
+ if (entity instanceof osmWay && target.classed("target")) {
+ var choice = geoChooseEdge(context.graph().childNodes(entity), loc2, 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")
+ );
+ context.validator().validate();
+ } else if (entity.type === "midpoint") {
+ context.perform(
+ actionAddMidpoint({ loc: entity.loc, edge: entity.edge }, osmNode()),
+ _t("operations.add.annotation.vertex")
+ );
+ context.validator().validate();
+ }
+ }
+ 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);
+ checkFocusedParent();
+ if (_focusedParentWayId) {
+ surface.selectAll(utilEntitySelector([_focusedParentWayId])).classed("related", true);
+ }
+ if (context.map().withinEditableZoom()) {
+ surface.selectAll(utilDeepMemberSelector(
+ selectedIDs,
+ context.graph(),
+ true
+ /* skipMultipolgonMembers */
+ )).classed("selected-member", true);
+ surface.selectAll(utilEntityOrDeepMemberSelector(selectedIDs, context.graph())).classed("selected", true);
+ }
+ }
+ function esc() {
+ if (context.container().select(".combobox").size())
+ return;
+ context.enter(modeBrowse(context));
+ }
+ function firstVertex(d3_event) {
+ d3_event.preventDefault();
+ var entity = singular();
+ var parentId = parentWayIdForVertexNavigation();
+ var way;
+ if (entity && entity.type === "way") {
+ way = entity;
+ } else if (parentId) {
+ way = context.entity(parentId);
+ }
+ _focusedParentWayId = way && way.id;
+ if (way) {
+ context.enter(
+ mode.selectedIDs([way.first()]).follow(true)
+ );
+ }
+ }
+ function lastVertex(d3_event) {
+ d3_event.preventDefault();
+ var entity = singular();
+ var parentId = parentWayIdForVertexNavigation();
+ var way;
+ if (entity && entity.type === "way") {
+ way = entity;
+ } else if (parentId) {
+ way = context.entity(parentId);
+ }
+ _focusedParentWayId = way && way.id;
+ if (way) {
+ context.enter(
+ mode.selectedIDs([way.last()]).follow(true)
+ );
+ }
+ }
+ 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;
+ 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)
+ );
+ }
+ }
+ 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;
+ 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)
+ );
+ }
+ }
+ function focusNextParent(d3_event) {
+ d3_event.preventDefault();
+ var parents = parentWaysIdsOfSelection(true);
+ if (!parents || parents.length < 2)
+ return;
+ var index = parents.indexOf(_focusedParentWayId);
+ if (index < 0 || index > parents.length - 2) {
+ _focusedParentWayId = parents[0];
+ } else {
+ _focusedParentWayId = parents[index + 1];
+ }
+ var surface = context.surface();
+ surface.selectAll(".related").classed("related", false);
+ if (_focusedParentWayId) {
+ surface.selectAll(utilEntitySelector([_focusedParentWayId])).classed("related", true);
+ }
+ }
+ 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)
+ );
+ _focusedVertexIds = currentSelectedIds;
+ }
+ function selectChild(d3_event) {
+ d3_event.preventDefault();
+ var currentSelectedIds = mode.selectedIDs();
+ var childIds = _focusedVertexIds ? _focusedVertexIds.filter((id2) => context.hasEntity(id2)) : childNodeIdsOfSelection(true);
+ if (!childIds || !childIds.length)
+ return;
+ if (currentSelectedIds.length === 1)
+ _focusedParentWayId = currentSelectedIds[0];
+ context.enter(
+ mode.selectedIDs(childIds)
+ );
+ }
+ };
+ mode.exit = function() {
+ _newFeature = false;
+ _focusedVertexIds = null;
+ _operations.forEach(function(operation) {
+ if (operation.behavior) {
+ context.uninstall(operation.behavior);
+ }
+ });
+ _operations = [];
+ _behaviors.forEach(context.uninstall);
+ select_default2(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();
+ 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)) {
+ var deleteAction = actionDeleteRelation(
+ entity.id,
+ true
+ /* don't delete untagged members */
+ );
+ context.perform(deleteAction, _t("operations.delete.annotation.relation"));
+ context.validator().validate();
+ }
+ };
+ return mode;
+ }
+
+ // modules/behavior/lasso.js
+ function behaviorLasso(context) {
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ var behavior = function(selection2) {
+ var lasso;
+ function pointerdown(d3_event) {
+ var button = 0;
+ if (d3_event.button === button && d3_event.shiftKey === true) {
+ lasso = null;
+ select_default2(window).on(_pointerPrefix + "move.lasso", pointermove).on(_pointerPrefix + "up.lasso", pointerup);
+ d3_event.stopPropagation();
+ }
+ }
+ function pointermove() {
+ if (!lasso) {
+ lasso = uiLasso(context);
+ context.surface().call(lasso);
+ }
+ lasso.p(context.map().mouse());
+ }
+ function normalize2(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])]
+ ];
+ }
+ function lassoed() {
+ if (!lasso)
+ return [];
+ var graph = context.graph();
+ var limitToNodes;
+ if (context.map().editableDataEnabled(
+ true
+ /* skipZoomCheck */
+ ) && context.map().isInWideSelection()) {
+ limitToNodes = new Set(utilGetAllNodes(context.selectedIDs(), graph));
+ } else if (!context.map().editableDataEnabled()) {
+ return [];
+ }
+ var bounds = lasso.extent().map(context.projection.invert);
+ var extent = geoExtent(normalize2(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));
+ });
+ intersects.sort(function(node1, node2) {
+ var parents1 = graph.parentWays(node1);
+ var parents2 = graph.parentWays(node2);
+ if (parents1.length && parents2.length) {
+ var sharedParents = utilArrayIntersection(parents1, parents2);
+ if (sharedParents.length) {
+ var sharedParentNodes = sharedParents[0].nodes;
+ return sharedParentNodes.indexOf(node1.id) - sharedParentNodes.indexOf(node2.id);
+ } else {
+ return Number(parents1[0].id.slice(1)) - Number(parents2[0].id.slice(1));
+ }
+ } else if (parents1.length || parents2.length) {
+ return parents1.length - parents2.length;
+ }
+ return node1.loc[0] - node2.loc[0];
+ });
+ return intersects.map(function(entity) {
+ return entity.id;
+ });
+ }
+ function pointerup() {
+ select_default2(window).on(_pointerPrefix + "move.lasso", null).on(_pointerPrefix + "up.lasso", null);
+ if (!lasso)
+ return;
+ var ids = lassoed();
+ lasso.close();
+ if (ids.length) {
+ context.enter(modeSelect(context, ids));
+ }
+ }
+ selection2.on(_pointerPrefix + "down.lasso", pointerdown);
+ };
+ behavior.off = function(selection2) {
+ selection2.on(_pointerPrefix + "down.lasso", null);
+ };
+ return behavior;
+ }
+
+ // modules/modes/browse.js
+ function modeBrowse(context) {
+ var mode = {
+ button: "browse",
+ id: "browse",
+ title: _t.append("modes.browse.title"),
+ description: _t.append("modes.browse.description")
+ };
+ var sidebar;
+ var _selectBehavior;
+ var _behaviors = [];
+ mode.selectBehavior = function(val) {
+ if (!arguments.length)
+ return _selectBehavior;
+ _selectBehavior = val;
+ return mode;
+ };
+ 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
+ ];
+ }
+ _behaviors.forEach(context.install);
+ if (document.activeElement && document.activeElement.blur) {
+ document.activeElement.blur();
+ }
+ if (sidebar) {
+ context.ui().sidebar.show(sidebar);
+ } else {
+ context.ui().sidebar.select(null);
+ }
+ };
+ mode.exit = function() {
+ context.ui().sidebar.hover.cancel();
+ _behaviors.forEach(context.uninstall);
+ if (sidebar) {
+ context.ui().sidebar.hide();
+ }
+ };
+ mode.sidebar = function(_) {
+ if (!arguments.length)
+ return sidebar;
+ sidebar = _;
+ return mode;
+ };
+ mode.operations = function() {
+ return [operationPaste(context)];
+ };
+ return mode;
+ }
+
+ // modules/behavior/add_way.js
+ function behaviorAddWay(context) {
+ var dispatch10 = dispatch_default("start", "startFromWay", "startFromNode");
+ var draw = behaviorDraw(context);
+ function behavior(surface) {
+ draw.on("click", function() {
+ dispatch10.apply("start", this, arguments);
+ }).on("clickWay", function() {
+ dispatch10.apply("startFromWay", this, arguments);
+ }).on("clickNode", function() {
+ dispatch10.apply("startFromNode", this, arguments);
+ }).on("cancel", behavior.cancel).on("finish", behavior.cancel);
+ context.map().dblclickZoomEnable(false);
+ surface.call(draw);
+ }
+ behavior.off = function(surface) {
+ surface.call(draw.off);
+ };
+ behavior.cancel = function() {
+ window.setTimeout(function() {
+ context.map().dblclickZoomEnable(true);
+ }, 1e3);
+ context.enter(modeBrowse(context));
+ };
+ return utilRebind(behavior, dispatch10, "on");
+ }
+
+ // modules/behavior/hash.js
+ function behaviorHash(context) {
+ var _cachedHash = null;
+ var _latitudeLimit = 90 - 1e-8;
+ function computedHashParameters() {
+ var map2 = context.map();
+ var center = map2.center();
+ var zoom = map2.zoom();
+ var precision2 = 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(id2) {
+ return context.hasEntity(id2);
+ });
+ if (selected.length) {
+ newParams.id = selected.join(",");
+ }
+ newParams.map = zoom.toFixed(2) + "/" + center[1].toFixed(precision2) + "/" + center[0].toFixed(precision2);
+ return Object.assign(oldParams, newParams);
+ }
+ 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(id2) {
+ return context.hasEntity(id2);
+ });
+ if (selected.length) {
+ var firstLabel = utilDisplayLabel(context.entity(selected[0]), context.graph());
+ if (selected.length > 1) {
+ contextual = _t("title.labeled_and_more", {
+ labeled: firstLabel,
+ count: selected.length - 1
+ });
+ } else {
+ contextual = firstLabel;
+ }
+ titleID = "context";
+ }
+ if (includeChangeCount) {
+ changeCount = context.history().difference().summary().length;
+ if (changeCount > 0) {
+ titleID = contextual ? "changes_context" : "changes";
+ }
+ }
+ if (titleID) {
+ return _t("title.format." + titleID, {
+ changes: changeCount,
+ base: baseTitle,
+ context: contextual
+ });
+ }
+ return baseTitle;
+ }
+ function updateTitle(includeChangeCount) {
+ if (!context.setsDocumentTitle())
+ return;
+ var newTitle = computedTitle(includeChangeCount);
+ if (document.title !== newTitle) {
+ document.title = newTitle;
+ }
+ }
+ function updateHashIfNeeded() {
+ if (context.inIntro())
+ return;
+ var latestHash = computedHash();
+ if (_cachedHash !== latestHash) {
+ _cachedHash = latestHash;
+ window.history.replaceState(null, computedTitle(
+ false
+ /* includeChangeCount */
+ ), latestHash);
+ updateTitle(
+ true
+ /* includeChangeCount */
+ );
+ const q = utilStringQs(latestHash);
+ if (q.map) {
+ corePreferences("map-location", q.map);
+ }
+ }
+ }
+ var _throttledUpdate = throttle_default(updateHashIfNeeded, 500);
+ var _throttledUpdateTitle = throttle_default(function() {
+ updateTitle(
+ true
+ /* includeChangeCount */
+ );
+ }, 500);
+ function hashchange() {
+ if (window.location.hash === _cachedHash)
+ return;
+ _cachedHash = window.location.hash;
+ var q = utilStringQs(_cachedHash);
+ var mapArgs = (q.map || "").split("/").map(Number);
+ if (mapArgs.length < 3 || mapArgs.some(isNaN)) {
+ updateHashIfNeeded();
+ } else {
+ if (_cachedHash === computedHash())
+ return;
+ var mode = context.mode();
+ context.map().centerZoom([mapArgs[2], Math.min(_latitudeLimit, Math.max(-_latitudeLimit, mapArgs[1]))], mapArgs[0]);
+ if (q.id && mode) {
+ var ids = q.id.split(",").filter(function(id2) {
+ return context.hasEntity(id2);
+ });
+ if (ids.length && (mode.id === "browse" || mode.id === "select" && !utilArrayIdentical(mode.selectedIDs(), ids))) {
+ context.enter(modeSelect(context, ids));
+ return;
+ }
+ }
+ var center = context.map().center();
+ var dist = geoSphericalDistance(center, [mapArgs[2], mapArgs[1]]);
+ var maxdist = 500;
+ if (mode && mode.id.match(/^draw/) !== null && dist > maxdist) {
+ context.enter(modeBrowse(context));
+ return;
+ }
+ }
+ }
+ function behavior() {
+ context.map().on("move.behaviorHash", _throttledUpdate);
+ context.history().on("change.behaviorHash", _throttledUpdateTitle);
+ context.on("enter.behaviorHash", _throttledUpdate);
+ select_default2(window).on("hashchange.behaviorHash", hashchange);
+ var q = utilStringQs(window.location.hash);
+ if (q.id) {
+ context.zoomToEntity(q.id.split(",")[0], !q.map);
+ }
+ if (q.walkthrough === "true") {
+ behavior.startWalkthrough = true;
+ }
+ if (q.map) {
+ behavior.hadLocation = true;
+ } else if (!q.id && corePreferences("map-location")) {
+ const mapArgs = corePreferences("map-location").split("/").map(Number);
+ context.map().centerZoom([mapArgs[2], Math.min(_latitudeLimit, Math.max(-_latitudeLimit, mapArgs[1]))], mapArgs[0]);
+ updateHashIfNeeded();
+ behavior.hadLocation = true;
+ }
+ hashchange();
+ updateTitle(false);
+ }
+ behavior.off = function() {
+ _throttledUpdate.cancel();
+ _throttledUpdateTitle.cancel();
+ context.map().on("move.behaviorHash", null);
+ context.on("enter.behaviorHash", null);
+ select_default2(window).on("hashchange.behaviorHash", null);
+ window.location.hash = "";
+ };
+ return behavior;
+ }
+
+ // node_modules/d3-brush/src/brush.js
+ var { abs: abs2, max: max2, min: min2 } = Math;
+ function number1(e) {
+ return [+e[0], +e[1]];
+ }
+ function number22(e) {
+ return [number1(e[0]), number1(e[1])];
+ }
+ var X = {
+ name: "x",
+ handles: ["w", "e"].map(type),
+ input: function(x, e) {
+ return x == null ? null : [[+x[0], e[0][1]], [+x[1], e[1][1]]];
+ },
+ output: function(xy) {
+ return xy && [xy[0][0], xy[1][0]];
+ }
+ };
+ var Y = {
+ name: "y",
+ handles: ["n", "s"].map(type),
+ input: function(y, e) {
+ return y == null ? null : [[e[0][0], +y[0]], [e[1][0], +y[1]]];
+ },
+ output: function(xy) {
+ return xy && [xy[0][1], xy[1][1]];
+ }
+ };
+ var XY = {
+ name: "xy",
+ handles: ["n", "w", "e", "s", "nw", "ne", "sw", "se"].map(type),
+ input: function(xy) {
+ return xy == null ? null : number22(xy);
+ },
+ output: function(xy) {
+ return xy;
+ }
+ };
+ function type(t) {
+ return { type: t };
+ }
+
+ // modules/index.js
+ var debug = false;
+ var d3 = {
+ dispatch: dispatch_default,
+ geoMercator: mercator_default,
+ geoProjection: projection,
+ polygonArea: area_default3,
+ polygonCentroid: centroid_default2,
+ select: select_default2,
+ selectAll: selectAll_default2,
+ timerFlush
+ };
+
+ // modules/id.js
+ window.requestIdleCallback = window.requestIdleCallback || function(cb) {
+ var start2 = Date.now();
+ return window.requestAnimationFrame(function() {
+ cb({
+ didTimeout: false,
+ timeRemaining: function() {
+ return Math.max(0, 50 - (Date.now() - start2));
+ }
+ });
+ });
+ };
+ window.cancelIdleCallback = window.cancelIdleCallback || function(id2) {
+ window.cancelAnimationFrame(id2);
+ };
+ window.iD = modules_exports;
+})();
+//# sourceMappingURL=iD.js.map