/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 = "{".concat(key, "}");
+ const regex = new RegExp(token, "g");
+ result = result.replace(regex, value);
+ }
}
}
- 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 i3 = 0; i3 < this.members.length; i3++) {
- result[i3] = Object.assign({}, this.members[i3], { index: i3 });
- }
- 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 i3 = 0; i3 < this.members.length; i3++) {
- if (this.members[i3].role === role) {
- return Object.assign({}, this.members[i3], { index: i3 });
+ if (typeof result === "string") {
+ return {
+ text: result,
+ locale: locale2
+ };
}
}
- },
- // Same as memberByRole, but returns all members with the given role
- membersByRole: function(role) {
- var result = [];
- for (var i3 = 0; i3 < this.members.length; i3++) {
- if (this.members[i3].role === role) {
- result.push(Object.assign({}, this.members[i3], { index: i3 }));
- }
+ let index = _localeCodes.indexOf(locale2);
+ if (index >= 0 && index < _localeCodes.length - 1) {
+ let fallback = _localeCodes[index + 1];
+ return localizer.tInfo(origStringId, replacements, fallback);
}
- 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 i3 = 0; i3 < this.members.length; i3++) {
- if (this.members[i3].id === id2) {
- return Object.assign({}, this.members[i3], { index: i3 });
+ if (replacements && "default" in replacements) {
+ return {
+ text: replacements.default,
+ locale: null
+ };
+ }
+ const missing = "Missing ".concat(locale2, " translation: ").concat(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 k2 in replacements) {
+ if (typeof replacements[k2] === "string") {
+ replacements[k2] = escape_default(replacements[k2]);
+ }
+ if (typeof replacements[k2] === "object" && typeof replacements[k2].html === "string") {
+ replacements[k2] = replacements[k2].html;
}
}
- },
- // 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 i3 = 0; i3 < this.members.length; i3++) {
- if (this.members[i3].id === id2 && this.members[i3].role === role) {
- return Object.assign({}, this.members[i3], { index: i3 });
+ const info = localizer.tInfo(stringId, replacements, locale2);
+ if (info.text) {
+ return '').concat(info.text, "");
+ } 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 && _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 && _languageNames[base]) {
+ const scriptCode = langInfo.script;
+ const script = _scriptNames && _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 });
+ }
}
}
- },
- 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(m2) {
- return m2.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 i3 = 0; i3 < this.members.length; i3++) {
- var member = this.members[i3];
- 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 code;
+ };
+ localizer.floatFormatter = (locale2) => {
+ if (!("Intl" in window && "NumberFormat" in Intl && "formatToParts" in Intl.NumberFormat.prototype)) {
+ return (number3, fractionDigits) => {
+ return fractionDigits === void 0 ? number3.toString() : number3.toFixed(fractionDigits);
+ };
+ } else {
+ return (number3, fractionDigits) => number3.toLocaleString(locale2, {
+ minimumFractionDigits: fractionDigits,
+ maximumFractionDigits: fractionDigits === void 0 ? 20 : fractionDigits
+ });
+ }
+ };
+ localizer.floatParser = (locale2) => {
+ const polyfill = (string) => +string.trim();
+ if (!("Intl" in window && "NumberFormat" in Intl))
+ return polyfill;
+ const format2 = new Intl.NumberFormat(locale2, { maximumFractionDigits: 20 });
+ if (!("formatToParts" in format2))
+ return polyfill;
+ const parts = format2.formatToParts(-12345.6);
+ const numerals = Array.from({ length: 10 }).map((_2, i3) => format2.format(i3));
+ const index = new Map(numerals.map((d2, i3) => [d2, i3]));
+ const literalPart = parts.find((d2) => d2.type === "literal");
+ const literal = literalPart && new RegExp("[".concat(literalPart.value, "]"), "g");
+ const groupPart = parts.find((d2) => d2.type === "group");
+ const group = groupPart && new RegExp("[".concat(groupPart.value, "]"), "g");
+ const decimalPart = parts.find((d2) => d2.type === "decimal");
+ const decimal = decimalPart && new RegExp("[".concat(decimalPart.value, "]"));
+ const numeral = new RegExp("[".concat(numerals.join(""), "]"), "g");
+ const getIndex = (d2) => index.get(d2);
+ return (string) => {
+ string = string.trim();
+ if (literal)
+ string = string.replace(literal, "");
+ if (group)
+ string = string.replace(group, "");
+ if (decimal)
+ string = string.replace(decimal, ".");
+ string = string.replace(numeral, getIndex);
+ return string ? +string : NaN;
+ };
+ };
+ localizer.decimalPlaceCounter = (locale2) => {
+ var literal, group, decimal;
+ if ("Intl" in window && "NumberFormat" in Intl) {
+ const format2 = new Intl.NumberFormat(locale2, { maximumFractionDigits: 20 });
+ if ("formatToParts" in format2) {
+ const parts = format2.formatToParts(-12345.6);
+ const literalPart = parts.find((d2) => d2.type === "literal");
+ literal = literalPart && new RegExp("[".concat(literalPart.value, "]"), "g");
+ const groupPart = parts.find((d2) => d2.type === "group");
+ group = groupPart && new RegExp("[".concat(groupPart.value, "]"), "g");
+ const decimalPart = parts.find((d2) => d2.type === "decimal");
+ decimal = decimalPart && new RegExp("[".concat(decimalPart.value, "]"));
}
}
- return this.update({ members });
- },
- asJXON: function(changeset_id) {
- var r2 = {
- 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)
+ return (string) => {
+ string = string.trim();
+ if (literal)
+ string = string.replace(literal, "");
+ if (group)
+ string = string.replace(group, "");
+ const parts = string.split(decimal || ".");
+ return parts && parts[1] && parts[1].length || 0;
+ };
+ };
+ 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((d2) => d2.id === id2);
+ if (found)
+ _memo[id2] = found;
+ return found;
+ };
+ _this.index = (id2) => _this.collection.findIndex((d2) => d2.id === id2);
+ _this.matchGeometry = (geometry) => {
+ return presetCollection(
+ _this.collection.filter((d2) => d2.matchGeometry(geometry))
+ );
+ };
+ _this.matchAllGeometry = (geometries) => {
+ return presetCollection(
+ _this.collection.filter((d2) => d2 && d2.matchAllGeometry(geometries))
+ );
+ };
+ _this.matchAnyGeometry = (geometries) => {
+ return presetCollection(
+ _this.collection.filter((d2) => geometries.some((geom) => d2.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(a2) {
+ const index = a2.indexOf(value);
+ return index === 0 || a2[index - 1] === " ";
+ }
+ function leadingStrict(a2) {
+ const index = a2.indexOf(value);
+ return index === 0;
+ }
+ function sortPresets(nameProp, aliasesProp) {
+ return function sortNames(a2, b2) {
+ let aCompare = a2[nameProp]();
+ let bCompare = b2[nameProp]();
+ if (aliasesProp) {
+ const findMatchingAlias = (strings) => {
+ if (strings.some((s2) => s2 === value)) {
+ return strings.find((s2) => s2 === value);
+ } else {
+ return strings.filter((s2) => s2.includes(value)).sort((a3, b3) => a3.length - b3.length)[0];
}
};
- }, this),
- tag: Object.keys(this.tags).map(function(k2) {
- return { keyAttributes: { k: k2, v: this.tags[k2] } };
- }, this)
- }
- };
- if (changeset_id) {
- r2.relation["@changeset"] = changeset_id;
+ aCompare = findMatchingAlias([aCompare].concat(a2[aliasesProp]()));
+ bCompare = findMatchingAlias([bCompare].concat(b2[aliasesProp]()));
+ }
+ if (value === aCompare)
+ return -1;
+ if (value === bCompare)
+ return 1;
+ let i3 = b2.originalScore - a2.originalScore;
+ if (i3 !== 0)
+ return i3;
+ i3 = aCompare.indexOf(value) - bCompare.indexOf(value);
+ if (i3 !== 0)
+ return i3;
+ return aCompare.length - bCompare.length;
+ };
}
- return r2;
- },
- 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));
+ let pool = _this.collection;
+ if (Array.isArray(loc)) {
+ const validHere = _sharedLocationManager.locationSetsAt(loc);
+ pool = pool.filter((a2) => !a2.locationSetID || validHere[a2.locationSetID]);
+ }
+ const searchable = pool.filter((a2) => a2.searchable !== false && a2.suggestion !== true);
+ const suggestions = pool.filter((a2) => a2.suggestion === true);
+ const leadingNames = searchable.filter((a2) => leading(a2.searchName()) || a2.searchAliases().some(leading)).sort(sortPresets("searchName", "searchAliases"));
+ const leadingSuggestions = suggestions.filter((a2) => leadingStrict(a2.searchName())).sort(sortPresets("searchName"));
+ const leadingNamesStripped = searchable.filter((a2) => leading(a2.searchNameStripped()) || a2.searchAliasesStripped().some(leading)).sort(sortPresets("searchNameStripped", "searchAliasesStripped"));
+ const leadingSuggestionsStripped = suggestions.filter((a2) => leadingStrict(a2.searchNameStripped())).sort(sortPresets("searchNameStripped"));
+ const leadingTerms = searchable.filter((a2) => (a2.terms() || []).some(leading));
+ const leadingSuggestionTerms = suggestions.filter((a2) => (a2.terms() || []).some(leading));
+ const leadingTagValues = searchable.filter((a2) => Object.values(a2.tags || {}).filter((val) => val !== "*").some(leading));
+ const similarName = searchable.map((a2) => ({ preset: a2, dist: utilEditDistance(value, a2.searchName()) })).filter((a2) => a2.dist + Math.min(value.length - a2.preset.searchName().length, 0) < 3).sort((a2, b2) => a2.dist - b2.dist).map((a2) => a2.preset);
+ const similarSuggestions = suggestions.map((a2) => ({ preset: a2, dist: utilEditDistance(value, a2.searchName()) })).filter((a2) => a2.dist + Math.min(value.length - a2.preset.searchName().length, 0) < 1).sort((a2, b2) => a2.dist - b2.dist).map((a2) => a2.preset);
+ const similarTerms = searchable.filter((a2) => {
+ return (a2.terms() || []).some((b2) => {
+ return utilEditDistance(value, b2) + Math.min(value.length - b2.length, 0) < 3;
+ });
});
- },
- isMultipolygon: function() {
- return this.tags.type === "multipolygon";
- },
- isComplete: function(resolver) {
- for (var i3 = 0; i3 < this.members.length; i3++) {
- if (!resolver.hasEntity(this.members[i3].id)) {
- return false;
+ let leadingTagKeyValues = [];
+ if (value.includes("=")) {
+ leadingTagKeyValues = searchable.filter((a2) => a2.tags && Object.keys(a2.tags).some((key) => key + "=" + a2.tags[key] === value)).concat(searchable.filter((a2) => a2.tags && Object.keys(a2.tags).some((key) => leading(key + "=" + a2.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 true;
- },
- hasFromViaTo: function() {
- return this.members.some(function(m2) {
- return m2.role === "from";
- }) && this.members.some(function(m2) {
- return m2.role === "via";
- }) && this.members.some(function(m2) {
- return m2.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(m2) {
- return m2.role === "from";
- });
- var vias = this.members.filter(function(m2) {
- return m2.role === "via";
- });
- var tos = this.members.filter(function(m2) {
- return m2.role === "to";
- });
- if (froms.length !== 1 && this.tags.restriction !== "no_entry")
- return false;
- if (froms.some(function(m2) {
- return m2.type !== "way";
- }))
- return false;
- if (tos.length !== 1 && this.tags.restriction !== "no_exit")
- return false;
- if (tos.some(function(m2) {
- return m2.type !== "way";
- }))
- return false;
- if (vias.length === 0)
- return false;
- if (vias.length > 1 && vias.some(function(m2) {
- return m2.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(m2) {
- return "outer" === (m2.role || "outer");
- });
- var inners = this.members.filter(function(m2) {
- return "inner" === m2.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 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 i3 in preset.geometry) {
+ const geometry = preset.geometry[i3];
+ if (acc.indexOf(geometry) === -1) {
+ acc.push(geometry);
}
- return sequence.nodes.map(function(node) {
- return node.loc;
- });
- };
- outers = outers.map(sequenceToLineString);
- inners = inners.map(sequenceToLineString);
- var result = outers.map(function(o3) {
- return [area_default({ type: "Polygon", coordinates: [o3] }) > 2 * Math.PI ? o3.reverse() : o3];
- });
- function findOuter(inner2) {
- var o3, outer;
- for (o3 = 0; o3 < outers.length; o3++) {
- outer = outers[o3];
- if (geoPolygonContainsPolygon(outer, inner2)) {
- return o3;
- }
+ }
+ 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.".concat(categoryID, ".name"), { "default": categoryID });
+ _this.nameLabel = () => _t.append("_tagging.presets.categories.".concat(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.".concat(fieldID, ".").concat(scope), options2);
+ _this.t.html = (scope, options2) => _t.html("_tagging.presets.fields.".concat(fieldID, ".").concat(scope), options2);
+ _this.t.append = (scope, options2) => _t.append("_tagging.presets.fields.".concat(fieldID, ".").concat(scope), options2);
+ _this.hasTextForStringId = (scope) => _mainLocalizer.hasTextForStringId("_tagging.presets.fields.".concat(fieldID, ".").concat(scope));
+ _this.resolveReference = (which) => {
+ const referenceRegex = /^\{(.*)\}$/;
+ const match = (field[which] || "").match(referenceRegex);
+ if (match) {
+ const field2 = allFields[match[1]];
+ if (field2) {
+ return field2;
}
- for (o3 = 0; o3 < outers.length; o3++) {
- outer = outers[o3];
- if (geoPolygonIntersectsPolygon(outer, inner2, false)) {
- return o3;
- }
+ console.error("Unable to resolve referenced field: ".concat(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
+ var import_lodash = __toESM(require_lodash());
+ function presetPreset(presetID, preset, addable, allFields, allPresets) {
+ allFields = allFields || {};
+ allPresets = allPresets || {};
+ let _this = Object.assign({}, preset);
+ let _addable = addable || false;
+ 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 = (loc) => resolveFields("fields", loc);
+ _this.moreFields = (loc) => resolveFields("moreFields", loc);
+ _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 k2 in tags) {
+ seen[k2] = true;
+ if (entityTags[k2] === tags[k2]) {
+ score += _this.originalScore;
+ } else if (tags[k2] === "*" && k2 in entityTags) {
+ score += _this.originalScore / 2;
+ } else {
+ return -1;
}
}
- for (var i3 = 0; i3 < inners.length; i3++) {
- var inner = inners[i3];
- if (area_default({ type: "Polygon", coordinates: [inner] }) < 2 * Math.PI) {
- inner = inner.reverse();
+ const addTags = _this.addTags;
+ for (let k2 in addTags) {
+ if (!seen[k2] && entityTags[k2] === addTags[k2]) {
+ score += _this.originalScore;
}
- var o2 = findOuter(inners[i3]);
- if (o2 !== void 0) {
- result[o2].push(inners[i3]);
- } else {
- result.push([inners[i3]]);
+ }
+ if (_this.searchable === false) {
+ score *= 0.999;
+ }
+ return score;
+ };
+ _this.t = (scope, options2) => {
+ const textID = "_tagging.presets.presets.".concat(presetID, ".").concat(scope);
+ return _t(textID, options2);
+ };
+ _this.t.append = (scope, options2) => {
+ const textID = "_tagging.presets.presets.".concat(presetID, ".").concat(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: ".concat(match[1]));
}
- return result;
+ return _this;
}
- });
-
- // modules/osm/qa_item.js
- var QAItem = class _QAItem {
- 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);
+ _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");
}
- }
- 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 i3;
- var best = 0;
- var idxB;
- function wrap2(index) {
- return utilWrap(index, nodes.length);
+ return null;
+ };
+ _this.subtitleLabel = () => {
+ if (_this.suggestion) {
+ let path = presetID.split("/");
+ path.pop();
+ return _t.append("_tagging.presets.presets." + path.join("/") + ".name");
}
- length = 0;
- for (i3 = wrap2(idxA + 1); i3 !== idxA; i3 = wrap2(i3 + 1)) {
- length += dist(graph, nodes[i3], nodes[wrap2(i3 - 1)]);
- lengths[i3] = length;
+ 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();
}
- length = 0;
- for (i3 = wrap2(idxA - 1); i3 !== idxA; i3 = wrap2(i3 - 1)) {
- length += dist(graph, nodes[i3], nodes[wrap2(i3 + 1)]);
- if (length < lengths[i3]) {
- lengths[i3] = length;
- }
+ return _searchName;
+ };
+ _this.searchNameStripped = () => {
+ if (!_searchNameStripped) {
+ _searchNameStripped = stripDiacritics(_this.searchName());
}
- for (i3 = 0; i3 < nodes.length; i3++) {
- var cost = lengths[i3] / dist(graph, nodes[idxA], nodes[i3]);
- if (cost > best) {
- idxB = i3;
- best = cost;
- }
+ return _searchNameStripped;
+ };
+ _this.searchAliases = () => {
+ if (!_searchAliases) {
+ _searchAliases = _this.aliases().map((alias) => alias.toLowerCase());
}
- return idxB;
- }
- function totalLengthBetweenNodes(graph, nodes) {
- var totalLength = 0;
- for (var i3 = 0; i3 < nodes.length - 1; i3++) {
- totalLength += dist(graph, nodes[i3], nodes[i3 + 1]);
+ return _searchAliases;
+ };
+ _this.searchAliasesStripped = () => {
+ if (!_searchAliasesStripped) {
+ _searchAliasesStripped = _this.searchAliases();
+ _searchAliasesStripped = _searchAliasesStripped.map(stripDiacritics);
}
- 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);
+ 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 };
}
- 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;
+ 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 {
- wayA = wayA.update({ nodes: nodesA });
- wayB = wayB.update({ nodes: nodesB });
+ return { key, value };
}
- 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 });
+ };
+ _this.unsetTags = (tags, geometry, ignoringKeys, skipFieldDefaults, loc) => {
+ let removeTags = ignoringKeys ? utilObjectOmit(_this.removeTags, ignoringKeys) : _this.removeTags;
+ tags = utilObjectOmit(tags, Object.keys(removeTags));
+ if (geometry && !skipFieldDefaults) {
+ _this.fields(loc).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, loc) => {
+ const addTags = _this.addTags;
+ tags = Object.assign({}, tags);
+ for (let k2 in addTags) {
+ if (addTags[k2] === "*") {
+ if (_this.tags[k2] || !tags[k2]) {
+ tags[k2] = "yes";
+ }
+ } else {
+ tags[k2] = addTags[k2];
}
}
- graph = graph.replace(wayA);
- graph = graph.replace(wayB);
- graph.parentRelations(wayA).forEach(function(relation) {
- var member;
- if (relation.hasFromViaTo()) {
- var f3 = relation.memberByRole("from");
- var v2 = relation.membersByRole("via");
- var t2 = relation.memberByRole("to");
- var i3;
- if (f3.id === wayA.id || t2.id === wayA.id) {
- var keepB = false;
- if (v2.length === 1 && v2[0].type === "node") {
- keepB = wayB.contains(v2[0].id);
- } else {
- for (i3 = 0; i3 < v2.length; i3++) {
- if (v2[i3].type === "way") {
- var wayVia = graph.hasEntity(v2[i3].id);
- if (wayVia && utilArrayIntersection(wayB.nodes, wayVia.nodes).length) {
- keepB = true;
- break;
- }
- }
- }
+ if (!addTags.hasOwnProperty("area")) {
+ delete tags.area;
+ if (geometry === "area") {
+ let needsAreaTag = true;
+ for (let k2 in addTags) {
+ if (_this.geometry.indexOf("line") === -1 && k2 in osmAreaKeys || k2 in osmAreaKeysExceptions && addTags[k2] in osmAreaKeysExceptions[k2]) {
+ needsAreaTag = false;
+ break;
}
- if (keepB) {
- relation = relation.replaceMember(wayA, wayB);
- graph = graph.replace(relation);
+ }
+ if (needsAreaTag) {
+ tags.area = "yes";
+ }
+ }
+ }
+ if (geometry && !skipFieldDefaults) {
+ _this.fields(loc).forEach((field) => {
+ if (field.matchGeometry(geometry) && field.key && !tags[field.key] && field.default) {
+ tags[field.key] = field.default;
+ }
+ });
+ }
+ return tags;
+ };
+ function resolveFields(which, loc) {
+ 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(allPresets[match[1]], which));
+ } else if (allFields[fieldID]) {
+ resolved.push(allFields[fieldID]);
+ } else {
+ console.log('Cannot resolve "'.concat(fieldID, '" found in ').concat(_this.id, ".").concat(which));
+ }
+ });
+ if (!resolved.length) {
+ const endIndex = _this.id.lastIndexOf("/");
+ const parentID = endIndex && _this.id.substring(0, endIndex);
+ if (parentID) {
+ let parent = allPresets[parentID];
+ if (loc) {
+ const validHere = _sharedLocationManager.locationSetsAt(loc);
+ if ((parent == null ? void 0 : parent.locationSetID) && !validHere[parent.locationSetID]) {
+ const candidateIDs = Object.keys(allPresets).filter((k2) => k2.startsWith(parentID));
+ parent = allPresets[candidateIDs.find((candidateID) => {
+ const candidate = allPresets[candidateID];
+ return validHere[candidate.locationSetID] && (0, import_lodash.isEqual)(candidate.tags, parent.tags);
+ })];
}
+ }
+ resolved = inheritFields(parent, which);
+ }
+ }
+ return utilArrayUniq(resolved);
+ function inheritFields(parent, which2) {
+ 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(s2) {
+ if (s2.normalize)
+ s2 = s2.normalize("NFD");
+ s2 = s2.replace(/[\u0300-\u036f]/g, "");
+ return s2;
+ }
+ return _this;
+ }
+
+ // modules/presets/index.js
+ var _mainPresetIndex = presetIndex();
+ function presetIndex() {
+ const dispatch14 = 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 _defaults2 = {
+ 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 = (d2) => {
+ let newLocationSets = [];
+ if (d2.fields) {
+ Object.keys(d2.fields).forEach((fieldID) => {
+ let f2 = d2.fields[fieldID];
+ if (f2) {
+ f2 = presetField(fieldID, f2, _fields);
+ if (f2.locationSet)
+ newLocationSets.push(f2);
+ _fields[fieldID] = f2;
} else {
- for (i3 = 0; i3 < v2.length; i3++) {
- if (v2[i3].type === "way" && v2[i3].id === wayA.id) {
- member = {
- id: wayB.id,
- type: "way",
- role: "via"
- };
- graph = actionAddMember(relation.id, member, v2[i3].index + 1)(graph);
- break;
- }
+ delete _fields[fieldID];
+ }
+ });
+ }
+ if (d2.presets) {
+ Object.keys(d2.presets).forEach((presetID) => {
+ let p2 = d2.presets[presetID];
+ if (p2) {
+ const isAddable = !_addablePresetIDs || _addablePresetIDs.has(presetID);
+ p2 = presetPreset(presetID, p2, isAddable, _fields, _presets);
+ if (p2.locationSet)
+ newLocationSets.push(p2);
+ _presets[presetID] = p2;
+ } else {
+ const existing = _presets[presetID];
+ if (existing && !existing.isFallback()) {
+ delete _presets[presetID];
}
}
- } else {
- if (relation === isOuter) {
- graph = graph.replace(relation.mergeTags(wayA.tags));
- graph = graph.replace(wayA.update({ tags: {} }));
- graph = graph.replace(wayB.update({ tags: {} }));
+ });
+ }
+ if (d2.categories) {
+ Object.keys(d2.categories).forEach((categoryID) => {
+ let c2 = d2.categories[categoryID];
+ if (c2) {
+ c2 = presetCategory(categoryID, c2, _presets);
+ if (c2.locationSet)
+ newLocationSets.push(c2);
+ _categories[categoryID] = c2;
+ } else {
+ delete _categories[categoryID];
+ }
+ });
+ }
+ _this.collection = Object.values(_presets).concat(Object.values(_categories));
+ if (d2.defaults) {
+ Object.keys(d2.defaults).forEach((geometry) => {
+ const def2 = d2.defaults[geometry];
+ if (Array.isArray(def2)) {
+ _defaults2[geometry] = presetCollection(
+ def2.map((id2) => _presets[id2] || _categories[id2]).filter(Boolean)
+ );
+ } else {
+ delete _defaults2[geometry];
+ }
+ });
+ }
+ _universal = Object.values(_fields).filter((field) => field.universal);
+ _geometryIndex = { point: {}, vertex: {}, line: {}, area: {}, relation: {} };
+ _this.collection.forEach((preset) => {
+ (preset.geometry || []).forEach((geometry) => {
+ let g3 = _geometryIndex[geometry];
+ for (let key in preset.tags) {
+ g3[key] = g3[key] || {};
+ let value = preset.tags[key];
+ (g3[key][value] = g3[key][value] || []).push(preset);
}
- 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: {} }));
+ });
+ if (d2.featureCollection && Array.isArray(d2.featureCollection.features)) {
+ _sharedLocationManager.mergeCustomGeoJSON(d2.featureCollection);
}
- _createdWayIDs.push(wayB.id);
- return graph;
- }
- var action = function(graph) {
- _createdWayIDs = [];
- var newWayIndex = 0;
- for (var i3 = 0; i3 < nodeIds.length; i3++) {
- var nodeId = nodeIds[i3];
- var candidates = action.waysForNode(nodeId, graph);
- for (var j3 = 0; j3 < candidates.length; j3++) {
- graph = split(graph, nodeId, candidates[j3], newWayIds && newWayIds[newWayIndex]);
- newWayIndex += 1;
- }
+ if (newLocationSets.length) {
+ _sharedLocationManager.mergeLocationSets(newLocationSets);
}
- return graph;
+ return _this;
};
- action.getCreatedWayIDs = function() {
- return _createdWayIDs;
+ _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());
+ });
};
- 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";
- });
+ _this.matchTags = (tags, geometry, loc) => {
+ const keyIndex = _geometryIndex[geometry];
+ let bestScore = -1;
+ let bestMatch;
+ let matchCandidates = [];
+ for (let k2 in tags) {
+ let indexMatches = [];
+ let valueIndex = keyIndex[k2];
+ if (!valueIndex)
+ continue;
+ let keyValueMatches = valueIndex[tags[k2]];
+ if (keyValueMatches)
+ indexMatches.push(...keyValueMatches);
+ let keyStarMatches = valueIndex["*"];
+ if (keyStarMatches)
+ indexMatches.push(...keyStarMatches);
+ if (indexMatches.length === 0)
+ continue;
+ for (let i3 = 0; i3 < indexMatches.length; i3++) {
+ const candidate = indexMatches[i3];
+ const score = candidate.matchScore(tags);
+ if (score === -1) {
+ continue;
+ }
+ matchCandidates.push({ score, candidate });
+ if (score > bestScore) {
+ bestScore = score;
+ bestMatch = candidate;
+ }
}
}
- return splittableParents;
- function isSplittable(parent) {
- if (_wayIDs && _wayIDs.indexOf(parent.id) === -1)
- return false;
- if (parent.isClosed())
- return true;
- for (var i3 = 1; i3 < parent.nodes.length - 1; i3++) {
- if (parent.nodes[i3] === nodeId)
- return true;
+ if (bestMatch && bestMatch.locationSetID && bestMatch.locationSetID !== "+[Q2]" && Array.isArray(loc)) {
+ const validHere = _sharedLocationManager.locationSetsAt(loc);
+ if (!validHere[bestMatch.locationSetID]) {
+ matchCandidates.sort((a2, b2) => a2.score < b2.score ? 1 : -1);
+ for (let i3 = 0; i3 < matchCandidates.length; i3++) {
+ const candidateScore = matchCandidates[i3];
+ if (!candidateScore.candidate.locationSetID || validHere[candidateScore.candidate.locationSetID]) {
+ bestMatch = candidateScore.candidate;
+ bestScore = candidateScore.score;
+ break;
+ }
+ }
+ }
+ }
+ if (!bestMatch || bestMatch.isFallback()) {
+ for (let k2 in tags) {
+ if (/^addr:/.test(k2) && keyIndex["addr:*"] && keyIndex["addr:*"]["*"]) {
+ bestMatch = keyIndex["addr:*"]["*"][0];
+ break;
+ }
}
- return false;
}
+ return bestMatch || _this.fallback(geometry);
};
- action.ways = function(graph) {
- return utilArrayUniq([].concat.apply([], nodeIds.map(function(nodeId) {
- return action.waysForNode(nodeId, graph);
- })));
+ _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;
+ });
};
- action.disabled = function(graph) {
- for (var i3 = 0; i3 < nodeIds.length; i3++) {
- var nodeId = nodeIds[i3];
- var candidates = action.waysForNode(nodeId, graph);
- if (candidates.length === 0 || _wayIDs && _wayIDs.length !== candidates.length) {
- return "not_eligible";
+ _this.areaKeys = () => {
+ const ignore = {
+ barrier: true,
+ highway: true,
+ footway: true,
+ railway: true,
+ junction: true,
+ type: true
+ };
+ let areaKeys = {};
+ const presets = _this.collection.filter((p2) => !p2.suggestion && !p2.replacement);
+ presets.forEach((p2) => {
+ const keys2 = p2.tags && Object.keys(p2.tags);
+ const key = keys2 && keys2.length && keys2[0];
+ if (!key)
+ return;
+ if (ignore[key])
+ return;
+ if (p2.geometry.indexOf("area") !== -1) {
+ areaKeys[key] = areaKeys[key] || {};
}
- }
+ });
+ presets.forEach((p2) => {
+ let key;
+ for (key in p2.addTags) {
+ const value = p2.addTags[key];
+ if (key in areaKeys && // probably an area...
+ p2.geometry.indexOf("line") !== -1 && // but sometimes a line
+ value !== "*") {
+ areaKeys[key][value] = true;
+ }
+ }
+ });
+ return areaKeys;
};
- action.limitWays = function(val) {
- if (!arguments.length)
- return _wayIDs;
- _wayIDs = val;
- return action;
+ _this.lineTags = () => {
+ return _this.collection.filter((lineTags, d2) => {
+ if (d2.suggestion || d2.replacement || d2.searchable === false)
+ return lineTags;
+ const keys2 = d2.tags && Object.keys(d2.tags);
+ const key = keys2 && keys2.length && keys2[0];
+ if (!key)
+ return lineTags;
+ if (d2.geometry.indexOf("line") !== -1) {
+ lineTags[key] = lineTags[key] || [];
+ lineTags[key].push(d2.tags);
+ }
+ return lineTags;
+ }, {});
};
- action.keepHistoryOn = function(val) {
- if (!arguments.length)
- return _keepHistoryOn;
- _keepHistoryOn = val;
- return action;
+ _this.pointTags = () => {
+ return _this.collection.reduce((pointTags, d2) => {
+ if (d2.suggestion || d2.replacement || d2.searchable === false)
+ return pointTags;
+ const keys2 = d2.tags && Object.keys(d2.tags);
+ const key = keys2 && keys2.length && keys2[0];
+ if (!key)
+ return pointTags;
+ if (d2.geometry.indexOf("point") !== -1) {
+ pointTags[key] = pointTags[key] || {};
+ pointTags[key][d2.tags[key]] = true;
+ }
+ return pointTags;
+ }, {});
};
- 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");
+ _this.vertexTags = () => {
+ return _this.collection.reduce((vertexTags, d2) => {
+ if (d2.suggestion || d2.replacement || d2.searchable === false)
+ return vertexTags;
+ const keys2 = d2.tags && Object.keys(d2.tags);
+ const key = keys2 && keys2.length && keys2[0];
+ if (!key)
+ return vertexTags;
+ if (d2.geometry.indexOf("vertex") !== -1) {
+ vertexTags[key] = vertexTags[key] || {};
+ vertexTags[key][d2.tags[key]] = true;
+ }
+ return vertexTags;
+ }, {});
+ };
+ _this.field = (id2) => _fields[id2];
+ _this.universal = () => _universal;
+ _this.defaults = (geometry, n3, startWithRecents, loc, extraPresets) => {
+ let recents = [];
+ if (startWithRecents) {
+ recents = _this.recent().matchGeometry(geometry).collection.slice(0, 4);
}
- 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];
+ let defaults;
+ if (_addablePresetIDs) {
+ defaults = Array.from(_addablePresetIDs).map(function(id2) {
+ var preset = _this.item(id2);
+ if (preset && preset.matchGeometry(geometry))
+ return preset;
+ return null;
+ }).filter(Boolean);
+ } else {
+ defaults = _defaults2[geometry].collection.concat(_this.fallback(geometry));
}
- 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);
+ let result = presetCollection(
+ utilArrayUniq(recents.concat(defaults).concat(extraPresets || [])).slice(0, n3 - 1)
+ );
+ if (Array.isArray(loc)) {
+ const validHere = _sharedLocationManager.locationSetsAt(loc);
+ result.collection = result.collection.filter((a2) => !a2.locationSetID || validHere[a2.locationSetID]);
}
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);
+ };
+ _this.addablePresetIDs = function(val) {
+ if (!arguments.length)
+ return _addablePresetIDs;
+ if (Array.isArray(val))
+ val = new Set(val);
+ _addablePresetIDs = val;
+ if (_addablePresetIDs) {
+ _this.collection.forEach((p2) => {
+ if (p2.addable)
+ p2.addable(_addablePresetIDs.has(p2.id));
+ });
+ } else {
+ _this.collection.forEach((p2) => {
+ if (p2.addable)
+ p2.addable(true);
+ });
}
- 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 _this;
+ };
+ _this.recent = () => {
+ return presetCollection(
+ utilArrayUniq(_this.getRecents().map((d2) => d2.preset).filter((d2) => d2.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(d2, source) {
+ if (d2 && d2.pID) {
+ const preset = _this.item(d2.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 [];
- var nodes = [];
- for (var i3 = 0; i3 < entity.nodes.length; i3++) {
- nodes[i3] = this.entity(entity.nodes[i3]);
+ 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((d2) => d2.minified());
+ corePreferences("preset_recents", JSON.stringify(minifiedItems));
+ dispatch14.call("recentsChange");
+ }
+ _this.getRecents = () => {
+ if (!_recents) {
+ _recents = (JSON.parse(corePreferences("preset_recents")) || []).reduce((acc, d2) => {
+ let item = ribbonItemForMinified(d2, "recent");
+ if (item && item.preset.addable())
+ acc.push(item);
+ return acc;
+ }, []);
}
- 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 i3, j3, k2, id2;
- for (i3 = 0; i3 < entities.length; i3++) {
- var entity = entities[i3];
- 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 (j3 = 0; j3 < entity.nodes.length; j3++) {
- id2 = entity.nodes[j3];
- for (k2 = 1; k2 < stack.length; k2++) {
- var ents = stack[k2].entities;
- if (ents.hasOwnProperty(id2) && ents[id2] === void 0) {
- delete ents[id2];
- }
- }
- }
+ 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 i3 in items) {
+ if (items[i3].matches(preset)) {
+ return items[i3];
}
}
- for (i3 = 0; i3 < stack.length; i3++) {
- stack[i3]._updateRebased();
+ 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();
}
- },
- _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, i3;
- 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 (i3 = 0; i3 < removed.length; i3++) {
- parentWays[removed[i3]] = new Set(parentWays[removed[i3]]);
- parentWays[removed[i3]].delete(oldentity.id);
- }
- for (i3 = 0; i3 < added.length; i3++) {
- parentWays[added[i3]] = new Set(parentWays[added[i3]]);
- parentWays[added[i3]].add(entity.id);
- }
- } else if (type2 === "relation") {
- var oldentityMemberIDs = oldentity ? oldentity.members.map(function(m2) {
- return m2.id;
- }) : [];
- var entityMemberIDs = entity ? entity.members.map(function(m2) {
- return m2.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 (i3 = 0; i3 < removed.length; i3++) {
- parentRels[removed[i3]] = new Set(parentRels[removed[i3]]);
- parentRels[removed[i3]].delete(oldentity.id);
- }
- for (i3 = 0; i3 < added.length; i3++) {
- parentRels[added[i3]] = new Set(parentRels[added[i3]]);
- parentRels[added[i3]].add(entity.id);
+ items.unshift(item);
+ setRecents(items);
+ };
+ function setFavorites(items) {
+ _favorites = items;
+ const minifiedItems = items.map((d2) => d2.minified());
+ corePreferences("preset_favorites", JSON.stringify(minifiedItems));
+ dispatch14.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"));
}
- },
- 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 i3 = 0; i3 < arguments.length; i3++) {
- arguments[i3].call(graph, graph);
+ setFavorites(favs);
+ };
+ _this.removeFavorite = (preset) => {
+ const item = _this.favoriteMatching(preset);
+ if (item) {
+ const items = _this.getFavorites();
+ items.splice(items.indexOf(item), 1);
+ setFavorites(items);
}
- 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 i3 in entities) {
- this.entities[i3] = entities[i3];
- this._updateCalculated(base.entities[i3], this.entities[i3]);
+ };
+ _this.getFavorites = () => {
+ if (!_favorites) {
+ let rawFavorites = JSON.parse(corePreferences("preset_favorites"));
+ if (!rawFavorites) {
+ rawFavorites = [];
+ corePreferences("preset_favorites", JSON.stringify(rawFavorites));
+ }
+ _favorites = rawFavorites.reduce((output, d2) => {
+ const item = ribbonItemForMinified(d2, "favorite");
+ if (item && item.preset.addable())
+ output.push(item);
+ return output;
+ }, []);
}
- return this;
- }
- };
+ 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, dispatch14, "on");
+ }
- // modules/osm/intersection.js
- function osmTurn(turn) {
- if (!(this instanceof osmTurn)) {
- return new osmTurn(turn);
+ // modules/util/util.js
+ function utilTagText(entity) {
+ var obj = entity && entity.tags || {};
+ return Object.keys(obj).map(function(k2) {
+ return k2 + "=" + obj[k2];
+ }).join(", ");
+ }
+ function utilTotalExtent(array2, graph) {
+ var extent = geoExtent();
+ var val, entity;
+ for (var i3 = 0; i3 < array2.length; i3++) {
+ val = array2[i3];
+ entity = typeof val === "string" ? graph.hasEntity(val) : val;
+ if (entity) {
+ extent._extend(entity.extent(graph));
+ }
}
- Object.assign(this, turn);
+ return extent;
}
- function osmIntersection(graph, startVertexId, maxDistance) {
- maxDistance = maxDistance || 30;
- var vgraph = coreGraph();
- var i3, j3, k2;
- function memberOfRestriction(entity) {
- return graph.parentRelations(entity).some(function(r2) {
- return r2.isRestriction();
+ function utilTagDiff(oldTags, newTags) {
+ var tagDiff = [];
+ var keys2 = utilArrayUnion(Object.keys(oldTags), Object.keys(newTags)).sort();
+ keys2.forEach(function(k2) {
+ var oldVal = oldTags[k2];
+ var newVal = newTags[k2];
+ if ((oldVal || oldVal === "") && (newVal === void 0 || newVal !== oldVal)) {
+ tagDiff.push({
+ type: "-",
+ key: k2,
+ oldVal,
+ newVal,
+ display: "- " + k2 + "=" + oldVal
+ });
+ }
+ if ((newVal || newVal === "") && (oldVal === void 0 || newVal !== oldVal)) {
+ tagDiff.push({
+ type: "+",
+ key: k2,
+ oldVal,
+ newVal,
+ display: "+ " + k2 + "=" + 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 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];
+ }
+ 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);
}
- 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 (i3 = 0; i3 < checkWays.length; i3++) {
- way = checkWays[i3];
- if (!isRoad(way) && !memberOfRestriction(way))
- continue;
- ways.push(way);
- hasWays = true;
- nodes = utilArrayUniq(graph.childNodes(way));
- for (j3 = 0; j3 < nodes.length; j3++) {
- node = nodes[j3];
- 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 (k2 = 0; k2 < parents.length; k2++) {
- parent = parents[k2];
- if (parent === way)
- continue;
- if (ways.indexOf(parent) !== -1)
- continue;
- if (!isRoad(parent))
- continue;
- hasParents = true;
- break;
- }
- if (hasParents) {
- checkVertices.push(node);
- }
- }
+ }
+ 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);
}
- if (hasWays) {
- vertices.push(vertex);
+ 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);
}
}
- 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));
- }
+ }
+ 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");
}
- });
- });
- ways.forEach(function(w2) {
- var way2 = vgraph.entity(w2.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(v2) {
- var splitAll = actionSplit([v2.id]).keepHistoryOn("first");
- if (!splitAll.disabled(vgraph)) {
- splitAll.ways(vgraph).forEach(function(way2) {
- var splitOne = actionSplit([v2.id]).limitWays([way2.id]).keepHistoryOn("first");
- actions.push(splitOne);
- vgraph = splitOne(vgraph);
- });
}
- });
- osmEntity.id.next.way = origCount;
- vertexIds = vertices.map(function(v2) {
- return v2.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(v2) {
- return v2.id;
- });
- wayIds = ways.map(function(w2) {
- return w2.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);
+ 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 keys2 = Object.keys(entity.tags).filter(Boolean);
+ keys2.forEach(function(key2) {
+ allKeys.add(key2);
+ });
});
- var keepGoing;
- var removeWayIds = [];
- var removeVertexIds = [];
- do {
- keepGoing = false;
- checkVertices = vertexIds.slice();
- for (i3 = 0; i3 < checkVertices.length; i3++) {
- var vertexId = checkVertices[i3];
- 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);
+ 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);
+ }
}
}
- if (parents.length === 2) {
- var a2 = parents[0];
- var b2 = parents[1];
- var aIsLeaf = a2 && !a2.__via;
- var bIsLeaf = b2 && !b2.__via;
- var leaf, survivor;
- if (aIsLeaf && !bIsLeaf) {
- leaf = a2;
- survivor = b2;
- } else if (!aIsLeaf && bIsLeaf) {
- leaf = b2;
- survivor = a2;
- }
- if (leaf && survivor) {
- survivor = withMetadata(survivor, vertexIds);
- vgraph = vgraph.replace(survivor).remove(leaf);
- removeWayIds.push(leaf.id);
- keepGoing = true;
- }
+ 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;
}
- 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 (val2 && val12) {
+ return val12.localeCompare(val2);
}
- if (parents.length < 1) {
- vgraph = vgraph.remove(vertex);
+ return val12 ? 1 : -1;
+ });
+ }
+ return tags;
+ }
+ function utilStringQs(str) {
+ var i3 = 0;
+ while (i3 < str.length && (str[i3] === "?" || str[i3] === "#"))
+ i3++;
+ str = str.slice(i3);
+ return str.split("&").reduce(function(obj, pair3) {
+ var parts = pair3.split("=");
+ if (parts.length === 2) {
+ obj[parts[0]] = null === parts[1] ? "" : decodeURIComponent(parts[1]);
+ }
+ return obj;
+ }, {});
+ }
+ function utilQsString(obj, noencode) {
+ function softEncode(s2) {
+ return encodeURIComponent(s2).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 i3 = -1;
+ var n3 = prefixes2.length;
+ var s2 = document.body;
+ if (property in s2)
+ return property;
+ property = property.slice(0, 1).toUpperCase() + property.slice(1);
+ while (++i3 < n3) {
+ if (prefixes2[i3] + property in s2) {
+ return prefixes2[i3] + property;
+ }
+ }
+ return false;
+ }
+ function utilPrefixCSSProperty(property) {
+ var prefixes2 = ["webkit", "ms", "Moz", "O"];
+ var i3 = -1;
+ var n3 = prefixes2.length;
+ var s2 = document.body.style;
+ if (property.toLowerCase() in s2) {
+ return property.toLowerCase();
+ }
+ while (++i3 < n3) {
+ if (prefixes2[i3] + property in s2) {
+ return "-" + prefixes2[i3].toLowerCase() + property.replace(/([A-Z])/g, "-$1").toLowerCase();
+ }
+ }
+ return false;
+ }
+ var transformProperty;
+ function utilSetTransform(el, x2, y2, scale) {
+ var prop = transformProperty = transformProperty || utilPrefixCSSProperty("Transform");
+ var translate = utilDetect().opera ? "translate(" + x2 + "px," + y2 + "px)" : "translate3d(" + x2 + "px," + y2 + "px,0)";
+ return el.style(prop, translate + (scale ? " scale(" + scale + ")" : ""));
+ }
+ function utilEditDistance(a2, b2) {
+ a2 = (0, import_diacritics.remove)(a2.toLowerCase());
+ b2 = (0, import_diacritics.remove)(b2.toLowerCase());
+ if (a2.length === 0)
+ return b2.length;
+ if (b2.length === 0)
+ return a2.length;
+ var matrix = [];
+ var i3, j2;
+ for (i3 = 0; i3 <= b2.length; i3++) {
+ matrix[i3] = [i3];
+ }
+ for (j2 = 0; j2 <= a2.length; j2++) {
+ matrix[0][j2] = j2;
+ }
+ for (i3 = 1; i3 <= b2.length; i3++) {
+ for (j2 = 1; j2 <= a2.length; j2++) {
+ if (b2.charAt(i3 - 1) === a2.charAt(j2 - 1)) {
+ matrix[i3][j2] = matrix[i3 - 1][j2 - 1];
+ } else {
+ matrix[i3][j2] = Math.min(
+ matrix[i3 - 1][j2 - 1] + 1,
+ // substitution
+ Math.min(
+ matrix[i3][j2 - 1] + 1,
+ // insertion
+ matrix[i3 - 1][j2] + 1
+ )
+ );
}
}
- } 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
+ }
+ return matrix[b2.length][a2.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(e3) {
+ return [
+ e3.clientX - rectLeft - clientLeft,
+ e3.clientY - rectTop - clientTop
+ ];
};
- intersection.turns = function(fromWayId, maxViaWay) {
- if (!fromWayId)
- return [];
- if (!maxViaWay)
- maxViaWay = 0;
- var vgraph2 = intersection.graph;
- var keyVertexIds = intersection.vertices.map(function(v2) {
- return v2.id;
+ }
+ function utilAsyncMap(inputs, func, callback) {
+ var remaining = inputs.length;
+ var results = [];
+ var errors = [];
+ inputs.forEach(function(d2, i3) {
+ func(d2, function done(err, data) {
+ errors[i3] = err;
+ results[i3] = data;
+ remaining--;
+ if (!remaining)
+ callback(errors, results);
});
- 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 utilWrap(index, length2) {
+ if (index < 0) {
+ index += Math.ceil(-index / length2) * length2;
+ }
+ return index % length2;
+ }
+ 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(str) {
+ var hash = 0;
+ if (str.length === 0) {
+ return hash;
+ }
+ for (var i3 = 0; i3 < str.length; i3++) {
+ var char = str.charCodeAt(i3);
+ hash = (hash << 5) - hash + char;
+ hash = hash & hash;
+ }
+ return hash;
+ }
+ function utilSafeClassName(str) {
+ return str.toLowerCase().replace(/[^a-z0-9]+/g, "_");
+ }
+ function utilUniqueDomId(val) {
+ return "ideditor-" + utilSafeClassName(val.toString()) + "-" + (/* @__PURE__ */ new Date()).getTime().toString();
+ }
+ function utilUnicodeCharsCount(str) {
+ return Array.from(str).length;
+ }
+ function utilUnicodeCharsTruncated(str, limit) {
+ return Array.from(str).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 i3 = 1; i3 < ids.length; i3++) {
+ var num = toNumericID(ids[i3]);
+ if (compareNumericIDs(oldestID, num) === 1) {
+ oldestIDIndex = i3;
+ oldestID = num;
}
- function stepNode(entity, currPath, currRestrictions) {
- var i4, j4;
- var parents2 = vgraph2.parentWays(entity);
- var nextWays = [];
- for (i4 = 0; i4 < parents2.length; i4++) {
- var way2 = parents2[i4];
- if (way2.__oneWay && way2.nodes[0] !== entity.id)
- continue;
- if (currPath.indexOf(way2.id) !== -1 && currPath.length >= 3)
- continue;
- var restrict = null;
- for (j4 = 0; j4 < currRestrictions.length; j4++) {
- var restriction = currRestrictions[j4];
- var f3 = restriction.memberByRole("from");
- var v2 = restriction.membersByRole("via");
- var t2 = restriction.memberByRole("to");
- var isNo = /^no_/.test(restriction.tags.restriction);
- var isOnly = /^only_/.test(restriction.tags.restriction);
- if (!(isNo || isOnly)) {
- continue;
- }
- var matchesFrom = f3.id === fromWayId;
- var matchesViaTo = false;
- var isAlongOnlyPath = false;
- if (t2.id === way2.id) {
- if (v2.length === 1 && v2[0].type === "node") {
- matchesViaTo = v2[0].id === entity.id && (matchesFrom && currPath.length === 2 || !matchesFrom && currPath.length > 2);
- } else {
- var pathVias = [];
- for (k2 = 2; k2 < currPath.length; k2 += 2) {
- pathVias.push(currPath[k2]);
- }
- var restrictionVias = [];
- for (k2 = 0; k2 < v2.length; k2++) {
- if (v2[k2].type === "way") {
- restrictionVias.push(v2[k2].id);
- }
- }
- var diff = utilArrayDifference(pathVias, restrictionVias);
- matchesViaTo = !diff.length;
- }
- } else if (isOnly) {
- for (k2 = 0; k2 < v2.length; k2++) {
- if (v2[k2].type === "way" && v2[k2].id === way2.id) {
- isAlongOnlyPath = true;
- break;
- }
- }
- }
- if (matchesViaTo) {
- if (isOnly) {
- restrict = { id: restriction.id, direct: matchesFrom, from: f3.id, only: true, end: true };
- } else {
- restrict = { id: restriction.id, direct: matchesFrom, from: f3.id, no: true, end: true };
- }
+ }
+ 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(d2) {
+ var oldKeys = Object.keys(d2.old);
+ if (oldKeys.length === 1) {
+ var oldKey = oldKeys[0];
+ var oldValue = d2.old[oldKey];
+ if (oldValue !== "*") {
+ if (!_deprecatedTagValuesByKey[oldKey]) {
+ _deprecatedTagValuesByKey[oldKey] = [oldValue];
} else {
- if (isAlongOnlyPath) {
- restrict = { id: restriction.id, direct: false, from: f3.id, only: true, end: false };
- } else if (isOnly) {
- restrict = { id: restriction.id, direct: false, from: f3.id, no: true, end: true };
- }
+ _deprecatedTagValuesByKey[oldKey].push(oldValue);
}
- 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 i4;
- if (currPath.length >= 3) {
- var turnPath = currPath.slice();
- if (matchedRestriction && matchedRestriction.direct === false) {
- for (i4 = 0; i4 < turnPath.length; i4++) {
- if (turnPath[i4] === matchedRestriction.from) {
- turnPath = turnPath.slice(i4);
- 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;
+ });
+ }
+ return _deprecatedTagValuesByKey;
+ };
+ osmEntity.prototype = {
+ tags: {},
+ initialize: function(sources) {
+ for (var i3 = 0; i3 < sources.length; ++i3) {
+ var source = sources[i3];
+ 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];
}
- turns.push(osmTurn(turn));
}
- if (currPath[0] === currPath[2])
- return;
- }
- if (matchedRestriction && matchedRestriction.end)
- return;
- var n1 = vgraph2.entity(entity.first());
- var n22 = vgraph2.entity(entity.last());
- var dist = geoSphericalDistance(n1.loc, n22.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(n22.id) !== -1 && // key vertex..
- currPath.indexOf(n22.id) === -1) {
- nextNodes.push(n22);
+ }
+ 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 k2 in tags) {
+ var t1 = merged[k2];
+ var t2 = tags[k2];
+ if (!t1) {
+ changed = true;
+ merged[k2] = t2;
+ } else if (t1 !== t2) {
+ changed = true;
+ merged[k2] = utilUnicodeCharsTruncated(
+ utilArrayUnion(t1.split(/;\s*/), t2.split(/;\s*/)).join(";"),
+ 255
+ // avoid exceeding character limit; see also context.maxCharsForTagValue()
+ );
}
- nextNodes.forEach(function(nextNode) {
- var fromRestrictions = vgraph2.parentRelations(entity).filter(function(r2) {
- if (!r2.isRestriction())
+ }
+ 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(k2) {
+ return k2 !== "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(d2) {
+ var oldKeys = Object.keys(d2.old);
+ if (d2.replace) {
+ var hasExistingValues = Object.keys(d2.replace).some(function(replaceKey) {
+ if (!tags[replaceKey] || d2.old[replaceKey])
return false;
- var f3 = r2.memberByRole("from");
- if (!f3 || f3.id !== entity.id)
+ var replaceValue = d2.replace[replaceKey];
+ if (replaceValue === "*")
return false;
- var isOnly = /^only_/.test(r2.tags.restriction);
- if (!isOnly)
- return true;
- var isOnlyVia = false;
- var v2 = r2.membersByRole("via");
- if (v2.length === 1 && v2[0].type === "node") {
- isOnlyVia = v2[0].id === nextNode.id;
- } else {
- for (var i5 = 0; i5 < v2.length; i5++) {
- if (v2[i5].type !== "way")
- continue;
- var viaWay = vgraph2.entity(v2[i5].id);
- if (viaWay.first() === nextNode.id || viaWay.last() === nextNode.id) {
- isOnlyVia = true;
- break;
- }
- }
- }
- return isOnlyVia;
+ if (replaceValue === tags[replaceKey])
+ return false;
+ return true;
});
- 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];
+ if (hasExistingValues)
+ return;
+ }
+ var matchesDeprecatedTags = oldKeys.every(function(oldKey) {
+ if (!tags[oldKey])
+ return false;
+ if (d2.old[oldKey] === "*")
+ return true;
+ if (d2.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(d2.old[oldKey]) !== -1;
} else {
- viaWayIds = path.filter(function(entityId) {
- return entityId[0] === "w";
- });
- viaWayIds = viaWayIds.slice(1, viaWayIds.length - 1);
+ if (tags[oldKey] === d2.old[oldKey]) {
+ if (d2.replace && d2.old[oldKey] === d2.replace[oldKey]) {
+ var replaceKeys = Object.keys(d2.replace);
+ return !replaceKeys.every(function(replaceKey) {
+ return tags[replaceKey] === d2.replace[replaceKey];
+ });
+ } else {
+ return true;
+ }
+ }
}
+ return false;
+ });
+ if (matchesDeprecatedTags) {
+ deprecated.push(d2);
}
- 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 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
};
- 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;
+ function getLaneCount(tags, isOneWay) {
+ var count;
+ if (tags.lanes) {
+ count = parseInt(tags.lanes, 10);
+ if (count > 0) {
+ return count;
+ }
}
- if (fromNode === toNode) {
- return "no_u_turn";
+ switch (tags.highway) {
+ case "trunk":
+ case "motorway":
+ count = isOneWay ? 2 : 4;
+ break;
+ default:
+ count = isOneWay ? 1 : 2;
+ break;
}
- if ((angle2 < 23 || angle2 > 336) && fromOneWay && toOneWay) {
- return "no_u_turn";
+ 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;
}
- if ((angle2 < 40 || angle2 > 319) && fromOneWay && toOneWay && turn.from.vertex !== turn.to.vertex) {
- return "no_u_turn";
+ return {
+ forward,
+ backward,
+ bothways
+ };
+ }
+ function parseTurnLanes(tag2) {
+ if (!tag2)
+ return;
+ var validValues = [
+ "left",
+ "slight_left",
+ "sharp_left",
+ "through",
+ "right",
+ "slight_right",
+ "sharp_right",
+ "reverse",
+ "merge_to_left",
+ "merge_to_right",
+ "none"
+ ];
+ return tag2.split("|").map(function(s2) {
+ if (s2 === "")
+ s2 = "none";
+ return s2.split(";").map(function(d2) {
+ return validValues.indexOf(d2) === -1 ? "unknown" : d2;
+ });
+ });
+ }
+ function parseMaxspeedLanes(tag2, maxspeed) {
+ if (!tag2)
+ return;
+ return tag2.split("|").map(function(s2) {
+ if (s2 === "none")
+ return s2;
+ var m2 = parseInt(s2, 10);
+ if (s2 === "" || m2 === maxspeed)
+ return null;
+ return isNaN(m2) ? "unknown" : m2;
+ });
+ }
+ function parseMiscLanes(tag2) {
+ if (!tag2)
+ return;
+ var validValues = [
+ "yes",
+ "no",
+ "designated"
+ ];
+ return tag2.split("|").map(function(s2) {
+ if (s2 === "")
+ s2 = "no";
+ return validValues.indexOf(s2) === -1 ? "unknown" : s2;
+ });
+ }
+ function parseBicycleWay(tag2) {
+ if (!tag2)
+ return;
+ var validValues = [
+ "yes",
+ "no",
+ "designated",
+ "lane"
+ ];
+ return tag2.split("|").map(function(s2) {
+ if (s2 === "")
+ s2 = "no";
+ return validValues.indexOf(s2) === -1 ? "unknown" : s2;
+ });
+ }
+ function mapToLanesObj(lanesObj, data, key) {
+ if (data.forward) {
+ data.forward.forEach(function(l2, i3) {
+ if (!lanesObj.forward[i3])
+ lanesObj.forward[i3] = {};
+ lanesObj.forward[i3][key] = l2;
+ });
}
- if (angle2 < 158) {
- return "no_right_turn";
+ if (data.backward) {
+ data.backward.forEach(function(l2, i3) {
+ if (!lanesObj.backward[i3])
+ lanesObj.backward[i3] = {};
+ lanesObj.backward[i3][key] = l2;
+ });
}
- if (angle2 > 202) {
- return "no_left_turn";
+ if (data.unspecified) {
+ data.unspecified.forEach(function(l2, i3) {
+ if (!lanesObj.unspecified[i3])
+ lanesObj.unspecified[i3] = {};
+ lanesObj.unspecified[i3][key] = l2;
+ });
}
- 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
- );
+ // modules/osm/way.js
+ function osmWay() {
+ if (!(this instanceof osmWay)) {
+ return new osmWay().initialize(arguments);
+ } else if (arguments.length) {
+ this.initialize(arguments);
}
- var action = function(graph) {
- var entities = groupEntities(graph);
- var polygons = entities.multipolygon.reduce(function(polygons2, m2) {
- return polygons2.concat(osmJoinWays(m2.members, graph));
- }, []).concat(entities.closedWay.map(function(d2) {
- var member = [{ id: d2.id }];
- member.nodes = graph.childNodes(d2);
- return member;
- }));
- var contained = polygons.map(function(w2, i3) {
- return polygons.map(function(d2, n3) {
- if (i3 === n3)
- return null;
- return geoPolygonContainsPolygon(
- d2.nodes.map(function(n4) {
- return n4.loc;
- }),
- w2.nodes.map(function(n4) {
- return n4.loc;
- })
- );
- });
- });
- var members = [];
- var outer = true;
- while (polygons.length) {
- extractUncontained(polygons);
- polygons = polygons.filter(isContained);
- contained = contained.filter(isContained).map(filterContained);
- }
- function isContained(d2, i3) {
- return contained[i3].some(function(val) {
- return val;
- });
- }
- function filterContained(d2) {
- return d2.filter(isContained);
- }
- function extractUncontained(polygons2) {
- polygons2.forEach(function(d2, i3) {
- if (!isContained(d2, i3)) {
- d2.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(m2) {
- if (m2.id !== relation.id) {
- relation = relation.mergeTags(m2.tags);
- graph = graph.remove(m2);
- }
+ }
+ 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;
});
- entities.closedWay.forEach(function(way) {
- function isThisOuter(m2) {
- return m2.id === way.id && m2.role !== "inner";
- }
- if (members.some(isThisOuter)) {
- relation = relation.mergeTags(way.tags);
- graph = graph.replace(way.update({ tags: {} }));
+ copy2 = copy2.update({ nodes });
+ copies[this.id] = copy2;
+ return copy2;
+ },
+ extent: function(resolver) {
+ return resolver.transient(this, "extent", function() {
+ var extent = geoExtent();
+ for (var i3 = 0; i3 < this.nodes.length; i3++) {
+ var node = resolver.hasEntity(this.nodes[i3]);
+ if (node) {
+ extent._extend(node.extent());
+ }
}
+ return extent;
});
- 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(r2) {
- return r2.isComplete(graph);
- })) {
- return "incomplete_relation";
+ },
+ 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 (!entities.multipolygon.length) {
- var sharedMultipolygons = [];
- entities.closedWay.forEach(function(way, i3) {
- if (i3 === 0) {
- sharedMultipolygons = graph.parentMultipolygons(way);
- } else {
- sharedMultipolygons = utilArrayIntersection(sharedMultipolygons, graph.parentMultipolygons(way));
+ 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,
+ busway: 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;
}
- });
- sharedMultipolygons = sharedMultipolygons.filter(function(relation) {
- return relation.members.length === entities.closedWay.length;
- });
- if (sharedMultipolygons.length) {
- return "not_eligible";
+ return width;
}
- } 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 j3 = 0; j3 < buffer2.length; j3++) {
- const item = buffer2[j3];
- if (equivalenceClasses[item]) {
- equivalenceClasses[item].push(j3);
- } else {
- equivalenceClasses[item] = [j3];
+ 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];
}
- }
- const NULLRESULT = { buffer1index: -1, buffer2index: -1, chain: null };
- let candidates = [NULLRESULT];
- for (let i3 = 0; i3 < buffer1.length; i3++) {
- const item = buffer1[i3];
- const buffer2indices = equivalenceClasses[item] || [];
- let r2 = 0;
- let c2 = candidates[0];
- for (let jx = 0; jx < buffer2indices.length; jx++) {
- const j3 = buffer2indices[jx];
- let s2;
- for (s2 = r2; s2 < candidates.length; s2++) {
- if (candidates[s2].buffer2index < j3 && (s2 === candidates.length - 1 || candidates[s2 + 1].buffer2index > j3)) {
- break;
- }
+ for (var key in this.tags) {
+ if (key in osmOneWayTags && this.tags[key] in osmOneWayTags[key]) {
+ return true;
}
- if (s2 < candidates.length) {
- const newCandidate = { buffer1index: i3, buffer2index: j3, chain: candidates[s2] };
- if (r2 === candidates.length) {
- candidates.push(c2);
+ }
+ 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 (const realKey in this.tags) {
+ const value = this.tags[realKey];
+ const key = osmRemoveLifecyclePrefix(realKey);
+ if (key in osmRightSideIsInsideTags && value in osmRightSideIsInsideTags[key]) {
+ if (osmRightSideIsInsideTags[key][value] === true) {
+ return key;
} else {
- candidates[r2] = c2;
- }
- r2 = s2 + 1;
- c2 = newCandidate;
- if (r2 === candidates.length) {
- break;
+ return osmRightSideIsInsideTags[key][value];
}
}
}
- candidates[r2] = c2;
- }
- 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)
- });
+ return null;
+ },
+ isSided: function() {
+ if (this.tags.two_sided === "yes") {
+ return false;
}
- }
- result.reverse();
- return result;
- }
- function diff3MergeRegions(a2, o2, b2) {
- let hunks = [];
- function addHunk(h2, ab) {
- hunks.push({
- ab,
- oStart: h2.buffer1[0],
- oLength: h2.buffer1[1],
- // length of o to remove
- abStart: h2.buffer2[0],
- abLength: h2.buffer2[1]
- // length of a/b to insert
- // abContent: (ab === 'a' ? a : b).slice(h.buffer2[0], h.buffer2[0] + h.buffer2[1])
+ 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(n3) {
+ return n3.loc;
});
- }
- diffIndices(o2, a2).forEach((item) => addHunk(item, "a"));
- diffIndices(o2, b2).forEach((item) => addHunk(item, "b"));
- hunks.sort((x2, y2) => x2.oStart - y2.oStart);
- let results = [];
- let currOffset = 0;
- function advanceTo(endOffset) {
- if (endOffset > currOffset) {
- results.push({
- stable: true,
- buffer: "o",
- bufferStart: currOffset,
- bufferLength: endOffset - currOffset,
- bufferContent: o2.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" ? a2 : b2;
- 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: [a2.length, -1, o2.length, -1],
- b: [b2.length, -1, o2.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 b3 = bounds[hunk.ab];
- b3[0] = Math.min(abStart, b3[0]);
- b3[1] = Math.max(abEnd, b3[1]);
- b3[2] = Math.min(oStart, b3[2]);
- b3[3] = Math.max(oEnd, b3[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: a2.slice(aStart, aEnd),
- oStart: regionStart,
- oLength: regionEnd - regionStart,
- oContent: o2.slice(regionStart, regionEnd),
- bStart,
- bLength: bEnd - bStart,
- bContent: b2.slice(bStart, bEnd)
- };
- results.push(result);
- }
- currOffset = regionEnd;
- }
- advanceTo(o2.length);
- return results;
- }
- function diff3Merge(a2, o2, b2, options2) {
- let defaults = {
- excludeFalseConflicts: true,
- stringSeparator: /\s+/
- };
- options2 = Object.assign(defaults, options2);
- if (typeof a2 === "string")
- a2 = a2.split(options2.stringSeparator);
- if (typeof o2 === "string")
- o2 = o2.split(options2.stringSeparator);
- if (typeof b2 === "string")
- b2 = b2.split(options2.stringSeparator);
- let results = [];
- const regions = diff3MergeRegions(a2, o2, b2);
- let okBuffer = [];
- function flushOk() {
- if (okBuffer.length) {
- results.push({ ok: okBuffer });
- }
- okBuffer = [];
- }
- function isFalseConflict(a3, b3) {
- if (a3.length !== b3.length)
- return false;
- for (let i3 = 0; i3 < a3.length; i3++) {
- if (a3[i3] !== b3[i3])
+ var curr = 0;
+ var prev = 0;
+ for (var i3 = 0; i3 < coords.length; i3++) {
+ var o2 = coords[(i3 + 1) % coords.length];
+ var a2 = coords[i3];
+ var b2 = coords[(i3 + 2) % coords.length];
+ var res = geoVecCross(a2, b2, o2);
+ curr = res > 0 ? 1 : res < 0 ? -1 : 0;
+ if (curr === 0) {
+ continue;
+ } else if (prev && curr !== prev) {
return false;
+ }
+ prev = curr;
}
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
- }
- });
+ },
+ // 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, n22) {
+ for (var i3 = 0; i3 < this.nodes.length; i3++) {
+ if (this.nodes[i3] === n1) {
+ if (this.nodes[i3 - 1] === n22)
+ return true;
+ if (this.nodes[i3 + 1] === n22)
+ return true;
}
}
- });
- flushOk();
- return results;
- }
-
- // modules/actions/merge_remote_changes.js
- var import_lodash2 = __toESM(require_lodash());
- function actionMergeRemoteChanges(id2, localGraph, remoteGraph, discardTags, formatUser) {
- discardTags = discardTags || {};
- var _option = "safe";
- var _conflicts = [];
- function user(d2) {
- return typeof formatUser === "function" ? formatUser(d2) : (0, import_lodash2.escape)(d2);
- }
- function mergeLocation(remote, target) {
- function pointEqual(a2, b2) {
- var epsilon3 = 1e-6;
- return Math.abs(a2[0] - b2[0]) < epsilon3 && Math.abs(a2[1] - b2[1]) < epsilon3;
- }
- if (_option === "force_local" || pointEqual(target.loc, remote.loc)) {
- return target;
+ 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 n22 = graph2.hasEntity(this.nodes[1]);
+ return n1 && n22 && geoExtent([
+ [
+ Math.min(n1.loc[0], n22.loc[0]),
+ Math.min(n1.loc[1], n22.loc[1])
+ ],
+ [
+ Math.max(n1.loc[0], n22.loc[0]),
+ Math.max(n1.loc[1], n22.loc[1])
+ ]
+ ]);
}
- if (_option === "force_remote") {
- return target.update({ loc: remote.loc });
+ return graph.transient(this, "segments", function() {
+ var segments = [];
+ for (var i3 = 0; i3 < this.nodes.length - 1; i3++) {
+ segments.push({
+ id: this.id + "-" + i3,
+ wayId: this.id,
+ index: i3,
+ nodes: [this.nodes[i3], this.nodes[i3 + 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 i3 = nodes.length - 1;
+ while (i3 > 0 && nodes.length > 1 && nodes[i3] === connector) {
+ nodes.splice(i3, 1);
+ i3 = nodes.length - 1;
}
- _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;
+ 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 (_option === "force_remote") {
- return target.update({ nodes: remote.nodes });
+ if (index < 0 || index > max3) {
+ throw new RangeError("index " + index + " out of range 0.." + max3);
}
- var ccount = _conflicts.length;
- var o2 = base.nodes || [];
- var a2 = target.nodes || [];
- var b2 = remote.nodes || [];
- var nodes = [];
- var hunks = diff3Merge(a2, o2, b2, { excludeFalseConflicts: true });
- for (var i3 = 0; i3 < hunks.length; i3++) {
- var hunk = hunks[i3];
- if (hunk.ok) {
- nodes.push.apply(nodes, hunk.ok);
- } else {
- var c2 = hunk.conflict;
- if ((0, import_fast_deep_equal.default)(c2.o, c2.a)) {
- nodes.push.apply(nodes, c2.b);
- } else if ((0, import_fast_deep_equal.default)(c2.o, c2.b)) {
- nodes.push.apply(nodes, c2.a);
- } else {
- _conflicts.push(_t.html("merge_remote_changes.conflict.nodelist", { user: { html: user(remote.user) } }));
- break;
- }
+ if (isClosed) {
+ var connector = this.first();
+ var i3 = 1;
+ while (i3 < nodes.length && nodes.length > 2 && nodes[i3] === connector) {
+ nodes.splice(i3, 1);
+ if (index > i3)
+ index--;
+ }
+ i3 = nodes.length - 1;
+ while (i3 > 0 && nodes.length > 1 && nodes[i3] === connector) {
+ nodes.splice(i3, 1);
+ if (index > i3)
+ index--;
+ i3 = nodes.length - 1;
}
}
- 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;
+ nodes.splice(index, 0, id2);
+ nodes = nodes.filter(noRepeatNodes);
+ if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+ nodes.push(nodes[0]);
}
- var ccount = _conflicts.length;
- for (var i3 = 0; i3 < children2.length; i3++) {
- var id3 = children2[i3];
- var node = graph.hasEntity(id3);
- if (targetWay.nodes.indexOf(id3) === -1) {
- if (node && !isUsed(node, targetWay)) {
- updates.removeIds.push(id3);
- }
- continue;
+ 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 i3 = 1;
+ while (i3 < nodes.length && nodes.length > 2 && nodes[i3] === connector) {
+ nodes.splice(i3, 1);
+ if (index > i3)
+ index--;
}
- 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);
+ i3 = nodes.length - 1;
+ while (i3 > 0 && nodes.length > 1 && nodes[i3] === connector) {
+ nodes.splice(i3, 1);
+ if (index === i3)
+ index = 0;
+ i3 = nodes.length - 1;
}
}
- return targetWay;
- }
- function updateChildren(updates, graph) {
- for (var i3 = 0; i3 < updates.replacements.length; i3++) {
- graph = graph.replace(updates.replacements[i3]);
- }
- if (updates.removeIds.length) {
- graph = actionDeleteMultiple(updates.removeIds)(graph);
+ 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 graph;
- }
- function mergeMembers(remote, target) {
- if (_option === "force_local" || (0, import_fast_deep_equal.default)(target.members, remote.members)) {
- return target;
+ 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 i3 = 0; i3 < nodes.length; i3++) {
+ if (nodes[i3] === needleID) {
+ nodes[i3] = replacementID;
+ }
}
- if (_option === "force_remote") {
- return target.update({ members: remote.members });
+ nodes = nodes.filter(noRepeatNodes);
+ if (isClosed && (nodes.length === 1 || nodes[0] !== nodes[nodes.length - 1])) {
+ nodes.push(nodes[0]);
}
- _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;
+ 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]);
}
- if (_option === "force_remote") {
- return target.update({ tags: remote.tags });
+ return this.update({ nodes });
+ },
+ asJXON: function(changeset_id) {
+ var r2 = {
+ 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(k2) {
+ return { keyAttributes: { k: k2, v: this.tags[k2] } };
+ }, this)
+ }
+ };
+ if (changeset_id) {
+ r2.way["@changeset"] = changeset_id;
}
- var ccount = _conflicts.length;
- var o2 = base.tags || {};
- var a2 = target.tags || {};
- var b2 = remote.tags || {};
- var keys2 = utilArrayUnion(utilArrayUnion(Object.keys(o2), Object.keys(a2)), Object.keys(b2)).filter(function(k3) {
- return !discardTags[k3];
+ return r2;
+ },
+ asGeoJSON: function(resolver) {
+ return resolver.transient(this, "GeoJSON", function() {
+ var coordinates = resolver.childNodes(this).map(function(n3) {
+ return n3.loc;
+ });
+ if (this.isArea() && this.isClosed()) {
+ return {
+ type: "Polygon",
+ coordinates: [coordinates]
+ };
+ } else {
+ return {
+ type: "LineString",
+ coordinates
+ };
+ }
});
- var tags = Object.assign({}, a2);
- var changed = false;
- for (var i3 = 0; i3 < keys2.length; i3++) {
- var k2 = keys2[i3];
- if (o2[k2] !== b2[k2] && a2[k2] !== b2[k2]) {
- if (o2[k2] !== a2[k2]) {
- _conflicts.push(_t.html(
- "merge_remote_changes.conflict.tags",
- { tag: k2, local: a2[k2], remote: b2[k2], user: { html: user(remote.user) } }
- ));
- } else {
- if (b2.hasOwnProperty(k2)) {
- tags[k2] = b2[k2];
- } else {
- delete tags[k2];
- }
- changed = true;
- }
+ },
+ area: function(resolver) {
+ return resolver.transient(this, "area", function() {
+ var nodes = resolver.childNodes(this);
+ var json = {
+ type: "Polygon",
+ coordinates: [nodes.map(function(n3) {
+ return n3.loc;
+ })]
+ };
+ if (!this.isClosed() && nodes.length) {
+ json.coordinates[0].push(nodes[0].loc);
}
- }
- return changed && _conflicts.length === ccount ? target.update({ tags }) : target;
+ 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;
+ });
}
- 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;
+ });
+ function noRepeatNodes(node, i3, arr) {
+ return i3 === 0 || node !== arr[i3 - 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;
}
}
- 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);
+ }
+ 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 i3 = 0; i3 < members.length; i3++) {
+ member = members[i3];
+ if (member.id === entity.id && member.role && member.role !== "outer") {
+ return false;
}
- target = mergeTags(base, remote, target);
- if (!_conflicts.length) {
- graph = updateChildren(updates, graph).replace(target);
+ if (member.id !== entity.id && (!member.role || member.role === "outer")) {
+ return false;
}
- return graph;
- };
- action.withOption = function(opt) {
- _option = opt;
- return action;
- };
- action.conflicts = function() {
- return _conflicts;
- };
- return action;
+ }
+ return parent;
}
-
- // 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 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 i3 = 0; i3 < members.length; i3++) {
+ member = members[i3];
+ if (!member.role || member.role === "outer") {
+ if (outerMember)
+ return false;
+ outerMember = member;
}
- function cacheEntities(ids) {
- for (var i3 = 0; i3 < ids.length; i3++) {
- var id2 = ids[i3];
- 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;
- }));
- }
- }
+ }
+ 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 i3;
+ var joinAsMembers = true;
+ for (i3 = 0; i3 < toJoin.length; i3++) {
+ if (toJoin[i3] instanceof osmWay) {
+ joinAsMembers = false;
+ break;
}
- function cacheIntersections(ids) {
- function isEndpoint(way2, id3) {
- return !way2.isClosed() && !!way2.affix(id3);
- }
- for (var i3 = 0; i3 < ids.length; i3++) {
- var id2 = ids[i3];
- var childNodes = graph.childNodes(graph.entity(id2));
- for (var j3 = 0; j3 < childNodes.length; j3++) {
- var node = childNodes[j3];
- var parents = graph.parentWays(node);
- if (parents.length !== 2)
- continue;
- var moved = graph.entity(id2);
- var unmoved = null;
- for (var k2 = 0; k2 < parents.length; k2++) {
- var way = parents[k2];
- 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)
- });
+ }
+ 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 (i3 = 0; i3 < toJoin.length; i3++) {
+ item = toJoin[i3];
+ 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 (!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;
+ 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(i3, 1);
}
+ currWays.nodes = currNodes;
+ sequences.push(currWays);
}
- 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;
+ 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 {
- len = way.nodes.length;
- prevIndex = movedIndex - 1;
- nextIndex = movedIndex + 1;
+ if (isPTv2 && isNaN(memberIndex)) {
+ memberIndex = 0;
+ }
+ graph = graph.replace(relation.addMember(member, memberIndex));
}
- 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];
+ return graph;
+ };
+ function addWayMember(relation, graph) {
+ var groups, tempWay, insertPairIsReversed, item, i3, j2, k2;
+ var PTv2members = [];
+ var members = [];
+ for (i3 = 0; i3 < relation.members.length; i3++) {
+ var m2 = relation.members[i3];
+ if (/stop|platform/.test(m2.role)) {
+ PTv2members.push(m2);
+ } else {
+ members.push(m2);
+ }
}
- var start2, end;
- if (delta) {
- start2 = projection2(cache.startLoc[nodeId]);
- end = projection2.invert(geoVecAdd(start2, delta));
+ 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 {
- 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();
+ groups = utilArrayGroupBy(relation.members, "type");
+ groups.way = groups.way || [];
+ groups.way.push(member);
}
- for (var i3 = 0; i3 < way.nodes.length; i3++) {
- curr = graph.entity(way.nodes[i3]);
- 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);
+ members = withIndex(groups.way);
+ var joined = osmJoinWays(members, graph);
+ for (i3 = 0; i3 < joined.length; i3++) {
+ var segment = joined[i3];
+ var nodes = segment.nodes.slice();
+ var startIndex = segment[0].index;
+ for (j2 = 0; j2 < members.length; j2++) {
+ if (members[j2].index === startIndex) {
+ break;
}
}
- 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(n3) {
- return n3 !== vertex;
- });
- var nodes2 = graph.childNodes(way2).filter(function(n3) {
- return n3 !== 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 i3 = 0; i3 < maxIter; i3++) {
- 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;
+ for (k2 = 0; k2 < segment.length; k2++) {
+ item = segment[k2];
+ 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 (k2 > 0) {
+ if (j2 + k2 >= members.length || item.index !== members[j2 + k2].index) {
+ moveMember(members, item.index, j2 + k2);
+ }
+ }
+ nodes.splice(0, way.nodes.length - 1);
}
- } 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 i3 = 0; i3 < cache.intersections.length; i3++) {
- var obj = cache.intersections[i3];
- 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);
+ if (tempWay) {
+ graph = graph.remove(tempWay);
}
- for (var i3 = 0; i3 < cache.intersections.length; i3++) {
- var obj = cache.intersections[i3];
- if (obj.movedIsEP && obj.unmovedIsEP)
- continue;
- if (!obj.movedIsEP)
+ var wayMembers = [];
+ for (i3 = 0; i3 < members.length; i3++) {
+ item = members[i3];
+ if (item.index === -1)
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(n3) {
- return moveNode(n3.loc);
- });
- var unmovedNodes = graph.childNodes(graph.entity(obj.unmovedId));
- var unmovedPath = unmovedNodes.map(function(n3) {
- return projection2(n3.loc);
- });
- var hits = geoPathIntersections(movedPath, unmovedPath);
- for (var j3 = 0; i3 < hits.length; i3++) {
- if (geoVecEqual(hits[j3], end))
- continue;
- var edge = geoChooseEdge(unmovedNodes, end, projection2);
- _delta = geoVecSubtract(projection2(edge.loc), start2);
+ if (item.pair) {
+ wayMembers.push(item.pair[0]);
+ wayMembers.push(item.pair[1]);
+ } else {
+ wayMembers.push(utilObjectOmit(item, ["index"]));
}
}
- }
- var action = function(graph) {
- if (_delta[0] === 0 && _delta[1] === 0)
- return graph;
- setupCache(graph);
- if (cache.intersections.length) {
- limitDelta(graph);
- }
- for (var i3 = 0; i3 < cache.nodes.length; i3++) {
- var node = graph.entity(cache.nodes[i3]);
- var start2 = projection2(node.loc);
- var end = geoVecAdd(start2, _delta);
- graph = graph.replace(node.move(projection2.invert(end)));
+ var newMembers = PTv2members.concat(groups.node || [], wayMembers, groups.relation || []);
+ return graph.replace(relation.update({ members: newMembers }));
+ function moveMember(arr, findIndex, toIndex) {
+ var i4;
+ for (i4 = 0; i4 < arr.length; i4++) {
+ if (arr[i4].index === findIndex) {
+ break;
+ }
+ }
+ var item2 = Object.assign({}, arr[i4]);
+ arr[i4].index = -1;
+ item2.index = toIndex;
+ arr.splice(toIndex, 0, item2);
}
- if (cache.intersections.length) {
- graph = cleanupIntersections(graph);
+ function withIndex(arr) {
+ var result = new Array(arr.length);
+ for (var i4 = 0; i4 < arr.length; i4++) {
+ result[i4] = Object.assign({}, arr[i4]);
+ result[i4].index = i4;
+ }
+ 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 i3 = 0; i3 < way.nodes.length - 1; i3++) {
+ if (geoEdgeEqual([way.nodes[i3], way.nodes[i3 + 1]], midpoint.edge)) {
+ graph = graph.replace(graph.entity(way.id).addNode(node.id, i3 + 1));
+ return;
+ }
+ }
+ });
return graph;
};
- action.delta = function() {
- return _delta;
+ }
+
+ // modules/actions/add_vertex.js
+ function actionAddVertex(wayId, nodeId, index) {
+ return function(graph) {
+ return graph.replace(graph.entity(wayId).addNode(nodeId, index));
};
- return action;
}
- // modules/actions/move_member.js
- function actionMoveMember(relationId, fromIndex, toIndex) {
+ // modules/actions/change_member.js
+ function actionChangeMember(relationId, member, memberIndex) {
return function(graph) {
- return graph.replace(graph.entity(relationId).moveMember(fromIndex, toIndex));
+ return graph.replace(graph.entity(relationId).updateMember(member, memberIndex));
};
}
- // modules/actions/move_node.js
- function actionMoveNode(nodeID, toLoc) {
- var action = function(graph, t2) {
- if (t2 === null || !isFinite(t2))
- t2 = 1;
- t2 = Math.min(Math.max(+t2, 0), 1);
- var node = graph.entity(nodeID);
- return graph.replace(
- node.move(geoVecInterp(node.loc, toLoc, t2))
- );
+ // 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;
+ const loc = entity.extent(graph).center();
+ var preserveKeys;
+ if (newPreset) {
+ preserveKeys = [];
+ if (newPreset.addTags) {
+ preserveKeys = preserveKeys.concat(Object.keys(newPreset.addTags));
+ }
+ if (oldPreset && !oldPreset.id.startsWith(newPreset.id)) {
+ newPreset.fields(loc).concat(newPreset.moreFields(loc)).filter((f2) => f2.matchGeometry(geometry)).map((f2) => f2.key).filter(Boolean).forEach((key) => preserveKeys.push(key));
+ }
+ }
+ if (oldPreset)
+ tags = oldPreset.unsetTags(tags, geometry, preserveKeys, false, loc);
+ if (newPreset)
+ tags = newPreset.setTags(tags, geometry, skipFieldDefaults, loc);
+ return graph.replace(entity.update({ tags }));
};
- action.transitionable = true;
- return action;
}
- // modules/actions/noop.js
- function actionNoop() {
+ // modules/actions/change_tags.js
+ function actionChangeTags(entityId, tags) {
return function(graph) {
- return graph;
+ var entity = graph.entity(entityId);
+ return graph.replace(entity.update({ tags }));
};
}
- // 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);
+ // 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 i3;
+ if (this.isHighwayIntersection(resolver) && (this.tags.stop || "").toLowerCase() === "all") {
+ val = "all";
+ } else {
+ val = (this.tags.direction || "").toLowerCase();
+ var re3 = /:direction$/i;
+ var keys2 = Object.keys(this.tags);
+ for (i3 = 0; i3 < keys2.length; i3++) {
+ if (re3.test(keys2[i3])) {
+ val = this.tags[keys2[i3]].toLowerCase();
+ break;
+ }
+ }
+ }
+ if (val === "")
+ return [];
+ var values = val.split(";");
+ var results = [];
+ values.forEach(function(v2) {
+ if (cardinal[v2] !== void 0) {
+ v2 = cardinal[v2];
+ }
+ if (v2 !== "" && !isNaN(+v2)) {
+ results.push(+v2);
+ return;
+ }
+ var lookBackward = this.tags["traffic_sign:backward"] || v2 === "backward" || v2 === "both" || v2 === "all";
+ var lookForward = this.tags["traffic_sign:forward"] || v2 === "forward" || v2 === "both" || v2 === "all";
+ if (!lookForward && !lookBackward)
+ return;
+ var nodeIds = {};
+ resolver.parentWays(this).forEach(function(parent) {
+ var nodes = parent.nodes;
+ for (i3 = 0; i3 < nodes.length; i3++) {
+ if (nodes[i3] === this.id) {
+ if (lookForward && i3 > 0) {
+ nodeIds[nodes[i3 - 1]] = true;
+ }
+ if (lookBackward && i3 < nodes.length - 1) {
+ nodeIds[nodes[i3 + 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 i3 in parents) {
+ if (parents[i3].geometry(resolver) === "line" && parents[i3].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 r2 = {
+ node: {
+ "@id": this.osmId(),
+ "@lon": this.loc[0],
+ "@lat": this.loc[1],
+ "@version": this.version || 0,
+ tag: Object.keys(this.tags).map(function(k2) {
+ return { keyAttributes: { k: k2, v: this.tags[k2] } };
+ }, this)
+ }
+ };
+ if (changeset_id)
+ r2.node["@changeset"] = changeset_id;
+ return r2;
+ },
+ 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, t2) {
if (t2 === null || !isFinite(t2))
t2 = 1;
t2 = Math.min(Math.max(+t2, 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 });
+ 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);
}
- 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 nodes = utilArrayUniq(graph.childNodes(way));
+ var keyNodes = nodes.filter(function(n3) {
+ return graph.parentWays(n3).length !== 1;
+ });
+ var points = nodes.map(function(n3) {
+ return projection2(n3.loc);
+ });
+ var keyPoints = keyNodes.map(function(n3) {
+ return projection2(n3.loc);
+ });
+ var centroid = points.length === 2 ? geoVecInterp(points[0], points[1], 0.5) : centroid_default2(points);
+ var radius = median(points, function(p2) {
+ return geoVecLength(centroid, p2);
+ });
+ var sign2 = area_default3(points) > 0 ? 1 : -1;
+ var ids, i3, j2, k2;
+ if (!keyNodes.length) {
+ keyNodes = [nodes[0]];
+ keyPoints = [points[0]];
}
- var nodeCount = {};
- var points = [];
- var corner = { i: 0, dotp: 1 };
- var node, point2, loc, score, motions, i3, j3;
- for (i3 = 0; i3 < nodes.length; i3++) {
- node = nodes[i3];
- nodeCount[node.id] = (nodeCount[node.id] || 0) + 1;
- points.push({ id: node.id, coord: projection2(node.loc) });
+ 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]);
}
- if (points.length === 3) {
- for (i3 = 0; i3 < 1e3; i3++) {
- motions = points.map(calcMotion);
- points[corner.i].coord = geoVecAdd(points[corner.i].coord, motions[corner.i]);
- score = corner.dotp;
- if (score < epsilon3) {
- break;
- }
+ for (i3 = 0; i3 < keyPoints.length; i3++) {
+ var nextKeyNodeIndex = (i3 + 1) % keyNodes.length;
+ var startNode = keyNodes[i3];
+ 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;
}
- node = graph.entity(nodes[corner.i].id);
- loc = projection2.invert(points[corner.i].coord);
- graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t2)));
- } else {
- var straights = [];
- var simplified = [];
- for (i3 = 0; i3 < points.length; i3++) {
- point2 = points[i3];
- var dotp = 0;
- if (isClosed || i3 > 0 && i3 < points.length - 1) {
- var a2 = points[(i3 - 1 + points.length) % points.length];
- var b2 = points[(i3 + 1) % points.length];
- dotp = Math.abs(geoOrthoNormalizedDotProduct(a2.coord, b2.coord, point2.coord));
- }
- if (dotp > upperThreshold) {
- straights.push(point2);
- } else {
- simplified.push(point2);
- }
+ var distance = geoVecLength(centroid, keyPoints[i3]) || 1e-4;
+ keyPoints[i3] = [
+ centroid[0] + (keyPoints[i3][0] - centroid[0]) / distance * radius,
+ centroid[1] + (keyPoints[i3][1] - centroid[1]) / distance * radius
+ ];
+ loc = projection2.invert(keyPoints[i3]);
+ node = keyNodes[i3];
+ origNode = origNodes[node.id];
+ node = node.move(geoVecInterp(origNode.loc, loc, t2));
+ graph = graph.replace(node);
+ startAngle = Math.atan2(keyPoints[i3][1] - centroid[1], keyPoints[i3][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));
}
- var bestPoints = clonePoints(simplified);
- var originalPoints = clonePoints(simplified);
- score = Infinity;
- for (i3 = 0; i3 < 1e3; i3++) {
- motions = simplified.map(calcMotion);
- for (j3 = 0; j3 < motions.length; j3++) {
- simplified[j3].coord = geoVecAdd(simplified[j3].coord, motions[j3]);
- }
- var newScore = geoOrthoCalcScore(simplified, isClosed, epsilon3, threshold);
- if (newScore < score) {
- bestPoints = clonePoints(simplified);
- score = newScore;
- }
- if (score < epsilon3) {
- break;
- }
+ 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, t2));
+ graph = graph.replace(node);
}
- var bestCoords = bestPoints.map(function(p2) {
- return p2.coord;
- });
- if (isClosed)
- bestCoords.push(bestCoords[0]);
- for (i3 = 0; i3 < bestPoints.length; i3++) {
- point2 = bestPoints[i3];
- if (!geoVecEqual(originalPoints[i3].coord, point2.coord)) {
- node = graph.entity(point2.id);
- loc = projection2.invert(point2.coord);
- graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t2)));
+ 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, t2) });
+ graph = graph.replace(node);
+ nodes.splice(endNodeIndex + j2, 0, node);
+ inBetweenNodes.push(node.id);
}
- for (i3 = 0; i3 < straights.length; i3++) {
- point2 = straights[i3];
- if (nodeCount[point2.id] > 1)
- continue;
- node = graph.entity(point2.id);
- if (t2 === 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, t2)));
+ 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[i3]);
+ 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 (k2 = 0; k2 < inBetweenNodes.length; k2++) {
+ sharedWay = sharedWay.addNode(inBetweenNodes[k2], insertAt + k2);
+ }
+ graph = graph.replace(sharedWay);
}
}
}
}
+ ids = nodes.map(function(n3) {
+ return n3.id;
+ });
+ ids.push(ids[0]);
+ way = way.update({ nodes: ids });
+ graph = graph.replace(way);
return graph;
- function clonePoints(array2) {
- return array2.map(function(p2) {
- return { id: p2.id, coord: [p2.coord[0], p2.coord[1]] };
- });
+ };
+ action.makeConvex = function(graph) {
+ var way = graph.entity(wayId);
+ var nodes = utilArrayUniq(graph.childNodes(way));
+ var points = nodes.map(function(n3) {
+ return projection2(n3.loc);
+ });
+ var sign2 = area_default3(points) > 0 ? 1 : -1;
+ var hull = hull_default(points);
+ var i3, j2;
+ if (sign2 === -1) {
+ nodes.reverse();
+ points.reverse();
}
- function calcMotion(point3, i4, array2) {
- if (!isClosed && (i4 === 0 || i4 === array2.length - 1))
- return [0, 0];
- if (nodeCount[array2[i4].id] > 1)
- return [0, 0];
- var a3 = array2[(i4 - 1 + array2.length) % array2.length].coord;
- var origin = point3.coord;
- var b3 = array2[(i4 + 1) % array2.length].coord;
- var p2 = geoVecSubtract(a3, origin);
- var q2 = geoVecSubtract(b3, origin);
- var scale = 2 * Math.min(geoVecLength(p2), geoVecLength(q2));
- p2 = geoVecNormalize(p2);
- q2 = geoVecNormalize(q2);
- var dotp2 = p2[0] * q2[0] + p2[1] * q2[1];
- var val = Math.abs(dotp2);
- if (val < lowerThreshold) {
- corner.i = i4;
- corner.dotp = val;
- var vec = geoVecNormalize(geoVecAdd(p2, q2));
- return geoVecScale(vec, 0.1 * dotp2 * scale);
+ for (i3 = 0; i3 < hull.length - 1; i3++) {
+ var startIndex = points.indexOf(hull[i3]);
+ var endIndex = points.indexOf(hull[i3 + 1]);
+ var indexRange = endIndex - startIndex;
+ if (indexRange < 0) {
+ indexRange += nodes.length;
}
- return [0, 0];
- }
- };
- function nodeSubset(nodes, vertexID2, isClosed) {
- var first = isClosed ? 0 : 1;
- var last = isClosed ? nodes.length : nodes.length - 1;
- for (var i3 = first; i3 < last; i3++) {
- if (nodes[i3].id === vertexID2) {
- return [
- nodes[(i3 - 1 + nodes.length) % nodes.length],
- nodes[i3],
- nodes[(i3 + 1) % nodes.length]
- ];
+ for (j2 = 1; j2 < indexRange; j2++) {
+ var point2 = geoVecInterp(hull[i3], hull[i3 + 1], j2 / indexRange);
+ var node = nodes[(j2 + startIndex) % nodes.length].move(projection2.invert(point2));
+ graph = graph.replace(node);
}
}
- return [];
- }
+ return graph;
+ };
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";
+ if (!graph.entity(wayId).isClosed()) {
+ return "not_closed";
}
- var coords = nodes.map(function(n3) {
+ var way = graph.entity(wayId);
+ var nodes = utilArrayUniq(graph.childNodes(way));
+ var points = nodes.map(function(n3) {
return projection2(n3.loc);
});
- var score = geoOrthoCanOrthogonalize(coords, isClosed, epsilon3, threshold, allowStraightAngles);
- if (score === null) {
- return "not_squarish";
- } else if (score === 0) {
- return "square_enough";
- } else {
+ var hull = hull_default(points);
+ var epsilonAngle = Math.PI / 180;
+ if (hull.length !== points.length || hull.length < 3) {
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" });
- });
+ var centroid = centroid_default2(points);
+ var radius = geoVecLengthSquare(centroid, points[0]);
+ var i3, actualPoint;
+ for (i3 = 0; i3 < hull.length; i3++) {
+ actualPoint = hull[i3];
+ var actualDist = geoVecLengthSquare(actualPoint, centroid);
+ var diff = Math.abs(actualDist - radius);
+ if (diff > 0.05 * radius) {
+ return false;
+ }
}
- 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);
- }
- });
+ for (i3 = 0; i3 < hull.length; i3++) {
+ actualPoint = hull[i3];
+ var nextPoint = hull[(i3 + 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;
}
- 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 "already_circular";
};
+ action.transitionable = true;
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(a2, o2, b2) {
- return geoVecDot(a2, b2, o2) / geoVecDot(b2, b2, o2);
- }
- 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];
+ // 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, t2) {
- if (t2 === null || !isFinite(t2))
- t2 = 1;
- t2 = Math.min(Math.max(+t2, 0), 1);
- var nodes = nodeIDs.map(function(id2) {
- return graph.entity(id2);
- });
- var points = nodes.map(function(n3) {
- return projection2(n3.loc);
- });
- var endpoints = getEndpoints(points);
- var startPoint = endpoints[0];
- var endPoint = endpoints[1];
- for (var i3 = 0; i3 < points.length; i3++) {
- var node = nodes[i3];
- var point2 = points[i3];
- var u2 = positionAlongWay(point2, startPoint, endPoint);
- var point22 = geoVecInterp(startPoint, endPoint, u2);
- var loc2 = projection2.invert(point22);
- graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t2)));
- }
- return graph;
- };
- action.disabled = function(graph) {
- var nodes = nodeIDs.map(function(id2) {
- return graph.entity(id2);
- });
- var points = nodes.map(function(n3) {
- return projection2(n3.loc);
+ 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);
+ }
});
- var endpoints = getEndpoints(points);
- var startPoint = endpoints[0];
- var endPoint = endpoints[1];
- var maxDistance = 0;
- for (var i3 = 0; i3 < points.length; i3++) {
- var point2 = points[i3];
- var u2 = positionAlongWay(point2, startPoint, endPoint);
- var p2 = geoVecInterp(startPoint, endPoint, u2);
- var dist = geoVecLength(p2, point2);
- if (!isNaN(dist) && dist > maxDistance) {
- maxDistance = dist;
+ 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);
}
- }
- if (maxDistance < 1e-4) {
- return "straight_enough";
- }
+ });
+ return graph.remove(way);
};
- action.transitionable = true;
return action;
}
- // modules/actions/straighten_way.js
- function actionStraightenWay(selectedIDs, projection2) {
- function positionAlongWay(a2, o2, b2) {
- return geoVecDot(a2, b2, o2) / geoVecDot(b2, b2, o2);
- }
- function allNodes(graph) {
- var nodes = [];
- var startNodes = [];
- var endNodes = [];
- var remainingWays = [];
- var selectedWays = selectedIDs.filter(function(w2) {
- return graph.entity(w2).type === "way";
- });
- var selectedNodes = selectedIDs.filter(function(n3) {
- return graph.entity(n3).type === "node";
+ // 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);
+ }
});
- for (var i3 = 0; i3 < selectedWays.length; i3++) {
- var way = graph.entity(selectedWays[i3]);
- nodes = way.nodes.slice(0);
- remainingWays.push(nodes);
- startNodes.push(nodes[0]);
- endNodes.push(nodes[nodes.length - 1]);
- }
- startNodes = startNodes.filter(function(n3) {
- return startNodes.indexOf(n3) === startNodes.lastIndexOf(n3);
+ 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);
+ }
});
- endNodes = endNodes.filter(function(n3) {
- return endNodes.indexOf(n3) === endNodes.lastIndexOf(n3);
+ var memberIDs = utilArrayUniq(relation.members.map(function(m2) {
+ return m2.id;
+ }));
+ memberIDs.forEach(function(memberID) {
+ graph = graph.replace(relation.removeMembersWithID(memberID));
+ var entity = graph.entity(memberID);
+ if (canDeleteEntity(entity, graph)) {
+ graph = actionDeleteMultiple([memberID])(graph);
+ }
});
- 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();
+ 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);
}
- 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(a2, b2) {
- return a2 - b2;
- });
- nodes = nodes.slice(sortedStartEnd[0], sortedStartEnd[1] + 1);
- }
- return nodes.map(function(n3) {
- return graph.entity(n3);
});
- }
- function shouldKeepNode(node, graph) {
- return graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags();
- }
- var action = function(graph, t2) {
- if (t2 === null || !isFinite(t2))
- t2 = 1;
- t2 = Math.min(Math.max(+t2, 0), 1);
- var nodes = allNodes(graph);
- var points = nodes.map(function(n3) {
- return projection2(n3.loc);
+ graph.parentRelations(node).forEach(function(parent) {
+ parent = parent.removeMembersWithID(nodeId);
+ graph = graph.replace(parent);
+ if (parent.isDegenerate()) {
+ graph = actionDeleteRelation(parent.id)(graph);
+ }
});
- var startPoint = points[0];
- var endPoint = points[points.length - 1];
- var toDelete = [];
- var i3;
- for (i3 = 1; i3 < points.length - 1; i3++) {
- var node = nodes[i3];
- var point2 = points[i3];
- if (t2 < 1 || shouldKeepNode(node, graph)) {
- var u2 = positionAlongWay(point2, startPoint, endPoint);
- var p2 = geoVecInterp(startPoint, endPoint, u2);
- var loc2 = projection2.invert(p2);
- graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t2)));
- } else {
- if (toDelete.indexOf(node) === -1) {
- toDelete.push(node);
+ return graph.remove(node);
+ };
+ return action;
+ }
+
+ // modules/actions/connect.js
+ function actionConnect(nodeIDs) {
+ var action = function(graph) {
+ var survivor;
+ var node;
+ var parents;
+ var i3, j2;
+ nodeIDs.reverse();
+ var interestingIDs = [];
+ for (i3 = 0; i3 < nodeIDs.length; i3++) {
+ node = graph.entity(nodeIDs[i3]);
+ if (node.hasInterestingTags()) {
+ if (!node.isNew()) {
+ interestingIDs.push(node.id);
}
}
}
- for (i3 = 0; i3 < toDelete.length; i3++) {
- graph = actionDeleteNode(toDelete[i3].id)(graph);
+ survivor = graph.entity(utilOldestID(interestingIDs.length > 0 ? interestingIDs : nodeIDs));
+ for (i3 = 0; i3 < nodeIDs.length; i3++) {
+ node = graph.entity(nodeIDs[i3]);
+ 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 (i3 = 0; i3 < parents.length; i3++) {
+ if (parents[i3].isDegenerate()) {
+ graph = actionDeleteWay(parents[i3].id)(graph);
+ }
}
return graph;
};
action.disabled = function(graph) {
- var nodes = allNodes(graph);
- var points = nodes.map(function(n3) {
- return projection2(n3.loc);
- });
- var startPoint = points[0];
- var endPoint = points[points.length - 1];
- var threshold = 0.2 * geoVecLength(startPoint, endPoint);
- var i3;
- if (threshold === 0) {
- return "too_bendy";
+ var seen = {};
+ var restrictionIDs = [];
+ var survivor;
+ var node, way;
+ var relations, relation, role;
+ var i3, j2, k2;
+ survivor = graph.entity(utilOldestID(nodeIDs));
+ for (i3 = 0; i3 < nodeIDs.length; i3++) {
+ node = graph.entity(nodeIDs[i3]);
+ 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;
+ }
+ }
}
- var maxDistance = 0;
- for (i3 = 1; i3 < points.length - 1; i3++) {
- var point2 = points[i3];
- var u2 = positionAlongWay(point2, startPoint, endPoint);
- var p2 = geoVecInterp(startPoint, endPoint, u2);
- var dist = geoVecLength(p2, point2);
- if (isNaN(dist) || dist > threshold) {
- return "too_bendy";
- } else if (dist > maxDistance) {
- maxDistance = dist;
+ for (i3 = 0; i3 < nodeIDs.length; i3++) {
+ node = graph.entity(nodeIDs[i3]);
+ var parents = graph.parentWays(node);
+ for (j2 = 0; j2 < parents.length; j2++) {
+ var parent = parents[j2];
+ relations = graph.parentRelations(parent);
+ for (k2 = 0; k2 < relations.length; k2++) {
+ relation = relations[k2];
+ if (relation.hasFromViaTo()) {
+ restrictionIDs.push(relation.id);
+ }
+ }
}
}
- var keepingAllNodes = nodes.every(function(node, i4) {
- return i4 === 0 || i4 === 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";
+ restrictionIDs = utilArrayUniq(restrictionIDs);
+ for (i3 = 0; i3 < restrictionIDs.length; i3++) {
+ relation = graph.entity(restrictionIDs[i3]);
+ if (!relation.isComplete(graph))
+ continue;
+ var memberWays = relation.members.filter(function(m2) {
+ return m2.type === "way";
+ }).map(function(m2) {
+ return graph.entity(m2.id);
+ });
+ memberWays = utilArrayUniq(memberWays);
+ var f2 = relation.memberByRole("from");
+ var t2 = relation.memberByRole("to");
+ var isUturn = f2.id === t2.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 n3 = nodeIDs[j2];
+ if (nodes.from.indexOf(n3) !== -1) {
+ connectFrom = true;
+ }
+ if (nodes.via.indexOf(n3) !== -1) {
+ connectVia = true;
+ }
+ if (nodes.to.indexOf(n3) !== -1) {
+ connectTo = true;
+ }
+ if (nodes.keyfrom.indexOf(n3) !== -1) {
+ connectKeyFrom = true;
+ }
+ if (nodes.keyto.indexOf(n3) !== -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 (k2 = 0; k2 < nodeIDs.length; k2++) {
+ if (nodeIDs[k2] === survivor.id)
+ continue;
+ if (way.areAdjacent(nodeIDs[k2], survivor.id)) {
+ way = way.removeNode(nodeIDs[k2]);
+ } else {
+ way = way.replaceNode(nodeIDs[k2], survivor.id);
+ }
+ }
+ if (way.isDegenerate()) {
+ return "restriction";
+ }
+ }
+ }
+ return false;
+ function hasDuplicates(n4, i4, arr) {
+ return arr.indexOf(n4) !== arr.lastIndexOf(n4);
+ }
+ function keyNodeFilter(froms, tos) {
+ return function(n4) {
+ return froms.indexOf(n4) === -1 && tos.indexOf(n4) === -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());
+ }
+ }
}
};
- 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, t2) {
- if (t2 === null || !isFinite(t2))
- t2 = 1;
- t2 = Math.min(Math.max(+t2, 0), 1);
- var nodes = utilGetAllNodes(reflectIds, graph);
- var points = nodes.map(function(n3) {
- return projection2(n3.loc);
+ // 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);
});
- 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 p3, q3;
- var isLong = geoVecLength(p1, q1) > geoVecLength(p2, q2);
- if (_useLongAxis && isLong || !_useLongAxis && !isLong) {
- p3 = p1;
- q3 = q1;
- } else {
- p3 = p2;
- q3 = q2;
- }
- var dx = q3[0] - p3[0];
- var dy = q3[1] - p3[1];
- var a2 = (dx * dx - dy * dy) / (dx * dx + dy * dy);
- var b2 = 2 * dx * dy / (dx * dx + dy * dy);
- for (var i3 = 0; i3 < nodes.length; i3++) {
- var node = nodes[i3];
- var c2 = projection2(node.loc);
- var c22 = [
- a2 * (c2[0] - p3[0]) + b2 * (c2[1] - p3[1]) + p3[0],
- b2 * (c2[0] - p3[0]) - a2 * (c2[1] - p3[1]) + p3[1]
- ];
- var loc2 = projection2.invert(c22);
- node = node.move(geoVecInterp(node.loc, loc2, t2));
- graph = graph.replace(node);
+ for (var id2 in _copies) {
+ graph = graph.replace(_copies[id2]);
}
return graph;
};
- action.useLongAxis = function(val) {
- if (!arguments.length)
- return _useLongAxis;
- _useLongAxis = val;
- return action;
+ action.copies = function() {
+ return _copies;
};
- action.transitionable = true;
return action;
}
- // modules/actions/upgrade_tags.js
- function actionUpgradeTags(entityId, oldTags, replaceTags) {
+ // modules/actions/delete_member.js
+ function actionDeleteMember(relationId, memberIndex) {
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];
+ 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(difference2, discardTags) {
+ discardTags = discardTags || {};
+ return (graph) => {
+ difference2.modified().forEach(checkTags);
+ difference2.created().forEach(checkTags);
+ return graph;
+ function checkTags(entity) {
+ const keys2 = Object.keys(entity.tags);
+ let didDiscard = false;
+ let tags = {};
+ for (let i3 = 0; i3 < keys2.length; i3++) {
+ const k2 = keys2[i3];
+ if (discardTags[k2] || !entity.tags[k2]) {
+ didDiscard = true;
} else {
- if (replaceTags && replaceTags[oldTagKey]) {
- semiIndex = oldIndex;
+ tags[k2] = entity.tags[k2];
+ }
+ }
+ 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 i3 = 0; i3 < parentWays.length; i3++) {
+ way = parentWays[i3];
+ 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 });
}
- 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(";");
+ 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 {
- tags[replaceKey] = replaceValue;
+ sharedRelation = relation;
}
+ } else {
+ seenRelationIds[relation.id] = way.id;
}
- }
- }
- return graph.replace(entity.update({ tags }));
+ });
+ });
+ if (sharedRelation)
+ return "relation";
};
- }
-
- // modules/behavior/edit.js
- function behaviorEdit(context) {
- function behavior() {
- context.map().minzoom(context.minEditableZoom());
- }
- behavior.off = function() {
- context.map().minzoom(0);
+ action.limitWays = function(val) {
+ if (!arguments.length)
+ return wayIds;
+ wayIds = val;
+ return action;
};
- return behavior;
+ return action;
}
- // modules/behavior/hover.js
- function behaviorHover(context) {
- var dispatch14 = 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);
- dispatch14.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);
- dispatch14.call("hover", this, _targets);
+ // 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 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 extractFromWayOrRelation(entity, graph) {
+ var fromGeometry = entity.geometry(graph);
+ var keysToCopyAndRetain = ["source", "wheelchair"];
+ var keysToRetain = ["area"];
+ var buildingKeysToRetain = ["architect", "building", "height", "layer", "nycdoitt:bin"];
+ 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();
}
- function pointerout(d3_event) {
- var target = eventTarget(d3_event);
- var index = _targets.indexOf(target);
- if (index !== -1) {
- _targets.splice(index);
- updateHover(d3_event, _targets);
+ 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;
}
- }
- function allowsVertex(d2) {
- return d2.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d2, 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";
+ if (keysToRetain.indexOf(key) !== -1) {
+ continue;
}
- 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;
+ if (isBuilding) {
+ if (buildingKeysToRetain.indexOf(key) !== -1 || key.match(/^building:.{1,}/) || key.match(/^roof:.{1,}/))
+ continue;
}
- 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 i3 in targets) {
- var datum2 = targets[i3];
- 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 j3 in datum2.members) {
- selector += ", ." + datum2.members[j3].id;
- }
- }
- }
+ if (isIndoorArea && key === "indoor") {
+ continue;
}
- 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);
+ pointTags[key] = entityTags[key];
+ if (keysToCopyAndRetain.indexOf(key) !== -1 || key.match(/^addr:.{1,}/)) {
+ continue;
+ } else if (isIndoorArea && key === "level") {
+ continue;
}
- dispatch14.call("hover", this, !suppressed && targets);
+ 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 }));
}
- 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;
+ action.getExtractedNodeID = function() {
+ return extractedNodeID;
};
- return utilRebind(behavior, dispatch14, "on");
+ return action;
}
- // modules/behavior/draw.js
- var _disableSpace = false;
- var _lastSpace = null;
- function behaviorDraw(context) {
- var dispatch14 = 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 d2 = element.__data__;
- return d2 && d2.properties && d2.properties.target ? d2 : {};
- }
- function pointerdown(d3_event) {
- if (_downPointer)
- return;
- var pointerLocGetter = utilFastMouse(this);
- _downPointer = {
- id: d3_event.pointerId || "mouse",
- pointerLocGetter,
- downTime: +/* @__PURE__ */ new Date(),
- downLoc: pointerLocGetter(d3_event)
- };
- dispatch14.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 = +/* @__PURE__ */ 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);
- }
+ // 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);
+ })
+ );
}
- 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;
- dispatch14.call("downcancel", this);
+ var action = function(graph) {
+ var ways = ids.map(graph.entity, graph);
+ var survivorID = utilOldestID(ways.map((way) => way.id));
+ ways.sort(function(a2, b2) {
+ var aSided = a2.isSided();
+ var bSided = b2.isSided();
+ return aSided && !bSided ? -1 : bSided && !aSided ? 1 : 0;
+ });
+ var sequences = osmJoinWays(ways, graph);
+ var joined = sequences[0];
+ graph = sequences.actions.reduce(function(g3, action2) {
+ return action2(g3);
+ }, graph);
+ var survivor = graph.entity(survivorID);
+ survivor = survivor.update({ nodes: joined.nodes.map(function(n3) {
+ return n3.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);
}
- 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;
- dispatch14.call("move", this, d3_event, datum2(d3_event));
- }
- function pointercancel(d3_event) {
- if (_downPointer && _downPointer.id === (d3_event.pointerId || "mouse")) {
- if (!_downPointer.isCancelled) {
- dispatch14.call("downcancel", this);
+ 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 i3;
+ var sortedParentRelations = function(id2) {
+ return graph.parentRelations(graph.entity(id2)).filter((rel) => !rel.isRestriction() && !rel.isConnectivity()).sort((a2, b2) => a2.id - b2.id);
+ };
+ var relsA = sortedParentRelations(ids[0]);
+ for (i3 = 1; i3 < ids.length; i3++) {
+ var relsB = sortedParentRelations(ids[i3]);
+ if (!utilArrayIdentical(relsA, relsB)) {
+ return "conflicting_relations";
}
- _downPointer = null;
}
- }
- function mouseenter() {
- _mouseLeave = false;
- }
- function mouseleave() {
- _mouseLeave = true;
- }
- function allowsVertex(d2) {
- return d2.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d2, context.graph());
- }
- function click(d3_event, loc) {
- var d2 = datum2(d3_event);
- var target = d2 && d2.properties && d2.properties.entity;
- var mode = context.mode();
- if (target && target.type === "node" && allowsVertex(target)) {
- dispatch14.call("clickNode", this, target, d2);
- 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]];
- dispatch14.call("clickWay", this, choice.loc, edge, d2);
- return;
+ for (i3 = 0; i3 < ids.length - 1; i3++) {
+ for (var j2 = i3 + 1; j2 < ids.length; j2++) {
+ var path1 = graph.childNodes(graph.entity(ids[i3])).map(function(e3) {
+ return e3.loc;
+ });
+ var path2 = graph.childNodes(graph.entity(ids[j2])).map(function(e3) {
+ return e3.loc;
+ });
+ var intersections = geoPathIntersections(path1, path2);
+ var common = utilArrayIntersection(
+ joined[0].nodes.map(function(n3) {
+ return n3.loc.toString();
+ }),
+ intersections.map(function(n3) {
+ return n3.toString();
+ })
+ );
+ if (common.length !== intersections.length) {
+ return "paths_intersect";
+ }
}
- } else if (mode.id !== "add-point" || mode.preset.matchGeometry("point")) {
- var locLatLng = context.projection.invert(loc);
- dispatch14.call("click", this, locLatLng, d2);
}
- }
- 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;
+ var nodeIds = joined[0].nodes.map(function(n3) {
+ return n3.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(m2) {
+ return nodeIds.indexOf(m2.id) >= 0;
+ })) {
+ relation = parent;
+ }
+ });
+ for (var k2 in way.tags) {
+ if (!(k2 in tags)) {
+ tags[k2] = way.tags[k2];
+ } else if (tags[k2] && osmIsInterestingTag(k2) && tags[k2] !== way.tags[k2]) {
+ conflicting = true;
+ }
}
- }
- 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();
- dispatch14.call("undo");
- }
- function del(d3_event) {
- d3_event.preventDefault();
- dispatch14.call("cancel");
- }
- function ret(d3_event) {
- d3_event.preventDefault();
- dispatch14.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, dispatch14, "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(x2) {
- return function() {
- return x2;
+ if (relation) {
+ return relation.isRestriction() ? "restriction" : "connectivity";
+ }
+ if (conflicting) {
+ return "conflicting_tags";
+ }
};
+ return action;
}
- // node_modules/d3-scale/src/number.js
- function number2(x2) {
- return +x2;
- }
-
- // node_modules/d3-scale/src/continuous.js
- var unit = [0, 1];
- function identity3(x2) {
- return x2;
- }
- function normalize(a2, b2) {
- return (b2 -= a2 = +a2) ? function(x2) {
- return (x2 - a2) / b2;
- } : constants(isNaN(b2) ? NaN : 0.5);
- }
- function clamper(a2, b2) {
- var t2;
- if (a2 > b2)
- t2 = a2, a2 = b2, b2 = t2;
- return function(x2) {
- return Math.max(a2, Math.min(b2, x2));
- };
- }
- 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(x2) {
- return r0(d0(x2));
- };
- }
- function polymap(domain2, range3, interpolate) {
- var j3 = Math.min(domain2.length, range3.length) - 1, d2 = new Array(j3), r2 = new Array(j3), i3 = -1;
- if (domain2[j3] < domain2[0]) {
- domain2 = domain2.slice().reverse();
- range3 = range3.slice().reverse();
- }
- while (++i3 < j3) {
- d2[i3] = normalize(domain2[i3], domain2[i3 + 1]);
- r2[i3] = interpolate(range3[i3], range3[i3 + 1]);
- }
- return function(x2) {
- var i4 = bisect_default(domain2, x2, 1, j3) - 1;
- return r2[i4](d2[i4](x2));
- };
- }
- 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 n3 = Math.min(domain2.length, range3.length);
- if (clamp3 !== identity3)
- clamp3 = clamper(domain2[0], domain2[n3 - 1]);
- piecewise = n3 > 2 ? polymap : bimap;
- output = input = null;
- return scale;
- }
- function scale(x2) {
- return x2 == null || isNaN(x2 = +x2) ? unknown : (output || (output = piecewise(domain2.map(transform2), range3, interpolate)))(transform2(clamp3(x2)));
+ // 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);
+ })
+ );
}
- scale.invert = function(y2) {
- return clamp3(untransform((input || (input = piecewise(range3, domain2.map(transform2), number_default)))(y2)));
- };
- scale.domain = function(_2) {
- return arguments.length ? (domain2 = Array.from(_2, number2), rescale()) : domain2.slice();
- };
- scale.range = function(_2) {
- return arguments.length ? (range3 = Array.from(_2), rescale()) : range3.slice();
- };
- scale.rangeRound = function(_2) {
- return range3 = Array.from(_2), interpolate = round_default, rescale();
- };
- scale.clamp = function(_2) {
- return arguments.length ? (clamp3 = _2 ? true : identity3, rescale()) : clamp3 !== identity3;
- };
- scale.interpolate = function(_2) {
- return arguments.length ? (interpolate = _2, rescale()) : interpolate;
- };
- scale.unknown = function(_2) {
- return arguments.length ? (unknown = _2, scale) : unknown;
+ 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 i3;
+ var node;
+ for (i3 = 0; i3 < nodes.length; i3++) {
+ node = nodes[i3];
+ if (canBeReplaced(node) && node.isNew()) {
+ replaceNode(node);
+ break;
+ }
+ }
+ if (!inserted && point2.hasInterestingTags()) {
+ for (i3 = 0; i3 < nodes.length; i3++) {
+ node = nodes[i3];
+ if (canBeReplaced(node) && !node.hasInterestingTags()) {
+ replaceNode(node);
+ break;
+ }
+ }
+ if (!inserted) {
+ for (i3 = 0; i3 < nodes.length; i3++) {
+ node = nodes[i3];
+ 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;
};
- return function(t2, u2) {
- transform2 = t2, untransform = u2;
- return rescale();
+ 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";
+ }
};
- }
- function continuous() {
- return transformer2()(identity3, identity3);
- }
-
- // node_modules/d3-format/src/formatDecimal.js
- function formatDecimal_default(x2) {
- return Math.abs(x2 = Math.round(x2)) >= 1e21 ? x2.toLocaleString("en").replace(/,/g, "") : x2.toString(10);
- }
- function formatDecimalParts(x2, p2) {
- if ((i3 = (x2 = p2 ? x2.toExponential(p2 - 1) : x2.toExponential()).indexOf("e")) < 0)
- return null;
- var i3, coefficient = x2.slice(0, i3);
- return [
- coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,
- +x2.slice(i3 + 1)
- ];
- }
-
- // node_modules/d3-format/src/exponent.js
- function exponent_default(x2) {
- return x2 = formatDecimalParts(Math.abs(x2)), x2 ? x2[1] : NaN;
+ return action;
}
- // node_modules/d3-format/src/formatGroup.js
- function formatGroup_default(grouping, thousands) {
- return function(value, width) {
- var i3 = value.length, t2 = [], j3 = 0, g3 = grouping[0], length = 0;
- while (i3 > 0 && g3 > 0) {
- if (length + g3 + 1 > width)
- g3 = Math.max(1, width - length);
- t2.push(value.substring(i3 -= g3, i3 + g3));
- if ((length += g3 + 1) > width)
- break;
- g3 = grouping[j3 = (j3 + 1) % grouping.length];
+ // 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 i3 = 0; i3 < nodeIDs.length; i3++) {
+ var node = graph.entity(nodeIDs[i3]);
+ if (node.hasInterestingTags()) {
+ interestingLoc = ++interestingCount === 1 ? node.loc : null;
+ }
+ sum = geoVecAdd(sum, node.loc);
}
- return t2.reverse().join(thousands);
+ 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 i3 = 0; i3 < nodeIDs.length; i3++) {
+ var node = graph.entity(nodeIDs[i3]);
+ if (node.loc !== toLoc) {
+ graph = graph.replace(node.move(toLoc));
+ }
+ }
+ return actionConnect(nodeIDs)(graph);
};
- }
-
- // node_modules/d3-format/src/formatNumerals.js
- function formatNumerals_default(numerals) {
- return function(value) {
- return value.replace(/[0-9]/g, function(i3) {
- return numerals[+i3];
- });
+ action.disabled = function(graph) {
+ if (nodeIDs.length < 2)
+ return "not_eligible";
+ for (var i3 = 0; i3 < nodeIDs.length; i3++) {
+ var entity = graph.entity(nodeIDs[i3]);
+ if (entity.type !== "node")
+ return "not_eligible";
+ }
+ return actionConnect(nodeIDs).disabled(graph);
};
+ return action;
}
- // 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 + "";
+ // modules/osm/changeset.js
+ function osmChangeset() {
+ if (!(this instanceof osmChangeset)) {
+ return new osmChangeset().initialize(arguments);
+ } else if (arguments.length) {
+ this.initialize(arguments);
+ }
}
- 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(s2) {
- out:
- for (var n3 = s2.length, i3 = 1, i0 = -1, i1; i3 < n3; ++i3) {
- switch (s2[i3]) {
- case ".":
- i0 = i1 = i3;
- break;
- case "0":
- if (i0 === 0)
- i0 = i3;
- i1 = i3;
- break;
- default:
- if (!+s2[i3])
- break out;
- if (i0 > 0)
- i0 = 0;
- break;
+ 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(k2) {
+ return { "@k": k2, "@v": this.tags[k2] };
+ }, 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(x2, order) {
+ var groups = {};
+ for (var i3 = 0; i3 < x2.length; i3++) {
+ var tagName = Object.keys(x2[i3])[0];
+ if (!groups[tagName])
+ groups[tagName] = [];
+ groups[tagName].push(x2[i3][tagName]);
}
+ var ordered = {};
+ order.forEach(function(o2) {
+ if (groups[o2])
+ ordered[o2] = groups[o2];
+ });
+ return ordered;
}
- return i0 > 0 ? s2.slice(0, i0) + s2.slice(i1 + 1) : s2;
- }
-
- // node_modules/d3-format/src/formatPrefixAuto.js
- var prefixExponent;
- function formatPrefixAuto_default(x2, p2) {
- var d2 = formatDecimalParts(x2, p2);
- if (!d2)
- return x2 + "";
- var coefficient = d2[0], exponent = d2[1], i3 = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1, n3 = coefficient.length;
- return i3 === n3 ? coefficient : i3 > n3 ? coefficient + new Array(i3 - n3 + 1).join("0") : i3 > 0 ? coefficient.slice(0, i3) + "." + coefficient.slice(i3) : "0." + new Array(1 - i3).join("0") + formatDecimalParts(x2, Math.max(0, p2 + i3 - 1))[0];
- }
-
- // node_modules/d3-format/src/formatRounded.js
- function formatRounded_default(x2, p2) {
- var d2 = formatDecimalParts(x2, p2);
- if (!d2)
- return x2 + "";
- var coefficient = d2[0], exponent = d2[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 = {
- "%": (x2, p2) => (x2 * 100).toFixed(p2),
- "b": (x2) => Math.round(x2).toString(2),
- "c": (x2) => x2 + "",
- "d": formatDecimal_default,
- "e": (x2, p2) => x2.toExponential(p2),
- "f": (x2, p2) => x2.toFixed(p2),
- "g": (x2, p2) => x2.toPrecision(p2),
- "o": (x2) => Math.round(x2).toString(8),
- "p": (x2, p2) => formatRounded_default(x2 * 100, p2),
- "r": formatRounded_default,
- "s": formatPrefixAuto_default,
- "X": (x2) => Math.round(x2).toString(16).toUpperCase(),
- "x": (x2) => Math.round(x2).toString(16)
- };
-
- // node_modules/d3-format/src/identity.js
- function identity_default3(x2) {
- return x2;
- }
-
- // 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, i3, n3, c2;
- 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) {
- i3 = -1, n3 = value.length;
- while (++i3 < n3) {
- if (c2 = value.charCodeAt(i3), 48 > c2 || c2 > 57) {
- valueSuffix = (c2 === 46 ? decimal + value.slice(i3 + 1) : value.slice(i3)) + valueSuffix;
- value = value.slice(0, i3);
- break;
- }
+ 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 i3 = 0; i3 < relations.length; i3++) {
+ var relation = relations[i3];
+ 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);
}
}
}
- 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);
+ changes2.relation = Object.values(sorted);
+ return changes2;
}
- format2.toString = function() {
- return specifier + "";
- };
- return format2;
- }
- function formatPrefix2(specifier, value) {
- var f3 = newFormat((specifier = formatSpecifier(specifier), specifier.type = "f", specifier)), e3 = Math.max(-8, Math.min(8, Math.floor(exponent_default(value) / 3))) * 3, k2 = Math.pow(10, -e3), prefix = prefixes[8 + e3 / 3];
- return function(value2) {
- return f3(k2 * value2) + prefix;
+ 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() {
+ return {};
}
- 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)));
+ // 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 i3 = 0; i3 < sources.length; ++i3) {
+ var source = sources[i3];
+ 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 });
+ }
+ });
- // 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;
+ // modules/osm/relation.js
+ function osmRelation() {
+ if (!(this instanceof osmRelation)) {
+ return new osmRelation().initialize(arguments);
+ } else if (arguments.length) {
+ this.initialize(arguments);
+ }
}
-
- // 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);
+ osmEntity.relation = osmRelation;
+ osmRelation.prototype = Object.create(osmEntity.prototype);
+ osmRelation.creationOrder = function(a2, b2) {
+ var aId = parseInt(osmEntity.id.toOSM(a2.id), 10);
+ var bId = parseInt(osmEntity.id.toOSM(b2.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 i3 = 0; i3 < this.members.length; i3++) {
+ var member = resolver.hasEntity(this.members[i3].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 i3 = 0; i3 < this.members.length; i3++) {
+ result[i3] = Object.assign({}, this.members[i3], { index: i3 });
}
- 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;
+ 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 i3 = 0; i3 < this.members.length; i3++) {
+ if (this.members[i3].role === role) {
+ return Object.assign({}, this.members[i3], { index: i3 });
+ }
}
- case "f":
- case "%": {
- if (specifier.precision == null && !isNaN(precision2 = precisionFixed_default(step)))
- specifier.precision = precision2 - (specifier.type === "%") * 2;
- break;
+ },
+ // Same as memberByRole, but returns all members with the given role
+ membersByRole: function(role) {
+ var result = [];
+ for (var i3 = 0; i3 < this.members.length; i3++) {
+ if (this.members[i3].role === role) {
+ result.push(Object.assign({}, this.members[i3], { index: i3 }));
+ }
}
- }
- return format(specifier);
- }
-
- // node_modules/d3-scale/src/linear.js
- function linearish(scale) {
- var domain2 = scale.domain;
- scale.ticks = function(count) {
- var d2 = domain2();
- return ticks(d2[0], d2[d2.length - 1], count == null ? 10 : count);
- };
- scale.tickFormat = function(count, specifier) {
- var d2 = domain2();
- return tickFormat(d2[0], d2[d2.length - 1], count == null ? 10 : count, specifier);
- };
- scale.nice = function(count) {
- if (count == null)
- count = 10;
- var d2 = domain2();
- var i0 = 0;
- var i1 = d2.length - 1;
- var start2 = d2[i0];
- var stop = d2[i1];
- var prestep;
- var step;
- var maxIter = 10;
- if (stop < start2) {
- step = start2, start2 = stop, stop = step;
- step = i0, i0 = i1, i1 = step;
+ 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 i3 = 0; i3 < this.members.length; i3++) {
+ if (this.members[i3].id === id2) {
+ return Object.assign({}, this.members[i3], { index: i3 });
+ }
}
- while (maxIter-- > 0) {
- step = tickIncrement(start2, stop, count);
- if (step === prestep) {
- d2[i0] = start2;
- d2[i1] = stop;
- return domain2(d2);
- } 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;
+ },
+ // 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 i3 = 0; i3 < this.members.length; i3++) {
+ if (this.members[i3].id === id2 && this.members[i3].role === role) {
+ return Object.assign({}, this.members[i3], { index: i3 });
}
- 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, n3 = 1, domain2 = [0.5], range3 = [0, 1], unknown;
- function scale(x2) {
- return x2 != null && x2 <= x2 ? range3[bisect_default(domain2, x2, 0, n3)] : unknown;
- }
- function rescale() {
- var i3 = -1;
- domain2 = new Array(n3);
- while (++i3 < n3)
- domain2[i3] = ((i3 + 1) * x12 - (i3 - n3) * x05) / (n3 + 1);
- return scale;
- }
- scale.domain = function(_2) {
- return arguments.length ? ([x05, x12] = _2, x05 = +x05, x12 = +x12, rescale()) : [x05, x12];
- };
- scale.range = function(_2) {
- return arguments.length ? (n3 = (range3 = Array.from(_2)).length - 1, rescale()) : range3.slice();
- };
- scale.invertExtent = function(y2) {
- var i3 = range3.indexOf(y2);
- return i3 < 0 ? [NaN, NaN] : i3 < 1 ? [x05, domain2[0]] : i3 >= n3 ? [domain2[n3 - 1], x12] : [domain2[i3 - 1], domain2[i3]];
- };
- scale.unknown = function(_2) {
- return arguments.length ? (unknown = _2, 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(a2, b2, steps2, units) {
- a2 = Number(a2);
- b2 = Number(b2);
- var sample = quantize().domain([0, 1]).range(quantize_default(number_default(a2, b2), steps2));
- return function(t2) {
- return String(sample(t2)) + (units || "");
+ },
+ 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(m2) {
+ return m2.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 i3 = 0; i3 < this.members.length; i3++) {
+ var member = this.members[i3];
+ 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 r2 = {
+ 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(k2) {
+ return { keyAttributes: { k: k2, v: this.tags[k2] } };
+ }, this)
+ }
};
- }
- 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(d2) {
- return ratchetyInterpolator(
- _params[d2.id][toFrom].opacity,
- _params[d2.id][fromTo].opacity,
- steps
- );
- }).styleTween("stroke-width", function(d2) {
- return ratchetyInterpolator(
- _params[d2.id][toFrom].width,
- _params[d2.id][fromTo].width,
- steps,
- "px"
- );
- }).styleTween("fill-opacity", function(d2) {
- return ratchetyInterpolator(
- _params[d2.id][toFrom].opacity,
- _params[d2.id][fromTo].opacity,
- steps
- );
- }).styleTween("r", function(d2) {
- return ratchetyInterpolator(
- _params[d2.id][toFrom].width,
- _params[d2.id][fromTo].width,
- steps,
- "px"
- );
+ if (changeset_id) {
+ r2.relation["@changeset"] = changeset_id;
+ }
+ return r2;
+ },
+ 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 i3 = 0; i3 < this.members.length; i3++) {
+ if (!resolver.hasEntity(this.members[i3].id)) {
+ return false;
+ }
+ }
+ return true;
+ },
+ hasFromViaTo: function() {
+ return this.members.some(function(m2) {
+ return m2.role === "from";
+ }) && this.members.some(function(m2) {
+ return m2.role === "via";
+ }) && this.members.some(function(m2) {
+ return m2.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(m2) {
+ return m2.role === "from";
+ });
+ var vias = this.members.filter(function(m2) {
+ return m2.role === "via";
+ });
+ var tos = this.members.filter(function(m2) {
+ return m2.role === "to";
+ });
+ if (froms.length !== 1 && this.tags.restriction !== "no_entry")
+ return false;
+ if (froms.some(function(m2) {
+ return m2.type !== "way";
+ }))
+ return false;
+ if (tos.length !== 1 && this.tags.restriction !== "no_exit")
+ return false;
+ if (tos.some(function(m2) {
+ return m2.type !== "way";
+ }))
+ return false;
+ if (vias.length === 0)
+ return false;
+ if (vias.length > 1 && vias.some(function(m2) {
+ return m2.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(m2) {
+ return "outer" === (m2.role || "outer");
});
- }
- function calcAnimationParams(selection2) {
- selection2.call(reset).each(function(d2) {
- var s2 = select_default2(this);
- var tag = s2.node().tagName;
- var p2 = { "from": {}, "to": {} };
- var opacity;
- var width;
- if (tag === "circle") {
- opacity = Number(s2.style("fill-opacity") || 0.5);
- width = Number(s2.style("r") || 15.5);
- } else {
- opacity = Number(s2.style("stroke-opacity") || 0.7);
- width = Number(s2.style("stroke-width") || 10);
- }
- p2.tag = tag;
- p2.from.opacity = opacity * 0.6;
- p2.to.opacity = opacity * 1.25;
- p2.from.width = width * 0.7;
- p2.to.width = width * (tag === "circle" ? 1.5 : 1);
- _params[d2.id] = p2;
+ var inners = this.members.filter(function(m2) {
+ return "inner" === m2.role;
});
- }
- 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));
+ 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(o3) {
+ return [area_default({ type: "Polygon", coordinates: [o3] }) > 2 * Math.PI ? o3.reverse() : o3];
});
- }
- function behavior(surface) {
- _done = false;
- _timer = timer(function() {
- if (surface.selectAll(selector).empty()) {
- return false;
+ function findOuter(inner2) {
+ var o3, outer;
+ for (o3 = 0; o3 < outers.length; o3++) {
+ outer = outers[o3];
+ if (geoPolygonContainsPolygon(outer, inner2)) {
+ return o3;
+ }
}
- surface.call(run, "from");
- _timer.stop();
- return true;
- }, 20);
- }
- behavior.restartIfNeeded = function(surface) {
- if (_selected.empty()) {
- surface.call(run, "from");
- if (_timer) {
- _timer.stop();
+ for (o3 = 0; o3 < outers.length; o3++) {
+ outer = outers[o3];
+ if (geoPolygonIntersectsPolygon(outer, inner2, false)) {
+ return o3;
+ }
}
}
- };
- behavior.off = function() {
- _done = true;
- if (_timer) {
- _timer.stop();
+ for (var i3 = 0; i3 < inners.length; i3++) {
+ var inner = inners[i3];
+ if (area_default({ type: "Polygon", coordinates: [inner] }) < 2 * Math.PI) {
+ inner = inner.reverse();
+ }
+ var o2 = findOuter(inners[i3]);
+ if (o2 !== void 0) {
+ result[o2].push(inners[i3]);
+ } else {
+ result.push([inners[i3]]);
+ }
}
- _selected.interrupt().call(reset);
- };
- return behavior;
- }
+ return result;
+ }
+ });
- // 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();
+ // modules/osm/qa_item.js
+ var QAItem = class _QAItem {
+ constructor(loc, service, itemType, id2, props) {
+ this.loc = loc;
+ this.service = service.title;
+ this.itemType = itemType;
+ this.id = id2 ? id2 : "".concat(_QAItem.id());
+ this.update(props);
+ if (service && typeof service.getIcon === "function") {
+ this.icon = service.getIcon(itemType);
}
}
- function behavior() {
- if (_operation && _operation.available()) {
- context.keybinding().on(_operation.keys, keypress);
- }
- return behavior;
+ 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;
}
- behavior.off = function() {
- context.keybinding().off(_operation.keys);
- };
- behavior.which = function(_2) {
- if (!arguments.length)
- return _operation;
- _operation = _2;
- return behavior;
- };
- return behavior;
- }
+ // Generic handling for newly created QAItems
+ static id() {
+ return this.nextId--;
+ }
+ };
+ QAItem.nextId = -1;
- // 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(n3) {
- return n3.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);
+ // 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;
}
- var operation = function() {
- if (!_actions.length)
- return;
- var combinedAction = function(graph, t2) {
- _actions.forEach(function(action) {
- if (!action.disabled(graph)) {
- graph = action(graph, t2);
- }
- });
- 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";
+ function splitArea(nodes, idxA, graph) {
+ var lengths = new Array(nodes.length);
+ var length2;
+ var i3;
+ var best = 0;
+ var idxB;
+ function wrap2(index) {
+ return utilWrap(index, nodes.length);
+ }
+ length2 = 0;
+ for (i3 = wrap2(idxA + 1); i3 !== idxA; i3 = wrap2(i3 + 1)) {
+ length2 += dist(graph, nodes[i3], nodes[wrap2(i3 - 1)]);
+ lengths[i3] = length2;
+ }
+ length2 = 0;
+ for (i3 = wrap2(idxA - 1); i3 !== idxA; i3 = wrap2(i3 - 1)) {
+ length2 += dist(graph, nodes[i3], nodes[wrap2(i3 + 1)]);
+ if (length2 < lengths[i3]) {
+ lengths[i3] = length2;
}
- 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;
- }
+ for (i3 = 0; i3 < nodes.length; i3++) {
+ var cost = lengths[i3] / dist(graph, nodes[idxA], nodes[i3]);
+ if (cost > best) {
+ idxB = i3;
+ best = cost;
}
- 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";
+ return idxB;
}
- var result = "", replacements = {
- "\u2318": "Ctrl",
- "\u21E7": "Shift",
- "\u2325": "Alt",
- "\u232B": "Backspace",
- "\u2326": "Delete"
- };
- for (var i3 = 0; i3 < code.length; i3++) {
- if (code[i3] in replacements) {
- result += replacements[code[i3]] + (i3 < code.length - 1 ? "+" : "");
- } else {
- result += code[i3];
+ function totalLengthBetweenNodes(graph, nodes) {
+ var totalLength = 0;
+ for (var i3 = 0; i3 < nodes.length - 1; i3++) {
+ totalLength += dist(graph, nodes[i3], nodes[i3 + 1]);
}
+ return totalLength;
}
- 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(n3) {
- return n3.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 i3 = nodes2.indexOf(id2);
- if (i3 === 0) {
- i3++;
- } else if (i3 === nodes2.length - 1) {
- i3--;
- } else {
- var a2 = geoSphericalDistance(entity.loc, context.entity(nodes2[i3 - 1]).loc);
- var b2 = geoSphericalDistance(entity.loc, context.entity(nodes2[i3 + 1]).loc);
- i3 = a2 < b2 ? i3 - 1 : i3 + 1;
- }
- nextSelectedID = nodes2[i3];
- 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));
+ 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 {
- context.map().centerEase(nextSelectedLoc);
- context.enter(modeBrowse(context));
+ nodesA = nodes.slice(idxA, idxB + 1);
+ nodesB = nodes.slice(idxB).concat(nodes.slice(0, idxA + 1));
}
} 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;
+ var idx = wayA.nodes.indexOf(nodeId, 1);
+ nodesA = wayA.nodes.slice(0, idx + 1);
+ nodesB = wayA.nodes.slice(idx);
}
- function incompleteRelation(id2) {
- var entity = context.entity(id2);
- return entity.type === "relation" && !entity.isComplete(context.graph());
+ 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 });
}
- function protectedMember(id2) {
- var entity = context.entity(id2);
- if (entity.type !== "way")
- return false;
- var parents = context.graph().parentRelations(entity);
- for (var i3 = 0; i3 < parents.length; i3++) {
- var parent = parents[i3];
- var type2 = parent.tags.type;
- var role = parent.memberById(id2).role || "outer";
- if (type2 === "route" || type2 === "boundary" || type2 === "multipolygon" && role === "outer") {
- return true;
- }
+ 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 });
}
- 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(n3) {
- return n3.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);
+ 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 v2 = relation.membersByRole("via");
+ var t2 = relation.memberByRole("to");
+ var i3;
+ if (f2.id === wayA.id || t2.id === wayA.id) {
+ var keepB = false;
+ if (v2.length === 1 && v2[0].type === "node") {
+ keepB = wayB.contains(v2[0].id);
+ } else {
+ for (i3 = 0; i3 < v2.length; i3++) {
+ if (v2[i3].type === "way") {
+ var wayVia = graph.hasEntity(v2[i3].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 (i3 = 0; i3 < v2.length; i3++) {
+ if (v2[i3].type === "way" && v2[i3].id === wayA.id) {
+ member = {
+ id: wayB.id,
+ type: "way",
+ role: "via"
+ };
+ graph = actionAddMember(relation.id, member, v2[i3].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: {} }));
}
- return null;
+ _createdWayIDs.push(wayB.id);
+ return graph;
}
- var operation = function() {
- if (!_actions.length)
- return;
- var combinedAction = function(graph, t2) {
- _actions.forEach(function(action) {
- if (!action.disabled(graph)) {
- graph = action(graph, t2);
- }
- });
- return graph;
- };
- combinedAction.transitionable = true;
- context.perform(combinedAction, operation.annotation());
- window.setTimeout(function() {
- context.validator().validate();
- }, 300);
+ var action = function(graph) {
+ _createdWayIDs = [];
+ var newWayIndex = 0;
+ for (var i3 = 0; i3 < nodeIds.length; i3++) {
+ var nodeId = nodeIds[i3];
+ 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;
};
- operation.available = function() {
- return _actions.length && selectedIDs.length === _actions.length;
+ action.getCreatedWayIDs = function() {
+ return _createdWayIDs;
};
- 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";
+ 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 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 splittableParents;
+ function isSplittable(parent) {
+ if (_wayIDs && _wayIDs.indexOf(parent.id) === -1)
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);
- });
+ if (parent.isClosed())
+ return true;
+ for (var i3 = 1; i3 < parent.nodes.length - 1; i3++) {
+ if (parent.nodes[i3] === nodeId)
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(n3) {
- return n3.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;
+ action.ways = function(graph) {
+ return utilArrayUniq([].concat.apply([], nodeIds.map(function(nodeId) {
+ return action.waysForNode(nodeId, graph);
+ })));
};
- 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;
- }
+ action.disabled = function(graph) {
+ for (var i3 = 0; i3 < nodeIds.length; i3++) {
+ var nodeId = nodeIds[i3];
+ var candidates = action.waysForNode(nodeId, graph);
+ if (candidates.length === 0 || _wayIDs && _wayIDs.length !== candidates.length) {
+ return "not_eligible";
}
- 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);
+ action.limitWays = function(val) {
+ if (!arguments.length)
+ return _wayIDs;
+ _wayIDs = val;
+ return action;
};
- operation.annotation = function() {
- return _t("operations.reflect.annotation." + axis + ".feature", { n: selectedIDs.length });
+ action.keepHistoryOn = function(val) {
+ if (!arguments.length)
+ return _keepHistoryOn;
+ _keepHistoryOn = val;
+ return action;
};
- 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;
+ return action;
}
- // 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(n3) {
- return n3.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";
+ // 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];
}
- 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;
+ if (!entity) {
+ throw new Error("entity " + id2 + " not found");
}
- function incompleteRelation(id2) {
- var entity = context.entity(id2);
- return entity.type === "relation" && !entity.isComplete(context.graph());
+ 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];
}
- };
- 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;
+ 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);
}
- 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(n3) {
- return projection2(n3.loc);
- });
- _pivot = getPivot(points);
- _prevAngle = void 0;
+ 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);
}
- 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 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 i3 = 0; i3 < entity.nodes.length; i3++) {
+ nodes[i3] = this.entity(entity.nodes[i3]);
+ }
+ 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 i3, j2, k2, id2;
+ for (i3 = 0; i3 < entities.length; i3++) {
+ var entity = entities[i3];
+ 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 (k2 = 1; k2 < stack.length; k2++) {
+ var ents = stack[k2].entities;
+ if (ents.hasOwnProperty(id2) && ents[id2] === void 0) {
+ delete ents[id2];
+ }
+ }
+ }
+ }
+ }
+ for (i3 = 0; i3 < stack.length; i3++) {
+ stack[i3]._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, i3;
+ 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 (i3 = 0; i3 < removed.length; i3++) {
+ parentWays[removed[i3]] = new Set(parentWays[removed[i3]]);
+ parentWays[removed[i3]].delete(oldentity.id);
+ }
+ for (i3 = 0; i3 < added.length; i3++) {
+ parentWays[added[i3]] = new Set(parentWays[added[i3]]);
+ parentWays[added[i3]].add(entity.id);
+ }
+ } else if (type2 === "relation") {
+ var oldentityMemberIDs = oldentity ? oldentity.members.map(function(m2) {
+ return m2.id;
+ }) : [];
+ var entityMemberIDs = entity ? entity.members.map(function(m2) {
+ return m2.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 (i3 = 0; i3 < removed.length; i3++) {
+ parentRels[removed[i3]] = new Set(parentRels[removed[i3]]);
+ parentRels[removed[i3]].delete(oldentity.id);
+ }
+ for (i3 = 0; i3 < added.length; i3++) {
+ parentRels[added[i3]] = new Set(parentRels[added[i3]]);
+ parentRels[added[i3]].add(entity.id);
}
}
- return _pivot2;
+ },
+ 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 i3 = 0; i3 < arguments.length; i3++) {
+ arguments[i3].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 i3 in entities) {
+ this.entities[i3] = entities[i3];
+ this._updateCalculated(base.entities[i3], this.entities[i3]);
+ }
+ return this;
}
- function finish(d3_event) {
- d3_event.stopPropagation();
- context.replace(actionNoop(), annotation);
- context.enter(modeSelect(context, entityIDs));
+ };
+
+ // modules/osm/intersection.js
+ function osmTurn(turn) {
+ if (!(this instanceof osmTurn)) {
+ return new osmTurn(turn);
}
- function cancel() {
- if (_prevGraph)
- context.pop();
- context.enter(modeSelect(context, entityIDs));
+ Object.assign(this, turn);
+ }
+ function osmIntersection(graph, startVertexId, maxDistance) {
+ maxDistance = maxDistance || 30;
+ var vgraph = coreGraph();
+ var i3, j2, k2;
+ function memberOfRestriction(entity) {
+ return graph.parentRelations(entity).some(function(r2) {
+ return r2.isRestriction();
+ });
}
- function undone() {
- context.enter(modeBrowse(context));
+ 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,
+ "busway": true,
+ "road": true,
+ "track": true
+ };
+ return roads[way2.tags.highway];
}
- 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(n3) {
- return n3.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;
+ 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 (i3 = 0; i3 < checkWays.length; i3++) {
+ way = checkWays[i3];
+ 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 (k2 = 0; k2 < parents.length; k2++) {
+ parent = parents[k2];
+ if (parent === way)
+ continue;
+ if (ways.indexOf(parent) !== -1)
+ continue;
+ if (!isRoad(parent))
+ continue;
+ hasParents = true;
+ break;
+ }
+ if (hasParents) {
+ checkVertices.push(node);
}
}
- 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;
+ if (hasWays) {
+ vertices.push(vertex);
}
- 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;
+ 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(w2) {
+ var way2 = vgraph.entity(w2.id);
+ if (way2.tags.oneway === "-1") {
+ var action = actionReverse(way2.id, { reverseOneway: true });
+ actions.push(action);
+ vgraph = action(vgraph);
}
- }
- function move() {
- doMove();
- var nudge = geoViewportEdge(context.map().mouse(), context.map().dimensions());
- if (nudge) {
- startNudge(nudge);
- } else {
- stopNudge();
+ });
+ var origCount = osmEntity.id.next.way;
+ vertices.forEach(function(v2) {
+ var splitAll = actionSplit([v2.id]).keepHistoryOn("first");
+ if (!splitAll.disabled(vgraph)) {
+ splitAll.ways(vgraph).forEach(function(way2) {
+ var splitOne = actionSplit([v2.id]).limitWays([way2.id]).keepHistoryOn("first");
+ actions.push(splitOne);
+ vgraph = splitOne(vgraph);
+ });
}
+ });
+ osmEntity.id.next.way = origCount;
+ vertexIds = vertices.map(function(v2) {
+ return v2.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(v2) {
+ return v2.id;
+ });
+ wayIds = ways.map(function(w2) {
+ return w2.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
+ });
}
- 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));
+ 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 (i3 = 0; i3 < checkVertices.length; i3++) {
+ var vertexId = checkVertices[i3];
+ 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 a2 = parents[0];
+ var b2 = parents[1];
+ var aIsLeaf = a2 && !a2.__via;
+ var bIsLeaf = b2 && !b2.__via;
+ var leaf, survivor;
+ if (aIsLeaf && !bIsLeaf) {
+ leaf = a2;
+ survivor = b2;
+ } else if (!aIsLeaf && bIsLeaf) {
+ leaf = b2;
+ survivor = a2;
+ }
+ 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);
+ }
}
- 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;
+ } 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 intersection2 = {
+ graph: vgraph,
+ actions,
+ vertices,
+ ways
};
- 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);
+ intersection2.turns = function(fromWayId, maxViaWay) {
+ if (!fromWayId)
+ return [];
+ if (!maxViaWay)
+ maxViaWay = 0;
+ var vgraph2 = intersection2.graph;
+ var keyVertexIds = intersection2.vertices.map(function(v2) {
+ return v2.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 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);
}
}
- 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 dispatch14 = 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];
+ function stepNode(entity, currPath, currRestrictions) {
+ var i4, j3;
+ var parents2 = vgraph2.parentWays(entity);
+ var nextWays = [];
+ for (i4 = 0; i4 < parents2.length; i4++) {
+ var way2 = parents2[i4];
+ 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 v2 = restriction.membersByRole("via");
+ var t2 = 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 (t2.id === way2.id) {
+ if (v2.length === 1 && v2[0].type === "node") {
+ matchesViaTo = v2[0].id === entity.id && (matchesFrom && currPath.length === 2 || !matchesFrom && currPath.length > 2);
+ } else {
+ var pathVias = [];
+ for (k2 = 2; k2 < currPath.length; k2 += 2) {
+ pathVias.push(currPath[k2]);
+ }
+ var restrictionVias = [];
+ for (k2 = 0; k2 < v2.length; k2++) {
+ if (v2[k2].type === "way") {
+ restrictionVias.push(v2[k2].id);
+ }
+ }
+ var diff = utilArrayDifference(pathVias, restrictionVias);
+ matchesViaTo = !diff.length;
+ }
+ } else if (isOnly) {
+ for (k2 = 0; k2 < v2.length; k2++) {
+ if (v2[k2].type === "way" && v2[k2].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);
+ });
}
- d3_event.stopPropagation();
- function pointermove(d3_event2) {
- if (_pointerId !== (d3_event2.pointerId || "mouse"))
- return;
- var p2 = pointerLocGetter(d3_event2);
- if (!started) {
- var dist = geoVecLength(startOrigin, p2);
- var tolerance = d3_event2.pointerType === "pen" ? _penTolerancePx : _tolerancePx;
- if (dist < tolerance)
+ function stepWay(entity, currPath, currRestrictions, matchedRestriction) {
+ var i4;
+ if (currPath.length >= 3) {
+ var turnPath = currPath.slice();
+ if (matchedRestriction && matchedRestriction.direct === false) {
+ for (i4 = 0; i4 < turnPath.length; i4++) {
+ if (turnPath[i4] === matchedRestriction.from) {
+ turnPath = turnPath.slice(i4);
+ 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;
- started = true;
- dispatch14.call("start", this, d3_event2, _targetEntity);
- } else {
- startOrigin = p2;
- d3_event2.stopPropagation();
- d3_event2.preventDefault();
- var dx = p2[0] - startOrigin[0];
- var dy = p2[1] - startOrigin[1];
- dispatch14.call("move", this, d3_event2, _targetEntity, [p2[0] + offset[0], p2[1] + offset[1]], [dx, dy]);
}
- }
- function pointerup(d3_event2) {
- if (_pointerId !== (d3_event2.pointerId || "mouse"))
+ if (matchedRestriction && matchedRestriction.end)
return;
- _pointerId = null;
- if (started) {
- dispatch14.call("end", this, d3_event2, _targetEntity);
- d3_event2.preventDefault();
+ var n1 = vgraph2.entity(entity.first());
+ var n22 = vgraph2.entity(entity.last());
+ var dist = geoSphericalDistance(n1.loc, n22.loc);
+ var nextNodes = [];
+ if (currPath.length > 1) {
+ if (dist > maxDistance)
+ return;
+ if (!entity.__via)
+ return;
}
- 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);
+ if (!entity.__oneWay && // bidirectional..
+ keyVertexIds.indexOf(n1.id) !== -1 && // key vertex..
+ currPath.indexOf(n1.id) === -1) {
+ nextNodes.push(n1);
+ }
+ if (keyVertexIds.indexOf(n22.id) !== -1 && // key vertex..
+ currPath.indexOf(n22.id) === -1) {
+ nextNodes.push(n22);
+ }
+ nextNodes.forEach(function(nextNode) {
+ var fromRestrictions = vgraph2.parentRelations(entity).filter(function(r2) {
+ if (!r2.isRestriction())
+ return false;
+ var f2 = r2.memberByRole("from");
+ if (!f2 || f2.id !== entity.id)
+ return false;
+ var isOnly = /^only_/.test(r2.tags.restriction);
+ if (!isOnly)
+ return true;
+ var isOnlyVia = false;
+ var v2 = r2.membersByRole("via");
+ if (v2.length === 1 && v2[0].type === "node") {
+ isOnlyVia = v2[0].id === nextNode.id;
+ } else {
+ for (var i5 = 0; i5 < v2.length; i5++) {
+ if (v2[i5].type !== "way")
+ continue;
+ var viaWay = vgraph2.entity(v2[i5].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];
+ }
}
- selection2.on(_pointerPrefix + "down.drag" + _selector, delegate);
- }
- behavior.off = function(selection2) {
- selection2.on(_pointerPrefix + "down.drag" + _selector, null);
- };
- behavior.selector = function(_2) {
- if (!arguments.length)
- return _selector;
- _selector = _2;
- return behavior;
- };
- behavior.origin = function(_2) {
- if (!arguments.length)
- return _origin;
- _origin = _2;
- 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(_2) {
- if (!arguments.length)
- return _targetNode;
- _targetNode = _2;
- return behavior;
};
- behavior.targetEntity = function(_2) {
- if (!arguments.length)
- return _targetEntity;
- _targetEntity = _2;
- return behavior;
- };
- behavior.surface = function(_2) {
- if (!arguments.length)
- return _surface;
- _surface = _2;
- return behavior;
- };
- return utilRebind(behavior, dispatch14, "on");
+ return intersection2;
}
-
- // 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 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;
}
- function stopNudge() {
- if (_nudgeInterval) {
- window.clearInterval(_nudgeInterval);
- _nudgeInterval = null;
- }
+ if (fromNode === toNode) {
+ return "no_u_turn";
}
- function moveAnnotation(entity) {
- return _t("operations.move.annotation." + entity.geometry(context.graph()));
+ if ((angle2 < 23 || angle2 > 336) && fromOneWay && toOneWay) {
+ return "no_u_turn";
}
- 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);
+ if ((angle2 < 40 || angle2 > 319) && fromOneWay && toOneWay && turn.from.vertex !== turn.to.vertex) {
+ return "no_u_turn";
}
- function shouldSnapToNode(target) {
- if (!_activeEntity)
- return false;
- return _activeEntity.geometry(context.graph()) !== "vertex" || (target.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(target, context.graph()));
+ if (angle2 < 158) {
+ return "no_right_turn";
}
- function origin(entity) {
- return context.projection(entity.loc);
+ if (angle2 > 202) {
+ return "no_left_turn";
}
- function keydown(d3_event) {
- if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
- if (context.surface().classed("nope")) {
- context.surface().classed("nope-suppressed", true);
+ 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";
}
- context.surface().classed("nope", false).classed("nope-disabled", true);
- }
+ });
+ return Object.assign(
+ { closedWay: [], multipolygon: [], other: [] },
+ geometryGroups
+ );
}
- 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);
+ var action = function(graph) {
+ var entities = groupEntities(graph);
+ var polygons = entities.multipolygon.reduce(function(polygons2, m2) {
+ return polygons2.concat(osmJoinWays(m2.members, graph));
+ }, []).concat(entities.closedWay.map(function(d2) {
+ var member = [{ id: d2.id }];
+ member.nodes = graph.childNodes(d2);
+ return member;
+ }));
+ var contained = polygons.map(function(w2, i3) {
+ return polygons.map(function(d2, n3) {
+ if (i3 === n3)
+ return null;
+ return geoPolygonContainsPolygon(
+ d2.nodes.map(function(n4) {
+ return n4.loc;
+ }),
+ w2.nodes.map(function(n4) {
+ return n4.loc;
+ })
+ );
+ });
+ });
+ var members = [];
+ var outer = true;
+ while (polygons.length) {
+ extractUncontained(polygons);
+ polygons = polygons.filter(isContained);
+ contained = contained.filter(isContained).map(filterContained);
}
- }
- 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();
+ function isContained(d2, i3) {
+ return contained[i3].some(function(val) {
+ return val;
+ });
}
- 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());
+ function filterContained(d2) {
+ return d2.filter(isContained);
}
- _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 {};
+ function extractUncontained(polygons2) {
+ polygons2.forEach(function(d2, i3) {
+ if (!isContained(d2, i3)) {
+ d2.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 {
- var d2 = d3_event.target.__data__;
- return d2 && d2.properties && d2.properties.target ? d2 : {};
+ relation = osmRelation({ id: newRelationId, tags: { type: "multipolygon" } });
}
- }
- 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 d2 = datum2(d3_event);
- target = d2 && d2.properties && d2.properties.entity;
- var targetLoc = target && target.loc;
- var targetNodes = d2 && d2.properties && d2.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;
- }
+ entities.multipolygon.forEach(function(m2) {
+ if (m2.id !== relation.id) {
+ relation = relation.mergeTags(m2.tags);
+ graph = graph.remove(m2);
}
+ });
+ entities.closedWay.forEach(function(way) {
+ function isThisOuter(m2) {
+ return m2.id === way.id && m2.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";
}
- 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());
+ if (!entities.multipolygon.every(function(r2) {
+ return r2.isComplete(graph);
+ })) {
+ return "incomplete_relation";
}
- 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("")();
+ if (!entities.multipolygon.length) {
+ var sharedMultipolygons = [];
+ entities.closedWay.forEach(function(way, i3) {
+ if (i3 === 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";
}
- var nopeDisabled = context.surface().classed("nope-disabled");
- if (nopeDisabled) {
- context.surface().classed("nope", false).classed("nope-suppressed", isInvalid);
+ };
+ 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 {
- 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;
+ equivalenceClasses[item] = [j2];
}
- var ids = [entity.id, target.id];
- return actionConnect(ids).disabled(testGraph);
}
- function hasInvalidGeometry(entity, graph) {
- var parents = graph.parentWays(entity);
- var i3, j3, k2;
- for (i3 = 0; i3 < parents.length; i3++) {
- var parent = parents[i3];
- var nodes = [];
- var activeIndex = null;
- var relations = graph.parentRelations(parent);
- for (j3 = 0; j3 < relations.length; j3++) {
- if (!relations[j3].isMultipolygon())
- continue;
- var rings = osmJoinWays(relations[j3].members, graph);
- for (k2 = 0; k2 < rings.length; k2++) {
- nodes = rings[k2].nodes;
- if (nodes.find(function(n3) {
- return n3.id === entity.id;
- })) {
- activeIndex = k2;
- if (geoHasSelfIntersections(nodes, entity.id)) {
- return "multipolygonMember";
- }
- }
- rings[k2].coords = nodes.map(function(n3) {
- return n3.loc;
- });
- }
- for (k2 = 0; k2 < rings.length; k2++) {
- if (k2 === activeIndex)
- continue;
- if (geoHasLineIntersections(rings[activeIndex].nodes, rings[k2].nodes, entity.id)) {
- return "multipolygonRing";
- }
+ const NULLRESULT = { buffer1index: -1, buffer2index: -1, chain: null };
+ let candidates = [NULLRESULT];
+ for (let i3 = 0; i3 < buffer1.length; i3++) {
+ const item = buffer1[i3];
+ const buffer2indices = equivalenceClasses[item] || [];
+ let r2 = 0;
+ let c2 = candidates[0];
+ for (let jx = 0; jx < buffer2indices.length; jx++) {
+ const j2 = buffer2indices[jx];
+ let s2;
+ for (s2 = r2; s2 < candidates.length; s2++) {
+ if (candidates[s2].buffer2index < j2 && (s2 === candidates.length - 1 || candidates[s2 + 1].buffer2index > j2)) {
+ break;
}
}
- if (activeIndex === null) {
- nodes = parent.nodes.map(function(nodeID) {
- return graph.entity(nodeID);
- });
- if (nodes.length && geoHasSelfIntersections(nodes, entity.id)) {
- return parent.geometry(graph);
+ if (s2 < candidates.length) {
+ const newCandidate = { buffer1index: i3, buffer2index: j2, chain: candidates[s2] };
+ if (r2 === candidates.length) {
+ candidates.push(c2);
+ } else {
+ candidates[r2] = c2;
+ }
+ r2 = s2 + 1;
+ c2 = newCandidate;
+ if (r2 === candidates.length) {
+ break;
}
}
}
- 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();
- }
+ candidates[r2] = c2;
}
- function end(d3_event, entity) {
- if (_isCancelled)
- return;
- var wasPoint = entity.geometry(context.graph()) === "point";
- var d2 = datum2(d3_event);
- var nope = d2 && d2.properties && d2.properties.nope || context.surface().classed("nope");
- var target = d2 && d2.properties && d2.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);
+ 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)
});
- 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, t2) {
- if (t2 === 1)
- context.pop();
- return moveNode(graph, t2);
- };
- 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(_2) {
- if (!arguments.length)
- return _restoreSelectedIDs;
- _restoreSelectedIDs = _2;
- 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));
+ result.reverse();
+ return result;
}
- 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 diff3MergeRegions(a2, o2, b2) {
+ let hunks = [];
+ function addHunk(h2, ab) {
+ hunks.push({
+ ab,
+ oStart: h2.buffer1[0],
+ oLength: h2.buffer1[1],
+ // length of o to remove
+ abStart: h2.buffer2[0],
+ abLength: h2.buffer2[1]
+ // length of a/b to insert
+ // abContent: (ab === 'a' ? a : b).slice(h.buffer2[0], h.buffer2[0] + h.buffer2[1])
+ });
}
- }
- function abortUnwantedRequests(cache, tiles) {
- Object.keys(cache.inflightTile).forEach((k2) => {
- const wanted = tiles.find((tile) => k2 === tile.id);
- if (!wanted) {
- abortRequest(cache.inflightTile[k2]);
- delete cache.inflightTile[k2];
+ diffIndices(o2, a2).forEach((item) => addHunk(item, "a"));
+ diffIndices(o2, b2).forEach((item) => addHunk(item, "b"));
+ hunks.sort((x2, y2) => x2.oStart - y2.oStart);
+ let results = [];
+ let currOffset = 0;
+ function advanceTo(endOffset) {
+ if (endOffset > currOffset) {
+ results.push({
+ stable: true,
+ buffer: "o",
+ bufferStart: currOffset,
+ bufferLength: endOffset - currOffset,
+ bufferContent: o2.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" ? a2 : b2;
+ 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: [a2.length, -1, o2.length, -1],
+ b: [b2.length, -1, o2.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 b3 = bounds[hunk.ab];
+ b3[0] = Math.min(abStart, b3[0]);
+ b3[1] = Math.max(abEnd, b3[1]);
+ b3[2] = Math.min(oStart, b3[2]);
+ b3[3] = Math.max(oEnd, b3[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: a2.slice(aStart, aEnd),
+ oStart: regionStart,
+ oLength: regionEnd - regionStart,
+ oContent: o2.slice(regionStart, regionEnd),
+ bStart,
+ bLength: bEnd - bStart,
+ bContent: b2.slice(bStart, bEnd)
+ };
+ results.push(result);
}
- });
- }
- function encodeIssueRtree(d2) {
- return { minX: d2.loc[0], minY: d2.loc[1], maxX: d2.loc[0], maxY: d2.loc[1], data: d2 };
- }
- function updateRtree(item, replace) {
- _cache.rtree.remove(item, (a2, b2) => a2.data.id === b2.data.id);
- if (replace) {
- _cache.rtree.insert(item);
+ currOffset = regionEnd;
}
+ advanceTo(o2.length);
+ return results;
}
- function tokenReplacements(d2) {
- if (!(d2 instanceof QAItem))
- return;
- const replacements = {};
- const issueTemplate = _krData.errorTypes[d2.whichType];
- if (!issueTemplate) {
- console.log("No Template: ", d2.whichType);
- console.log(" ", d2.description);
- return;
+ function diff3Merge(a2, o2, b2, options2) {
+ let defaults = {
+ excludeFalseConflicts: true,
+ stringSeparator: /\s+/
+ };
+ options2 = Object.assign(defaults, options2);
+ if (typeof a2 === "string")
+ a2 = a2.split(options2.stringSeparator);
+ if (typeof o2 === "string")
+ o2 = o2.split(options2.stringSeparator);
+ if (typeof b2 === "string")
+ b2 = b2.split(options2.stringSeparator);
+ let results = [];
+ const regions = diff3MergeRegions(a2, o2, b2);
+ let okBuffer = [];
+ function flushOk() {
+ if (okBuffer.length) {
+ results.push({ ok: okBuffer });
+ }
+ okBuffer = [];
}
- if (!issueTemplate.regex)
- return;
- const errorRegex = new RegExp(issueTemplate.regex, "i");
- const errorMatch = errorRegex.exec(d2.description);
- if (!errorMatch) {
- console.log("Unmatched: ", d2.whichType);
- console.log(" ", d2.description);
- console.log(" ", errorRegex);
- return;
+ function isFalseConflict(a3, b3) {
+ if (a3.length !== b3.length)
+ return false;
+ for (let i3 = 0; i3 < a3.length; i3++) {
+ if (a3[i3] !== b3[i3])
+ return false;
+ }
+ return true;
}
- for (let i3 = 1; i3 < errorMatch.length; i3++) {
- let capture = errorMatch[i3];
- let idType;
- idType = "IDs" in issueTemplate ? issueTemplate.IDs[i3 - 1] : "";
- if (idType && capture) {
- capture = parseError(capture, idType);
+ regions.forEach((region) => {
+ if (region.stable) {
+ okBuffer.push(...region.bufferContent);
} else {
- const compare = capture.toLowerCase();
- if (_krData.localizeStrings[compare]) {
- capture = _t("QA.keepRight.error_parts." + _krData.localizeStrings[compare]);
+ if (options2.excludeFalseConflicts && isFalseConflict(region.aContent, region.bContent)) {
+ okBuffer.push(...region.aContent);
} else {
- capture = unescape_default(capture);
+ flushOk();
+ results.push({
+ conflict: {
+ a: region.aContent,
+ aIndex: region.aStart,
+ o: region.oContent,
+ oIndex: region.oStart,
+ b: region.bContent,
+ bIndex: region.bStart
+ }
+ });
}
}
- replacements["var" + i3] = capture;
- }
- return replacements;
+ });
+ flushOk();
+ return results;
}
- function parseError(capture, idType) {
- const compare = capture.toLowerCase();
- if (_krData.localizeStrings[compare]) {
- capture = _t("QA.keepRight.error_parts." + _krData.localizeStrings[compare]);
+
+ // modules/actions/merge_remote_changes.js
+ var import_lodash2 = __toESM(require_lodash());
+ function actionMergeRemoteChanges(id2, localGraph, remoteGraph, discardTags, formatUser) {
+ discardTags = discardTags || {};
+ var _option = "safe";
+ var _conflicts = [];
+ function user(d2) {
+ return typeof formatUser === "function" ? formatUser(d2) : (0, import_lodash2.escape)(d2);
}
- 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;
+ function mergeLocation(remote, target) {
+ function pointEqual(a2, b2) {
+ var epsilon3 = 1e-6;
+ return Math.abs(a2[0] - b2[0]) < epsilon3 && Math.abs(a2[1] - b2[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;
}
- return capture;
- function linkErrorObject2(d2) {
- return { html: `${d2}` };
+ 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 o2 = base.nodes || [];
+ var a2 = target.nodes || [];
+ var b2 = remote.nodes || [];
+ var nodes = [];
+ var hunks = diff3Merge(a2, o2, b2, { excludeFalseConflicts: true });
+ for (var i3 = 0; i3 < hunks.length; i3++) {
+ var hunk = hunks[i3];
+ if (hunk.ok) {
+ nodes.push.apply(nodes, hunk.ok);
+ } else {
+ var c2 = hunk.conflict;
+ if ((0, import_fast_deep_equal.default)(c2.o, c2.a)) {
+ nodes.push.apply(nodes, c2.b);
+ } else if ((0, import_fast_deep_equal.default)(c2.o, c2.b)) {
+ nodes.push.apply(nodes, c2.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 linkEntity2(d2) {
- return { html: `${d2}` };
+ 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 i3 = 0; i3 < children2.length; i3++) {
+ var id3 = children2[i3];
+ 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 linkURL(d2) {
- return { html: `${d2}` };
+ function updateChildren(updates, graph) {
+ for (var i3 = 0; i3 < updates.replacements.length; i3++) {
+ graph = graph.replace(updates.replacements[i3]);
+ }
+ if (updates.removeIds.length) {
+ graph = actionDeleteMultiple(updates.removeIds)(graph);
+ }
+ return graph;
}
- function parse211(capture2) {
- let newList = [];
- const items = capture2.split(", ");
- items.forEach((item) => {
- let id2 = linkEntity2("n" + item.slice(1));
- newList.push(id2);
+ 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 o2 = base.tags || {};
+ var a2 = target.tags || {};
+ var b2 = remote.tags || {};
+ var keys2 = utilArrayUnion(utilArrayUnion(Object.keys(o2), Object.keys(a2)), Object.keys(b2)).filter(function(k3) {
+ return !discardTags[k3];
});
- return newList.join(", ");
+ var tags = Object.assign({}, a2);
+ var changed = false;
+ for (var i3 = 0; i3 < keys2.length; i3++) {
+ var k2 = keys2[i3];
+ if (o2[k2] !== b2[k2] && a2[k2] !== b2[k2]) {
+ if (o2[k2] !== a2[k2]) {
+ _conflicts.push(_t.html(
+ "merge_remote_changes.conflict.tags",
+ { tag: k2, local: a2[k2], remote: b2[k2], user: { html: user(remote.user) } }
+ ));
+ } else {
+ if (b2.hasOwnProperty(k2)) {
+ tags[k2] = b2[k2];
+ } else {
+ delete tags[k2];
+ }
+ changed = true;
+ }
+ }
+ }
+ return changed && _conflicts.length === ccount ? target.update({ tags }) : target;
}
- 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] })
- );
+ 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;
}
- });
- return newList.join(", ");
+ }
+ 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 i3 = 0; i3 < ids.length; i3++) {
+ var id2 = ids[i3];
+ 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 i3 = 0; i3 < ids.length; i3++) {
+ var id2 = ids[i3];
+ 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 k2 = 0; k2 < parents.length; k2++) {
+ var way = parents[k2];
+ 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 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}`);
+ 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 i3 = 0; i3 < way.nodes.length; i3++) {
+ curr = graph.entity(way.nodes[i3]);
+ 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(intersection2, graph) {
+ var vertex = graph.entity(intersection2.nodeId);
+ var way1 = graph.entity(intersection2.movedId);
+ var way2 = graph.entity(intersection2.unmovedId);
+ var isEP1 = intersection2.movedIsEP;
+ var isEP2 = intersection2.unmovedIsEP;
+ if (isEP1 && isEP2)
+ return graph;
+ var nodes1 = graph.childNodes(way1).filter(function(n3) {
+ return n3 !== vertex;
});
- return newList.join(", ");
+ var nodes2 = graph.childNodes(way2).filter(function(n3) {
+ return n3 !== 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 i3 = 0; i3 < maxIter; i3++) {
+ 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 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] });
+ function cleanupIntersections(graph) {
+ for (var i3 = 0; i3 < cache.intersections.length; i3++) {
+ var obj = cache.intersections[i3];
+ 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 "";
+ return graph;
}
- 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(", ");
+ function limitDelta(graph) {
+ function moveNode(loc) {
+ return geoVecAdd(projection2(loc), _delta);
+ }
+ for (var i3 = 0; i3 < cache.intersections.length; i3++) {
+ var obj = cache.intersections[i3];
+ 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(n3) {
+ return moveNode(n3.loc);
+ });
+ var unmovedNodes = graph.childNodes(graph.entity(obj.unmovedId));
+ var unmovedPath = unmovedNodes.map(function(n3) {
+ return projection2(n3.loc);
+ });
+ var hits = geoPathIntersections(movedPath, unmovedPath);
+ for (var j2 = 0; i3 < hits.length; i3++) {
+ 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 i3 = 0; i3 < cache.nodes.length; i3++) {
+ var node = graph.entity(cache.nodes[i3]);
+ 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;
}
- var keepRight_default = {
- title: "keepRight",
- init() {
- _mainFileFetcher.get("keepRight").then((d2) => _krData = d2);
- if (!_cache) {
- this.reset();
+
+ // 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, t2) {
+ if (t2 === null || !isFinite(t2))
+ t2 = 1;
+ t2 = Math.min(Math.max(+t2, 0), 1);
+ var node = graph.entity(nodeID);
+ return graph.replace(
+ node.move(geoVecInterp(node.loc, toLoc, t2))
+ );
+ };
+ 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, t2) {
+ if (t2 === null || !isFinite(t2))
+ t2 = 1;
+ t2 = Math.min(Math.max(+t2, 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 });
}
- this.event = utilRebind(this, dispatch2, "on");
- },
- reset() {
- if (_cache) {
- Object.values(_cache.inflightTile).forEach(abortRequest);
+ 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;
}
- _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");
+ var nodeCount = {};
+ var points = [];
+ var corner = { i: 0, dotp: 1 };
+ var node, point2, loc, score, motions, i3, j2;
+ for (i3 = 0; i3 < nodes.length; i3++) {
+ node = nodes[i3];
+ nodeCount[node.id] = (nodeCount[node.id] || 0) + 1;
+ points.push({ id: node.id, coord: projection2(node.loc) });
+ }
+ if (points.length === 3) {
+ for (i3 = 0; i3 < 1e3; i3++) {
+ motions = points.map(calcMotion);
+ points[corner.i].coord = geoVecAdd(points[corner.i].coord, motions[corner.i]);
+ score = corner.dotp;
+ if (score < epsilon3) {
+ break;
}
- 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 d2 = new QAItem(loc, this, itemType, id2, {
- comment,
- description,
- whichType,
- parentIssueType,
- severity: whichTemplate.severity || "error",
- objectId,
- objectType,
- schema,
- title
- });
- d2.replacements = tokenReplacements(d2);
- _cache.data[id2] = d2;
- _cache.rtree.insert(encodeIssueRtree(d2));
- });
- dispatch2.call("loaded");
- }).catch(() => {
- delete _cache.inflightTile[tile.id];
- _cache.loadedTile[tile.id] = true;
+ }
+ node = graph.entity(nodes[corner.i].id);
+ loc = projection2.invert(points[corner.i].coord);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t2)));
+ } else {
+ var straights = [];
+ var simplified = [];
+ for (i3 = 0; i3 < points.length; i3++) {
+ point2 = points[i3];
+ var dotp = 0;
+ if (isClosed || i3 > 0 && i3 < points.length - 1) {
+ var a2 = points[(i3 - 1 + points.length) % points.length];
+ var b2 = points[(i3 + 1) % points.length];
+ dotp = Math.abs(geoOrthoNormalizedDotProduct(a2.coord, b2.coord, point2.coord));
+ }
+ if (dotp > upperThreshold) {
+ straights.push(point2);
+ } else {
+ simplified.push(point2);
+ }
+ }
+ var bestPoints = clonePoints(simplified);
+ var originalPoints = clonePoints(simplified);
+ score = Infinity;
+ for (i3 = 0; i3 < 1e3; i3++) {
+ 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(p2) {
+ return p2.coord;
});
- });
- },
- postUpdate(d2, callback) {
- if (_cache.inflightPost[d2.id]) {
- return callback({ message: "Error update already inflight", status: -2 }, d2);
+ if (isClosed)
+ bestCoords.push(bestCoords[0]);
+ for (i3 = 0; i3 < bestPoints.length; i3++) {
+ point2 = bestPoints[i3];
+ if (!geoVecEqual(originalPoints[i3].coord, point2.coord)) {
+ node = graph.entity(point2.id);
+ loc = projection2.invert(point2.coord);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc, t2)));
+ }
+ }
+ for (i3 = 0; i3 < straights.length; i3++) {
+ point2 = straights[i3];
+ if (nodeCount[point2.id] > 1)
+ continue;
+ node = graph.entity(point2.id);
+ if (t2 === 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, t2)));
+ }
+ }
+ }
}
- const params = { schema: d2.schema, id: d2.id };
- if (d2.newStatus) {
- params.st = d2.newStatus;
+ return graph;
+ function clonePoints(array2) {
+ return array2.map(function(p2) {
+ return { id: p2.id, coord: [p2.coord[0], p2.coord[1]] };
+ });
}
- if (d2.newComment !== void 0) {
- params.co = d2.newComment;
+ function calcMotion(point3, i4, array2) {
+ if (!isClosed && (i4 === 0 || i4 === array2.length - 1))
+ return [0, 0];
+ if (nodeCount[array2[i4].id] > 1)
+ return [0, 0];
+ var a3 = array2[(i4 - 1 + array2.length) % array2.length].coord;
+ var origin = point3.coord;
+ var b3 = array2[(i4 + 1) % array2.length].coord;
+ var p2 = geoVecSubtract(a3, origin);
+ var q2 = geoVecSubtract(b3, origin);
+ var scale = 2 * Math.min(geoVecLength(p2), geoVecLength(q2));
+ p2 = geoVecNormalize(p2);
+ q2 = geoVecNormalize(q2);
+ var dotp2 = p2[0] * q2[0] + p2[1] * q2[1];
+ var val = Math.abs(dotp2);
+ if (val < lowerThreshold) {
+ corner.i = i4;
+ corner.dotp = val;
+ var vec = geoVecNormalize(geoVecAdd(p2, q2));
+ return geoVecScale(vec, 0.1 * dotp2 * scale);
+ }
+ return [0, 0];
}
- const url = `${_krUrlRoot}/comment.php?` + utilQsString(params);
- const controller = new AbortController();
- _cache.inflightPost[d2.id] = controller;
- json_default(url, { signal: controller.signal }).finally(() => {
- delete _cache.inflightPost[d2.id];
- if (d2.newStatus === "ignore") {
- this.removeItem(d2);
- } else if (d2.newStatus === "ignore_t") {
- this.removeItem(d2);
- _cache.closed[`${d2.schema}:${d2.id}`] = true;
- } else {
- d2 = this.replaceItem(d2.update({
- comment: d2.newComment,
- newComment: void 0,
- newState: void 0
- }));
+ };
+ function nodeSubset(nodes, vertexID2, isClosed) {
+ var first = isClosed ? 0 : 1;
+ var last = isClosed ? nodes.length : nodes.length - 1;
+ for (var i3 = first; i3 < last; i3++) {
+ if (nodes[i3].id === vertexID2) {
+ return [
+ nodes[(i3 - 1 + nodes.length) % nodes.length],
+ nodes[i3],
+ nodes[(i3 + 1) % nodes.length]
+ ];
}
- if (callback)
- callback(null, d2);
- });
- },
- // 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((d2) => d2.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();
+ }
+ 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(n3) {
+ return projection2(n3.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/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(i3) {
- Object.values(i3).forEach((controller) => {
- if (controller) {
- controller.abort();
+ // 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
+ }));
+ };
}
- function abortUnwantedRequests2(cache, tiles) {
- Object.keys(cache.inflightTile).forEach((k2) => {
- const wanted = tiles.find((tile) => k2 === tile.id);
- if (!wanted) {
- abortRequest2(cache.inflightTile[k2]);
- delete cache.inflightTile[k2];
+
+ // 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);
+ }
+ });
}
- });
- }
- function encodeIssueRtree2(d2) {
- return { minX: d2.loc[0], minY: d2.loc[1], maxX: d2.loc[0], maxY: d2.loc[1], data: d2 };
- }
- function updateRtree2(item, replace) {
- _cache2.rtree.remove(item, (a2, b2) => a2.data.id === b2.data.id);
- if (replace) {
- _cache2.rtree.insert(item);
- }
+ return graph.revert(id2);
+ };
+ return action;
}
- function linkErrorObject(d2) {
- return { html: `${d2}` };
+
+ // 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;
}
- function linkEntity(d2) {
- return { html: `${d2}` };
+
+ // 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)));
+ });
+ });
+ };
}
- 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];
+
+ // modules/actions/straighten_nodes.js
+ function actionStraightenNodes(nodeIDs, projection2) {
+ function positionAlongWay(a2, o2, b2) {
+ return geoVecDot(a2, b2, o2) / geoVecDot(b2, b2, o2);
}
- }
- function relativeBearing(p1, p2) {
- let angle2 = Math.atan2(p2.lon - p1.lon, p2.lat - p1.lat);
- if (angle2 < 0) {
- angle2 += 2 * Math.PI;
+ 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];
}
- 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"
+ var action = function(graph, t2) {
+ if (t2 === null || !isFinite(t2))
+ t2 = 1;
+ t2 = Math.min(Math.max(+t2, 0), 1);
+ var nodes = nodeIDs.map(function(id2) {
+ return graph.entity(id2);
+ });
+ var points = nodes.map(function(n3) {
+ return projection2(n3.loc);
+ });
+ var endpoints = getEndpoints(points);
+ var startPoint = endpoints[0];
+ var endPoint = endpoints[1];
+ for (var i3 = 0; i3 < points.length; i3++) {
+ var node = nodes[i3];
+ var point2 = points[i3];
+ var u2 = positionAlongWay(point2, startPoint, endPoint);
+ var point22 = geoVecInterp(startPoint, endPoint, u2);
+ var loc2 = projection2.invert(point22);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t2)));
+ }
+ return graph;
};
- 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((d2) => _impOsmData = d2.improveOSM);
- if (!_cache2) {
- this.reset();
+ action.disabled = function(graph) {
+ var nodes = nodeIDs.map(function(id2) {
+ return graph.entity(id2);
+ });
+ var points = nodes.map(function(n3) {
+ return projection2(n3.loc);
+ });
+ var endpoints = getEndpoints(points);
+ var startPoint = endpoints[0];
+ var endPoint = endpoints[1];
+ var maxDistance = 0;
+ for (var i3 = 0; i3 < points.length; i3++) {
+ var point2 = points[i3];
+ var u2 = positionAlongWay(point2, startPoint, endPoint);
+ var p2 = geoVecInterp(startPoint, endPoint, u2);
+ var dist = geoVecLength(p2, point2);
+ if (!isNaN(dist) && dist > maxDistance) {
+ maxDistance = dist;
+ }
}
- this.event = utilRebind(this, dispatch3, "on");
- },
- reset() {
- if (_cache2) {
- Object.values(_cache2.inflightTile).forEach(abortRequest2);
+ if (maxDistance < 1e-4) {
+ return "straight_enough";
}
- _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
+ };
+ action.transitionable = true;
+ return action;
+ }
+
+ // modules/actions/straighten_way.js
+ function actionStraightenWay(selectedIDs, projection2) {
+ function positionAlongWay(a2, o2, b2) {
+ return geoVecDot(a2, b2, o2) / geoVecDot(b2, b2, o2);
+ }
+ function allNodes(graph) {
+ var nodes = [];
+ var startNodes = [];
+ var endNodes = [];
+ var remainingWays = [];
+ var selectedWays = selectedIDs.filter(function(w2) {
+ return graph.entity(w2).type === "way";
+ });
+ var selectedNodes = selectedIDs.filter(function(n3) {
+ return graph.entity(n3).type === "node";
+ });
+ for (var i3 = 0; i3 < selectedWays.length; i3++) {
+ var way = graph.entity(selectedWays[i3]);
+ nodes = way.nodes.slice(0);
+ remainingWays.push(nodes);
+ startNodes.push(nodes[0]);
+ endNodes.push(nodes[nodes.length - 1]);
+ }
+ startNodes = startNodes.filter(function(n3) {
+ return startNodes.indexOf(n3) === startNodes.lastIndexOf(n3);
+ });
+ endNodes = endNodes.filter(function(n3) {
+ return endNodes.indexOf(n3) === endNodes.lastIndexOf(n3);
+ });
+ 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];
};
- 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((k2) => {
- const kParams = Object.assign(
- {},
- params,
- k2 === "mr" ? { type: "PARKING,ROAD,BOTH,PATH" } : { confidenceLevel: "C1" }
- );
- const url = `${_impOsmUrls[k2]}/search?` + utilQsString(kParams);
- const controller = new AbortController();
- requests[k2] = controller;
- json_default(url, { signal: controller.signal }).then((data) => {
- delete _cache2.inflightTile[tile.id][k2];
- 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 d2 = new QAItem(loc, this, k2, itemId, {
- issueKey: k2,
- // used as a category
- identifier: {
- // used to post changes
- wayId,
- fromNodeId,
- toNodeId
- },
- objectId: wayId,
- objectType: "way"
- });
- d2.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[d2.id] = d2;
- _cache2.rtree.insert(encodeIssueRtree2(d2));
- });
- }
- if (data.tiles) {
- data.tiles.forEach((feature3) => {
- const { type: type2, x: x2, y: y2, numberOfTrips } = feature3;
- const geoType = type2.toLowerCase();
- const itemId = `${geoType}${x2}${y2}${numberOfTrips}`;
- let loc = pointAverage(feature3.points);
- loc = preventCoincident(loc, false);
- let d2 = new QAItem(loc, this, `${k2}-${geoType}`, itemId, {
- issueKey: k2,
- identifier: { x: x2, y: y2 }
- });
- d2.replacements = {
- num_trips: numberOfTrips,
- geometry_type: _t(`QA.improveOSM.geometry_types.${geoType}`)
- };
- if (numberOfTrips === -1) {
- d2.desc = _t("QA.improveOSM.error_types.mr.description_alt", d2.replacements);
- }
- _cache2.data[d2.id] = d2;
- _cache2.rtree.insert(encodeIssueRtree2(d2));
- });
- }
- 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 d2 = new QAItem(loc, this, k2, itemId, {
- issueKey: k2,
- identifier: id2,
- objectId: via_node,
- objectType: "node"
- });
- const [p1, p2] = segments[0].points;
- const dir_of_travel = cardinalDirection(relativeBearing(p1, p2));
- d2.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[d2.id] = d2;
- _cache2.rtree.insert(encodeIssueRtree2(d2));
- dispatch3.call("loaded");
- });
- }
- }).catch(() => {
- delete _cache2.inflightTile[tile.id][k2];
- if (!Object.keys(_cache2.inflightTile[tile.id]).length) {
- delete _cache2.inflightTile[tile.id];
- _cache2.loadedTile[tile.id] = true;
- }
- });
+ 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(a2, b2) {
+ return a2 - b2;
});
- _cache2.inflightTile[tile.id] = requests;
+ nodes = nodes.slice(sortedStartEnd[0], sortedStartEnd[1] + 1);
+ }
+ return nodes.map(function(n3) {
+ return graph.entity(n3);
});
- },
- getComments(item) {
- if (item.comments) {
- return Promise.resolve(item);
+ }
+ function shouldKeepNode(node, graph) {
+ return graph.parentWays(node).length > 1 || graph.parentRelations(node).length || node.hasInterestingTags();
+ }
+ var action = function(graph, t2) {
+ if (t2 === null || !isFinite(t2))
+ t2 = 1;
+ t2 = Math.min(Math.max(+t2, 0), 1);
+ var nodes = allNodes(graph);
+ var points = nodes.map(function(n3) {
+ return projection2(n3.loc);
+ });
+ var startPoint = points[0];
+ var endPoint = points[points.length - 1];
+ var toDelete = [];
+ var i3;
+ for (i3 = 1; i3 < points.length - 1; i3++) {
+ var node = nodes[i3];
+ var point2 = points[i3];
+ if (t2 < 1 || shouldKeepNode(node, graph)) {
+ var u2 = positionAlongWay(point2, startPoint, endPoint);
+ var p2 = geoVecInterp(startPoint, endPoint, u2);
+ var loc2 = projection2.invert(p2);
+ graph = graph.replace(node.move(geoVecInterp(node.loc, loc2, t2)));
+ } else {
+ if (toDelete.indexOf(node) === -1) {
+ toDelete.push(node);
+ }
+ }
}
- 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;
+ for (i3 = 0; i3 < toDelete.length; i3++) {
+ graph = actionDeleteNode(toDelete[i3].id)(graph);
}
- 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(d2, callback) {
- if (!osm_default.authenticated()) {
- return callback({ message: "Not Authenticated", status: -3 }, d2);
+ return graph;
+ };
+ action.disabled = function(graph) {
+ var nodes = allNodes(graph);
+ var points = nodes.map(function(n3) {
+ return projection2(n3.loc);
+ });
+ var startPoint = points[0];
+ var endPoint = points[points.length - 1];
+ var threshold = 0.2 * geoVecLength(startPoint, endPoint);
+ var i3;
+ if (threshold === 0) {
+ return "too_bendy";
}
- if (_cache2.inflightPost[d2.id]) {
- return callback({ message: "Error update already inflight", status: -2 }, d2);
+ var maxDistance = 0;
+ for (i3 = 1; i3 < points.length - 1; i3++) {
+ var point2 = points[i3];
+ var u2 = positionAlongWay(point2, startPoint, endPoint);
+ var p2 = geoVecInterp(startPoint, endPoint, u2);
+ var dist = geoVecLength(p2, point2);
+ if (isNaN(dist) || dist > threshold) {
+ return "too_bendy";
+ } else if (dist > maxDistance) {
+ maxDistance = dist;
+ }
+ }
+ var keepingAllNodes = nodes.every(function(node, i4) {
+ return i4 === 0 || i4 === 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, t2) {
+ if (t2 === null || !isFinite(t2))
+ t2 = 1;
+ t2 = Math.min(Math.max(+t2, 0), 1);
+ var nodes = utilGetAllNodes(reflectIds, graph);
+ var points = nodes.map(function(n3) {
+ return projection2(n3.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 p3, q3;
+ var isLong = geoVecLength(p1, q1) > geoVecLength(p2, q2);
+ if (_useLongAxis && isLong || !_useLongAxis && !isLong) {
+ p3 = p1;
+ q3 = q1;
+ } else {
+ p3 = p2;
+ q3 = q2;
+ }
+ var dx = q3[0] - p3[0];
+ var dy = q3[1] - p3[1];
+ var a2 = (dx * dx - dy * dy) / (dx * dx + dy * dy);
+ var b2 = 2 * dx * dy / (dx * dx + dy * dy);
+ for (var i3 = 0; i3 < nodes.length; i3++) {
+ var node = nodes[i3];
+ var c2 = projection2(node.loc);
+ var c22 = [
+ a2 * (c2[0] - p3[0]) + b2 * (c2[1] - p3[1]) + p3[0],
+ b2 * (c2[0] - p3[0]) - a2 * (c2[1] - p3[1]) + p3[1]
+ ];
+ var loc2 = projection2.invert(c22);
+ node = node.move(geoVecInterp(node.loc, loc2, t2));
+ graph = graph.replace(node);
}
- osm_default.userDetails(sendPayload.bind(this));
- function sendPayload(err, user) {
- if (err) {
- return callback(err, d2);
- }
- const key = d2.issueKey;
- const url = `${_impOsmUrls[key]}/comment`;
- const payload = {
- username: user.display_name,
- targetIds: [d2.identifier]
- };
- if (d2.newStatus) {
- payload.status = d2.newStatus;
- payload.text = "status changed";
- }
- if (d2.newComment) {
- payload.text = d2.newComment;
+ 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(";");
+ }
}
- const controller = new AbortController();
- _cache2.inflightPost[d2.id] = controller;
- const options2 = {
- method: "POST",
- signal: controller.signal,
- body: JSON.stringify(payload)
- };
- json_default(url, options2).then(() => {
- delete _cache2.inflightPost[d2.id];
- if (!d2.newStatus) {
- const now3 = /* @__PURE__ */ new Date();
- let comments = d2.comments ? d2.comments : [];
- comments.push({
- username: payload.username,
- text: payload.text,
- timestamp: now3.getTime() / 1e3
- });
- this.replaceItem(d2.update({
- comments,
- newComment: void 0
- }));
+ }
+ 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 {
- this.removeItem(d2);
- if (d2.newStatus === "SOLVED") {
- if (!(d2.issueKey in _cache2.closed)) {
- _cache2.closed[d2.issueKey] = 0;
+ 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(";");
}
- _cache2.closed[d2.issueKey] += 1;
+ } else {
+ tags[replaceKey] = replaceValue;
}
}
- if (callback)
- callback(null, d2);
- }).catch((err2) => {
- delete _cache2.inflightPost[d2.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((d2) => d2.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: false,
- headerPrefix: "",
- highlight: null,
- hooks: null,
- langPrefix: "language-",
- mangle: false,
- pedantic: false,
- renderer: null,
- sanitize: false,
- sanitizer: null,
- silent: false,
- smartypants: false,
- tokenizer: null,
- walkTokens: null,
- xhtml: false
+ return graph.replace(entity.update({ tags }));
};
}
- var _defaults = _getDefaults();
- function changeDefaults(newDefaults) {
- _defaults = newDefaults;
+
+ // modules/behavior/edit.js
+ function behaviorEdit(context) {
+ function behavior() {
+ context.map().minzoom(context.minEditableZoom());
+ }
+ behavior.off = function() {
+ context.map().minzoom(0);
+ };
+ return behavior;
}
- var escapeTest = /[&<>"']/;
- var escapeReplace = new RegExp(escapeTest.source, "g");
- var escapeTestNoEncode = /[<>"']|&(?!(#\d{1,7}|#[Xx][a-fA-F0-9]{1,6}|\w+);)/;
- var escapeReplaceNoEncode = new RegExp(escapeTestNoEncode.source, "g");
- var escapeReplacements = {
- "&": "&",
- "<": "<",
- ">": ">",
- '"': """,
- "'": "'"
- };
- var getEscapeReplacement = (ch) => escapeReplacements[ch];
- function escape4(html2, encode) {
- if (encode) {
- if (escapeTest.test(html2)) {
- return html2.replace(escapeReplace, getEscapeReplacement);
+
+ // modules/behavior/hover.js
+ function behaviorHover(context) {
+ var dispatch14 = 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);
+ dispatch14.call("hover", this, null);
}
- } else {
- if (escapeTestNoEncode.test(html2)) {
- return html2.replace(escapeReplaceNoEncode, getEscapeReplacement);
+ }
+ 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);
+ dispatch14.call("hover", this, _targets);
}
}
- return html2;
- }
- var unescapeTest = /&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/ig;
- function unescape3(html2) {
- return html2.replace(unescapeTest, (_2, n3) => {
- n3 = n3.toLowerCase();
- if (n3 === "colon")
- return ":";
- if (n3.charAt(0) === "#") {
- return n3.charAt(1) === "x" ? String.fromCharCode(parseInt(n3.substring(2), 16)) : String.fromCharCode(+n3.substring(1));
+ function behavior(selection2) {
+ _selection = selection2;
+ _targets = [];
+ if (_initialNodeID) {
+ _newNodeId = _initialNodeID;
+ _initialNodeID = null;
+ } else {
+ _newNodeId = null;
}
- return "";
- });
- }
- var caret = /(^|[^\[])\^/g;
- function edit(regex, opt) {
- regex = typeof regex === "string" ? regex : regex.source;
- opt = opt || "";
- const obj = {
- replace: (name, val) => {
- val = typeof val === "object" && "source" in val ? val.source : val;
- val = val.replace(caret, "$1");
- regex = regex.replace(name, val);
- return obj;
- },
- getRegex: () => {
- return new RegExp(regex, opt);
+ _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;
}
- };
- 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 (e3) {
- return null;
+ 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);
+ }
}
- if (prot.indexOf("javascript:") === 0 || prot.indexOf("vbscript:") === 0 || prot.indexOf("data:") === 0) {
- return null;
+ function pointerout(d3_event) {
+ var target = eventTarget(d3_event);
+ var index = _targets.indexOf(target);
+ if (index !== -1) {
+ _targets.splice(index);
+ updateHover(d3_event, _targets);
+ }
}
- }
- if (base && !originIndependentUrl.test(href)) {
- href = resolveUrl(base, href);
- }
- try {
- href = encodeURI(href).replace(/%25/g, "%");
- } catch (e3) {
- 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);
+ function allowsVertex(d2) {
+ return d2.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d2, context.graph());
}
- }
- base = baseUrls[" " + base];
- const relativeBase = base.indexOf(":") === -1;
- if (href.substring(0, 2) === "//") {
- if (relativeBase) {
- return href;
+ 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;
}
- return base.replace(protocol, "$1") + href;
- } else if (href.charAt(0) === "/") {
- if (relativeBase) {
- return href;
+ 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 i3 in targets) {
+ var datum2 = targets[i3];
+ 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);
+ }
+ dispatch14.call("hover", this, !suppressed && targets);
}
- return base.replace(domain, "$1") + href;
- } else {
- return base + href;
}
+ 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, dispatch14, "on");
}
- var noopTest = { exec: () => null };
- 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 "|";
+
+ // modules/behavior/draw.js
+ var _disableSpace = false;
+ var _lastSpace = null;
+ function behaviorDraw(context) {
+ var dispatch14 = 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 {
- return " |";
+ element = d3_event.target;
}
- }), cells = row.split(/ \|/);
- let i3 = 0;
- if (!cells[0].trim()) {
- cells.shift();
+ var d2 = element.__data__;
+ return d2 && d2.properties && d2.properties.target ? d2 : {};
}
- if (cells.length > 0 && !cells[cells.length - 1].trim()) {
- cells.pop();
+ function pointerdown(d3_event) {
+ if (_downPointer)
+ return;
+ var pointerLocGetter = utilFastMouse(this);
+ _downPointer = {
+ id: d3_event.pointerId || "mouse",
+ pointerLocGetter,
+ downTime: +/* @__PURE__ */ new Date(),
+ downLoc: pointerLocGetter(d3_event)
+ };
+ dispatch14.call("down", this, d3_event, datum2(d3_event));
}
- if (count) {
- if (cells.length > count) {
- cells.splice(count);
- } else {
- while (cells.length < count)
- cells.push("");
+ 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 = +/* @__PURE__ */ 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);
}
}
- for (; i3 < cells.length; i3++) {
- cells[i3] = cells[i3].trim().replace(/\\\|/g, "|");
+ 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;
+ dispatch14.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;
+ dispatch14.call("move", this, d3_event, datum2(d3_event));
}
- return cells;
- }
- function rtrim(str2, c2, invert) {
- const l2 = str2.length;
- if (l2 === 0) {
- return "";
+ function pointercancel(d3_event) {
+ if (_downPointer && _downPointer.id === (d3_event.pointerId || "mouse")) {
+ if (!_downPointer.isCancelled) {
+ dispatch14.call("downcancel", this);
+ }
+ _downPointer = null;
+ }
}
- let suffLen = 0;
- while (suffLen < l2) {
- const currChar = str2.charAt(l2 - suffLen - 1);
- if (currChar === c2 && !invert) {
- suffLen++;
- } else if (currChar !== c2 && invert) {
- suffLen++;
- } else {
- break;
+ function mouseenter() {
+ _mouseLeave = false;
+ }
+ function mouseleave() {
+ _mouseLeave = true;
+ }
+ function allowsVertex(d2) {
+ return d2.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d2, context.graph());
+ }
+ function click(d3_event, loc) {
+ var d2 = datum2(d3_event);
+ var target = d2 && d2.properties && d2.properties.entity;
+ var mode = context.mode();
+ if (target && target.type === "node" && allowsVertex(target)) {
+ dispatch14.call("clickNode", this, target, d2);
+ 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]];
+ dispatch14.call("clickWay", this, choice.loc, edge, d2);
+ return;
+ }
+ } else if (mode.id !== "add-point" || mode.preset.matchGeometry("point")) {
+ var locLatLng = context.projection.invert(loc);
+ dispatch14.call("click", this, locLatLng, d2);
}
}
- return str2.slice(0, l2 - suffLen);
- }
- function findClosingBracket(str2, b2) {
- if (str2.indexOf(b2[1]) === -1) {
- return -1;
- }
- const l2 = str2.length;
- let level = 0, i3 = 0;
- for (; i3 < l2; i3++) {
- if (str2[i3] === "\\") {
- i3++;
- } else if (str2[i3] === b2[0]) {
- level++;
- } else if (str2[i3] === b2[1]) {
- level--;
- if (level < 0) {
- return i3;
+ 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);
}
- return -1;
- }
- function checkDeprecations(opt, callback) {
- if (!opt || opt.silent) {
- return;
- }
- if (callback) {
- console.warn("marked(): callback is deprecated since version 5.0.0, should not be used and will be removed in the future. Read more here: https://marked.js.org/using_pro#async");
- }
- if (opt.sanitize || opt.sanitizer) {
- 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");
- }
- if (opt.highlight || opt.langPrefix !== "language-") {
- console.warn("marked(): highlight and langPrefix parameters are deprecated since version 5.0.0, should not be used and will be removed in the future. Instead use https://www.npmjs.com/package/marked-highlight.");
- }
- if (opt.mangle) {
- console.warn("marked(): mangle parameter is enabled by default, but is deprecated since version 5.0.0, and will be removed in the future. To clear this warning, install https://www.npmjs.com/package/marked-mangle, or disable by setting `{mangle: false}`.");
- }
- if (opt.baseUrl) {
- console.warn("marked(): baseUrl parameter is deprecated since version 5.0.0, should not be used and will be removed in the future. Instead use https://www.npmjs.com/package/marked-base-url.");
+ function backspace(d3_event) {
+ d3_event.preventDefault();
+ dispatch14.call("undo");
}
- if (opt.smartypants) {
- console.warn("marked(): smartypants parameter is deprecated since version 5.0.0, should not be used and will be removed in the future. Instead use https://www.npmjs.com/package/marked-smartypants.");
+ function del(d3_event) {
+ d3_event.preventDefault();
+ dispatch14.call("cancel");
}
- if (opt.xhtml) {
- console.warn("marked(): xhtml parameter is deprecated since version 5.0.0, should not be used and will be removed in the future. Instead use https://www.npmjs.com/package/marked-xhtml.");
+ function ret(d3_event) {
+ d3_event.preventDefault();
+ dispatch14.call("finish");
}
- if (opt.headerIds || opt.headerPrefix) {
- console.warn("marked(): headerIds and headerPrefix parameters enabled by default, but are deprecated since version 5.0.0, and will be removed in the future. To clear this warning, install https://www.npmjs.com/package/marked-gfm-heading-id, or disable by setting `{headerIds: false}`.");
+ 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, dispatch14, "on");
}
- 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;
+
+ // modules/behavior/breathe.js
+ var import_fast_deep_equal2 = __toESM(require_fast_deep_equal());
+
+ // node_modules/d3-scale/src/init.js
+ function initRange(domain, range3) {
+ switch (arguments.length) {
+ case 0:
+ break;
+ case 1:
+ this.range(domain);
+ break;
+ default:
+ this.range(range3).domain(domain);
+ break;
}
- return {
- type: "image",
- raw,
- href,
- title,
- text: escape4(text2)
+ return this;
+ }
+
+ // node_modules/d3-scale/src/constant.js
+ function constants(x2) {
+ return function() {
+ return x2;
};
}
- 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");
+
+ // node_modules/d3-scale/src/number.js
+ function number2(x2) {
+ return +x2;
}
- var _Tokenizer = class {
- options;
- rules;
- lexer;
- 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
- };
- }
+
+ // node_modules/d3-scale/src/continuous.js
+ var unit = [0, 1];
+ function identity3(x2) {
+ return x2;
+ }
+ function normalize(a2, b2) {
+ return (b2 -= a2 = +a2) ? function(x2) {
+ return (x2 - a2) / b2;
+ } : constants(isNaN(b2) ? NaN : 0.5);
+ }
+ function clamper(a2, b2) {
+ var t2;
+ if (a2 > b2)
+ t2 = a2, a2 = b2, b2 = t2;
+ return function(x2) {
+ return Math.max(a2, Math.min(b2, x2));
+ };
+ }
+ function bimap(domain, range3, interpolate) {
+ var d0 = domain[0], d1 = domain[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(x2) {
+ return r0(d0(x2));
+ };
+ }
+ function polymap(domain, range3, interpolate) {
+ var j2 = Math.min(domain.length, range3.length) - 1, d2 = new Array(j2), r2 = new Array(j2), i3 = -1;
+ if (domain[j2] < domain[0]) {
+ domain = domain.slice().reverse();
+ range3 = range3.slice().reverse();
}
- 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
- };
- }
+ while (++i3 < j2) {
+ d2[i3] = normalize(domain[i3], domain[i3 + 1]);
+ r2[i3] = interpolate(range3[i3], range3[i3 + 1]);
}
- 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)
- };
- }
+ return function(x2) {
+ var i4 = bisect_default(domain, x2, 1, j2) - 1;
+ return r2[i4](d2[i4](x2));
+ };
+ }
+ function copy(source, target) {
+ return target.domain(source.domain()).range(source.range()).interpolate(source.interpolate()).clamp(source.clamp()).unknown(source.unknown());
+ }
+ function transformer2() {
+ var domain = unit, range3 = unit, interpolate = value_default, transform2, untransform, unknown, clamp3 = identity3, piecewise, output, input;
+ function rescale() {
+ var n3 = Math.min(domain.length, range3.length);
+ if (clamp3 !== identity3)
+ clamp3 = clamper(domain[0], domain[n3 - 1]);
+ piecewise = n3 > 2 ? polymap : bimap;
+ output = input = null;
+ return scale;
}
- hr(src) {
- const cap = this.rules.block.hr.exec(src);
- if (cap) {
- return {
- type: "hr",
- raw: cap[0]
- };
- }
+ function scale(x2) {
+ return x2 == null || isNaN(x2 = +x2) ? unknown : (output || (output = piecewise(domain.map(transform2), range3, interpolate)))(transform2(clamp3(x2)));
}
- blockquote(src) {
- const cap = this.rules.block.blockquote.exec(src);
- if (cap) {
- const text2 = cap[0].replace(/^ *>[ \t]?/gm, "");
- const top = this.lexer.state.top;
- this.lexer.state.top = true;
- const tokens = this.lexer.blockTokens(text2);
- this.lexer.state.top = top;
- return {
- type: "blockquote",
- raw: cap[0],
- tokens,
- text: text2
- };
+ scale.invert = function(y2) {
+ return clamp3(untransform((input || (input = piecewise(range3, domain.map(transform2), number_default)))(y2)));
+ };
+ scale.domain = function(_2) {
+ return arguments.length ? (domain = Array.from(_2, number2), rescale()) : domain.slice();
+ };
+ scale.range = function(_2) {
+ return arguments.length ? (range3 = Array.from(_2), rescale()) : range3.slice();
+ };
+ scale.rangeRound = function(_2) {
+ return range3 = Array.from(_2), interpolate = round_default, rescale();
+ };
+ scale.clamp = function(_2) {
+ return arguments.length ? (clamp3 = _2 ? true : identity3, rescale()) : clamp3 !== identity3;
+ };
+ scale.interpolate = function(_2) {
+ return arguments.length ? (interpolate = _2, rescale()) : interpolate;
+ };
+ scale.unknown = function(_2) {
+ return arguments.length ? (unknown = _2, scale) : unknown;
+ };
+ return function(t2, u2) {
+ transform2 = t2, untransform = u2;
+ return rescale();
+ };
+ }
+ function continuous() {
+ return transformer2()(identity3, identity3);
+ }
+
+ // node_modules/d3-format/src/formatDecimal.js
+ function formatDecimal_default(x2) {
+ return Math.abs(x2 = Math.round(x2)) >= 1e21 ? x2.toLocaleString("en").replace(/,/g, "") : x2.toString(10);
+ }
+ function formatDecimalParts(x2, p2) {
+ if ((i3 = (x2 = p2 ? x2.toExponential(p2 - 1) : x2.toExponential()).indexOf("e")) < 0)
+ return null;
+ var i3, coefficient = x2.slice(0, i3);
+ return [
+ coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,
+ +x2.slice(i3 + 1)
+ ];
+ }
+
+ // node_modules/d3-format/src/exponent.js
+ function exponent_default(x2) {
+ return x2 = formatDecimalParts(Math.abs(x2)), x2 ? x2[1] : NaN;
+ }
+
+ // node_modules/d3-format/src/formatGroup.js
+ function formatGroup_default(grouping, thousands) {
+ return function(value, width) {
+ var i3 = value.length, t2 = [], j2 = 0, g3 = grouping[0], length2 = 0;
+ while (i3 > 0 && g3 > 0) {
+ if (length2 + g3 + 1 > width)
+ g3 = Math.max(1, width - length2);
+ t2.push(value.substring(i3 -= g3, i3 + g3));
+ if ((length2 += g3 + 1) > width)
+ break;
+ g3 = grouping[j2 = (j2 + 1) % grouping.length];
}
- }
- list(src) {
- let cap = this.rules.block.list.exec(src);
- if (cap) {
- let raw, istask, ischecked, indent2, i3, 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))) {
+ return t2.reverse().join(thousands);
+ };
+ }
+
+ // node_modules/d3-format/src/formatNumerals.js
+ function formatNumerals_default(numerals) {
+ return function(value) {
+ return value.replace(/[0-9]/g, function(i3) {
+ return numerals[+i3];
+ });
+ };
+ }
+
+ // 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(s2) {
+ out:
+ for (var n3 = s2.length, i3 = 1, i0 = -1, i1; i3 < n3; ++i3) {
+ switch (s2[i3]) {
+ case ".":
+ i0 = i1 = i3;
break;
- }
- if (this.rules.block.hr.test(src)) {
+ case "0":
+ if (i0 === 0)
+ i0 = i3;
+ i1 = i3;
break;
- }
- raw = cap[0];
- src = src.substring(raw.length);
- line = cap[2].split("\n", 1)[0].replace(/^\t+/, (t2) => " ".repeat(3 * t2.length));
- 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];
- nextLine = rawLine;
- if (this.options.pedantic) {
- nextLine = nextLine.replace(/^ {1,4}(?=( {4})*[^ ])/g, " ");
- }
- if (fencesBeginRegex.test(nextLine)) {
- break;
- }
- if (headingBeginRegex.test(nextLine)) {
- break;
- }
- if (nextBulletRegex.test(nextLine)) {
- break;
- }
- if (hrRegex.test(src)) {
+ default:
+ if (!+s2[i3])
+ break out;
+ if (i0 > 0)
+ i0 = 0;
+ break;
+ }
+ }
+ return i0 > 0 ? s2.slice(0, i0) + s2.slice(i1 + 1) : s2;
+ }
+
+ // node_modules/d3-format/src/formatPrefixAuto.js
+ var prefixExponent;
+ function formatPrefixAuto_default(x2, p2) {
+ var d2 = formatDecimalParts(x2, p2);
+ if (!d2)
+ return x2 + "";
+ var coefficient = d2[0], exponent = d2[1], i3 = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1, n3 = coefficient.length;
+ return i3 === n3 ? coefficient : i3 > n3 ? coefficient + new Array(i3 - n3 + 1).join("0") : i3 > 0 ? coefficient.slice(0, i3) + "." + coefficient.slice(i3) : "0." + new Array(1 - i3).join("0") + formatDecimalParts(x2, Math.max(0, p2 + i3 - 1))[0];
+ }
+
+ // node_modules/d3-format/src/formatRounded.js
+ function formatRounded_default(x2, p2) {
+ var d2 = formatDecimalParts(x2, p2);
+ if (!d2)
+ return x2 + "";
+ var coefficient = d2[0], exponent = d2[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 = {
+ "%": (x2, p2) => (x2 * 100).toFixed(p2),
+ "b": (x2) => Math.round(x2).toString(2),
+ "c": (x2) => x2 + "",
+ "d": formatDecimal_default,
+ "e": (x2, p2) => x2.toExponential(p2),
+ "f": (x2, p2) => x2.toFixed(p2),
+ "g": (x2, p2) => x2.toPrecision(p2),
+ "o": (x2) => Math.round(x2).toString(8),
+ "p": (x2, p2) => formatRounded_default(x2 * 100, p2),
+ "r": formatRounded_default,
+ "s": formatPrefixAuto_default,
+ "X": (x2) => Math.round(x2).toString(16).toUpperCase(),
+ "x": (x2) => Math.round(x2).toString(16)
+ };
+
+ // node_modules/d3-format/src/identity.js
+ function identity_default4(x2) {
+ return x2;
+ }
+
+ // 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_default4 : 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_default4 : 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, precision3 = specifier.precision, trim = specifier.trim, type2 = specifier.type;
+ if (type2 === "n")
+ comma = true, type2 = "g";
+ else if (!formatTypes_default[type2])
+ precision3 === void 0 && (precision3 = 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);
+ precision3 = precision3 === void 0 ? 6 : /[gprs]/.test(type2) ? Math.max(1, Math.min(21, precision3)) : Math.max(0, Math.min(20, precision3));
+ function format2(value) {
+ var valuePrefix = prefix, valueSuffix = suffix, i3, n3, c2;
+ 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), precision3);
+ 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) {
+ i3 = -1, n3 = value.length;
+ while (++i3 < n3) {
+ if (c2 = value.charCodeAt(i3), 48 > c2 || c2 > 57) {
+ valueSuffix = (c2 === 46 ? decimal + value.slice(i3 + 1) : value.slice(i3)) + valueSuffix;
+ value = value.slice(0, i3);
break;
- }
- if (nextLine.search(/[^ ]/) >= indent2 || !nextLine.trim()) {
- itemContents += "\n" + nextLine.slice(indent2);
- } else {
- if (blankLine) {
- break;
- }
- if (line.search(/[^ ]/) >= 4) {
- break;
- }
- if (fencesBeginRegex.test(line)) {
- break;
- }
- if (headingBeginRegex.test(line)) {
- break;
- }
- if (hrRegex.test(line)) {
- break;
- }
- itemContents += "\n" + nextLine;
- }
- if (!blankLine && !nextLine.trim()) {
- blankLine = true;
- }
- raw += rawLine + "\n";
- src = src.substring(rawLine.length + 1);
- line = nextLine.slice(indent2);
- }
- }
- 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 l2 = list.items.length;
- for (i3 = 0; i3 < l2; i3++) {
- this.lexer.state.top = false;
- list.items[i3].tokens = this.lexer.blockTokens(list.items[i3].text, []);
- if (!list.loose) {
- const spacers = list.items[i3].tokens.filter((t2) => t2.type === "space");
- const hasMultipleLineBreaks = spacers.length > 0 && spacers.some((t2) => /\n.*\n/.test(t2.raw));
- list.loose = hasMultipleLineBreaks;
- }
- }
- if (list.loose) {
- for (i3 = 0; i3 < l2; i3++) {
- list.items[i3].loose = true;
- }
- }
- return list;
- }
- }
- html(src) {
- const cap = this.rules.block.html.exec(src);
- if (cap) {
- const token = {
- type: "html",
- block: true,
- 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]);
- const paragraph = token;
- paragraph.type = "paragraph";
- paragraph.text = text2;
- paragraph.tokens = this.lexer.inline(text2);
}
- return token;
- }
- }
- def(src) {
- const cap = this.rules.block.def.exec(src);
- if (cap) {
- const tag = cap[1].toLowerCase().replace(/\s+/g, " ");
- const href = cap[2] ? cap[2].replace(/^<(.*)>$/, "$1").replace(this.rules.inline._escapes, "$1") : "";
- const title = cap[3] ? cap[3].substring(1, cap[3].length - 1).replace(this.rules.inline._escapes, "$1") : cap[3];
- return {
- type: "def",
- tag,
- raw: cap[0],
- href,
- title
- };
- }
- }
- table(src) {
- const cap = this.rules.block.table.exec(src);
- if (cap) {
- const item = {
- type: "table",
- raw: cap[0],
- header: splitCells(cap[1]).map((c2) => {
- return { text: c2 };
- }),
- 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) {
- let l2 = item.align.length;
- let i3, j3, k2, row;
- for (i3 = 0; i3 < l2; i3++) {
- if (/^ *-+: *$/.test(item.align[i3])) {
- item.align[i3] = "right";
- } else if (/^ *:-+: *$/.test(item.align[i3])) {
- item.align[i3] = "center";
- } else if (/^ *:-+ *$/.test(item.align[i3])) {
- item.align[i3] = "left";
- } else {
- item.align[i3] = null;
- }
- }
- l2 = item.rows.length;
- for (i3 = 0; i3 < l2; i3++) {
- item.rows[i3] = splitCells(item.rows[i3], item.header.length).map((c2) => {
- return { text: c2 };
- });
- }
- l2 = item.header.length;
- for (j3 = 0; j3 < l2; j3++) {
- item.header[j3].tokens = this.lexer.inline(item.header[j3].text);
- }
- l2 = item.rows.length;
- for (j3 = 0; j3 < l2; j3++) {
- row = item.rows[j3];
- for (k2 = 0; k2 < row.length; k2++) {
- row[k2].tokens = this.lexer.inline(row[k2].text);
- }
- }
- return item;
+ if (comma && !zero3)
+ value = group(value, Infinity);
+ var length2 = valuePrefix.length + value.length + valueSuffix.length, padding = length2 < width ? new Array(width - length2 + 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, length2 = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length2);
+ break;
+ default:
+ value = padding + valuePrefix + value + valueSuffix;
+ break;
}
+ return numerals(value);
}
+ format2.toString = function() {
+ return specifier + "";
+ };
+ return format2;
}
- 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])
- };
- }
+ function formatPrefix2(specifier, value) {
+ var f2 = newFormat((specifier = formatSpecifier(specifier), specifier.type = "f", specifier)), e3 = Math.max(-8, Math.min(8, Math.floor(exponent_default(value) / 3))) * 3, k2 = Math.pow(10, -e3), prefix = prefixes[8 + e3 / 3];
+ return function(value2) {
+ return f2(k2 * value2) + prefix;
+ };
}
- 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)
- };
+ 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), precision3;
+ 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(precision3 = precisionPrefix_default(step, value)))
+ specifier.precision = precision3;
+ return formatPrefix(specifier, value);
}
- }
- 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])
- };
+ case "":
+ case "e":
+ case "g":
+ case "p":
+ case "r": {
+ if (specifier.precision == null && !isNaN(precision3 = precisionRound_default(step, Math.max(Math.abs(start2), Math.abs(stop)))))
+ specifier.precision = precision3 - (specifier.type === "e");
+ break;
}
- }
- escape(src) {
- const cap = this.rules.inline.escape.exec(src);
- if (cap) {
- return {
- type: "escape",
- raw: cap[0],
- text: escape4(cap[1])
- };
+ case "f":
+ case "%": {
+ if (specifier.precision == null && !isNaN(precision3 = precisionFixed_default(step)))
+ specifier.precision = precision3 - (specifier.type === "%") * 2;
+ break;
}
}
- tag(src) {
- const cap = this.rules.inline.tag.exec(src);
- if (cap) {
- if (!this.lexer.state.inLink && /^/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,
- block: false,
- text: this.options.sanitize ? this.options.sanitizer ? this.options.sanitizer(cap[0]) : escape4(cap[0]) : cap[0]
- };
+ return format(specifier);
+ }
+
+ // node_modules/d3-scale/src/linear.js
+ function linearish(scale) {
+ var domain = scale.domain;
+ scale.ticks = function(count) {
+ var d2 = domain();
+ return ticks(d2[0], d2[d2.length - 1], count == null ? 10 : count);
+ };
+ scale.tickFormat = function(count, specifier) {
+ var d2 = domain();
+ return tickFormat(d2[0], d2[d2.length - 1], count == null ? 10 : count, specifier);
+ };
+ scale.nice = function(count) {
+ if (count == null)
+ count = 10;
+ var d2 = domain();
+ var i0 = 0;
+ var i1 = d2.length - 1;
+ var start2 = d2[i0];
+ var stop = d2[i1];
+ var prestep;
+ var step;
+ var maxIter = 10;
+ if (stop < start2) {
+ step = start2, start2 = stop, stop = step;
+ step = i0, i0 = i1, i1 = step;
}
- }
- link(src) {
- const cap = this.rules.inline.link.exec(src);
- if (cap) {
- const trimmedUrl = cap[2].trim();
- if (!this.options.pedantic && /^$/.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];
- }
+ while (maxIter-- > 0) {
+ step = tickIncrement(start2, stop, count);
+ if (step === prestep) {
+ d2[i0] = start2;
+ d2[i1] = stop;
+ return domain(d2);
+ } 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 {
- title = cap[3] ? cap[3].slice(1, -1) : "";
- }
- href = href.trim();
- if (/^$/.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) {
- const text2 = cap[0].charAt(0);
- return {
- type: "text",
- raw: text2,
- text: text2
- };
+ break;
}
- return outputLink(cap, link2, cap[0], this.lexer);
+ 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, n3 = 1, domain = [0.5], range3 = [0, 1], unknown;
+ function scale(x2) {
+ return x2 != null && x2 <= x2 ? range3[bisect_default(domain, x2, 0, n3)] : unknown;
}
- 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 || !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 + rLength + 1).join("");
- 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)
- };
- }
- }
+ function rescale() {
+ var i3 = -1;
+ domain = new Array(n3);
+ while (++i3 < n3)
+ domain[i3] = ((i3 + 1) * x12 - (i3 - n3) * x05) / (n3 + 1);
+ return scale;
}
- 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
- };
- }
+ scale.domain = function(_2) {
+ return arguments.length ? ([x05, x12] = _2, x05 = +x05, x12 = +x12, rescale()) : [x05, x12];
+ };
+ scale.range = function(_2) {
+ return arguments.length ? (n3 = (range3 = Array.from(_2)).length - 1, rescale()) : range3.slice();
+ };
+ scale.invertExtent = function(y2) {
+ var i3 = range3.indexOf(y2);
+ return i3 < 0 ? [NaN, NaN] : i3 < 1 ? [x05, domain[0]] : i3 >= n3 ? [domain[n3 - 1], x12] : [domain[i3 - 1], domain[i3]];
+ };
+ scale.unknown = function(_2) {
+ return arguments.length ? (unknown = _2, scale) : scale;
+ };
+ scale.thresholds = function() {
+ return domain.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(a2, b2, steps2, units) {
+ a2 = Number(a2);
+ b2 = Number(b2);
+ var sample = quantize().domain([0, 1]).range(quantize_default(number_default(a2, b2), steps2));
+ return function(t2) {
+ return String(sample(t2)) + (units || "");
+ };
}
- br(src) {
- const cap = this.rules.inline.br.exec(src);
- if (cap) {
- return {
- type: "br",
- raw: cap[0]
- };
- }
+ function reset(selection2) {
+ selection2.style("stroke-opacity", null).style("stroke-width", null).style("fill-opacity", null).style("r", null);
}
- 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])
- };
- }
+ function setAnimationParams(transition2, fromTo) {
+ var toFrom = fromTo === "from" ? "to" : "from";
+ transition2.styleTween("stroke-opacity", function(d2) {
+ return ratchetyInterpolator(
+ _params[d2.id][toFrom].opacity,
+ _params[d2.id][fromTo].opacity,
+ steps
+ );
+ }).styleTween("stroke-width", function(d2) {
+ return ratchetyInterpolator(
+ _params[d2.id][toFrom].width,
+ _params[d2.id][fromTo].width,
+ steps,
+ "px"
+ );
+ }).styleTween("fill-opacity", function(d2) {
+ return ratchetyInterpolator(
+ _params[d2.id][toFrom].opacity,
+ _params[d2.id][fromTo].opacity,
+ steps
+ );
+ }).styleTween("r", function(d2) {
+ return ratchetyInterpolator(
+ _params[d2.id][toFrom].width,
+ _params[d2.id][fromTo].width,
+ steps,
+ "px"
+ );
+ });
}
- 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;
+ function calcAnimationParams(selection2) {
+ selection2.call(reset).each(function(d2) {
+ var s2 = select_default2(this);
+ var tag2 = s2.node().tagName;
+ var p2 = { "from": {}, "to": {} };
+ var opacity;
+ var width;
+ if (tag2 === "circle") {
+ opacity = Number(s2.style("fill-opacity") || 0.5);
+ width = Number(s2.style("r") || 15.5);
} else {
- text2 = escape4(cap[1]);
- href = text2;
+ opacity = Number(s2.style("stroke-opacity") || 0.7);
+ width = Number(s2.style("stroke-width") || 10);
}
- return {
- type: "link",
- raw: cap[0],
- text: text2,
- href,
- tokens: [
- {
- type: "text",
- raw: text2,
- text: text2
- }
- ]
- };
+ p2.tag = tag2;
+ p2.from.opacity = opacity * 0.6;
+ p2.to.opacity = opacity * 1.25;
+ p2.from.width = width * 0.7;
+ p2.to.width = width * (tag2 === "circle" ? 1.5 : 1);
+ _params[d2.id] = p2;
+ });
+ }
+ 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));
+ }
+ });
}
- 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://" + cap[0];
- } else {
- href = cap[0];
- }
+ function behavior(surface) {
+ _done = false;
+ _timer = timer(function() {
+ if (surface.selectAll(selector).empty()) {
+ return false;
}
- return {
- type: "link",
- raw: cap[0],
- text: text2,
- href,
- tokens: [
- {
- type: "text",
- raw: text2,
- text: text2
- }
- ]
- };
- }
+ surface.call(run, "from");
+ _timer.stop();
+ return true;
+ }, 20);
}
- 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]);
+ behavior.restartIfNeeded = function(surface) {
+ if (_selected.empty()) {
+ surface.call(run, "from");
+ if (_timer) {
+ _timer.stop();
}
- 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*|$)|\\n*|$)|\\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][^\s]*|<.*?>)(?:(?: +(?:\n *)?| *\n *)(title))? *(?:\n+|$)/,
- table: noopTest,
- lheading: /^((?:(?!^bull ).|\n(?!\n|bull ))+?)\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 = /|$)/;
- 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.lheading = edit(block.lheading).replace(/bull/g, block.bullet).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 = { ...block };
- block.gfm = {
- ...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 = {
- ...block.normal,
- html: edit(`^ *(?:comment *(?:\\n|\\s*$)|<(tag)[\\s\\S]+?\\1> *(?:\\n{2,}|\\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,
- lheading: /^(.+?)\n {0,3}(=+|-+) *(?:\n+|$)/,
- 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]*?\\?>|^|^",
- 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]*?(?:(?=[\\`^|~";
- inline.punctuation = edit(inline.punctuation, "u").replace(/punctuation/g, inline._punctuation).getRegex();
- inline.blockSkip = /\[[^[\]]*?\]\([^\(\)]*?\)|`[^`]*?`|<[^<>]*?>/g;
- inline.anyPunctuation = /\\[punct]/g;
- inline._escapes = /\\([punct])/g;
- inline._comment = edit(block._comment).replace("(?:-->|$)", "-->").getRegex();
- inline.emStrong.lDelim = edit(inline.emStrong.lDelim, "u").replace(/punct/g, inline._punctuation).getRegex();
- inline.emStrong.rDelimAst = edit(inline.emStrong.rDelimAst, "gu").replace(/punct/g, inline._punctuation).getRegex();
- inline.emStrong.rDelimUnd = edit(inline.emStrong.rDelimUnd, "gu").replace(/punct/g, inline._punctuation).getRegex();
- inline.anyPunctuation = edit(inline.anyPunctuation, "gu").replace(/punct/g, inline._punctuation).getRegex();
- inline._escapes = edit(inline._escapes, "gu").replace(/punct/g, inline._punctuation).getRegex();
- 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 = { ...inline };
- inline.pedantic = {
- ...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 = {
- ...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]*?(?:(?=[\\ 0.5) {
- ch = "x" + ch.toString(16);
+ };
+ behavior.off = function() {
+ _done = true;
+ if (_timer) {
+ _timer.stop();
}
- out += "" + ch + ";";
- }
- return out;
+ _selected.interrupt().call(reset);
+ };
+ return behavior;
}
- var _Lexer = class __Lexer {
- tokens;
- options;
- state;
- tokenizer;
- inlineQueue;
- 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;
- }
+
+ // 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();
}
- 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);
+ function behavior() {
+ if (_operation && _operation.available()) {
+ context.keybinding().on(_operation.keys, keypress);
}
- return this.tokens;
+ return behavior;
}
- blockTokens(src, tokens = []) {
- if (this.options.pedantic) {
- src = src.replace(/\t/g, " ").replace(/^ +$/gm, "");
+ behavior.off = function() {
+ context.keybinding().off(_operation.keys);
+ };
+ behavior.which = function(_2) {
+ if (!arguments.length)
+ return _operation;
+ _operation = _2;
+ 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(n3) {
+ return n3.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 {
- src = src.replace(/^( *)(\t+)/gm, (_2, leading, tabs) => {
- return leading + " ".repeat(tabs.length);
- });
+ _extent = _extent.extend(entity.extent(context.graph()));
}
- let token;
- let lastToken;
- let cutSrc;
- let 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 actionCircularize(entityID, context.projection);
+ }
+ var operation2 = function() {
+ if (!_actions.length)
+ return;
+ var combinedAction = function(graph, t2) {
+ _actions.forEach(function(action) {
+ if (!action.disabled(graph)) {
+ graph = action(graph, t2);
}
- return false;
- })) {
- continue;
+ });
+ return graph;
+ };
+ combinedAction.transitionable = true;
+ context.perform(combinedAction, operation2.annotation());
+ window.setTimeout(function() {
+ context.validator().validate();
+ }, 300);
+ };
+ operation2.available = function() {
+ return _actions.length && selectedIDs.length === _actions.length;
+ };
+ operation2.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";
}
- 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);
+ 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;
}
- 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;
+ return false;
+ }
+ };
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
+ return disable ? _t.append("operations.circularize." + disable + "." + _amount) : _t.append("operations.circularize.description." + _amount);
+ };
+ operation2.annotation = function() {
+ return _t("operations.circularize.annotation.feature", { n: _actions.length });
+ };
+ operation2.id = "circularize";
+ operation2.keys = [_t("operations.circularize.key")];
+ operation2.title = _t.append("operations.circularize.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
+ }
+
+ // 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 i3 = 0; i3 < code.length; i3++) {
+ if (code[i3] in replacements) {
+ result += replacements[code[i3]] + (i3 < code.length - 1 ? "+" : "");
+ } else {
+ result += code[i3];
+ }
+ }
+ 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(n3) {
+ return n3.loc;
+ });
+ var extent = utilTotalExtent(selectedIDs, context.graph());
+ var operation2 = 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 i3 = nodes2.indexOf(id2);
+ if (i3 === 0) {
+ i3++;
+ } else if (i3 === nodes2.length - 1) {
+ i3--;
} 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
- };
+ var a2 = geoSphericalDistance(entity.loc, context.entity(nodes2[i3 - 1]).loc);
+ var b2 = geoSphericalDistance(entity.loc, context.entity(nodes2[i3 + 1]).loc);
+ i3 = a2 < b2 ? i3 - 1 : i3 + 1;
}
- continue;
- }
- if (token = this.tokenizer.table(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
+ nextSelectedID = nodes2[i3];
+ nextSelectedLoc = context.entity(nextSelectedID).loc;
}
- if (token = this.tokenizer.lheading(src)) {
- src = src.substring(token.raw.length);
- tokens.push(token);
- continue;
+ }
+ context.perform(action, operation2.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));
}
- 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((getStartIndex) => {
- tempStart = getStartIndex.call({ lexer: this }, tempSrc);
- if (typeof tempStart === "number" && tempStart >= 0) {
- startIndex = Math.min(startIndex, tempStart);
- }
+ } else {
+ context.enter(modeBrowse(context));
+ }
+ };
+ operation2.available = function() {
+ return true;
+ };
+ operation2.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 (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);
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
}
- 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);
+ 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 i3 = 0; i3 < parents.length; i3++) {
+ var parent = parents[i3];
+ var type2 = parent.tags.type;
+ var role = parent.memberById(id2).role || "outer";
+ if (type2 === "route" || type2 === "boundary" || type2 === "multipolygon" && role === "outer") {
+ return true;
}
- 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 false;
+ }
+ };
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
+ return disable ? _t.append("operations.delete." + disable + "." + multi) : _t.append("operations.delete.description." + multi);
+ };
+ operation2.annotation = function() {
+ return selectedIDs.length === 1 ? _t("operations.delete.annotation." + context.graph().geometry(selectedIDs[0])) : _t("operations.delete.annotation.feature", { n: selectedIDs.length });
+ };
+ operation2.id = "delete";
+ operation2.keys = [uiCmd("\u2318\u232B"), uiCmd("\u2318\u2326"), uiCmd("\u2326")];
+ operation2.title = _t.append("operations.delete.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
+ }
+
+ // 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(n3) {
+ return n3.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);
}
}
}
- this.state.top = true;
- return tokens;
- }
- inline(src, tokens = []) {
- this.inlineQueue.push({ src, tokens });
- return tokens;
+ return null;
}
- /**
- * 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) + "[" + "a".repeat(match[0].length - 2) + "]" + maskedSrc.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex);
- }
+ var operation2 = function() {
+ if (!_actions.length)
+ return;
+ var combinedAction = function(graph, t2) {
+ _actions.forEach(function(action) {
+ if (!action.disabled(graph)) {
+ graph = action(graph, t2);
}
+ });
+ return graph;
+ };
+ combinedAction.transitionable = true;
+ context.perform(combinedAction, operation2.annotation());
+ window.setTimeout(function() {
+ context.validator().validate();
+ }, 300);
+ };
+ operation2.available = function() {
+ return _actions.length && selectedIDs.length === _actions.length;
+ };
+ operation2.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";
}
- while ((match = this.tokenizer.rules.inline.blockSkip.exec(maskedSrc)) != null) {
- maskedSrc = maskedSrc.slice(0, match.index) + "[" + "a".repeat(match[0].length - 2) + "]" + maskedSrc.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);
- }
- while ((match = this.tokenizer.rules.inline.anyPunctuation.exec(maskedSrc)) != null) {
- maskedSrc = maskedSrc.slice(0, match.index) + "++" + maskedSrc.slice(this.tokenizer.rules.inline.anyPunctuation.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;
+ function someMissing() {
+ if (context.inIntro())
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((getStartIndex) => {
- tempStart = getStartIndex.call({ lexer: this }, tempSrc);
- if (typeof tempStart === "number" && tempStart >= 0) {
- startIndex = Math.min(startIndex, tempStart);
- }
+ var osm = context.connection();
+ if (osm) {
+ var missing = _coords.filter(function(loc) {
+ return !osm.isDataLoaded(loc);
});
- if (startIndex < Infinity && startIndex >= 0) {
- cutSrc = src.substring(0, startIndex + 1);
+ if (missing.length) {
+ missing.forEach(function(loc) {
+ context.loadTileAtLoc(loc);
+ });
+ return true;
}
}
- 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);
+ return false;
+ }
+ };
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
+ return disable ? _t.append("operations.orthogonalize." + disable + "." + _amount) : _t.append("operations.orthogonalize.description." + _type + "." + _amount);
+ };
+ operation2.annotation = function() {
+ return _t("operations.orthogonalize.annotation." + _type, { n: _actions.length });
+ };
+ operation2.id = "orthogonalize";
+ operation2.keys = [_t("operations.orthogonalize.key")];
+ operation2.title = _t.append("operations.orthogonalize.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
+ }
+
+ // 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(n3) {
+ return n3.loc;
+ });
+ var extent = utilTotalExtent(selectedIDs, context.graph());
+ var operation2 = function() {
+ var action = actionReflect(selectedIDs, context.projection).useLongAxis(Boolean(axis === "long"));
+ context.perform(action, operation2.annotation());
+ window.setTimeout(function() {
+ context.validator().validate();
+ }, 300);
+ };
+ operation2.available = function() {
+ return nodes.length >= 3;
+ };
+ operation2.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;
}
- 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 false;
+ }
+ function incompleteRelation(id2) {
+ var entity = context.entity(id2);
+ return entity.type === "relation" && !entity.isComplete(context.graph());
+ }
+ };
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
+ return disable ? _t.append("operations.reflect." + disable + "." + multi) : _t.append("operations.reflect.description." + axis + "." + multi);
+ };
+ operation2.annotation = function() {
+ return _t("operations.reflect.annotation." + axis + ".feature", { n: selectedIDs.length });
+ };
+ operation2.id = "reflect-" + axis;
+ operation2.keys = [_t("operations.reflect.key." + axis)];
+ operation2.title = _t.append("operations.reflect.title." + axis);
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
+ }
+
+ // 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(n3) {
+ return n3.loc;
+ });
+ var extent = utilTotalExtent(selectedIDs, context.graph());
+ var operation2 = function() {
+ context.enter(modeMove(context, selectedIDs));
+ };
+ operation2.available = function() {
+ return selectedIDs.length > 0;
+ };
+ operation2.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;
}
- return tokens;
- }
- };
- var _Renderer = class {
- options;
- 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;
- }
+ function incompleteRelation(id2) {
+ var entity = context.entity(id2);
+ return entity.type === "relation" && !entity.isComplete(context.graph());
}
- code = code.replace(/\n$/, "") + "\n";
- if (!lang) {
- return "" + (escaped ? code : escape4(code, true)) + "
\n";
+ };
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
+ return disable ? _t.append("operations.move." + disable + "." + multi) : _t.append("operations.move.description." + multi);
+ };
+ operation2.annotation = function() {
+ return selectedIDs.length === 1 ? _t("operations.move.annotation." + context.graph().geometry(selectedIDs[0])) : _t("operations.move.annotation.feature", { n: selectedIDs.length });
+ };
+ operation2.id = "move";
+ operation2.keys = [_t("operations.move.key")];
+ operation2.title = _t.append("operations.move.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ operation2.mouseOnly = true;
+ return operation2;
+ }
+
+ // 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;
}
- return '' + (escaped ? code : escape4(code, true)) + "
\n";
- }
- blockquote(quote2) {
- return `
-${quote2}
-`;
- }
- html(html2, block2) {
- return html2;
- }
- heading(text2, level, raw, slugger) {
- if (this.options.headerIds) {
- const id2 = this.options.headerPrefix + slugger.slug(raw);
- return `${text2}
-`;
+ 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(n3) {
+ return projection2(n3.loc);
+ });
+ _pivot = getPivot(points);
+ _prevAngle = void 0;
}
- return `${text2}
-`;
- }
- hr() {
- return this.options.xhtml ? "
\n" : "
\n";
- }
- list(body, ordered, start2) {
- const type2 = ordered ? "ol" : "ul", startatt = ordered && start2 !== 1 ? ' start="' + start2 + '"' : "";
- return "<" + type2 + startatt + ">\n" + body + "" + type2 + ">\n";
- }
- listitem(text2, task, checked) {
- return `${text2}
-`;
- }
- checkbox(checked) {
- return " ";
- }
- paragraph(text2) {
- return `${text2}
-`;
- }
- table(header, body) {
- if (body)
- body = `${body}`;
- return "\n\n" + header + "\n" + body + "
\n";
- }
- tablerow(content) {
- return `
-${content}
-`;
+ 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();
}
- tablecell(content, flags) {
- const type2 = flags.header ? "th" : "td";
- const tag = flags.align ? `<${type2} align="${flags.align}">` : `<${type2}>`;
- return tag + content + `${type2}>
-`;
+ 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;
}
- /**
- * span level renderer
- */
- strong(text2) {
- return `${text2}`;
+ function finish(d3_event) {
+ d3_event.stopPropagation();
+ context.replace(actionNoop(), annotation);
+ context.enter(modeSelect(context, entityIDs));
}
- em(text2) {
- return `${text2}`;
+ function cancel() {
+ if (_prevGraph)
+ context.pop();
+ context.enter(modeSelect(context, entityIDs));
}
- codespan(text2) {
- return `${text2}
`;
+ function undone() {
+ context.enter(modeBrowse(context));
}
- br() {
- return this.options.xhtml ? "
" : "
";
+ 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(n3) {
+ return n3.loc;
+ });
+ var extent = utilTotalExtent(selectedIDs, context.graph());
+ var operation2 = function() {
+ context.enter(modeRotate(context, selectedIDs));
+ };
+ operation2.available = function() {
+ return nodes.length >= 2;
+ };
+ operation2.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());
+ }
+ };
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
+ return disable ? _t.append("operations.rotate." + disable + "." + multi) : _t.append("operations.rotate.description." + multi);
+ };
+ operation2.annotation = function() {
+ return selectedIDs.length === 1 ? _t("operations.rotate.annotation." + context.graph().geometry(selectedIDs[0])) : _t("operations.rotate.annotation.feature", { n: selectedIDs.length });
+ };
+ operation2.id = "rotate";
+ operation2.keys = [_t("operations.rotate.key")];
+ operation2.title = _t.append("operations.rotate.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ operation2.mouseOnly = true;
+ return operation2;
+ }
+
+ // 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 _cache5;
+ var _origin;
+ var _nudgeInterval;
+ var _pointerPrefix = "PointerEvent" in window ? "pointer" : "mouse";
+ function doMove(nudge) {
+ nudge = nudge || [0, 0];
+ var fn;
+ if (_prevGraph !== context.graph()) {
+ _cache5 = {};
+ _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, _cache5));
+ _prevGraph = context.graph();
}
- del(text2) {
- return `${text2}`;
+ function startNudge(nudge) {
+ if (_nudgeInterval)
+ window.clearInterval(_nudgeInterval);
+ _nudgeInterval = window.setInterval(function() {
+ context.map().pan(nudge);
+ doMove(nudge);
+ }, 50);
}
- link(href, title, text2) {
- href = cleanUrl(this.options.sanitize, this.options.baseUrl, href);
- if (href === null) {
- return text2;
- }
- let out = '" + text2 + "";
- return out;
}
- image(href, title, text2) {
- href = cleanUrl(this.options.sanitize, this.options.baseUrl, href);
- if (href === null) {
- return text2;
- }
- let out = `" : ">";
- 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 {
- seen;
- constructor() {
- this.seen = {};
}
- serialize(value) {
- return value.toLowerCase().trim().replace(/<[!\/a-z].*?>/ig, "").replace(/[\u2000-\u206F\u2E00-\u2E7F\\'!"#$%&()*+,./:;<=>?@[\]^`{|}~]/g, "").replace(/\s/g, "-");
+ function finish(d3_event) {
+ d3_event.stopPropagation();
+ context.replace(actionNoop(), annotation);
+ context.enter(modeSelect(context, entityIDs));
+ stopNudge();
}
- /**
- * Finds the next safe (unique) slug to use
- */
- 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;
+ function cancel() {
+ if (baseGraph) {
+ while (context.graph() !== baseGraph)
+ context.pop();
+ context.enter(modeBrowse(context));
+ } else {
+ if (_prevGraph)
+ context.pop();
+ context.enter(modeSelect(context, entityIDs));
}
- return slug;
- }
- /**
- * Convert string to unique id
- */
- slug(value, options2 = {}) {
- const slug = this.serialize(value);
- return this.getNextSafeSlug(slug, options2.dryrun);
+ stopNudge();
}
- };
- var _Parser = class __Parser {
- options;
- renderer;
- textRenderer;
- slugger;
- 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();
+ function undone() {
+ context.enter(modeBrowse(context));
}
- /**
- * Static Parse Method
- */
- static parse(tokens, options2) {
- const parser3 = new __Parser(options2);
- return parser3.parse(tokens);
+ mode.enter = function() {
+ _origin = context.map().mouseCoordinates();
+ _prevGraph = null;
+ _cache5 = {};
+ 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));
}
- /**
- * Static Parse Inline Method
- */
- static parseInline(tokens, options2) {
- const parser3 = new __Parser(options2);
- return parser3.parseInline(tokens);
+ function behavior() {
+ context.keybinding().on(uiCmd("\u2318V"), doPaste);
+ return behavior;
}
- /**
- * Parse Loop
- */
- parse(tokens, top = true) {
- let out = "", i3, j3, k2, l2, l3, row, cell, header, body, token, ordered, start2, loose, itemBody, item, checked, task, checkbox, ret;
- const l4 = tokens.length;
- for (i3 = 0; i3 < l4; i3++) {
- token = tokens[i3];
- 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;
- }
+ behavior.off = function() {
+ context.keybinding().off(uiCmd("\u2318V"));
+ };
+ return behavior;
+ }
+
+ // modules/behavior/drag.js
+ function behaviorDrag() {
+ var dispatch14 = 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 p2 = pointerLocGetter(d3_event2);
+ if (!started) {
+ var dist = geoVecLength(startOrigin, p2);
+ var tolerance = d3_event2.pointerType === "pen" ? _penTolerancePx : _tolerancePx;
+ if (dist < tolerance)
+ return;
+ started = true;
+ dispatch14.call("start", this, d3_event2, _targetEntity);
+ } else {
+ startOrigin = p2;
+ d3_event2.stopPropagation();
+ d3_event2.preventDefault();
+ var dx = p2[0] - startOrigin[0];
+ var dy = p2[1] - startOrigin[1];
+ dispatch14.call("move", this, d3_event2, _targetEntity, [p2[0] + offset[0], p2[1] + offset[1]], [dx, dy]);
}
- 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 (j3 = 0; j3 < l2; j3++) {
- cell += this.renderer.tablecell(this.parseInline(token.header[j3].tokens), { header: true, align: token.align[j3] });
- }
- header += this.renderer.tablerow(cell);
- body = "";
- l2 = token.rows.length;
- for (j3 = 0; j3 < l2; j3++) {
- row = token.rows[j3];
- cell = "";
- l3 = row.length;
- for (k2 = 0; k2 < l3; k2++) {
- cell += this.renderer.tablecell(this.parseInline(row[k2].tokens), { header: false, align: token.align[k2] });
- }
- 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 (j3 = 0; j3 < l2; j3++) {
- item = token.items[j3];
- 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, token.block);
- continue;
- }
- case "paragraph": {
- out += this.renderer.paragraph(this.parseInline(token.tokens));
- continue;
- }
- case "text": {
- body = token.tokens ? this.parseInline(token.tokens) : token.text;
- while (i3 + 1 < l4 && tokens[i3 + 1].type === "text") {
- token = tokens[++i3];
- 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);
- }
- }
+ }
+ function pointerup(d3_event2) {
+ if (_pointerId !== (d3_event2.pointerId || "mouse"))
+ return;
+ _pointerId = null;
+ if (started) {
+ dispatch14.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();
}
- return out;
}
- /**
- * Parse Inline Tokens
- */
- parseInline(tokens, renderer) {
- renderer = renderer || this.renderer;
- let out = "", i3, token, ret;
- const l2 = tokens.length;
- for (i3 = 0; i3 < l2; i3++) {
- token = tokens[i3];
- 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);
+ 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);
}
}
- }
+ };
}
- return out;
+ selection2.on(_pointerPrefix + "down.drag" + _selector, delegate);
}
- };
- var _Hooks = class {
- options;
- constructor(options2) {
- this.options = options2 || _defaults;
+ behavior.off = function(selection2) {
+ selection2.on(_pointerPrefix + "down.drag" + _selector, null);
+ };
+ behavior.selector = function(_2) {
+ if (!arguments.length)
+ return _selector;
+ _selector = _2;
+ return behavior;
+ };
+ behavior.origin = function(_2) {
+ if (!arguments.length)
+ return _origin;
+ _origin = _2;
+ 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(_2) {
+ if (!arguments.length)
+ return _targetNode;
+ _targetNode = _2;
+ return behavior;
+ };
+ behavior.targetEntity = function(_2) {
+ if (!arguments.length)
+ return _targetEntity;
+ _targetEntity = _2;
+ return behavior;
+ };
+ behavior.surface = function(_2) {
+ if (!arguments.length)
+ return _surface;
+ _surface = _2;
+ return behavior;
+ };
+ return utilRebind(behavior, dispatch14, "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);
}
- static passThroughHooks = /* @__PURE__ */ new Set([
- "preprocess",
- "postprocess"
- ]);
- /**
- * Process markdown before marked
- */
- preprocess(markdown) {
- return markdown;
+ 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);
}
- /**
- * Process HTML after marked is finished
- */
- postprocess(html2) {
- return html2;
+ function shouldSnapToNode(target) {
+ if (!_activeEntity)
+ return false;
+ return _activeEntity.geometry(context.graph()) !== "vertex" || (target.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(target, context.graph()));
}
- };
- var Marked = class {
- defaults = _getDefaults();
- options = this.setOptions;
- parse = this.#parseMarkdown(_Lexer.lex, _Parser.parse);
- parseInline = this.#parseMarkdown(_Lexer.lexInline, _Parser.parseInline);
- Parser = _Parser;
- parser = _Parser.parse;
- Renderer = _Renderer;
- TextRenderer = _TextRenderer;
- Lexer = _Lexer;
- lexer = _Lexer.lex;
- Tokenizer = _Tokenizer;
- Slugger = _Slugger;
- Hooks = _Hooks;
- constructor(...args) {
- this.use(...args);
+ function origin(entity) {
+ return context.projection(entity.loc);
}
- /**
- * Run callback for every token
- */
- walkTokens(tokens, callback) {
- let values = [];
- for (const token of tokens) {
- values = values.concat(callback.call(this, token));
- switch (token.type) {
- case "table": {
- for (const cell of token.header) {
- values = values.concat(this.walkTokens(cell.tokens, callback));
- }
- for (const row of token.rows) {
- for (const cell of row) {
- values = values.concat(this.walkTokens(cell.tokens, callback));
- }
- }
- break;
- }
- case "list": {
- values = values.concat(this.walkTokens(token.items, callback));
- break;
- }
- default: {
- if (this.defaults.extensions && this.defaults.extensions.childTokens && this.defaults.extensions.childTokens[token.type]) {
- this.defaults.extensions.childTokens[token.type].forEach((childTokens) => {
- values = values.concat(this.walkTokens(token[childTokens], callback));
- });
- } else if (token.tokens) {
- values = values.concat(this.walkTokens(token.tokens, callback));
- }
- }
+ 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);
}
- return values;
}
- use(...args) {
- const extensions = this.defaults.extensions || { renderers: {}, childTokens: {} };
- args.forEach((pack) => {
- const opts = { ...pack };
- opts.async = this.defaults.async || opts.async || false;
- if (pack.extensions) {
- pack.extensions.forEach((ext) => {
- if (!ext.name) {
- throw new Error("extension name required");
- }
- if ("renderer" in ext) {
- const prevRenderer = extensions.renderers[ext.name];
- 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 ("tokenizer" in ext) {
- 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 ("childTokens" in ext && ext.childTokens) {
- extensions.childTokens[ext.name] = ext.childTokens;
- }
- });
- opts.extensions = extensions;
+ function keyup(d3_event) {
+ if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
+ if (context.surface().classed("nope-suppressed")) {
+ context.surface().classed("nope", true);
}
- if (pack.renderer) {
- const renderer = this.defaults.renderer || new _Renderer(this.defaults);
- for (const prop in pack.renderer) {
- const rendererFunc = pack.renderer[prop];
- const rendererKey = prop;
- const prevRenderer = renderer[rendererKey];
- renderer[rendererKey] = (...args2) => {
- let ret = rendererFunc.apply(renderer, args2);
- if (ret === false) {
- ret = prevRenderer.apply(renderer, args2);
- }
- return ret || "";
- };
- }
- opts.renderer = renderer;
+ 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"))();
}
- if (pack.tokenizer) {
- const tokenizer = this.defaults.tokenizer || new _Tokenizer(this.defaults);
- for (const prop in pack.tokenizer) {
- const tokenizerFunc = pack.tokenizer[prop];
- const tokenizerKey = prop;
- const prevTokenizer = tokenizer[tokenizerKey];
- tokenizer[tokenizerKey] = (...args2) => {
- let ret = tokenizerFunc.apply(tokenizer, args2);
- if (ret === false) {
- ret = prevTokenizer.apply(tokenizer, args2);
- }
- return ret;
- };
+ 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 d2 = d3_event.target.__data__;
+ return d2 && d2.properties && d2.properties.target ? d2 : {};
+ }
+ }
+ 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 d2 = datum2(d3_event);
+ target = d2 && d2.properties && d2.properties.entity;
+ var targetLoc = target && target.loc;
+ var targetNodes = d2 && d2.properties && d2.properties.nodes;
+ if (targetLoc) {
+ if (shouldSnapToNode(target)) {
+ loc = targetLoc;
}
- opts.tokenizer = tokenizer;
- }
- if (pack.hooks) {
- const hooks = this.defaults.hooks || new _Hooks();
- for (const prop in pack.hooks) {
- const hooksFunc = pack.hooks[prop];
- const hooksKey = prop;
- const prevHook = hooks[hooksKey];
- if (_Hooks.passThroughHooks.has(prop)) {
- hooks[hooksKey] = (arg) => {
- if (this.defaults.async) {
- return Promise.resolve(hooksFunc.call(hooks, arg)).then((ret2) => {
- return prevHook.call(hooks, ret2);
- });
- }
- const ret = hooksFunc.call(hooks, arg);
- return prevHook.call(hooks, ret);
- };
- } else {
- hooks[hooksKey] = (...args2) => {
- let ret = hooksFunc.apply(hooks, args2);
- if (ret === false) {
- ret = prevHook.apply(hooks, args2);
- }
- return ret;
- };
- }
+ } else if (targetNodes) {
+ edge = geoChooseEdge(targetNodes, context.map().mouse(), context.projection, end.id);
+ if (edge) {
+ loc = edge.loc;
}
- opts.hooks = hooks;
}
- if (pack.walkTokens) {
- const walkTokens2 = this.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;
- };
+ }
+ 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() }
+ ))();
}
- this.defaults = { ...this.defaults, ...opts };
- });
- return this;
+ } 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;
}
- setOptions(opt) {
- this.defaults = { ...this.defaults, ...opt };
- return this;
+ 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);
}
- #parseMarkdown(lexer2, parser3) {
- return (src, optOrCallback, callback) => {
- if (typeof optOrCallback === "function") {
- callback = optOrCallback;
- optOrCallback = null;
- }
- const origOpt = { ...optOrCallback };
- const opt = { ...this.defaults, ...origOpt };
- const throwError = this.#onError(!!opt.silent, !!opt.async, callback);
- if (typeof src === "undefined" || src === null) {
- return throwError(new Error("marked(): input parameter is undefined or null"));
- }
- if (typeof src !== "string") {
- return throwError(new Error("marked(): input parameter is of type " + Object.prototype.toString.call(src) + ", string expected"));
- }
- checkDeprecations(opt, callback);
- if (opt.hooks) {
- opt.hooks.options = opt;
- }
- if (callback) {
- const highlight = opt.highlight;
- let tokens;
- try {
- if (opt.hooks) {
- src = opt.hooks.preprocess(src);
- }
- tokens = lexer2(src, opt);
- } catch (e3) {
- return throwError(e3);
- }
- const done = (err) => {
- let out;
- if (!err) {
- try {
- if (opt.walkTokens) {
- this.walkTokens(tokens, opt.walkTokens);
- }
- out = parser3(tokens, opt);
- if (opt.hooks) {
- out = opt.hooks.postprocess(out);
- }
- } catch (e3) {
- err = e3;
+ function hasInvalidGeometry(entity, graph) {
+ var parents = graph.parentWays(entity);
+ var i3, j2, k2;
+ for (i3 = 0; i3 < parents.length; i3++) {
+ var parent = parents[i3];
+ 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 (k2 = 0; k2 < rings.length; k2++) {
+ nodes = rings[k2].nodes;
+ if (nodes.find(function(n3) {
+ return n3.id === entity.id;
+ })) {
+ activeIndex = k2;
+ if (geoHasSelfIntersections(nodes, entity.id)) {
+ return "multipolygonMember";
}
}
- opt.highlight = highlight;
- return err ? throwError(err) : callback(null, out);
- };
- if (!highlight || highlight.length < 3) {
- return done();
- }
- delete opt.highlight;
- if (!tokens.length)
- return done();
- let pending = 0;
- this.walkTokens(tokens, (token) => {
- if (token.type === "code") {
- pending++;
- setTimeout(() => {
- highlight(token.text, token.lang, (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);
+ rings[k2].coords = nodes.map(function(n3) {
+ return n3.loc;
+ });
+ }
+ for (k2 = 0; k2 < rings.length; k2++) {
+ if (k2 === activeIndex)
+ continue;
+ if (geoHasLineIntersections(rings[activeIndex].nodes, rings[k2].nodes, entity.id)) {
+ return "multipolygonRing";
}
- });
- if (pending === 0) {
- done();
}
- return;
}
- if (opt.async) {
- return Promise.resolve(opt.hooks ? opt.hooks.preprocess(src) : src).then((src2) => lexer2(src2, opt)).then((tokens) => opt.walkTokens ? Promise.all(this.walkTokens(tokens, opt.walkTokens)).then(() => tokens) : tokens).then((tokens) => parser3(tokens, opt)).then((html2) => opt.hooks ? opt.hooks.postprocess(html2) : html2).catch(throwError);
- }
- try {
- if (opt.hooks) {
- src = opt.hooks.preprocess(src);
- }
- const tokens = lexer2(src, opt);
- if (opt.walkTokens) {
- this.walkTokens(tokens, opt.walkTokens);
- }
- let html2 = parser3(tokens, opt);
- if (opt.hooks) {
- html2 = opt.hooks.postprocess(html2);
+ 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 html2;
- } catch (e3) {
- return throwError(e3);
}
- };
+ }
+ return false;
}
- #onError(silent, async, callback) {
- return (e3) => {
- e3.message += "\nPlease report this to https://github.com/markedjs/marked.";
- if (silent) {
- const msg = "An error occurred:
" + escape4(e3.message + "", true) + "
";
- if (async) {
- return Promise.resolve(msg);
- }
- if (callback) {
- callback(null, msg);
- return;
- }
- return msg;
- }
- if (async) {
- return Promise.reject(e3);
- }
- if (callback) {
- callback(e3);
- return;
+ 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 d2 = datum2(d3_event);
+ var nope = d2 && d2.properties && d2.properties.nope || context.surface().classed("nope");
+ var target = d2 && d2.properties && d2.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));
}
- throw e3;
+ }
+ }
+ function _actionBounceBack(nodeID, toLoc) {
+ var moveNode = actionMoveNode(nodeID, toLoc);
+ var action = function(graph, t2) {
+ if (t2 === 1)
+ context.pop();
+ return moveNode(graph, t2);
};
+ action.transitionable = true;
+ return action;
}
- };
- var markedInstance = new Marked();
- function marked(src, opt, callback) {
- return markedInstance.parse(src, opt, callback);
+ 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(_2) {
+ if (!arguments.length)
+ return _restoreSelectedIDs;
+ _restoreSelectedIDs = _2;
+ return mode;
+ };
+ mode.behavior = drag;
+ return mode;
}
- marked.options = marked.setOptions = function(options2) {
- markedInstance.setOptions(options2);
- marked.defaults = markedInstance.defaults;
- changeDefaults(marked.defaults);
- return marked;
- };
- marked.getDefaults = _getDefaults;
- marked.defaults = _defaults;
- marked.use = function(...args) {
- markedInstance.use(...args);
- marked.defaults = markedInstance.defaults;
- changeDefaults(marked.defaults);
- return marked;
- };
- marked.walkTokens = function(tokens, callback) {
- return markedInstance.walkTokens(tokens, callback);
- };
- marked.parseInline = markedInstance.parseInline;
- 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.Hooks = _Hooks;
- 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) {
+ // 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((text) => new DOMParser().parseFromString(text, 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 abortUnwantedRequests3(cache, tiles) {
+ function abortUnwantedRequests(cache, tiles) {
Object.keys(cache.inflightTile).forEach((k2) => {
- let wanted = tiles.find((tile) => k2 === tile.id);
+ const wanted = tiles.find((tile) => k2 === tile.id);
if (!wanted) {
- abortRequest3(cache.inflightTile[k2]);
+ abortRequest(cache.inflightTile[k2]);
delete cache.inflightTile[k2];
}
});
}
- function encodeIssueRtree3(d2) {
+ function encodeIssueRtree(d2) {
return { minX: d2.loc[0], minY: d2.loc[1], maxX: d2.loc[0], maxY: d2.loc[1], data: d2 };
}
- function updateRtree3(item, replace) {
- _cache3.rtree.remove(item, (a2, b2) => a2.data.id === b2.data.id);
+ function updateRtree(item, replace) {
+ _cache.rtree.remove(item, (a2, b2) => a2.data.id === b2.data.id);
if (replace) {
- _cache3.rtree.insert(item);
+ _cache.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((d2) => {
- _osmoseData = d2.osmose;
- _osmoseData.items = Object.keys(d2.osmose.icons).map((s2) => s2.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 [x2, y2, z2] = tile.xyz;
- let url = `${_osmoseUrlRoot}/issues/${z2}/${x2}/${y2}.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 d2 = new QAItem(loc, this, itemType, id2, { item });
- if (item === 8300 || item === 8360) {
- d2.elems = [];
- }
- _cache3.data[d2.id] = d2;
- _cache3.rtree.insert(encodeIssueRtree3(d2));
- }
- });
- }
- 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((e3) => e3.type.substring(0, 1) + e3.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((d2) => d2.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}`;
+ function tokenReplacements(d2) {
+ if (!(d2 instanceof QAItem))
+ return;
+ const replacements = {};
+ const issueTemplate = _krData.errorTypes[d2.whichType];
+ if (!issueTemplate) {
+ console.log("No Template: ", d2.whichType);
+ console.log(" ", d2.description);
+ return;
}
- };
-
- // 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 tiler9 = utilTiler().zoomExtent([minZoom, maxZoom2]).skipNullIsland(true);
- const tiles = tiler9.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])
+ if (!issueTemplate.regex)
+ return;
+ const errorRegex = new RegExp(issueTemplate.regex, "i");
+ const errorMatch = errorRegex.exec(d2.description);
+ if (!errorMatch) {
+ console.log("Unmatched: ", d2.whichType);
+ console.log(" ", d2.description);
+ console.log(" ", errorRegex);
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, i3, feature3, loc, d2;
- if (vectorTile.layers.hasOwnProperty("image")) {
- features = [];
- cache = _mlyCache.images;
- layer = vectorTile.layers.image;
- for (i3 = 0; i3 < layer.length; i3++) {
- feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
- loc = feature3.geometry.coordinates;
- d2 = {
- 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[d2.id] = d2;
- features.push({
- minX: loc[0],
- minY: loc[1],
- maxX: loc[0],
- maxY: loc[1],
- data: d2
- });
- }
- if (cache.rtree) {
- cache.rtree.load(features);
- }
}
- if (vectorTile.layers.hasOwnProperty("sequence")) {
- features = [];
- cache = _mlyCache.sequences;
- layer = vectorTile.layers.sequence;
- for (i3 = 0; i3 < layer.length; i3++) {
- feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
- if (cache.lineString[feature3.properties.id]) {
- cache.lineString[feature3.properties.id].push(feature3);
+ for (let i3 = 1; i3 < errorMatch.length; i3++) {
+ let capture = errorMatch[i3];
+ let idType;
+ idType = "IDs" in issueTemplate ? issueTemplate.IDs[i3 - 1] : "";
+ if (idType && capture) {
+ capture = parseError(capture, idType);
+ } else {
+ const compare2 = capture.toLowerCase();
+ if (_krData.localizeStrings[compare2]) {
+ capture = _t("QA.keepRight.error_parts." + _krData.localizeStrings[compare2]);
} else {
- cache.lineString[feature3.properties.id] = [feature3];
+ capture = unescape_default(capture);
}
}
+ replacements["var" + i3] = capture;
}
- if (vectorTile.layers.hasOwnProperty("point")) {
- features = [];
- cache = _mlyCache[which];
- layer = vectorTile.layers.point;
- for (i3 = 0; i3 < layer.length; i3++) {
- feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
- loc = feature3.geometry.coordinates;
- d2 = {
- 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: d2
- });
- }
- if (cache.rtree) {
- cache.rtree.load(features);
- }
+ return replacements;
+ }
+ function parseError(capture, idType) {
+ const compare2 = capture.toLowerCase();
+ if (_krData.localizeStrings[compare2]) {
+ capture = _t("QA.keepRight.error_parts." + _krData.localizeStrings[compare2]);
+ }
+ 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(d2) {
+ return { html: ''.concat(d2, "") };
+ }
+ function linkEntity2(d2) {
+ return { html: ''.concat(d2, "") };
+ }
+ function linkURL(d2) {
+ return { html: '').concat(d2, "") };
+ }
+ 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 = '"'.concat(item[0], '"');
+ const idType2 = item[1].slice(0, 1);
+ let id2 = item[2].slice(1);
+ id2 = linkEntity2(idType2 + id2);
+ newList.push("".concat(role, " ").concat(item[1], " ").concat(id2));
+ });
+ return newList.join(", ");
}
- if (vectorTile.layers.hasOwnProperty("traffic_sign")) {
- features = [];
- cache = _mlyCache[which];
- layer = vectorTile.layers.traffic_sign;
- for (i3 = 0; i3 < layer.length; i3++) {
- feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
- loc = feature3.geometry.coordinates;
- d2 = {
- 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: d2
- });
- }
- if (cache.rtree) {
- cache.rtree.load(features);
+ 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 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 z2 = geoScaleToZoom(projection2.scale());
- const z22 = Math.ceil(z2 * 2) / 2 + 2.5;
- const tiler9 = utilTiler().zoomExtent([z22, z22]);
- return tiler9.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(d2) {
- return d2.data;
+ function parse20(capture2) {
+ let newList = [];
+ const items = capture2.split(",");
+ items.forEach((item) => {
+ const id2 = linkEntity2("n" + item.slice(1));
+ newList.push(id2);
});
- return found.length ? result.concat(found) : result;
- }, []);
+ return newList.join(", ");
+ }
}
- var mapillary_default = {
- // Initialize Mapillary
- init: function() {
- if (!_mlyCache) {
+ var keepRight_default = {
+ title: "keepRight",
+ init() {
+ _mainFileFetcher.get("keepRight").then((d2) => _krData = d2);
+ if (!_cache) {
this.reset();
}
- this.event = utilRebind(this, dispatch5, "on");
+ this.event = utilRebind(this, dispatch2, "on");
},
- // Reset cache and state
- reset: function() {
- if (_mlyCache) {
- Object.values(_mlyCache.requests.inflight).forEach(function(request3) {
- request3.abort();
- });
+ reset() {
+ if (_cache) {
+ Object.values(_cache.inflightTile).forEach(abortRequest);
}
- _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: {} }
+ _cache = {
+ data: {},
+ loadedTile: {},
+ inflightTile: {},
+ inflightPost: {},
+ closed: {},
+ rtree: new import_rbush.default()
};
- _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(d2) {
- if (d2.data.sequence_id) {
- sequenceIds[d2.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();
+ // 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 = "".concat(_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: ".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;
+ }
+ 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 d2 = new QAItem(loc, this, itemType, id2, {
+ comment,
+ description,
+ whichType,
+ parentIssueType,
+ severity: whichTemplate.severity || "error",
+ objectId,
+ objectType,
+ schema,
+ title
+ });
+ d2.replacements = tokenReplacements(d2);
+ _cache.data[id2] = d2;
+ _cache.rtree.insert(encodeIssueRtree(d2));
+ });
+ dispatch2.call("loaded");
+ }).catch(() => {
+ delete _cache.inflightTile[tile.id];
+ _cache.loadedTile[tile.id] = true;
});
- }).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);
+ postUpdate(d2, callback) {
+ if (_cache.inflightPost[d2.id]) {
+ return callback({ message: "Error update already inflight", status: -2 }, d2);
}
- _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();
+ const params = { schema: d2.schema, id: d2.id };
+ if (d2.newStatus) {
+ params.st = d2.newStatus;
}
- return this;
- },
- // Hide the image viewer and resets map markers
- hideViewer: function(context) {
- _mlyActiveImage = null;
- if (!_mlyFallback && _mlyViewer) {
- _mlyViewer.getComponent("sequence").stop();
+ if (d2.newComment !== void 0) {
+ params.co = d2.newComment;
}
- 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;
+ const url = "".concat(_krUrlRoot, "/comment.php?") + utilQsString(params);
+ const controller = new AbortController();
+ _cache.inflightPost[d2.id] = controller;
+ json_default(url, { signal: controller.signal }).finally(() => {
+ delete _cache.inflightPost[d2.id];
+ if (d2.newStatus === "ignore") {
+ this.removeItem(d2);
+ } else if (d2.newStatus === "ignore_t") {
+ this.removeItem(d2);
+ _cache.closed["".concat(d2.schema, ":").concat(d2.id)] = true;
} else {
- delete hash.photo;
+ d2 = this.replaceItem(d2.update({
+ comment: d2.newComment,
+ newComment: void 0,
+ newState: void 0
+ }));
}
- 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();
+ if (callback)
+ callback(null, d2);
});
- 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(e3) {
- dispatch5.call("bearingChanged", void 0, e3);
- }
- },
- // Move to an image
- selectImage: function(context, imageId) {
- if (_mlyViewer && imageId) {
- _mlyViewer.moveTo(imageId).catch(function(e3) {
- console.error("mly3", e3);
- });
- }
- 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;
- }
+ // 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((d2) => d2.data);
},
- // 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(d2) {
- return d2.sequence_id === selectedSequenceId || d2.id === hoveredImageId;
- }).classed("hovered", function(d2) {
- return d2.id === hoveredImageId;
- });
- context.container().selectAll(".layer-mapillary .sequence").classed("highlighted", function(d2) {
- return d2.properties.id === hoveredSequenceId;
- }).classed("currentView", function(d2) {
- return d2.properties.id === selectedSequenceId;
- });
- return this;
+ // Get a QAItem from cache
+ // NOTE: Don't change method name until UI v3 is merged
+ getError(id2) {
+ return _cache.data[id2];
},
- // Get detections for the current image and shows them in the image viewer
- updateDetections: function(imageId, url) {
- if (!_mlyViewer || _mlyFallback)
+ // Replace a single QAItem in the cache
+ replaceItem(item) {
+ if (!(item instanceof QAItem) || !item.id)
return;
- if (!imageId)
+ _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;
- 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 i3 = 0; i3 < decodedGeometry.length; i3++) {
- uintArray[i3] = decodedGeometry.charCodeAt(i3);
- }
- 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;
- }
+ delete _cache.data[item.id];
+ updateRtree(encodeIssueRtree(item), false);
},
- // Return the current cache
- cache: function() {
- return _mlyCache;
+ 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() {
+ return Object.keys(_cache.closed).sort();
}
};
- // 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);
+ // 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(i3) {
+ Object.values(i3).forEach((controller) => {
+ if (controller) {
+ controller.abort();
}
- if (this.entityIds) {
- var entityKeys = this.entityIds.slice().sort();
- parts.push.apply(parts, entityKeys);
+ });
+ }
+ function abortUnwantedRequests2(cache, tiles) {
+ Object.keys(cache.inflightTile).forEach((k2) => {
+ const wanted = tiles.find((tile) => k2 === tile.id);
+ if (!wanted) {
+ abortRequest2(cache.inflightTile[k2]);
+ delete cache.inflightTile[k2];
}
- return parts.join(":");
+ });
+ }
+ function encodeIssueRtree2(d2) {
+ return { minX: d2.loc[0], minY: d2.loc[1], maxX: d2.loc[0], maxY: d2.loc[1], data: d2 };
+ }
+ function updateRtree2(item, replace) {
+ _cache2.rtree.remove(item, (a2, b2) => a2.data.id === b2.data.id);
+ if (replace) {
+ _cache2.rtree.insert(item);
}
- function generateKey() {
- return this.id + ":" + Date.now().toString();
+ }
+ function linkErrorObject(d2) {
+ return { html: ''.concat(d2, "") };
+ }
+ function linkEntity(d2) {
+ return { html: ''.concat(d2, "") };
+ }
+ 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];
}
- 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;
+ 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;
}
-
- // modules/services/maprules.js
- var buildRuleChecks = function() {
- return {
- equals: function(equals) {
- return function(tags) {
- return Object.keys(equals).every(function(k2) {
- return equals[k2] === tags[k2];
- });
- };
- },
- notEquals: function(notEquals) {
- return function(tags) {
- return Object.keys(notEquals).some(function(k2) {
- return notEquals[k2] !== tags[k2];
- });
- };
- },
- 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]);
- };
- }
+ 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"
};
- };
- var buildLineKeys = function() {
- return {
- highway: {
- rest_area: true,
- services: true
- },
- railway: {
- roundhouse: true,
- station: true,
- traverser: true,
- turntable: true,
- wash: true
+ return _t("QA.improveOSM.directions.".concat(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((d2) => _impOsmData = d2.improveOSM);
+ if (!_cache2) {
+ this.reset();
}
- };
- };
- 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;
- }, []);
+ this.event = utilRebind(this, dispatch3, "on");
},
- // builds tagMap from mapcss-parse selector object...
- buildTagMap: function(selector) {
- var getRegexValues = function(regexes) {
- return regexes.map(function(regex) {
- return regex.replace(/\$|\^/g, "");
- });
+ reset() {
+ if (_cache2) {
+ Object.values(_cache2.inflightTile).forEach(abortRequest2);
+ }
+ _cache2 = {
+ data: {},
+ loadedTile: {},
+ inflightTile: {},
+ inflightPost: {},
+ closed: {},
+ rtree: new import_rbush2.default()
};
- var tagMap = Object.keys(selector).reduce(function(expectedTags, key) {
- var values;
- var isRegex = /regex/gi.test(key);
- var isEqual4 = /equals/gi.test(key);
- if (isRegex || isEqual4) {
- Object.keys(selector[key]).forEach(function(selectorKey) {
- values = isEqual4 ? [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;
+ loadIssues(projection2) {
+ const options2 = {
+ client: "iD",
+ status: "OPEN",
+ zoom: "19"
+ // Use a high zoom so that clusters aren't returned
};
- 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);
+ 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((k2) => {
+ const kParams = Object.assign(
+ {},
+ params,
+ k2 === "mr" ? { type: "PARKING,ROAD,BOTH,PATH" } : { confidenceLevel: "C1" }
+ );
+ const url = "".concat(_impOsmUrls[k2], "/search?") + utilQsString(kParams);
+ const controller = new AbortController();
+ requests[k2] = controller;
+ json_default(url, { signal: controller.signal }).then((data) => {
+ delete _cache2.inflightTile[tile.id][k2];
+ 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 = "".concat(wayId).concat(fromNodeId).concat(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 d2 = new QAItem(loc, this, k2, itemId, {
+ issueKey: k2,
+ // used as a category
+ identifier: {
+ // used to post changes
+ wayId,
+ fromNodeId,
+ toNodeId
+ },
+ objectId: wayId,
+ objectType: "way"
+ });
+ d2.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[d2.id] = d2;
+ _cache2.rtree.insert(encodeIssueRtree2(d2));
+ });
+ }
+ if (data.tiles) {
+ data.tiles.forEach((feature3) => {
+ const { type: type2, x: x2, y: y2, numberOfTrips } = feature3;
+ const geoType = type2.toLowerCase();
+ const itemId = "".concat(geoType).concat(x2).concat(y2).concat(numberOfTrips);
+ let loc = pointAverage(feature3.points);
+ loc = preventCoincident(loc, false);
+ let d2 = new QAItem(loc, this, "".concat(k2, "-").concat(geoType), itemId, {
+ issueKey: k2,
+ identifier: { x: x2, y: y2 }
+ });
+ d2.replacements = {
+ num_trips: numberOfTrips,
+ geometry_type: _t("QA.improveOSM.geometry_types.".concat(geoType))
+ };
+ if (numberOfTrips === -1) {
+ d2.desc = _t("QA.improveOSM.error_types.mr.description_alt", d2.replacements);
+ }
+ _cache2.data[d2.id] = d2;
+ _cache2.rtree.insert(encodeIssueRtree2(d2));
+ });
+ }
+ if (data.entities) {
+ data.entities.forEach((feature3) => {
+ const { point: point2, id: id2, segments, numberOfPasses, turnType } = feature3;
+ const itemId = "".concat(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 d2 = new QAItem(loc, this, k2, itemId, {
+ issueKey: k2,
+ identifier: id2,
+ objectId: via_node,
+ objectType: "node"
+ });
+ const [p1, p2] = segments[0].points;
+ const dir_of_travel = cardinalDirection(relativeBearing(p1, p2));
+ d2.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[d2.id] = d2;
+ _cache2.rtree.insert(encodeIssueRtree2(d2));
+ dispatch3.call("loaded");
+ });
+ }
+ }).catch(() => {
+ delete _cache2.inflightTile[tile.id][k2];
+ if (!Object.keys(_cache2.inflightTile[tile.id]).length) {
+ delete _cache2.inflightTile[tile.id];
+ _cache2.loadedTile[tile.id] = true;
+ }
});
- },
- // 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;
+ });
+ _cache2.inflightTile[tile.id] = requests;
+ });
},
- // returns ruleChecks
- ruleChecks: function() {
- return this._ruleChecks;
- }
- };
-
- // modules/services/nominatim.js
- var import_rbush7 = __toESM(require_rbush_min());
-
- // 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 h2 = head.entities[id2];
- var b2 = base.entities[id2];
- if (h2 === b2)
- return;
- if (_changes[id2])
- return;
- if (!h2 && b2) {
- _changes[id2] = { base: b2, head: h2 };
- _didChange.deletion = true;
- return;
+ getComments(item) {
+ if (item.comments) {
+ return Promise.resolve(item);
}
- if (h2 && !b2) {
- _changes[id2] = { base: b2, head: h2 };
- _didChange.addition = true;
- return;
+ 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;
}
- if (h2 && b2) {
- if (h2.members && b2.members && !(0, import_fast_deep_equal3.default)(h2.members, b2.members)) {
- _changes[id2] = { base: b2, head: h2 };
- _didChange.geometry = true;
- _didChange.properties = true;
- return;
- }
- if (h2.loc && b2.loc && !geoVecEqual(h2.loc, b2.loc)) {
- _changes[id2] = { base: b2, head: h2 };
- _didChange.geometry = true;
- }
- if (h2.nodes && b2.nodes && !(0, import_fast_deep_equal3.default)(h2.nodes, b2.nodes)) {
- _changes[id2] = { base: b2, head: h2 };
- _didChange.geometry = true;
- }
- if (h2.tags && b2.tags && !(0, import_fast_deep_equal3.default)(h2.tags, b2.tags)) {
- _changes[id2] = { base: b2, head: h2 };
- _didChange.properties = true;
- }
+ const url = "".concat(_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(d2, callback) {
+ if (!osm_default.authenticated()) {
+ return callback({ message: "Not Authenticated", status: -3 }, d2);
}
- }
- function load() {
- var ids = utilArrayUniq(Object.keys(head.entities).concat(Object.keys(base.entities)));
- for (var i3 = 0; i3 < ids.length; i3++) {
- checkEntityID(ids[i3]);
+ if (_cache2.inflightPost[d2.id]) {
+ return callback({ message: "Error update already inflight", status: -2 }, d2);
}
- }
- 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 h2 = _changes[id2].head;
- var b2 = _changes[id2].base;
- var entity = h2 || b2;
- if (includeRelMembers && entity.type === "relation") {
- var mh = h2 ? h2.members.map(function(m2) {
- return m2.id;
- }) : [];
- var mb = b2 ? b2.members.map(function(m2) {
- return m2.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 keys2 = Object.keys(_changes);
- for (var i3 = 0; i3 < keys2.length; i3++) {
- var change = _changes[keys2[i3]];
- 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");
+ osm_default.userDetails(sendPayload.bind(this));
+ function sendPayload(err, user) {
+ if (err) {
+ return callback(err, d2);
}
- }
- return Object.values(relevant);
- function addEntity(entity, graph, changeType) {
- relevant[entity.id] = {
- entity,
- graph,
- changeType
+ const key = d2.issueKey;
+ const url = "".concat(_impOsmUrls[key], "/comment");
+ const payload = {
+ username: user.display_name,
+ targetIds: [d2.identifier]
};
- }
- function addParents(entity) {
- var parents = head.parentWays(entity);
- for (var j3 = parents.length - 1; j3 >= 0; j3--) {
- var parent = parents[j3];
- 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 h2 = change.head;
- var b2 = change.base;
- var entity = h2 || b2;
- var i3;
- if (extent && (!h2 || !h2.intersects(extent, head)) && (!b2 || !b2.intersects(extent, base))) {
- continue;
+ if (d2.newStatus) {
+ payload.status = d2.newStatus;
+ payload.text = "status changed";
}
- result[id2] = h2;
- if (entity.type === "way") {
- var nh = h2 ? h2.nodes : [];
- var nb = b2 ? b2.nodes : [];
- var diff;
- diff = utilArrayDifference(nh, nb);
- for (i3 = 0; i3 < diff.length; i3++) {
- result[diff[i3]] = head.hasEntity(diff[i3]);
- }
- diff = utilArrayDifference(nb, nh);
- for (i3 = 0; i3 < diff.length; i3++) {
- result[diff[i3]] = head.hasEntity(diff[i3]);
- }
+ if (d2.newComment) {
+ payload.text = d2.newComment;
}
- if (entity.type === "relation" && entity.isMultipolygon()) {
- var mh = h2 ? h2.members.map(function(m2) {
- return m2.id;
- }) : [];
- var mb = b2 ? b2.members.map(function(m2) {
- return m2.id;
- }) : [];
- var ids = utilArrayUnion(mh, mb);
- for (i3 = 0; i3 < ids.length; i3++) {
- var member = head.hasEntity(ids[i3]);
- if (!member)
- continue;
- if (extent && !member.intersects(extent, head))
- continue;
- result[ids[i3]] = member;
+ const controller = new AbortController();
+ _cache2.inflightPost[d2.id] = controller;
+ const options2 = {
+ method: "POST",
+ signal: controller.signal,
+ body: JSON.stringify(payload)
+ };
+ json_default(url, options2).then(() => {
+ delete _cache2.inflightPost[d2.id];
+ if (!d2.newStatus) {
+ const now3 = /* @__PURE__ */ new Date();
+ let comments = d2.comments ? d2.comments : [];
+ comments.push({
+ username: payload.username,
+ text: payload.text,
+ timestamp: now3.getTime() / 1e3
+ });
+ this.replaceItem(d2.update({
+ comments,
+ newComment: void 0
+ }));
+ } else {
+ this.removeItem(d2);
+ if (d2.newStatus === "SOLVED") {
+ if (!(d2.issueKey in _cache2.closed)) {
+ _cache2.closed[d2.issueKey] = 0;
+ }
+ _cache2.closed[d2.issueKey] += 1;
+ }
}
- }
- addParents(head.parentWays(entity), result);
- addParents(head.parentRelations(entity), result);
- }
- return result;
- function addParents(parents, result2) {
- for (var i4 = 0; i4 < parents.length; i4++) {
- var parent = parents[i4];
- if (parent.id in result2)
- continue;
- result2[parent.id] = parent;
- addParents(head.parentRelations(parent), result2);
- }
- }
- };
- return _diff;
- }
-
- // modules/core/tree.js
- var import_rbush5 = __toESM(require_rbush_min());
- function coreTree(head) {
- var _rtree = new import_rbush5.default();
- var _bboxes = {};
- var _segmentsRTree = new import_rbush5.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];
+ if (callback)
+ callback(null, d2);
+ }).catch((err2) => {
+ delete _cache2.inflightPost[d2.id];
+ if (callback)
+ callback(err2.message);
});
- 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 i3 = 0; i3 < entities.length; i3++) {
- var entity = entities[i3];
- 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)
+ },
+ // 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((d2) => d2.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;
- var diff = coreDifference(head, graph);
- head = graph;
- var changed = diff.didChange;
- if (!changed.addition && !changed.deletion && !changed.geometry)
+ _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;
- var insertions = {};
- if (changed.deletion) {
- diff.deleted().forEach(function(entity) {
- removeEntity(entity);
- });
+ 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,
+ breaks: false,
+ extensions: null,
+ gfm: true,
+ hooks: null,
+ pedantic: false,
+ renderer: null,
+ silent: false,
+ tokenizer: null,
+ walkTokens: null
+ };
+ }
+ var _defaults = _getDefaults();
+ function changeDefaults(newDefaults) {
+ _defaults = newDefaults;
+ }
+ var escapeTest = /[&<>"']/;
+ var escapeReplace = new RegExp(escapeTest.source, "g");
+ var escapeTestNoEncode = /[<>"']|&(?!(#\d{1,7}|#[Xx][a-fA-F0-9]{1,6}|\w+);)/;
+ var escapeReplaceNoEncode = new RegExp(escapeTestNoEncode.source, "g");
+ var escapeReplacements = {
+ "&": "&",
+ "<": "<",
+ ">": ">",
+ '"': """,
+ "'": "'"
+ };
+ var getEscapeReplacement = (ch) => escapeReplacements[ch];
+ function escape$1(html3, encode) {
+ if (encode) {
+ if (escapeTest.test(html3)) {
+ return html3.replace(escapeReplace, getEscapeReplacement);
}
- if (changed.geometry) {
- diff.modified().forEach(function(entity) {
- removeEntity(entity);
- insertions[entity.id] = entity;
- updateParents(entity, insertions, {});
- });
+ } else {
+ if (escapeTestNoEncode.test(html3)) {
+ return html3.replace(escapeReplaceNoEncode, getEscapeReplacement);
}
- if (changed.addition) {
- diff.created().forEach(function(entity) {
- insertions[entity.id] = entity;
- });
+ }
+ return html3;
+ }
+ var unescapeTest = /&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/ig;
+ function unescape2(html3) {
+ return html3.replace(unescapeTest, (_2, n3) => {
+ n3 = n3.toLowerCase();
+ if (n3 === "colon")
+ return ":";
+ if (n3.charAt(0) === "#") {
+ return n3.charAt(1) === "x" ? String.fromCharCode(parseInt(n3.substring(2), 16)) : String.fromCharCode(+n3.substring(1));
}
- 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;
+ return "";
+ });
}
-
- // 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);
+ var caret = /(^|[^\[])\^/g;
+ function edit(regex, opt) {
+ let source = typeof regex === "string" ? regex : regex.source;
+ opt = opt || "";
+ const obj = {
+ replace: (name, val) => {
+ let valSource = typeof val === "string" ? val : val.source;
+ valSource = valSource.replace(caret, "$1");
+ source = source.replace(name, valSource);
+ return obj;
+ },
+ getRegex: () => {
+ return new RegExp(source, opt);
+ }
};
+ return obj;
}
-
- // 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);
+ function cleanUrl(href) {
+ try {
+ href = encodeURI(href).replace(/%25/g, "%");
+ } catch (e3) {
+ return null;
}
- return shaded;
- function moveFocusToFirst() {
- let node = modal.select("a, button, input:not(.keytrap), select, textarea").node();
- if (node) {
- node.focus();
+ return href;
+ }
+ var noopTest = { exec: () => null };
+ function splitCells(tableRow, count) {
+ const row = tableRow.replace(/\|/g, (match, offset, str) => {
+ let escaped = false;
+ let curr = offset;
+ while (--curr >= 0 && str[curr] === "\\")
+ escaped = !escaped;
+ if (escaped) {
+ return "|";
} else {
- select_default2(this).node().blur();
+ return " |";
}
+ }), cells = row.split(/ \|/);
+ let i3 = 0;
+ if (!cells[0].trim()) {
+ cells.shift();
}
- function moveFocusToLast() {
- let nodes = modal.selectAll("a, button, input:not(.keytrap), select, textarea").nodes();
- if (nodes.length) {
- nodes[nodes.length - 1].focus();
+ if (cells.length > 0 && !cells[cells.length - 1].trim()) {
+ cells.pop();
+ }
+ if (count) {
+ if (cells.length > count) {
+ cells.splice(count);
} else {
- select_default2(this).node().blur();
+ while (cells.length < count)
+ cells.push("");
}
}
+ for (; i3 < cells.length; i3++) {
+ cells[i3] = cells[i3].trim().replace(/\\\|/g, "|");
+ }
+ return cells;
}
-
- // 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 dispatch14 = 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, t2) {
- actions = Array.prototype.slice.call(actions);
- var annotation;
- if (typeof actions[actions.length - 1] !== "function") {
- annotation = actions.pop();
+ function rtrim(str, c2, invert) {
+ const l2 = str.length;
+ if (l2 === 0) {
+ return "";
+ }
+ let suffLen = 0;
+ while (suffLen < l2) {
+ const currChar = str.charAt(l2 - suffLen - 1);
+ if (currChar === c2 && !invert) {
+ suffLen++;
+ } else if (currChar !== c2 && invert) {
+ suffLen++;
+ } else {
+ break;
}
- var graph = _stack[_index].graph;
- for (var i3 = 0; i3 < actions.length; i3++) {
- graph = actions[i3](graph, t2);
+ }
+ return str.slice(0, l2 - suffLen);
+ }
+ function findClosingBracket(str, b2) {
+ if (str.indexOf(b2[1]) === -1) {
+ return -1;
+ }
+ let level = 0;
+ for (let i3 = 0; i3 < str.length; i3++) {
+ if (str[i3] === "\\") {
+ i3++;
+ } else if (str[i3] === b2[0]) {
+ level++;
+ } else if (str[i3] === b2[1]) {
+ level--;
+ if (level < 0) {
+ return i3;
+ }
}
- return {
- graph,
- annotation,
- imageryUsed: _imageryUsed,
- photoOverlaysUsed: _photoOverlaysUsed,
- transform: context.projection.transform(),
- selectedIDs: context.selectedIDs()
+ }
+ return -1;
+ }
+ function outputLink(cap, link3, raw, lexer2) {
+ const href = link3.href;
+ const title = link3.title ? escape$1(link3.title) : null;
+ const text = cap[1].replace(/\\([\[\]])/g, "$1");
+ if (cap[0].charAt(0) !== "!") {
+ lexer2.state.inLink = true;
+ const token = {
+ type: "link",
+ raw,
+ href,
+ title,
+ text,
+ tokens: lexer2.inlineTokens(text)
};
+ lexer2.state.inLink = false;
+ return token;
}
- function _perform(args, t2) {
- var previous = _stack[_index].graph;
- _stack = _stack.slice(0, _index + 1);
- var actionResult = _act(args, t2);
- _stack.push(actionResult);
- _index++;
- return change(previous);
+ return {
+ type: "image",
+ raw,
+ href,
+ title,
+ text: escape$1(text)
+ };
+ }
+ function indentCodeCompensation(raw, text) {
+ const matchIndentToCode = raw.match(/^(\s+)(?:```)/);
+ if (matchIndentToCode === null) {
+ return text;
}
- function _replace(args, t2) {
- var previous = _stack[_index].graph;
- var actionResult = _act(args, t2);
- _stack[_index] = actionResult;
- return change(previous);
+ const indentToCode = matchIndentToCode[1];
+ return text.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 {
+ // set by the lexer
+ constructor(options2) {
+ __publicField(this, "options");
+ __publicField(this, "rules");
+ // set by the lexer
+ __publicField(this, "lexer");
+ this.options = options2 || _defaults;
}
- function _overwrite(args, t2) {
- var previous = _stack[_index].graph;
- if (_index > 0) {
- _index--;
- _stack.pop();
+ space(src) {
+ const cap = this.rules.block.newline.exec(src);
+ if (cap && cap[0].length > 0) {
+ return {
+ type: "space",
+ raw: cap[0]
+ };
}
- _stack = _stack.slice(0, _index + 1);
- var actionResult = _act(args, t2);
- _stack.push(actionResult);
- _index++;
- return change(previous);
}
- function change(previous) {
- var difference = coreDifference(previous, history.graph());
- if (!_pausedGraph) {
- dispatch14.call("change", this, difference);
+ code(src) {
+ const cap = this.rules.block.code.exec(src);
+ if (cap) {
+ const text = cap[0].replace(/^ {1,4}/gm, "");
+ return {
+ type: "code",
+ raw: cap[0],
+ codeBlockStyle: "indented",
+ text: !this.options.pedantic ? rtrim(text, "\n") : text
+ };
}
- return difference;
}
- function getKey(n3) {
- return "iD_" + window.location.origin + "_" + n3;
+ fences(src) {
+ const cap = this.rules.block.fences.exec(src);
+ if (cap) {
+ const raw = cap[0];
+ const text = indentCodeCompensation(raw, cap[3] || "");
+ return {
+ type: "code",
+ raw,
+ lang: cap[2] ? cap[2].trim().replace(this.rules.inline.anyPunctuation, "$1") : cap[2],
+ text
+ };
+ }
}
- 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);
- dispatch14.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(t2) {
- if (t2 < 1)
- _overwrite([action0], t2);
- };
- }).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(n3) {
- select_default2(document).interrupt("history.perform");
- var previous = _stack[_index].graph;
- if (isNaN(+n3) || +n3 < 0) {
- n3 = 1;
- }
- while (n3-- > 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;
- }
- dispatch14.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;
- dispatch14.call("redone", this, _stack[_index], previousStack);
- break;
+ heading(src) {
+ const cap = this.rules.block.heading.exec(src);
+ if (cap) {
+ let text = cap[2].trim();
+ if (/#$/.test(text)) {
+ const trimmed = rtrim(text, "#");
+ if (this.options.pedantic) {
+ text = trimmed.trim();
+ } else if (!trimmed || / $/.test(trimmed)) {
+ text = trimmed.trim();
}
}
- 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 i3 = _index;
- while (i3 >= 0) {
- if (_stack[i3].annotation)
- return _stack[i3].annotation;
- i3--;
- }
- },
- redoAnnotation: function() {
- var i3 = _index + 1;
- while (i3 <= _stack.length - 1) {
- if (_stack[i3].annotation)
- return _stack[i3].annotation;
- i3++;
- }
- },
- // 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 s2 = /* @__PURE__ */ new Set();
- _stack.slice(1, _index + 1).forEach(function(state) {
- state.imageryUsed.forEach(function(source) {
- if (source !== "Custom") {
- s2.add(source);
- }
- });
- });
- return Array.from(s2);
- }
- },
- photoOverlaysUsed: function(sources) {
- if (sources) {
- _photoOverlaysUsed = sources;
- return history;
- } else {
- var s2 = /* @__PURE__ */ new Set();
- _stack.slice(1, _index + 1).forEach(function(state) {
- if (state.photoOverlaysUsed && Array.isArray(state.photoOverlaysUsed)) {
- state.photoOverlaysUsed.forEach(function(photoOverlay) {
- s2.add(photoOverlay);
- });
- }
- });
- return Array.from(s2);
- }
- },
- // 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 = {};
- }
- dispatch14.call("reset");
- dispatch14.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;
- });
+ type: "heading",
+ raw: cap[0],
+ depth: cap[1].length,
+ text,
+ tokens: this.lexer.inline(text)
+ };
+ }
+ }
+ 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 text = rtrim(cap[0].replace(/^ *>[ \t]?/gm, ""), "\n");
+ const top = this.lexer.state.top;
+ this.lexer.state.top = true;
+ const tokens = this.lexer.blockTokens(text);
+ this.lexer.state.top = top;
+ return {
+ type: "blockquote",
+ raw: cap[0],
+ tokens,
+ text
+ };
+ }
+ }
+ list(src) {
+ let cap = this.rules.block.list.exec(src);
+ if (cap) {
+ let bull = cap[1].trim();
+ const isordered = bull.length > 1;
+ const list2 = {
+ type: "list",
+ raw: "",
+ ordered: isordered,
+ start: isordered ? +bull.slice(0, -1) : "",
+ loose: false,
+ items: []
+ };
+ bull = isordered ? "\\d{1,9}\\".concat(bull.slice(-1)) : "\\".concat(bull);
+ if (this.options.pedantic) {
+ bull = isordered ? bull : "[*+-]";
+ }
+ const itemRegex = new RegExp("^( {0,3}".concat(bull, ")((?:[ ][^\\n]*)?(?:\\n|$))"));
+ let raw = "";
+ let itemContents = "";
+ let endsWithBlankLine = false;
+ while (src) {
+ let endEarly = false;
+ if (!(cap = itemRegex.exec(src))) {
+ break;
}
- });
- 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 (this.rules.block.hr.test(src)) {
+ break;
}
- if (Array.isArray(copy2.loc)) {
- copy2.loc[0] = +copy2.loc[0].toFixed(6);
- copy2.loc[1] = +copy2.loc[1].toFixed(6);
+ raw = cap[0];
+ src = src.substring(raw.length);
+ let line = cap[2].split("\n", 1)[0].replace(/^\t+/, (t2) => " ".repeat(3 * t2.length));
+ let nextLine = src.split("\n", 1)[0];
+ let indent = 0;
+ if (this.options.pedantic) {
+ indent = 2;
+ itemContents = line.trimStart();
+ } else {
+ indent = cap[2].search(/[^ ]/);
+ indent = indent > 4 ? 1 : indent;
+ itemContents = line.slice(indent);
+ indent += cap[1].length;
}
- 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;
+ let blankLine = false;
+ if (!line && /^ *$/.test(nextLine)) {
+ raw += nextLine + "\n";
+ src = src.substring(nextLine.length + 1);
+ endEarly = true;
}
- return copy2;
- }
- },
- toJSON: function() {
- if (!this.hasChanges())
- return;
- var allEntities = {};
- var baseEntities = {};
- var base = _stack[0];
- var s2 = _stack.map(function(i3) {
- var modified = [];
- var deleted = [];
- Object.keys(i3.graph.entities).forEach(function(id2) {
- var entity = i3.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];
+ if (!endEarly) {
+ const nextBulletRegex = new RegExp("^ {0,".concat(Math.min(3, indent - 1), "}(?:[*+-]|\\d{1,9}[.)])((?:[ ][^\\n]*)?(?:\\n|$))"));
+ const hrRegex = new RegExp("^ {0,".concat(Math.min(3, indent - 1), "}((?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$)"));
+ const fencesBeginRegex = new RegExp("^ {0,".concat(Math.min(3, indent - 1), "}(?:```|~~~)"));
+ const headingBeginRegex = new RegExp("^ {0,".concat(Math.min(3, indent - 1), "}#"));
+ while (src) {
+ const rawLine = src.split("\n", 1)[0];
+ nextLine = rawLine;
+ if (this.options.pedantic) {
+ nextLine = nextLine.replace(/^ {1,4}(?=( {4})*[^ ])/g, " ");
+ }
+ if (fencesBeginRegex.test(nextLine)) {
+ break;
+ }
+ if (headingBeginRegex.test(nextLine)) {
+ break;
+ }
+ if (nextBulletRegex.test(nextLine)) {
+ break;
+ }
+ if (hrRegex.test(src)) {
+ break;
+ }
+ if (nextLine.search(/[^ ]/) >= indent || !nextLine.trim()) {
+ itemContents += "\n" + nextLine.slice(indent);
+ } else {
+ if (blankLine) {
+ break;
}
- });
- }
- var baseParents = base.graph._parentWays[id2];
- if (baseParents) {
- baseParents.forEach(function(parentID) {
- if (parentID in base.graph.entities) {
- baseEntities[parentID] = base.graph.entities[parentID];
+ if (line.search(/[^ ]/) >= 4) {
+ break;
}
- });
- }
- });
- var x2 = {};
- if (modified.length)
- x2.modified = modified;
- if (deleted.length)
- x2.deleted = deleted;
- if (i3.imageryUsed)
- x2.imageryUsed = i3.imageryUsed;
- if (i3.photoOverlaysUsed)
- x2.photoOverlaysUsed = i3.photoOverlaysUsed;
- if (i3.annotation)
- x2.annotation = i3.annotation;
- if (i3.transform)
- x2.transform = i3.transform;
- if (i3.selectedIDs)
- x2.selectedIDs = i3.selectedIDs;
- return x2;
- });
- return JSON.stringify({
- version: 3,
- entities: Object.values(allEntities),
- baseEntities: Object.values(baseEntities),
- stack: s2,
- nextIDs: osmEntity.id.next,
- index: _index,
- // note the time the changes were saved
- timestamp: (/* @__PURE__ */ new Date()).getTime()
- });
- },
- fromJSON: function(json, loadChildNodes) {
- var h2 = JSON.parse(json);
- var loadComplete = true;
- osmEntity.id.next = h2.nextIDs;
- _index = h2.index;
- if (h2.version === 2 || h2.version === 3) {
- var allEntities = {};
- h2.entities.forEach(function(entity) {
- allEntities[osmEntity.key(entity)] = osmEntity(entity);
- });
- if (h2.version === 3) {
- var baseEntities = h2.baseEntities.map(function(d2) {
- return osmEntity(d2);
- });
- 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(e3) {
- return e3.type === "way";
- });
- var nodeIDs = baseWays.reduce(function(acc, way) {
- return utilArrayUnion(acc, way.nodes);
- }, []);
- var missing = nodeIDs.filter(function(n3) {
- return !_stack[0].graph.hasEntity(n3);
- });
- 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);
- dispatch14.call("change");
- dispatch14.call("restore", this);
- }
- };
- osm.loadMultiple(missing, childNodesLoaded);
+ if (fencesBeginRegex.test(line)) {
+ break;
+ }
+ if (headingBeginRegex.test(line)) {
+ break;
+ }
+ if (hrRegex.test(line)) {
+ break;
+ }
+ itemContents += "\n" + nextLine;
+ }
+ if (!blankLine && !nextLine.trim()) {
+ blankLine = true;
}
+ raw += rawLine + "\n";
+ src = src.substring(rawLine.length + 1);
+ line = nextLine.slice(indent);
}
}
- _stack = h2.stack.map(function(d2) {
- var entities = {}, entity;
- if (d2.modified) {
- d2.modified.forEach(function(key) {
- entity = allEntities[key];
- entities[entity.id] = entity;
- });
+ if (!list2.loose) {
+ if (endsWithBlankLine) {
+ list2.loose = true;
+ } else if (/\n *\n *$/.test(raw)) {
+ endsWithBlankLine = true;
}
- if (d2.deleted) {
- d2.deleted.forEach(function(id2) {
- entities[id2] = void 0;
- });
+ }
+ let istask = null;
+ let ischecked;
+ if (this.options.gfm) {
+ istask = /^\[[ xX]\] /.exec(itemContents);
+ if (istask) {
+ ischecked = istask[0] !== "[ ] ";
+ itemContents = itemContents.replace(/^\[[ xX]\] +/, "");
}
- return {
- graph: coreGraph(_stack[0].graph).load(entities),
- annotation: d2.annotation,
- imageryUsed: d2.imageryUsed,
- photoOverlaysUsed: d2.photoOverlaysUsed,
- transform: d2.transform,
- selectedIDs: d2.selectedIDs
- };
+ }
+ list2.items.push({
+ type: "list_item",
+ raw,
+ task: !!istask,
+ checked: ischecked,
+ loose: false,
+ text: itemContents,
+ tokens: []
});
+ list2.raw += raw;
+ }
+ list2.items[list2.items.length - 1].raw = raw.trimEnd();
+ list2.items[list2.items.length - 1].text = itemContents.trimEnd();
+ list2.raw = list2.raw.trimEnd();
+ for (let i3 = 0; i3 < list2.items.length; i3++) {
+ this.lexer.state.top = false;
+ list2.items[i3].tokens = this.lexer.blockTokens(list2.items[i3].text, []);
+ if (!list2.loose) {
+ const spacers = list2.items[i3].tokens.filter((t2) => t2.type === "space");
+ const hasMultipleLineBreaks = spacers.length > 0 && spacers.some((t2) => /\n.*\n/.test(t2.raw));
+ list2.loose = hasMultipleLineBreaks;
+ }
+ }
+ if (list2.loose) {
+ for (let i3 = 0; i3 < list2.items.length; i3++) {
+ list2.items[i3].loose = true;
+ }
+ }
+ return list2;
+ }
+ }
+ html(src) {
+ const cap = this.rules.block.html.exec(src);
+ if (cap) {
+ const token = {
+ type: "html",
+ block: true,
+ raw: cap[0],
+ pre: cap[1] === "pre" || cap[1] === "script" || cap[1] === "style",
+ text: cap[0]
+ };
+ return token;
+ }
+ }
+ def(src) {
+ const cap = this.rules.block.def.exec(src);
+ if (cap) {
+ const tag2 = cap[1].toLowerCase().replace(/\s+/g, " ");
+ const href = cap[2] ? cap[2].replace(/^<(.*)>$/, "$1").replace(this.rules.inline.anyPunctuation, "$1") : "";
+ const title = cap[3] ? cap[3].substring(1, cap[3].length - 1).replace(this.rules.inline.anyPunctuation, "$1") : cap[3];
+ return {
+ type: "def",
+ tag: tag2,
+ raw: cap[0],
+ href,
+ title
+ };
+ }
+ }
+ table(src) {
+ const cap = this.rules.block.table.exec(src);
+ if (!cap) {
+ return;
+ }
+ if (!/[:|]/.test(cap[2])) {
+ return;
+ }
+ const headers = splitCells(cap[1]);
+ const aligns = cap[2].replace(/^\||\| *$/g, "").split("|");
+ const rows = cap[3] && cap[3].trim() ? cap[3].replace(/\n[ \t]*$/, "").split("\n") : [];
+ const item = {
+ type: "table",
+ raw: cap[0],
+ header: [],
+ align: [],
+ rows: []
+ };
+ if (headers.length !== aligns.length) {
+ return;
+ }
+ for (const align of aligns) {
+ if (/^ *-+: *$/.test(align)) {
+ item.align.push("right");
+ } else if (/^ *:-+: *$/.test(align)) {
+ item.align.push("center");
+ } else if (/^ *:-+ *$/.test(align)) {
+ item.align.push("left");
} else {
- _stack = h2.stack.map(function(d2) {
- var entities = {};
- for (var i3 in d2.entities) {
- var entity = d2.entities[i3];
- entities[i3] = entity === "undefined" ? void 0 : osmEntity(entity);
- }
- d2.graph = coreGraph(_stack[0].graph).load(entities);
- return d2;
- });
+ item.align.push(null);
}
- var transform2 = _stack[_index].transform;
- if (transform2) {
- context.map().transformEase(transform2, 0);
+ }
+ for (const header of headers) {
+ item.header.push({
+ text: header,
+ tokens: this.lexer.inline(header)
+ });
+ }
+ for (const row of rows) {
+ item.rows.push(splitCells(row, item.header.length).map((cell) => {
+ return {
+ text: cell,
+ tokens: this.lexer.inline(cell)
+ };
+ }));
+ }
+ 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 text = cap[1].charAt(cap[1].length - 1) === "\n" ? cap[1].slice(0, -1) : cap[1];
+ return {
+ type: "paragraph",
+ raw: cap[0],
+ text,
+ tokens: this.lexer.inline(text)
+ };
+ }
+ }
+ 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: escape$1(cap[1])
+ };
+ }
+ }
+ tag(src) {
+ const cap = this.rules.inline.tag.exec(src);
+ if (cap) {
+ if (!this.lexer.state.inLink && /^/i.test(cap[0])) {
+ this.lexer.state.inLink = false;
}
- if (loadComplete) {
- dispatch14.call("change");
- dispatch14.call("restore", this);
+ 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 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)
- dispatch14.call("storage_error");
+ return {
+ type: "html",
+ raw: cap[0],
+ inLink: this.lexer.state.inLink,
+ inRawBlock: this.lexer.state.inRawBlock,
+ block: false,
+ text: 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)) {
+ 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] = "";
+ }
}
- 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);
+ let href = cap[2];
+ let title = "";
+ if (this.options.pedantic) {
+ const link3 = /^([^'"]*[^\s])\s+(['"])(.*)\2/.exec(href);
+ if (link3) {
+ href = link3[1];
+ title = link3[3];
+ }
+ } else {
+ title = cap[3] ? cap[3].slice(1, -1) : "";
}
- 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);
+ href = href.trim();
+ if (/^$/.test(trimmedUrl)) {
+ href = href.slice(1);
+ } else {
+ href = href.slice(1, -1);
+ }
}
- },
- _getKey: getKey
- };
- history.reset();
- return utilRebind(history, dispatch14, "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];
+ return outputLink(cap, {
+ href: href ? href.replace(this.rules.inline.anyPunctuation, "$1") : href,
+ title: title ? title.replace(this.rules.inline.anyPunctuation, "$1") : title
+ }, cap[0], this.lexer);
+ }
}
- function isTaggedAsNotContinuing(node) {
- return node.tags.noexit === "yes" || node.tags.amenity === "parking_entrance" || node.tags.entrance && node.tags.entrance !== "no";
+ reflink(src, links) {
+ let cap;
+ if ((cap = this.rules.inline.reflink.exec(src)) || (cap = this.rules.inline.nolink.exec(src))) {
+ const linkString = (cap[2] || cap[1]).replace(/\s+/g, " ");
+ const link3 = links[linkString.toLowerCase()];
+ if (!link3) {
+ const text = cap[0].charAt(0);
+ return {
+ type: "text",
+ raw: text,
+ text
+ };
+ }
+ return outputLink(cap, link3, cap[0], this.lexer);
+ }
}
- 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
- );
+ emStrong(src, maskedSrc, prevChar = "") {
+ let match = this.rules.inline.emStrongLDelim.exec(src);
+ if (!match)
+ return;
+ if (match[3] && prevChar.match(/[\p{L}\p{N}]/u))
+ return;
+ const nextChar = match[1] || match[2] || "";
+ if (!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.emStrongRDelimAst : this.rules.inline.emStrongRDelimUnd;
+ 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;
}
}
- })];
- 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")
- );
- }
- }));
+ delimTotal -= rLength;
+ if (delimTotal > 0)
+ continue;
+ rLength = Math.min(rLength, rLength + delimTotal + midDelimTotal);
+ const lastCharLength = [...match[0]][0].length;
+ const raw = src.slice(0, lLength + match.index + lastCharLength + rLength);
+ if (Math.min(lLength, rLength) % 2) {
+ const text2 = raw.slice(1, -1);
+ return {
+ type: "em",
+ raw,
+ text: text2,
+ tokens: this.lexer.inlineTokens(text2)
+ };
+ }
+ const text = raw.slice(2, -2);
+ return {
+ type: "strong",
+ raw,
+ text,
+ tokens: this.lexer.inlineTokens(text)
+ };
}
- 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;
+ }
+ codespan(src) {
+ const cap = this.rules.inline.code.exec(src);
+ if (cap) {
+ let text = cap[2].replace(/\n/g, " ");
+ const hasNonSpaceChars = /[^ ]/.test(text);
+ const hasSpaceCharsOnBothEnds = /^ /.test(text) && / $/.test(text);
+ if (hasNonSpaceChars && hasSpaceCharsOnBothEnds) {
+ text = text.substring(1, text.length - 1);
}
- if (isTaggedAsNotContinuing(node) || graph.parentWays(node).length !== 1) {
- return false;
+ text = escape$1(text, true);
+ return {
+ type: "codespan",
+ raw: cap[0],
+ text
+ };
+ }
+ }
+ 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) {
+ const cap = this.rules.inline.autolink.exec(src);
+ if (cap) {
+ let text, href;
+ if (cap[2] === "@") {
+ text = escape$1(cap[1]);
+ href = "mailto:" + text;
+ } else {
+ text = escape$1(cap[1]);
+ href = text;
}
- let occurrences = 0;
- for (const index in way.nodes) {
- if (way.nodes[index] === node.id) {
- occurrences += 1;
- if (occurrences > 1) {
- return false;
+ return {
+ type: "link",
+ raw: cap[0],
+ text,
+ href,
+ tokens: [
+ {
+ type: "text",
+ raw: text,
+ text
}
+ ]
+ };
+ }
+ }
+ url(src) {
+ var _a2, _b;
+ let cap;
+ if (cap = this.rules.inline.url.exec(src)) {
+ let text, href;
+ if (cap[2] === "@") {
+ text = escape$1(cap[0]);
+ href = "mailto:" + text;
+ } else {
+ let prevCapZero;
+ do {
+ prevCapZero = cap[0];
+ cap[0] = (_b = (_a2 = this.rules.inline._backpedal.exec(cap[0])) == null ? void 0 : _a2[0]) != null ? _b : "";
+ } while (prevCapZero !== cap[0]);
+ text = escape$1(cap[0]);
+ if (cap[1] === "www.") {
+ href = "http://" + cap[0];
+ } else {
+ href = cap[0];
}
}
- return true;
+ return {
+ type: "link",
+ raw: cap[0],
+ text,
+ href,
+ tokens: [
+ {
+ type: "text",
+ raw: text,
+ text
+ }
+ ]
+ };
}
- 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;
+ }
+ inlineText(src) {
+ const cap = this.rules.inline.text.exec(src);
+ if (cap) {
+ let text;
+ if (this.lexer.state.inRawBlock) {
+ text = cap[0];
+ } else {
+ text = escape$1(cap[0]);
+ }
+ return {
+ type: "text",
+ raw: cap[0],
+ text
+ };
}
- function findNearbyEndNodes(node, way) {
- return [
- way.nodes[0],
- way.nodes[way.nodes.length - 1]
- ].map((d2) => graph.entity(d2)).filter((d2) => {
- return d2.id !== node.id && geoSphericalDistance(node.loc, d2.loc) <= CLOSE_NODE_TH;
- });
+ }
+ };
+ var newline = /^(?: *(?:\n|$))+/;
+ var blockCode = /^( {4}[^\n]+(?:\n(?: *(?:\n|$))*)?)+/;
+ var fences = /^ {0,3}(`{3,}(?=[^`\n]*(?:\n|$))|~{3,})([^\n]*)(?:\n|$)(?:|([\s\S]*?)(?:\n|$))(?: {0,3}\1[~`]* *(?=\n|$)|$)/;
+ var hr = /^ {0,3}((?:-[\t ]*){3,}|(?:_[ \t]*){3,}|(?:\*[ \t]*){3,})(?:\n+|$)/;
+ var heading = /^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/;
+ var bullet = /(?:[*+-]|\d{1,9}[.)])/;
+ var lheading = edit(/^(?!bull )((?:.|\n(?!\s*?\n|bull ))+?)\n {0,3}(=+|-+) *(?:\n+|$)/).replace(/bull/g, bullet).getRegex();
+ var _paragraph = /^([^\n]+(?:\n(?!hr|heading|lheading|blockquote|fences|list|html|table| +\n)[^\n]+)*)/;
+ var blockText = /^[^\n]+/;
+ var _blockLabel = /(?!\s*\])(?:\\.|[^\[\]\\])+/;
+ var def = edit(/^ {0,3}\[(label)\]: *(?:\n *)?([^<\s][^\s]*|<.*?>)(?:(?: +(?:\n *)?| *\n *)(title))? *(?:\n+|$)/).replace("label", _blockLabel).replace("title", /(?:"(?:\\"?|[^"\\])*"|'[^'\n]*(?:\n[^'\n]+)*\n?'|\([^()]*\))/).getRegex();
+ var list = edit(/^( {0,3}bull)([ \t][^\n]+?)?(?:\n|$)/).replace(/bull/g, bullet).getRegex();
+ var _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|search|section|summary|table|tbody|td|tfoot|th|thead|title|tr|track|ul";
+ var _comment = /|$))/;
+ var html2 = edit("^ {0,3}(?:<(script|pre|style|textarea)[\\s>][\\s\\S]*?(?:\\1>[^\\n]*\\n+|$)|comment[^\\n]*(\\n+|$)|<\\?[\\s\\S]*?(?:\\?>\\n*|$)|\\n*|$)|\\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|$))", "i").replace("comment", _comment).replace("tag", _tag).replace("attribute", / +[a-zA-Z:_][\w.:-]*(?: *= *"[^"\n]*"| *= *'[^'\n]*'| *= *[^\s"'=<>`]+)?/).getRegex();
+ var paragraph = edit(_paragraph).replace("hr", hr).replace("heading", " {0,3}#{1,6}(?:\\s|$)").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", _tag).getRegex();
+ var blockquote = edit(/^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/).replace("paragraph", paragraph).getRegex();
+ var blockNormal = {
+ blockquote,
+ code: blockCode,
+ def,
+ fences,
+ heading,
+ hr,
+ html: html2,
+ lheading,
+ list,
+ newline,
+ paragraph,
+ table: noopTest,
+ text: blockText
+ };
+ var gfmTable = edit("^ *([^\\n ].*)\\n {0,3}((?:\\| *)?:?-+:? *(?:\\| *:?-+:? *)*(?:\\| *)?)(?:\\n((?:(?! *\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)").replace("hr", hr).replace("heading", " {0,3}#{1,6}(?:\\s|$)").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", _tag).getRegex();
+ var blockGfm = {
+ ...blockNormal,
+ table: gfmTable,
+ paragraph: edit(_paragraph).replace("hr", hr).replace("heading", " {0,3}#{1,6}(?:\\s|$)").replace("|lheading", "").replace("table", gfmTable).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", _tag).getRegex()
+ };
+ var blockPedantic = {
+ ...blockNormal,
+ html: edit("^ *(?:comment *(?:\\n|\\s*$)|<(tag)[\\s\\S]+?\\1> *(?:\\n{2,}|\\s*$)|\\s]*)*?/?> *(?:\\n{2,}|\\s*$))").replace("comment", _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
+ lheading: /^(.+?)\n {0,3}(=+|-+) *(?:\n+|$)/,
+ paragraph: edit(_paragraph).replace("hr", hr).replace("heading", " *#{1,6} *[^\n]").replace("lheading", lheading).replace("|table", "").replace("blockquote", " {0,3}>").replace("|fences", "").replace("|list", "").replace("|html", "").replace("|tag", "").getRegex()
+ };
+ var escape4 = /^\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/;
+ var inlineCode = /^(`+)([^`]|[^`][\s\S]*?[^`])\1(?!`)/;
+ var br = /^( {2,}|\\)\n(?!\s*$)/;
+ var inlineText = /^(`+|[^`])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\]*?>/g;
+ var emStrongLDelim = edit(/^(?:\*+(?:((?!\*)[punct])|[^\s*]))|^_+(?:((?!_)[punct])|([^\s_]))/, "u").replace(/punct/g, _punctuation).getRegex();
+ var emStrongRDelimAst = edit("^[^_*]*?__[^_*]*?\\*[^_*]*?(?=__)|[^*]+(?=[^*])|(?!\\*)[punct](\\*+)(?=[\\s]|$)|[^punct\\s](\\*+)(?!\\*)(?=[punct\\s]|$)|(?!\\*)[punct\\s](\\*+)(?=[^punct\\s])|[\\s](\\*+)(?!\\*)(?=[punct])|(?!\\*)[punct](\\*+)(?!\\*)(?=[punct])|[^punct\\s](\\*+)(?=[^punct\\s])", "gu").replace(/punct/g, _punctuation).getRegex();
+ var emStrongRDelimUnd = edit("^[^_*]*?\\*\\*[^_*]*?_[^_*]*?(?=\\*\\*)|[^_]+(?=[^_])|(?!_)[punct](_+)(?=[\\s]|$)|[^punct\\s](_+)(?!_)(?=[punct\\s]|$)|(?!_)[punct\\s](_+)(?=[^punct\\s])|[\\s](_+)(?!_)(?=[punct])|(?!_)[punct](_+)(?!_)(?=[punct])", "gu").replace(/punct/g, _punctuation).getRegex();
+ var anyPunctuation = edit(/\\([punct])/, "gu").replace(/punct/g, _punctuation).getRegex();
+ var autolink = edit(/^<(scheme:[^\s\x00-\x1f<>]*|email)>/).replace("scheme", /[a-zA-Z][a-zA-Z0-9+.-]{1,31}/).replace("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])?)+(?![-_])/).getRegex();
+ var _inlineComment = edit(_comment).replace("(?:-->|$)", "-->").getRegex();
+ var tag = edit("^comment|^[a-zA-Z][\\w:-]*\\s*>|^<[a-zA-Z][\\w-]*(?:attribute)*?\\s*/?>|^<\\?[\\s\\S]*?\\?>|^|^").replace("comment", _inlineComment).replace("attribute", /\s+[a-zA-Z:_][\w.:-]*(?:\s*=\s*"[^"]*"|\s*=\s*'[^']*'|\s*=\s*[^\s"'=<>`]+)?/).getRegex();
+ var _inlineLabel = /(?:\[(?:\\.|[^\[\]\\])*\]|\\.|`[^`]*`|[^\[\]\\`])*?/;
+ var link2 = edit(/^!?\[(label)\]\(\s*(href)(?:\s+(title))?\s*\)/).replace("label", _inlineLabel).replace("href", /<(?:\\.|[^\n<>\\])+>|[^\s\x00-\x1f]*/).replace("title", /"(?:\\"?|[^"\\])*"|'(?:\\'?|[^'\\])*'|\((?:\\\)?|[^)\\])*\)/).getRegex();
+ var reflink = edit(/^!?\[(label)\]\[(ref)\]/).replace("label", _inlineLabel).replace("ref", _blockLabel).getRegex();
+ var nolink = edit(/^!?\[(ref)\](?:\[\])?/).replace("ref", _blockLabel).getRegex();
+ var reflinkSearch = edit("reflink|nolink(?!\\()", "g").replace("reflink", reflink).replace("nolink", nolink).getRegex();
+ var inlineNormal = {
+ _backpedal: noopTest,
+ // only used for GFM url
+ anyPunctuation,
+ autolink,
+ blockSkip,
+ br,
+ code: inlineCode,
+ del: noopTest,
+ emStrongLDelim,
+ emStrongRDelimAst,
+ emStrongRDelimUnd,
+ escape: escape4,
+ link: link2,
+ nolink,
+ punctuation,
+ reflink,
+ reflinkSearch,
+ tag,
+ text: inlineText,
+ url: noopTest
+ };
+ var inlinePedantic = {
+ ...inlineNormal,
+ link: edit(/^!?\[(label)\]\((.*?)\)/).replace("label", _inlineLabel).getRegex(),
+ reflink: edit(/^!?\[(label)\]\s*\[([^\]]*)\]/).replace("label", _inlineLabel).getRegex()
+ };
+ var inlineGfm = {
+ ...inlineNormal,
+ escape: edit(escape4).replace("])", "~|])").getRegex(),
+ url: edit(/^((?:ftp|https?):\/\/|www\.)(?:[a-zA-Z0-9\-]+\.?)+[^\s<]*|^email/, "i").replace("email", /[A-Za-z0-9._+-]+(@)[a-zA-Z0-9-_]+(?:\.[a-zA-Z0-9-_]*[a-zA-Z0-9])+(?![-_])/).getRegex(),
+ _backpedal: /(?:[^?!.,:;*_'"~()&]+|\([^)]*\)|&(?![a-zA-Z0-9]+;$)|[?!.,:;*_'"~)]+(?!$))+/,
+ del: /^(~~?)(?=[^\s~])([\s\S]*?[^\s~])\1(?=[^~]|$)/,
+ text: /^([`~]+|[^`~])(?:(?= {2,}\n)|(?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)|[\s\S]*?(?:(?=[\\ {
- 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;
+ 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);
+ for (let i3 = 0; i3 < this.inlineQueue.length; i3++) {
+ const next = this.inlineQueue[i3];
+ this.inlineTokens(next.src, next.tokens);
}
- function hasTag(tags, key) {
- return tags[key] !== void 0 && tags[key] !== "no";
+ this.inlineQueue = [];
+ return this.tokens;
+ }
+ blockTokens(src, tokens = []) {
+ if (this.options.pedantic) {
+ src = src.replace(/\t/g, " ").replace(/^ +$/gm, "");
+ } else {
+ src = src.replace(/^( *)(\t+)/gm, (_2, leading, tabs) => {
+ return leading + " ".repeat(tabs.length);
+ });
}
- 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)
+ let token;
+ let lastToken;
+ let cutSrc;
+ let 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;
- 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 t2 = EXTEND_TH_METERS / edgeLen + 1;
- const extTipLoc = geoVecInterp(midNode.loc, tipNode.loc, t2);
- const segmentInfos = tree.waySegments(queryExtent, graph);
- for (let i3 = 0; i3 < segmentInfos.length; i3++) {
- let segmentInfo = segmentInfos[i3];
- 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
+ })) {
+ 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((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);
+ }
}
- 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);
+ 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;
}
- }
- 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 i3 in parentRelations) {
- var relation = parentRelations[i3];
- 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";
+ 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;
}
- 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 i3 = 0; i3 < nodes.length - 1; i3++) {
- var node1 = nodes[i3];
- var node2 = nodes[i3 + 1];
- var issue = getWayIssueIfAny(node1, node2, way);
- if (issue)
- issues.push(issue);
+ 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 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 i3 = 0; i3 < parentWays.length; i3++) {
- var parentWay = parentWays[i3];
- if (!shouldCheckWay(parentWay))
- continue;
- var lastIndex = parentWay.nodes.length - 1;
- for (var j3 = 0; j3 < parentWay.nodes.length; j3++) {
- if (j3 !== 0) {
- if (parentWay.nodes[j3 - 1] === node.id) {
- checkForCloseness(node, graph.entity(parentWay.nodes[j3]), parentWay);
- }
- }
- if (j3 !== lastIndex) {
- if (parentWay.nodes[j3 + 1] === node.id) {
- checkForCloseness(graph.entity(parentWay.nodes[j3]), node, parentWay);
- }
+ 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) + "[" + "a".repeat(match[0].length - 2) + "]" + maskedSrc.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex);
}
}
}
- 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;
+ while ((match = this.tokenizer.rules.inline.blockSkip.exec(maskedSrc)) != null) {
+ maskedSrc = maskedSrc.slice(0, match.index) + "[" + "a".repeat(match[0].length - 2) + "]" + maskedSrc.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);
}
- 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 j3 = 0; j3 < intersected.length; j3++) {
- var nearby = intersected[j3];
- 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")
- })
- ];
- }
- }));
+ while ((match = this.tokenizer.rules.inline.anyPunctuation.exec(maskedSrc)) != null) {
+ maskedSrc = maskedSrc.slice(0, match.index) + "++" + maskedSrc.slice(this.tokenizer.rules.inline.anyPunctuation.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;
}
- 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);
+ if (token = this.tokenizer.escape(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
}
- }
- function getWayIssueIfAny(node1, node2, way) {
- if (node1.id === node2.id || node1.hasInterestingTags() && node2.hasInterestingTags()) {
- return null;
+ 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 (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;
+ if (token = this.tokenizer.link(src)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
}
- 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")
- })
- ];
+ 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);
}
- });
- 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);
+ continue;
}
- }
- };
- validation.type = type2;
- return validation;
- }
-
- // modules/validations/crossing_ways.js
- var import_lodash3 = __toESM(require_lodash());
- function validationCrossingWays(context) {
- var type2 = "crossing_ways";
- function getFeatureWithFeatureTypeTagsForWay(way, graph) {
- if (getFeatureType(way, graph) === null) {
- var parentRels = graph.parentRelations(way);
- for (var i3 = 0; i3 < parentRels.length; i3++) {
- var rel = parentRels[i3];
- if (getFeatureType(rel, graph) !== null) {
- return rel;
- }
+ if (token = this.tokenizer.emStrong(src, maskedSrc, prevChar)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
}
- }
- 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
- };
- function tagsForConnectionNodeIfAllowed(entity1, entity2, graph, lessLikelyTags) {
- 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;
- var pathFeature = entity1IsPath ? entity1 : entity2;
- if (roadFeature.tags.highway === "track") {
- return {};
- }
- if (!lessLikelyTags && roadFeature.tags.highway === "service" && pathFeature.tags.highway === "footway" && pathFeature.tags.footway === "sidewalk") {
- return {};
- }
- if (["marked", "unmarked", "traffic_signals", "uncontrolled"].indexOf(pathFeature.tags.crossing) !== -1) {
- return bothLines ? { highway: "crossing", crossing: pathFeature.tags.crossing } : {};
+ 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)) {
+ src = src.substring(token.raw.length);
+ tokens.push(token);
+ continue;
+ }
+ if (!this.state.inLink && (token = this.tokenizer.url(src))) {
+ 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((getStartIndex) => {
+ tempStart = getStartIndex.call({ lexer: this }, tempSrc);
+ if (typeof tempStart === "number" && tempStart >= 0) {
+ startIndex = Math.min(startIndex, tempStart);
}
- return bothLines ? { highway: "crossing" } : {};
+ });
+ if (startIndex < Infinity && startIndex >= 0) {
+ cutSrc = src.substring(0, startIndex + 1);
}
- 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 (token = this.tokenizer.inlineText(cutSrc)) {
+ 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);
}
- 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" } : {};
+ 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 null;
+ return tokens;
}
- 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 i3, j3;
- var extent;
- var n1, n22, nA, nB, nAId, nBId;
- var segment1, segment2;
- var oneOnly;
- var segmentInfos, segment2Info, way2, taggedFeature2, way2FeatureType;
- var way1Nodes = graph.childNodes(way1);
- var comparedWays = {};
- for (i3 = 0; i3 < way1Nodes.length - 1; i3++) {
- n1 = way1Nodes[i3];
- n22 = way1Nodes[i3 + 1];
- extent = geoExtent([
- [
- Math.min(n1.loc[0], n22.loc[0]),
- Math.min(n1.loc[1], n22.loc[1])
- ],
- [
- Math.max(n1.loc[0], n22.loc[0]),
- Math.max(n1.loc[1], n22.loc[1])
- ]
- ]);
- segmentInfos = tree.waySegments(extent, graph);
- for (j3 = 0; j3 < segmentInfos.length; j3++) {
- segment2Info = segmentInfos[j3];
- if (segment2Info.wayId === way1.id)
+ };
+ var _Renderer = class {
+ constructor(options2) {
+ __publicField(this, "options");
+ this.options = options2 || _defaults;
+ }
+ code(code, infostring, escaped) {
+ var _a2;
+ const lang = (_a2 = (infostring || "").match(/^\S*/)) == null ? void 0 : _a2[0];
+ code = code.replace(/\n$/, "") + "\n";
+ if (!lang) {
+ return "" + (escaped ? code : escape$1(code, true)) + "
\n";
+ }
+ return '' + (escaped ? code : escape$1(code, true)) + "
\n";
+ }
+ blockquote(quote) {
+ return "\n".concat(quote, "
\n");
+ }
+ html(html3, block2) {
+ return html3;
+ }
+ heading(text, level, raw) {
+ return "").concat(text, "\n");
+ }
+ hr() {
+ return "
\n";
+ }
+ list(body, ordered, start2) {
+ const type2 = ordered ? "ol" : "ul";
+ const startatt = ordered && start2 !== 1 ? ' start="' + start2 + '"' : "";
+ return "<" + type2 + startatt + ">\n" + body + "" + type2 + ">\n";
+ }
+ listitem(text, task, checked) {
+ return "".concat(text, "\n");
+ }
+ checkbox(checked) {
+ return "';
+ }
+ paragraph(text) {
+ return "".concat(text, "
\n");
+ }
+ table(header, body) {
+ if (body)
+ body = "".concat(body, "");
+ return "\n\n" + header + "\n" + body + "
\n";
+ }
+ tablerow(content) {
+ return "\n".concat(content, "
\n");
+ }
+ tablecell(content, flags) {
+ const type2 = flags.header ? "th" : "td";
+ const tag2 = flags.align ? "<".concat(type2, ' align="').concat(flags.align, '">') : "<".concat(type2, ">");
+ return tag2 + content + "".concat(type2, ">\n");
+ }
+ /**
+ * span level renderer
+ */
+ strong(text) {
+ return "".concat(text, "");
+ }
+ em(text) {
+ return "".concat(text, "");
+ }
+ codespan(text) {
+ return "".concat(text, "
");
+ }
+ br() {
+ return "
";
+ }
+ del(text) {
+ return "".concat(text, "");
+ }
+ link(href, title, text) {
+ const cleanHref = cleanUrl(href);
+ if (cleanHref === null) {
+ return text;
+ }
+ href = cleanHref;
+ let out = '" + text + "";
+ return out;
+ }
+ image(href, title, text) {
+ const cleanHref = cleanUrl(href);
+ if (cleanHref === null) {
+ return text;
+ }
+ href = cleanHref;
+ let out = '";
+ return out;
+ }
+ text(text) {
+ return text;
+ }
+ };
+ var _TextRenderer = class {
+ // no need for block level renderers
+ strong(text) {
+ return text;
+ }
+ em(text) {
+ return text;
+ }
+ codespan(text) {
+ return text;
+ }
+ del(text) {
+ return text;
+ }
+ html(text) {
+ return text;
+ }
+ text(text) {
+ return text;
+ }
+ link(href, title, text) {
+ return "" + text;
+ }
+ image(href, title, text) {
+ return "" + text;
+ }
+ br() {
+ return "";
+ }
+ };
+ var _Parser = class __Parser {
+ constructor(options2) {
+ __publicField(this, "options");
+ __publicField(this, "renderer");
+ __publicField(this, "textRenderer");
+ 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();
+ }
+ /**
+ * 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 = "";
+ for (let i3 = 0; i3 < tokens.length; i3++) {
+ const token = tokens[i3];
+ if (this.options.extensions && this.options.extensions.renderers && this.options.extensions.renderers[token.type]) {
+ const genericToken = token;
+ const ret = this.options.extensions.renderers[genericToken.type].call({ parser: this }, genericToken);
+ if (ret !== false || !["space", "hr", "heading", "code", "table", "blockquote", "list", "html", "paragraph", "text"].includes(genericToken.type)) {
+ out += ret || "";
continue;
- if (checkedSingleCrossingWays[segment2Info.wayId])
+ }
+ }
+ switch (token.type) {
+ case "space": {
continue;
- comparedWays[segment2Info.wayId] = true;
- way2 = graph.hasEntity(segment2Info.wayId);
- if (!way2)
+ }
+ case "hr": {
+ out += this.renderer.hr();
continue;
- taggedFeature2 = getFeatureWithFeatureTypeTagsForWay(way2, graph);
- way2FeatureType = getFeatureType(taggedFeature2, graph);
- if (way2FeatureType === null || isLegitCrossing(taggedFeature1.tags, way1FeatureType, taggedFeature2.tags, way2FeatureType)) {
+ }
+ case "heading": {
+ const headingToken = token;
+ out += this.renderer.heading(this.parseInline(headingToken.tokens), headingToken.depth, unescape2(this.parseInline(headingToken.tokens, this.textRenderer)));
continue;
}
- oneOnly = way1FeatureType === "building" || way2FeatureType === "building";
- nAId = segment2Info.nodes[0];
- nBId = segment2Info.nodes[1];
- if (nAId === n1.id || nAId === n22.id || nBId === n1.id || nBId === n22.id) {
+ case "code": {
+ const codeToken = token;
+ out += this.renderer.code(codeToken.text, codeToken.lang, !!codeToken.escaped);
continue;
}
- nA = graph.hasEntity(nAId);
- if (!nA)
+ case "table": {
+ const tableToken = token;
+ let header = "";
+ let cell = "";
+ for (let j2 = 0; j2 < tableToken.header.length; j2++) {
+ cell += this.renderer.tablecell(this.parseInline(tableToken.header[j2].tokens), { header: true, align: tableToken.align[j2] });
+ }
+ header += this.renderer.tablerow(cell);
+ let body = "";
+ for (let j2 = 0; j2 < tableToken.rows.length; j2++) {
+ const row = tableToken.rows[j2];
+ cell = "";
+ for (let k2 = 0; k2 < row.length; k2++) {
+ cell += this.renderer.tablecell(this.parseInline(row[k2].tokens), { header: false, align: tableToken.align[k2] });
+ }
+ body += this.renderer.tablerow(cell);
+ }
+ out += this.renderer.table(header, body);
continue;
- nB = graph.hasEntity(nBId);
- if (!nB)
+ }
+ case "blockquote": {
+ const blockquoteToken = token;
+ const body = this.parse(blockquoteToken.tokens);
+ out += this.renderer.blockquote(body);
continue;
- segment1 = [n1.loc, n22.loc];
- segment2 = [nA.loc, nB.loc];
- var point2 = geoLineIntersection(segment1, segment2);
- if (point2) {
- edgeCrossInfos.push({
- wayInfos: [
- {
- way: way1,
- featureType: way1FeatureType,
- edge: [n1.id, n22.id]
- },
- {
- way: way2,
- featureType: way2FeatureType,
- edge: [nA.id, nB.id]
+ }
+ case "list": {
+ const listToken = token;
+ const ordered = listToken.ordered;
+ const start2 = listToken.start;
+ const loose = listToken.loose;
+ let body = "";
+ for (let j2 = 0; j2 < listToken.items.length; j2++) {
+ const item = listToken.items[j2];
+ const checked = item.checked;
+ const task = item.task;
+ let itemBody = "";
+ if (item.task) {
+ const 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 + " ";
}
- ],
- crossPoint: point2
- });
- if (oneOnly) {
- checkedSingleCrossingWays[way2.id] = true;
- break;
+ }
+ itemBody += this.parse(item.tokens, loose);
+ body += this.renderer.listitem(itemBody, task, !!checked);
}
+ out += this.renderer.list(body, ordered, start2);
+ continue;
}
- }
- }
- 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);
+ case "html": {
+ const htmlToken = token;
+ out += this.renderer.html(htmlToken.text, htmlToken.block);
+ continue;
+ }
+ case "paragraph": {
+ const paragraphToken = token;
+ out += this.renderer.paragraph(this.parseInline(paragraphToken.tokens));
+ continue;
+ }
+ case "text": {
+ let textToken = token;
+ let body = textToken.tokens ? this.parseInline(textToken.tokens) : textToken.text;
+ while (i3 + 1 < tokens.length && tokens[i3 + 1].type === "text") {
+ textToken = tokens[++i3];
+ body += "\n" + (textToken.tokens ? this.parseInline(textToken.tokens) : textToken.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 array2;
- }, []);
+ }
}
- return [];
+ return out;
}
- 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));
+ /**
+ * Parse Inline Tokens
+ */
+ parseInline(tokens, renderer) {
+ renderer = renderer || this.renderer;
+ let out = "";
+ for (let i3 = 0; i3 < tokens.length; i3++) {
+ const token = tokens[i3];
+ if (this.options.extensions && this.options.extensions.renderers && this.options.extensions.renderers[token.type]) {
+ const 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;
+ }
}
- }
- 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;
+ switch (token.type) {
+ case "escape": {
+ const escapeToken = token;
+ out += renderer.text(escapeToken.text);
+ break;
+ }
+ case "html": {
+ const tagToken = token;
+ out += renderer.html(tagToken.text);
+ break;
+ }
+ case "link": {
+ const linkToken = token;
+ out += renderer.link(linkToken.href, linkToken.title, this.parseInline(linkToken.tokens, renderer));
+ break;
+ }
+ case "image": {
+ const imageToken = token;
+ out += renderer.image(imageToken.href, imageToken.title, imageToken.text);
+ break;
+ }
+ case "strong": {
+ const strongToken = token;
+ out += renderer.strong(this.parseInline(strongToken.tokens, renderer));
+ break;
+ }
+ case "em": {
+ const emToken = token;
+ out += renderer.em(this.parseInline(emToken.tokens, renderer));
+ break;
+ }
+ case "codespan": {
+ const codespanToken = token;
+ out += renderer.codespan(codespanToken.text);
+ break;
+ }
+ case "br": {
+ out += renderer.br();
+ break;
+ }
+ case "del": {
+ const delToken = token;
+ out += renderer.del(this.parseInline(delToken.tokens, renderer));
+ break;
+ }
+ case "text": {
+ const textToken = token;
+ out += renderer.text(textToken.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 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));
- let lessLikelyConnectionTags = tagsForConnectionNodeIfAllowed(entities[0], entities[1], graph, true);
- if (lessLikelyConnectionTags && !(0, import_lodash3.isEqual)(connectionTags, lessLikelyConnectionTags)) {
- fixes.push(makeConnectWaysFix(lessLikelyConnectionTags));
+ return out;
+ }
+ };
+ var _Hooks = class {
+ constructor(options2) {
+ __publicField(this, "options");
+ this.options = options2 || _defaults;
+ }
+ /**
+ * Process markdown before marked
+ */
+ preprocess(markdown) {
+ return markdown;
+ }
+ /**
+ * Process HTML after marked is finished
+ */
+ postprocess(html3) {
+ return html3;
+ }
+ /**
+ * Process all tokens before walk tokens
+ */
+ processAllTokens(tokens) {
+ return tokens;
+ }
+ };
+ __publicField(_Hooks, "passThroughHooks", /* @__PURE__ */ new Set([
+ "preprocess",
+ "postprocess",
+ "processAllTokens"
+ ]));
+ var _parseMarkdown, parseMarkdown_fn, _onError, onError_fn;
+ var Marked = class {
+ constructor(...args) {
+ __privateAdd(this, _parseMarkdown);
+ __privateAdd(this, _onError);
+ __publicField(this, "defaults", _getDefaults());
+ __publicField(this, "options", this.setOptions);
+ __publicField(this, "parse", __privateMethod(this, _parseMarkdown, parseMarkdown_fn).call(this, _Lexer.lex, _Parser.parse));
+ __publicField(this, "parseInline", __privateMethod(this, _parseMarkdown, parseMarkdown_fn).call(this, _Lexer.lexInline, _Parser.parseInline));
+ __publicField(this, "Parser", _Parser);
+ __publicField(this, "Renderer", _Renderer);
+ __publicField(this, "TextRenderer", _TextRenderer);
+ __publicField(this, "Lexer", _Lexer);
+ __publicField(this, "Tokenizer", _Tokenizer);
+ __publicField(this, "Hooks", _Hooks);
+ this.use(...args);
+ }
+ /**
+ * Run callback for every token
+ */
+ walkTokens(tokens, callback) {
+ var _a2, _b;
+ let values = [];
+ for (const token of tokens) {
+ values = values.concat(callback.call(this, token));
+ switch (token.type) {
+ case "table": {
+ const tableToken = token;
+ for (const cell of tableToken.header) {
+ values = values.concat(this.walkTokens(cell.tokens, callback));
}
- }
- 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"));
+ for (const row of tableToken.rows) {
+ for (const cell of row) {
+ values = values.concat(this.walkTokens(cell.tokens, callback));
+ }
}
- var skipTunnelFix = otherFeatureType === "waterway" && selectedFeatureType !== "waterway";
- if (allowsTunnel(selectedFeatureType) && !skipTunnelFix) {
- fixes.push(makeAddBridgeOrTunnelFix("add_a_tunnel", "temaki-tunnel", "tunnel"));
+ break;
+ }
+ case "list": {
+ const listToken = token;
+ values = values.concat(this.walkTokens(listToken.items, callback));
+ break;
+ }
+ default: {
+ const genericToken = token;
+ if ((_b = (_a2 = this.defaults.extensions) == null ? void 0 : _a2.childTokens) == null ? void 0 : _b[genericToken.type]) {
+ this.defaults.extensions.childTokens[genericToken.type].forEach((childTokens) => {
+ const tokens2 = genericToken[childTokens].flat(Infinity);
+ values = values.concat(this.walkTokens(tokens2, callback));
+ });
+ } else if (genericToken.tokens) {
+ values = values.concat(this.walkTokens(genericToken.tokens, callback));
}
}
- 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"));
}
+ return values;
}
- 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();
+ use(...args) {
+ const extensions = this.defaults.extensions || { renderers: {}, childTokens: {} };
+ args.forEach((pack) => {
+ const opts = { ...pack };
+ opts.async = this.defaults.async || opts.async || false;
+ if (pack.extensions) {
+ pack.extensions.forEach((ext) => {
+ if (!ext.name) {
+ throw new Error("extension name required");
}
- if (structLengthMeters) {
- if (getFeatureType(crossedWay, graph) === "railway") {
- structLengthMeters *= 2;
+ if ("renderer" in ext) {
+ const prevRenderer = extensions.renderers[ext.name];
+ 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;
}
- } 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])
- ];
+ if ("tokenizer" in ext) {
+ if (!ext.level || ext.level !== "block" && ext.level !== "inline") {
+ throw new Error("extension level must be 'block' or 'inline'");
+ }
+ const extLevel = extensions[ext.level];
+ if (extLevel) {
+ extLevel.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];
+ }
+ }
+ }
}
- function projToGeom(projPoint) {
- var lat = geoMetersToLat(projPoint[1]);
- return [
- geoMetersToLon(projPoint[0], lat),
- lat
- ];
+ if ("childTokens" in ext && ext.childTokens) {
+ extensions.childTokens[ext.name] = ext.childTokens;
}
- 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
- ]);
+ });
+ opts.extensions = extensions;
+ }
+ if (pack.renderer) {
+ const renderer = this.defaults.renderer || new _Renderer(this.defaults);
+ for (const prop in pack.renderer) {
+ if (!(prop in renderer)) {
+ throw new Error("renderer '".concat(prop, "' does not exist"));
}
- var endpointLocGetter1 = function(lengthMeters) {
- return locSphericalDistanceFromCrossingLoc(projectedAngle, lengthMeters);
+ if (prop === "options") {
+ continue;
+ }
+ const rendererProp = prop;
+ const rendererFunc = pack.renderer[rendererProp];
+ const prevRenderer = renderer[rendererProp];
+ renderer[rendererProp] = (...args2) => {
+ let ret = rendererFunc.apply(renderer, args2);
+ if (ret === false) {
+ ret = prevRenderer.apply(renderer, args2);
+ }
+ return ret || "";
};
- var endpointLocGetter2 = function(lengthMeters) {
- return locSphericalDistanceFromCrossingLoc(projectedAngle + Math.PI, lengthMeters);
+ }
+ opts.renderer = renderer;
+ }
+ if (pack.tokenizer) {
+ const tokenizer = this.defaults.tokenizer || new _Tokenizer(this.defaults);
+ for (const prop in pack.tokenizer) {
+ if (!(prop in tokenizer)) {
+ throw new Error("tokenizer '".concat(prop, "' does not exist"));
+ }
+ if (["options", "rules", "lexer"].includes(prop)) {
+ continue;
+ }
+ const tokenizerProp = prop;
+ const tokenizerFunc = pack.tokenizer[tokenizerProp];
+ const prevTokenizer = tokenizer[tokenizerProp];
+ tokenizer[tokenizerProp] = (...args2) => {
+ let ret = tokenizerFunc.apply(tokenizer, args2);
+ if (ret === false) {
+ ret = prevTokenizer.apply(tokenizer, args2);
+ }
+ return ret;
};
- 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;
- }
- }
+ }
+ opts.tokenizer = tokenizer;
+ }
+ if (pack.hooks) {
+ const hooks = this.defaults.hooks || new _Hooks();
+ for (const prop in pack.hooks) {
+ if (!(prop in hooks)) {
+ throw new Error("hook '".concat(prop, "' does not exist"));
+ }
+ if (prop === "options") {
+ continue;
+ }
+ const hooksProp = prop;
+ const hooksFunc = pack.hooks[hooksProp];
+ const prevHook = hooks[hooksProp];
+ if (_Hooks.passThroughHooks.has(prop)) {
+ hooks[hooksProp] = (arg) => {
+ if (this.defaults.async) {
+ return Promise.resolve(hooksFunc.call(hooks, arg)).then((ret2) => {
+ return prevHook.call(hooks, ret2);
});
- });
- 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";
+ const ret = hooksFunc.call(hooks, arg);
+ return prevHook.call(hooks, ret);
+ };
} else {
- var tunnelValue = "yes";
- if (getFeatureType(structureWay, graph) === "waterway") {
- tunnelValue = "culvert";
- }
- tags.tunnel = tunnelValue;
- tags.layer = "-1";
+ hooks[hooksProp] = (...args2) => {
+ let ret = hooksFunc.apply(hooks, args2);
+ if (ret === false) {
+ ret = prevHook.apply(hooks, args2);
+ }
+ return ret;
+ };
}
- graph = actionChangeTags(structureWay.id, tags)(graph);
- return graph;
+ }
+ opts.hooks = hooks;
+ }
+ if (pack.walkTokens) {
+ const walkTokens2 = this.defaults.walkTokens;
+ const packWalktokens = pack.walkTokens;
+ opts.walkTokens = function(token) {
+ let values = [];
+ values.push(packWalktokens.call(this, token));
+ if (walkTokens2) {
+ values = values.concat(walkTokens2.call(this, token));
+ }
+ return values;
};
- context2.perform(action, _t("issues.fix." + fixTitleID + ".annotation"));
- context2.enter(modeSelect(context2, resultWayIDs));
}
+ this.defaults = { ...this.defaults, ...opts };
});
+ return this;
}
- function makeConnectWaysFix(connectionTags) {
- var fixTitleID = "connect_features";
- var fixIcon = "iD-icon-crossing";
- if (connectionTags.highway === "crossing") {
- fixTitleID = "connect_using_crossing";
- fixIcon = "temaki-pedestrian";
+ setOptions(opt) {
+ this.defaults = { ...this.defaults, ...opt };
+ return this;
+ }
+ lexer(src, options2) {
+ return _Lexer.lex(src, options2 != null ? options2 : this.defaults);
+ }
+ parser(tokens, options2) {
+ return _Parser.parse(tokens, options2 != null ? options2 : this.defaults);
+ }
+ };
+ _parseMarkdown = new WeakSet();
+ parseMarkdown_fn = function(lexer2, parser3) {
+ return (src, options2) => {
+ const origOpt = { ...options2 };
+ const opt = { ...this.defaults, ...origOpt };
+ if (this.defaults.async === true && origOpt.async === false) {
+ if (!opt.silent) {
+ console.warn("marked(): The async option was set to true by an extension. The async: false option sent to parse will be ignored.");
+ }
+ opt.async = true;
}
- if (connectionTags.ford) {
- fixTitleID = "connect_using_ford";
- fixIcon = "roentgen-ford";
+ const throwError = __privateMethod(this, _onError, onError_fn).call(this, !!opt.silent, !!opt.async);
+ if (typeof src === "undefined" || src === null) {
+ return throwError(new Error("marked(): input parameter is undefined or null"));
}
- const fix = new validationIssueFix({
- icon: fixIcon,
- title: _t.append("issues.fix." + fixTitleID + ".title"),
- onClick: function(context2) {
- var loc = this.issue.loc;
- var edges = this.issue.data.edges;
- context2.perform(
- function actionConnectCrossingWays(graph) {
- var node = osmNode({ loc, tags: connectionTags });
- graph = graph.replace(node);
- var nodesToMerge = [node.id];
- var mergeThresholdInMeters = 0.75;
- edges.forEach(function(edge) {
- var edgeNodes = [graph.entity(edge[0]), graph.entity(edge[1])];
- var nearby = geoSphericalClosestNode(edgeNodes, loc);
- if ((!nearby.node.hasInterestingTags() || nearby.node.isCrossing()) && nearby.distance < mergeThresholdInMeters) {
- nodesToMerge.push(nearby.node.id);
- } else {
- graph = actionAddMidpoint({ loc, edge }, node)(graph);
+ if (typeof src !== "string") {
+ return throwError(new Error("marked(): input parameter is of type " + Object.prototype.toString.call(src) + ", string expected"));
+ }
+ if (opt.hooks) {
+ opt.hooks.options = opt;
+ }
+ if (opt.async) {
+ return Promise.resolve(opt.hooks ? opt.hooks.preprocess(src) : src).then((src2) => lexer2(src2, opt)).then((tokens) => opt.hooks ? opt.hooks.processAllTokens(tokens) : tokens).then((tokens) => opt.walkTokens ? Promise.all(this.walkTokens(tokens, opt.walkTokens)).then(() => tokens) : tokens).then((tokens) => parser3(tokens, opt)).then((html3) => opt.hooks ? opt.hooks.postprocess(html3) : html3).catch(throwError);
+ }
+ try {
+ if (opt.hooks) {
+ src = opt.hooks.preprocess(src);
+ }
+ let tokens = lexer2(src, opt);
+ if (opt.hooks) {
+ tokens = opt.hooks.processAllTokens(tokens);
+ }
+ if (opt.walkTokens) {
+ this.walkTokens(tokens, opt.walkTokens);
+ }
+ let html3 = parser3(tokens, opt);
+ if (opt.hooks) {
+ html3 = opt.hooks.postprocess(html3);
+ }
+ return html3;
+ } catch (e3) {
+ return throwError(e3);
+ }
+ };
+ };
+ _onError = new WeakSet();
+ onError_fn = function(silent, async) {
+ return (e3) => {
+ e3.message += "\nPlease report this to https://github.com/markedjs/marked.";
+ if (silent) {
+ const msg = "An error occurred:
" + escape$1(e3.message + "", true) + "
";
+ if (async) {
+ return Promise.resolve(msg);
+ }
+ return msg;
+ }
+ if (async) {
+ return Promise.reject(e3);
+ }
+ throw e3;
+ };
+ };
+ var markedInstance = new Marked();
+ function marked(src, opt) {
+ return markedInstance.parse(src, opt);
+ }
+ marked.options = marked.setOptions = function(options2) {
+ markedInstance.setOptions(options2);
+ marked.defaults = markedInstance.defaults;
+ changeDefaults(marked.defaults);
+ return marked;
+ };
+ marked.getDefaults = _getDefaults;
+ marked.defaults = _defaults;
+ marked.use = function(...args) {
+ markedInstance.use(...args);
+ marked.defaults = markedInstance.defaults;
+ changeDefaults(marked.defaults);
+ return marked;
+ };
+ marked.walkTokens = function(tokens, callback) {
+ return markedInstance.walkTokens(tokens, callback);
+ };
+ marked.parseInline = markedInstance.parseInline;
+ marked.Parser = _Parser;
+ marked.parser = _Parser.parse;
+ marked.Renderer = _Renderer;
+ marked.TextRenderer = _TextRenderer;
+ marked.Lexer = _Lexer;
+ marked.lexer = _Lexer.lex;
+ marked.Tokenizer = _Tokenizer;
+ marked.Hooks = _Hooks;
+ 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((k2) => {
+ let wanted = tiles.find((tile) => k2 === tile.id);
+ if (!wanted) {
+ abortRequest3(cache.inflightTile[k2]);
+ delete cache.inflightTile[k2];
+ }
+ });
+ }
+ function encodeIssueRtree3(d2) {
+ return { minX: d2.loc[0], minY: d2.loc[1], maxX: d2.loc[0], maxY: d2.loc[1], data: d2 };
+ }
+ function updateRtree3(item, replace) {
+ _cache3.rtree.remove(item, (a2, b2) => a2.data.id === b2.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((d2) => {
+ _osmoseData = d2.osmose;
+ _osmoseData.items = Object.keys(d2.osmose.icons).map((s2) => s2.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 [x2, y2, z2] = tile.xyz;
+ let url = "".concat(_osmoseUrlRoot, "/issues/").concat(z2, "/").concat(x2, "/").concat(y2, ".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 = "".concat(item, "-").concat(cl);
+ if (itemType in _osmoseData.icons) {
+ let loc = issue.geometry.coordinates;
+ loc = preventCoincident2(loc);
+ let d2 = new QAItem(loc, this, itemType, id2, { item });
+ if (item === 8300 || item === 8360) {
+ d2.elems = [];
}
- });
- if (nodesToMerge.length > 1) {
- graph = actionMergeNodes(nodesToMerge, loc)(graph);
+ _cache3.data[d2.id] = d2;
+ _cache3.rtree.insert(encodeIssueRtree3(d2));
}
- return graph;
- },
- _t("issues.fix.connect_crossing_features.annotation")
- );
- }
+ });
+ }
+ dispatch4.call("loaded");
+ }).catch(() => {
+ delete _cache3.inflightTile[tile.id];
+ _cache3.loadedTile[tile.id] = true;
+ });
});
- fix._connectionTags = connectionTags;
- return fix;
- }
- 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)
+ },
+ loadIssueDetail(issue) {
+ if (issue.elems !== void 0) {
+ return Promise.resolve(issue);
+ }
+ const url = "".concat(_osmoseUrlRoot, "/issue/").concat(issue.id, "?langs=").concat(_mainLocalizer.localeCode());
+ const cacheDetails = (data) => {
+ issue.elems = data.elems.map((e3) => e3.type.substring(0, 1) + e3.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 (".concat(itemType, ") had unexpected data"));
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")
- );
+ 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 = "".concat(_osmoseUrlRoot, "/items/").concat(item, "/class/").concat(cl, "?langs=").concat(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 = "".concat(_osmoseUrlRoot, "/issue/").concat(issue.id, "/").concat(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((d2) => d2.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/".concat(item.id);
}
- validation.type = type2;
- return validation;
- }
+ };
- // modules/behavior/draw_way.js
- function behaviorDrawWay(context, wayID, mode, startGraph) {
- const keybinding = utilKeybinding("drawWay");
- var dispatch14 = 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);
+ // 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 = "".concat(baseTileUrl, "/mly_map_feature_point/2/{z}/{x}/{y}?access_token=").concat(accessToken);
+ var tileUrl = "".concat(baseTileUrl, "/mly1_public/2/{z}/{x}/{y}?access_token=").concat(accessToken);
+ var trafficSignTileUrl = "".concat(baseTileUrl, "/mly_map_feature_traffic_sign/2/{z}/{x}/{y}?access_token=").concat(accessToken);
+ var viewercss = "mapillary-js/mapillary.css";
+ var viewerjs = "mapillary-js/mapillary.js";
+ var minZoom = 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 tiler9 = utilTiler().zoomExtent([minZoom, maxZoom2]).skipNullIsland(true);
+ const tiles = tiler9.getTiles(projection2);
+ tiles.forEach(function(tile) {
+ loadTile(which, url, tile);
+ });
+ }
+ function loadTile(which, url, tile) {
+ const cache = _mlyCache.requests;
+ const tileId = "".concat(tile.id, "-").concat(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, i3, feature3, loc, d2;
+ if (vectorTile.layers.hasOwnProperty("image")) {
+ features = [];
+ cache = _mlyCache.images;
+ layer = vectorTile.layers.image;
+ for (i3 = 0; i3 < layer.length; i3++) {
+ feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+ loc = feature3.geometry.coordinates;
+ d2 = {
+ 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[d2.id] = d2;
+ features.push({
+ minX: loc[0],
+ minY: loc[1],
+ maxX: loc[0],
+ maxY: loc[1],
+ data: d2
+ });
+ }
+ if (cache.rtree) {
+ cache.rtree.load(features);
}
}
- function keyup(d3_event) {
- if (d3_event.keyCode === utilKeybinding.modifierCodes.alt) {
- if (context.surface().classed("nope-suppressed")) {
- context.surface().classed("nope", true);
+ if (vectorTile.layers.hasOwnProperty("sequence")) {
+ features = [];
+ cache = _mlyCache.sequences;
+ layer = vectorTile.layers.sequence;
+ for (i3 = 0; i3 < layer.length; i3++) {
+ feature3 = layer.feature(i3).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];
}
- context.surface().classed("nope-suppressed", false).classed("nope-disabled", false);
}
}
- function allowsVertex(d2) {
- return d2.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d2, 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;
- }
+ if (vectorTile.layers.hasOwnProperty("point")) {
+ features = [];
+ cache = _mlyCache[which];
+ layer = vectorTile.layers.point;
+ for (i3 = 0; i3 < layer.length; i3++) {
+ feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+ loc = feature3.geometry.coordinates;
+ d2 = {
+ 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: d2
+ });
+ }
+ if (cache.rtree) {
+ cache.rtree.load(features);
}
- 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);
+ if (vectorTile.layers.hasOwnProperty("traffic_sign")) {
+ features = [];
+ cache = _mlyCache[which];
+ layer = vectorTile.layers.traffic_sign;
+ for (i3 = 0; i3 < layer.length; i3++) {
+ feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
+ loc = feature3.geometry.coordinates;
+ d2 = {
+ 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: d2
+ });
+ }
+ if (cache.rtree) {
+ cache.rtree.load(features);
}
}
- 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();
+ }
+ 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 z2 = geoScaleToZoom(projection2.scale());
+ const z22 = Math.ceil(z2 * 2) / 2 + 2.5;
+ const tiler9 = utilTiler().zoomExtent([z22, z22]);
+ return tiler9.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(d2) {
+ return d2.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(d2) {
+ if (d2.data.sequence_id) {
+ sequenceIds[d2.data.sequence_id] = true;
}
- } else {
- if (parentWay.isClosed()) {
- if (nodes.length < 3)
- return false;
- if (_drawNode)
- nodes.splice(-2, 1);
- testNode = nodes[nodes.length - 2];
+ });
+ 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 {
- return false;
+ delete hash.photo;
}
+ window.location.replace("#" + utilQsString(hash, true));
}
- 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;
+ },
+ // Highlight the detection in the viewer that is related to the clicked map feature
+ highlightDetection: function(detection) {
+ if (detection) {
+ _mlyHighlightedDetection = detection.id;
}
- context.perform(actionNoop());
- context.pop(1);
- context.resumeChangeDispatch();
- context.enter(nextMode);
- }
- function setActiveElements() {
- if (!_drawNode)
+ return this;
+ },
+ // Initialize image viewer (Mapillar JS)
+ initViewer: function(context) {
+ const that = this;
+ if (!window.mapillary)
return;
- context.surface().selectAll("." + _drawNode.id).classed("active", true);
- }
- function resetToStartGraph() {
- while (context.graph() !== startGraph) {
- context.pop();
+ 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
+ };
}
- }
- 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];
+ _mlyViewer = new mapillary.Viewer(opts);
+ _mlyViewer.on("image", imageChanged);
+ _mlyViewer.on("bearing", bearingChanged);
+ if (_mlyViewerFilter) {
+ _mlyViewer.setFilter(_mlyViewerFilter);
}
- _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();
+ 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, "".concat(apiUrl, "/").concat(image.id, "/detections?access_token=").concat(accessToken, "&fields=id,image,geometry,value"));
+ }
+ dispatch5.call("imageChanged");
}
- _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(d2, loc, doAdd) {
- if (_drawNode) {
- context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
- _drawNode = context.entity(_drawNode.id);
- } else {
- createDrawNode(loc);
+ function bearingChanged(e3) {
+ dispatch5.call("bearingChanged", void 0, e3);
}
- checkGeometry(
- true
- /* includeDrawNode */
- );
- if (d2 && d2.properties && d2.properties.nope || context.surface().classed("nope")) {
- if (!_pointerHasMoved) {
- removeDrawNode();
- }
- dispatch14.call("rejectedSelfIntersection", this);
- return;
+ },
+ // Move to an image
+ selectImage: function(context, imageId) {
+ if (_mlyViewer && imageId) {
+ _mlyViewer.moveTo(imageId).catch(function(e3) {
+ console.error("mly3", e3);
+ });
}
- context.pauseChangeDispatch();
- doAdd();
- _didResolveTempEdit = true;
- context.resumeChangeDispatch();
- context.enter(mode);
- }
- drawWay.add = function(loc, d2) {
- attemptAdd(d2, loc, function() {
+ 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("".concat(apiUrl, "/").concat(id2, "/detections?access_token=").concat(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(d2) {
+ return d2.sequence_id === selectedSequenceId || d2.id === hoveredImageId;
+ }).classed("hovered", function(d2) {
+ return d2.id === hoveredImageId;
});
- };
- drawWay.addWay = function(loc, edge, d2) {
- attemptAdd(d2, loc, function() {
- context.replace(
- actionAddMidpoint({ loc, edge }, _drawNode),
- _annotation
- );
+ context.container().selectAll(".layer-mapillary .sequence").classed("highlighted", function(d2) {
+ return d2.properties.id === hoveredSequenceId;
+ }).classed("currentView", function(d2) {
+ return d2.properties.id === selectedSequenceId;
});
- };
- drawWay.addNode = function(node, d2) {
- if (node.id === _headNodeID || // or the first node when drawing an area
- _origWay.isClosed() && node.id === _origWay.first()) {
- drawWay.finish();
+ return this;
+ },
+ // Get detections for the current image and shows them in the image viewer
+ updateDetections: function(imageId, url) {
+ if (!_mlyViewer || _mlyFallback)
return;
- }
- attemptAdd(d2, 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)
+ if (!imageId)
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((w2) => w2.id !== wayID);
- const secondLastNodesParents = historyGraph.parentWays(historyGraph.entity(secondLastNodeId)).filter((w2) => w2.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((n3) => n3.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 }
+ 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] || []);
});
- } 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")) {
- dispatch14.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, dispatch14, "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")
- }));
+ function showDetections(detections) {
+ const tagComponent = _mlyViewer.getComponent("tag");
+ detections.forEach(function(data) {
+ const tag2 = makeTag(data);
+ if (tag2) {
+ tagComponent.add([tag2]);
}
- 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);
- }
- });
+ function makeTag(data) {
+ const valueParts = data.value.split("--");
+ if (!valueParts.length)
+ return;
+ let tag2;
+ let text;
+ let color2 = 16777215;
+ if (_mlyHighlightedDetection === data.id) {
+ color2 = 16776960;
+ 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 (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;
+ var decodedGeometry = window.atob(data.geometry);
+ var uintArray = new Uint8Array(decodedGeometry.length);
+ for (var i3 = 0; i3 < decodedGeometry.length; i3++) {
+ uintArray[i3] = decodedGeometry.charCodeAt(i3);
}
- while (waysToCheck.length) {
- var wayToCheck = waysToCheck.pop();
- var childNodes = graph.childNodes(wayToCheck);
- for (var i3 in childNodes) {
- var vertex = childNodes[i3];
- if (isConnectedVertex(vertex)) {
- return null;
- }
- if (isRoutableNode(vertex)) {
- routingIsland.add(vertex);
- }
- queueParentWays(vertex);
+ 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]));
+ tag2 = new mapillary.OutlineTag(
+ data.id,
+ new mapillary.PolygonGeometry(polygon2[0]),
+ {
+ text,
+ textColor: color2,
+ lineColor: color2,
+ lineWidth: 2,
+ fillColor: color2,
+ fillOpacity: 0.3
}
- }
- return routingIsland;
+ );
+ return tag2;
+ }
+ },
+ // 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);
}
- 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;
+ if (this.subtype) {
+ parts.push(this.subtype);
}
- function isRoutableNode(node) {
- if (node.tags.highway === "elevator")
- return true;
- return false;
+ if (this.entityIds) {
+ var entityKeys = this.entityIds.slice().sort();
+ parts.push.apply(parts, entityKeys);
}
- function isRoutableWay(way, ignoreInnerWays) {
- if (isTaggedAsHighway(way) || way.tags.route === "ferry")
- return true;
- return graph.parentRelations(way).some(function(parentRelation) {
- if (parentRelation.tags.type === "route" && parentRelation.tags.route === "ferry")
- return true;
- if (parentRelation.isMultipolygon() && isTaggedAsHighway(parentRelation) && (!ignoreInnerWays || parentRelation.memberById(way.id).role !== "inner"))
- return true;
- return false;
- });
+ return parts.join(":");
+ }
+ function generateKey() {
+ return this.id + ":" + Date.now().toString();
+ }
+ this.extent = function(resolver) {
+ if (this.loc) {
+ return geoExtent(this.loc);
}
- 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)
- );
- }
- });
+ if (this.entityIds && this.entityIds.length) {
+ return this.entityIds.reduce(function(extent, entityId) {
+ return extent.extend(resolver.entity(entityId).extent(resolver));
+ }, geoExtent());
}
+ return null;
};
- 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"));
+ 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);
+ }
+ }));
}
- if (entity.tags.email) {
- var emails = entity.tags.email.split(";").map(function(s2) {
- return s2.trim();
- }).filter(function(x2) {
- return !isValidEmail(x2);
- });
- 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" : ""
- }));
+ fixes.forEach(function(fix) {
+ fix.id = fix.title.stringId;
+ fix.issue = issue;
+ if (fix.autoArgs) {
+ issue.autoFix = fix;
}
- }
- return issues;
+ });
+ return fixes;
};
- validation.type = type2;
- return validation;
+ }
+ 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/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"));
+ // modules/services/maprules.js
+ var buildRuleChecks = function() {
+ return {
+ equals: function(equals) {
+ return function(tags) {
+ return Object.keys(equals).every(function(k2) {
+ return equals[k2] === tags[k2];
+ });
+ };
+ },
+ notEquals: function(notEquals) {
+ return function(tags) {
+ return Object.keys(notEquals).some(function(k2) {
+ return notEquals[k2] !== tags[k2];
+ });
+ };
+ },
+ 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]);
+ };
}
};
- 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;
+ };
+ var buildLineKeys = function() {
+ return {
+ highway: {
+ rest_area: true,
+ services: true
+ },
+ railway: {
+ roundhouse: true,
+ station: true,
+ traverser: true,
+ turntable: true,
+ wash: true
}
- 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;
- }
+ };
+ };
+ 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 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 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 isEqual4 = /equals/gi.test(key);
+ if (isRegex || isEqual4) {
+ Object.keys(selector[key]).forEach(function(selectorKey) {
+ values = isEqual4 ? [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 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 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";
}
- 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 [];
+ for (var key in tagMap) {
+ if (key in _areaKeys && !keyValueDoesNotImplyArea(key)) {
+ return "area";
}
- var placement = isFirst ? "start" : "end", messageID = wayType + ".", referenceID = wayType + ".";
- if (wayType === "waterway") {
- messageID += "connected." + placement;
- referenceID += "connected";
- } else {
- messageID += placement;
- referenceID += placement;
+ if (key in _lineKeys && keyValueImpliesLine(key)) {
+ return "area";
}
- 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"));
- };
+ }
+ 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]
+ }));
+ }
}
- }
- };
- 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)
- );
+ };
+ this._validationRules.push(rule);
+ },
+ clearRules: function() {
+ this._validationRules = [];
+ },
+ // returns validationRules...
+ validationRules: function() {
+ return this._validationRules;
+ },
+ // returns ruleChecks
+ ruleChecks: function() {
+ return this._ruleChecks;
}
- 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/services/nominatim.js
+ var import_rbush7 = __toESM(require_rbush_min());
- // 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 i3 = 0; i3 < rules.length; i3++) {
- var rule = rules[i3];
- rule.findIssues(entity, graph, issues);
+ // 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 h2 = head.entities[id2];
+ var b2 = base.entities[id2];
+ if (h2 === b2)
+ return;
+ if (_changes[id2])
+ return;
+ if (!h2 && b2) {
+ _changes[id2] = { base: b2, head: h2 };
+ _didChange.deletion = true;
+ return;
}
- 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;
+ if (h2 && !b2) {
+ _changes[id2] = { base: b2, head: h2 };
+ _didChange.addition = true;
+ return;
}
- var asLine = _mainPresetIndex.matchTags(tagSuggestingArea, "line");
- var asArea = _mainPresetIndex.matchTags(tagSuggestingArea, "area");
- if (asLine && asArea && (0, import_fast_deep_equal4.default)(asLine.tags, asArea.tags)) {
- return null;
+ if (h2 && b2) {
+ if (h2.members && b2.members && !(0, import_fast_deep_equal3.default)(h2.members, b2.members)) {
+ _changes[id2] = { base: b2, head: h2 };
+ _didChange.geometry = true;
+ _didChange.properties = true;
+ return;
+ }
+ if (h2.loc && b2.loc && !geoVecEqual(h2.loc, b2.loc)) {
+ _changes[id2] = { base: b2, head: h2 };
+ _didChange.geometry = true;
+ }
+ if (h2.nodes && b2.nodes && !(0, import_fast_deep_equal3.default)(h2.nodes, b2.nodes)) {
+ _changes[id2] = { base: b2, head: h2 };
+ _didChange.geometry = true;
+ }
+ if (h2.tags && b2.tags && !(0, import_fast_deep_equal3.default)(h2.tags, b2.tags)) {
+ _changes[id2] = { base: b2, head: h2 };
+ _didChange.properties = true;
+ }
}
- 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")
- );
- };
+ function load() {
+ var ids = utilArrayUniq(Object.keys(head.entities).concat(Object.keys(base.entities)));
+ for (var i3 = 0; i3 < ids.length; i3++) {
+ checkEntityID(ids[i3]);
+ }
+ }
+ load();
+ _diff.length = function length2() {
+ 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 h2 = _changes[id2].head;
+ var b2 = _changes[id2].base;
+ var entity = h2 || b2;
+ if (includeRelMembers && entity.type === "relation") {
+ var mh = h2 ? h2.members.map(function(m2) {
+ return m2.id;
+ }) : [];
+ var mb = b2 ? b2.members.map(function(m2) {
+ return m2.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 keys2 = Object.keys(_changes);
+ for (var i3 = 0; i3 < keys2.length; i3++) {
+ var change = _changes[keys2[i3]];
+ 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");
}
}
- 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")
- );
+ return Object.values(relevant);
+ function addEntity(entity, graph, changeType) {
+ relevant[entity.id] = {
+ entity,
+ graph,
+ changeType
};
}
- }
- 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;
+ 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");
+ }
}
}
- 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
- }));
+ };
+ _diff.complete = function complete(extent) {
+ var result = {};
+ var id2, change;
+ for (id2 in _changes) {
+ change = _changes[id2];
+ var h2 = change.head;
+ var b2 = change.base;
+ var entity = h2 || b2;
+ var i3;
+ if (extent && (!h2 || !h2.intersects(extent, head)) && (!b2 || !b2.intersects(extent, base))) {
+ continue;
+ }
+ result[id2] = h2;
+ if (entity.type === "way") {
+ var nh = h2 ? h2.nodes : [];
+ var nb = b2 ? b2.nodes : [];
+ var diff;
+ diff = utilArrayDifference(nh, nb);
+ for (i3 = 0; i3 < diff.length; i3++) {
+ result[diff[i3]] = head.hasEntity(diff[i3]);
+ }
+ diff = utilArrayDifference(nb, nh);
+ for (i3 = 0; i3 < diff.length; i3++) {
+ result[diff[i3]] = head.hasEntity(diff[i3]);
}
- 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"));
+ if (entity.type === "relation" && entity.isMultipolygon()) {
+ var mh = h2 ? h2.members.map(function(m2) {
+ return m2.id;
+ }) : [];
+ var mb = b2 ? b2.members.map(function(m2) {
+ return m2.id;
+ }) : [];
+ var ids = utilArrayUnion(mh, mb);
+ for (i3 = 0; i3 < ids.length; i3++) {
+ var member = head.hasEntity(ids[i3]);
+ if (!member)
+ continue;
+ if (extent && !member.intersects(extent, head))
+ continue;
+ result[ids[i3]] = member;
+ }
+ }
+ addParents(head.parentWays(entity), result);
+ addParents(head.parentRelations(entity), result);
+ }
+ return result;
+ function addParents(parents, result2) {
+ for (var i4 = 0; i4 < parents.length; i4++) {
+ var parent = parents[i4];
+ if (parent.id in result2)
+ continue;
+ result2[parent.id] = parent;
+ addParents(head.parentRelations(parent), result2);
+ }
}
+ };
+ return _diff;
+ }
+
+ // modules/core/tree.js
+ var import_rbush5 = __toESM(require_rbush_min());
+ function coreTree(head) {
+ var _rtree = new import_rbush5.default();
+ var _bboxes = {};
+ var _segmentsRTree = new import_rbush5.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 vertexPointIssue(entity, graph) {
- if (entity.type !== "node")
- return null;
- if (Object.keys(entity.tags).length === 0)
- return null;
- if (entity.isOnAddressLine(graph))
+ function segmentBBox(segment) {
+ var extent = segment.extent(head);
+ if (!extent)
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
+ 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];
}
- 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] === "*";
+ 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 (!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
+ 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);
});
}
- 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;
+ tree.rebase = function(entities, force) {
+ var insertions = {};
+ for (var i3 = 0; i3 < entities.length; i3++) {
+ var entity = entities[i3];
+ if (!entity.visible)
+ continue;
+ if (head.entities.hasOwnProperty(entity.id) || _bboxes[entity.id]) {
+ if (!force) {
+ continue;
+ } else if (_bboxes[entity.id]) {
+ removeEntity(entity);
}
- context2.perform(
- actionChangeTags(entityId2, tags2),
- _t("issues.fix.convert_to_line.annotation")
- );
- };
+ }
+ insertions[entity.id] = entity;
+ updateParents(entity, insertions, {});
}
- 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()]));
- };
+ 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);
+ });
}
- 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 i3 in sequences) {
- var sequence = sequences[i3];
- 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()
+ if (changed.geometry) {
+ diff.modified().forEach(function(entity) {
+ removeEntity(entity);
+ insertions[entity.id] = entity;
+ updateParents(entity, insertions, {});
});
- 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"));
+ if (changed.addition) {
+ diff.created().forEach(function(entity) {
+ insertions[entity.id] = entity;
+ });
}
+ loadEntities(Object.values(insertions));
}
- 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);
+ tree.intersects = function(extent, graph) {
+ updateToGraph(graph);
+ return _rtree.search(extent.bbox()).map(function(bbox2) {
+ return graph.entity(bbox2.id);
+ });
};
- validation.type = type2;
- return validation;
+ tree.waySegments = function(extent, graph) {
+ updateToGraph(graph);
+ return _segmentsRTree.search(extent.bbox()).map(function(bbox2) {
+ return bbox2.segment;
+ });
+ };
+ return tree;
}
- // 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));
- }
- });
+ // 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();
}
- return issues;
+ }
+ 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;
};
- function isMissingRole(member) {
- return !member.role || !member.role.trim().length;
+ 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 dispatch14 = 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, t2) {
+ 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 i3 = 0; i3 < actions.length; i3++) {
+ graph = actions[i3](graph, t2);
+ }
+ return {
+ graph,
+ annotation,
+ imageryUsed: _imageryUsed,
+ photoOverlaysUsed: _photoOverlaysUsed,
+ transform: context.projection.transform(),
+ selectedIDs: context.selectedIDs()
+ };
}
- 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 _perform(args, t2) {
+ var previous = _stack[_index].graph;
+ _stack = _stack.slice(0, _index + 1);
+ var actionResult = _act(args, t2);
+ _stack.push(actionResult);
+ _index++;
+ return change(previous);
+ }
+ function _replace(args, t2) {
+ var previous = _stack[_index].graph;
+ var actionResult = _act(args, t2);
+ _stack[_index] = actionResult;
+ return change(previous);
+ }
+ function _overwrite(args, t2) {
+ var previous = _stack[_index].graph;
+ if (_index > 0) {
+ _index--;
+ _stack.pop();
+ }
+ _stack = _stack.slice(0, _index + 1);
+ var actionResult = _act(args, t2);
+ _stack.push(actionResult);
+ _index++;
+ return change(previous);
+ }
+ function change(previous) {
+ var difference2 = coreDifference(previous, history.graph());
+ if (!_pausedGraph) {
+ dispatch14.call("change", this, difference2);
+ }
+ return difference2;
+ }
+ function getKey(n3) {
+ return "iD_" + window.location.origin + "_" + n3;
+ }
+ 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);
+ dispatch14.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(t2) {
+ if (t2 < 1)
+ _overwrite([action0], t2);
+ };
+ }).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(n3) {
+ select_default2(document).interrupt("history.perform");
+ var previous = _stack[_index].graph;
+ if (isNaN(+n3) || +n3 < 0) {
+ n3 = 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
- })
- );
+ while (n3-- > 0 && _index > 0) {
+ _index--;
+ _stack.pop();
}
- });
- }
- 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(k2) {
- if (k2 === "area" || !osmIsInterestingTag(k2))
- return false;
- return !onlyAttributeKeys.some(function(attributeKey) {
- return k2 === attributeKey || k2.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";
+ 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;
}
- }
- 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();
- }
- };
+ dispatch14.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;
+ dispatch14.call("redone", this, _stack[_index], previousStack);
+ break;
}
- 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((d2) => _dataDeprecated = d2).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);
+ 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 i3 = _index;
+ while (i3 >= 0) {
+ if (_stack[i3].annotation)
+ return _stack[i3].annotation;
+ i3--;
+ }
+ },
+ redoAnnotation: function() {
+ var i3 = _index + 1;
+ while (i3 <= _stack.length - 1) {
+ if (_stack[i3].annotation)
+ return _stack[i3].annotation;
+ i3++;
+ }
+ },
+ // 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 difference2 = coreDifference(base, head);
+ return {
+ modified: difference2.modified(),
+ created: difference2.created(),
+ deleted: difference2.deleted()
+ };
+ },
+ hasChanges: function() {
+ return this.difference().length() > 0;
+ },
+ imageryUsed: function(sources) {
+ if (sources) {
+ _imageryUsed = sources;
+ return history;
+ } else {
+ var s2 = /* @__PURE__ */ new Set();
+ _stack.slice(1, _index + 1).forEach(function(state) {
+ state.imageryUsed.forEach(function(source) {
+ if (source !== "Custom") {
+ s2.add(source);
+ }
+ });
});
- entity = graph.entity(entity.id);
+ return Array.from(s2);
}
- }
- let newTags = Object.assign({}, entity.tags);
- if (preset.tags !== preset.addTags) {
- Object.keys(preset.addTags).forEach((k2) => {
- if (!newTags[k2]) {
- if (preset.addTags[k2] === "*") {
- newTags[k2] = "yes";
- } else {
- newTags[k2] = preset.addTags[k2];
+ },
+ photoOverlaysUsed: function(sources) {
+ if (sources) {
+ _photoOverlaysUsed = sources;
+ return history;
+ } else {
+ var s2 = /* @__PURE__ */ new Set();
+ _stack.slice(1, _index + 1).forEach(function(state) {
+ if (state.photoOverlaysUsed && Array.isArray(state.photoOverlaysUsed)) {
+ state.photoOverlaysUsed.forEach(function(photoOverlay) {
+ s2.add(photoOverlay);
+ });
}
+ });
+ return Array.from(s2);
+ }
+ },
+ // 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 = {};
+ }
+ dispatch14.call("reset");
+ dispatch14.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];
}
});
- }
- 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";
+ 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;
}
- }
- let issues = [];
- issues.provisional = _waitingForDeprecated || waitingForNsi;
- const tagDiff = utilTagDiff(oldTags, newTags);
- if (!tagDiff.length)
- return issues;
- const isOnlyAddingTags = tagDiff.every((d2) => d2.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"));
+ },
+ toJSON: function() {
+ if (!this.hasChanges())
+ return;
+ var allEntities = {};
+ var baseEntities = {};
+ var base = _stack[0];
+ var s2 = _stack.map(function(i3) {
+ var modified = [];
+ var deleted = [];
+ Object.keys(i3.graph.entities).forEach(function(id2) {
+ var entity = i3.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 x2 = {};
+ if (modified.length)
+ x2.modified = modified;
+ if (deleted.length)
+ x2.deleted = deleted;
+ if (i3.imageryUsed)
+ x2.imageryUsed = i3.imageryUsed;
+ if (i3.photoOverlaysUsed)
+ x2.photoOverlaysUsed = i3.photoOverlaysUsed;
+ if (i3.annotation)
+ x2.annotation = i3.annotation;
+ if (i3.transform)
+ x2.transform = i3.transform;
+ if (i3.selectedIDs)
+ x2.selectedIDs = i3.selectedIDs;
+ return x2;
+ });
+ return JSON.stringify({
+ version: 3,
+ entities: Object.values(allEntities),
+ baseEntities: Object.values(baseEntities),
+ stack: s2,
+ nextIDs: osmEntity.id.next,
+ index: _index,
+ // note the time the changes were saved
+ timestamp: (/* @__PURE__ */ new Date()).getTime()
+ });
+ },
+ fromJSON: function(json, loadChildNodes) {
+ var h2 = JSON.parse(json);
+ var loadComplete = true;
+ osmEntity.id.next = h2.nextIDs;
+ _index = h2.index;
+ if (h2.version === 2 || h2.version === 3) {
+ var allEntities = {};
+ h2.entities.forEach(function(entity) {
+ allEntities[osmEntity.key(entity)] = osmEntity(entity);
+ });
+ if (h2.version === 3) {
+ var baseEntities = h2.baseEntities.map(function(d2) {
+ return osmEntity(d2);
+ });
+ 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(e3) {
+ return e3.type === "way";
+ });
+ var nodeIDs = baseWays.reduce(function(acc, way) {
+ return utilArrayUnion(acc, way.nodes);
+ }, []);
+ var missing = nodeIDs.filter(function(n3) {
+ return !_stack[0].graph.hasEntity(n3);
+ });
+ 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);
+ dispatch14.call("change");
+ dispatch14.call("restore", this);
+ }
+ };
+ osm.loadMultiple(missing, childNodesLoaded);
}
- })
- ];
- 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;
+ _stack = h2.stack.map(function(d2) {
+ var entities = {}, entity;
+ if (d2.modified) {
+ d2.modified.forEach(function(key) {
+ entity = allEntities[key];
+ entities[entity.id] = entity;
+ });
+ }
+ if (d2.deleted) {
+ d2.deleted.forEach(function(id2) {
+ entities[id2] = void 0;
+ });
+ }
+ return {
+ graph: coreGraph(_stack[0].graph).load(entities),
+ annotation: d2.annotation,
+ imageryUsed: d2.imageryUsed,
+ photoOverlaysUsed: d2.photoOverlaysUsed,
+ transform: d2.transform,
+ selectedIDs: d2.selectedIDs
+ };
+ });
+ } else {
+ _stack = h2.stack.map(function(d2) {
+ var entities = {};
+ for (var i3 in d2.entities) {
+ var entity = d2.entities[i3];
+ entities[i3] = entity === "undefined" ? void 0 : osmEntity(entity);
+ }
+ d2.graph = coreGraph(_stack[0].graph).load(entities);
+ return d2;
+ });
}
- }));
- 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];
+ var transform2 = _stack[_index].transform;
+ if (transform2) {
+ context.map().transformEase(transform2, 0);
}
- 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";
+ if (loadComplete) {
+ dispatch14.call("change");
+ dispatch14.call("restore", this);
}
- 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", (d2) => {
- let klass = d2.type === "+" ? "add" : "remove";
- return `tagDiff-cell tagDiff-cell-${klass}`;
- }).html((d2) => d2.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"));
- }
- })
- ];
+ 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)
+ dispatch14.call("storage_error");
}
- })];
- 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 k2 in tags) {
- if (publicKeys[k2])
- return [];
- if (!personalTags[k2]) {
- keepTags[k2] = tags[k2];
+ 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);
}
- }
- 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"));
- }
- })
- ];
+ 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);
}
- })];
- 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(d2) {
- var klass = d2.type === "+" ? "add" : "remove";
- return "tagDiff-cell tagDiff-cell-" + klass;
- }).html(function(d2) {
- return d2.display;
- });
- }
+ },
+ _getKey: getKey
};
- validation.type = type2;
- return validation;
+ history.reset();
+ return utilRebind(history, dispatch14, "on");
}
- // 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 i3 = 0; i3 < keysToTestForGenericValues.length; i3++) {
- let key = keysToTestForGenericValues[i3];
- 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;
+ // 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,
+ validationMutuallyExclusiveTags: () => validationMutuallyExclusiveTags,
+ 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 isGenericName(name, tags) {
- name = name.toLowerCase();
- return nameMatchesRawTag(name, tags) || isGenericMatchInNsi(tags);
+ function isTaggedAsNotContinuing(node) {
+ return node.tags.noexit === "yes" || node.tags.amenity === "parking_entrance" || node.tags.entrance && node.tags.entrance !== "no";
}
- 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")
- );
- }
- })
- ];
- }
+ 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
+ }));
});
- 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")
+ 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.generic_name.reference"));
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.almost_junction.highway-highway.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 m2 = key.match(/^name(?:(?::)([a-zA-Z_-]+))?$/);
- if (!m2)
- continue;
- const langCode = m2.length >= 2 ? m2[1] : null;
- const value = tags[key];
- if (notNames2.length) {
- for (let i3 in notNames2) {
- const notName = notNames2[i3];
- if (notName && value === notName) {
- issues.push(makeIncorrectNameIssue(entity.id, key, value, langCode));
- continue;
+ 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;
}
}
}
- if (isGenericName(value, tags)) {
- issues.provisional = _waitingForNsi;
- issues.push(makeGenericNameIssue(entity.id, key, value, langCode));
- }
+ return true;
}
- 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";
- });
+ 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);
});
- });
- 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 results;
}
- 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 findNearbyEndNodes(node, way) {
+ return [
+ way.nodes[0],
+ way.nodes[way.nodes.length - 1]
+ ].map((d2) => graph.entity(d2)).filter((d2) => {
+ return d2.id !== node.id && geoSphericalDistance(node.loc, d2.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 t2 = EXTEND_TH_METERS / edgeLen + 1;
+ const extTipLoc = geoVecInterp(midNode.loc, tipNode.loc, t2);
+ const segmentInfos = tree.waySegments(queryExtent, graph);
+ for (let i3 = 0; i3 < segmentInfos.length; i3++) {
+ let segmentInfo = segmentInfos[i3];
+ 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
+ };
+ }
}
- })];
- function showReference(selection2) {
- selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.unsquare_way.buildings.reference"));
+ return null;
}
};
validation.type = type2;
return validation;
}
- // modules/core/validator.js
- function coreValidator(context) {
- let dispatch14 = dispatch_default("validated", "focusedIssue");
- let validator = utilRebind({}, dispatch14, "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((k2) => _disabledRules[k2] = true);
+ // 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);
}
- };
- 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();
- dispatch14.call("validated");
- };
- validator.revalidateUnsquare = () => {
- revalidateUnsquare(_headCache);
- revalidateUnsquare(_baseCache);
- dispatch14.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);
- });
+ return [];
+ function getIssuesForNode(node) {
+ var parentWays = graph.parentWays(node);
+ if (parentWays.length) {
+ return getIssuesForVertex(node, parentWays);
+ } else {
+ return getIssuesForDetachedPoint(node);
+ }
}
- if (opts.what === "all") {
- Object.values(_baseCache.issuesByIssueID).forEach((issue) => {
- if (!filter2(issue))
- return;
- seen.add(issue.id);
- results.push(issue);
- });
+ 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 i3 in parentRelations) {
+ var relation = parentRelations[i3];
+ 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";
}
- 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))
+ function shouldCheckWay(way) {
+ if (way.nodes.length <= 2 || way.isClosed() && way.nodes.length <= 4)
return false;
- if ((issue.entityIds || []).some((id2) => !context.hasEntity(id2)))
+ var bbox2 = way.extent(graph).bbox();
+ var hypotenuseMeters = geoSphericalDistance([bbox2.minX, bbox2.minY], [bbox2.maxX, bbox2.maxY]);
+ if (hypotenuseMeters < 1.5)
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();
+ function getIssuesForWay(way) {
+ if (!shouldCheckWay(way))
+ return [];
+ var issues = [], nodes = graph.childNodes(way);
+ for (var i3 = 0; i3 < nodes.length - 1; i3++) {
+ var node1 = nodes[i3];
+ var node2 = nodes[i3 + 1];
+ var issue = getWayIssueIfAny(node1, node2, way);
+ if (issue)
+ issues.push(issue);
+ }
+ return issues;
}
- 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();
+ function getIssuesForVertex(node, parentWays) {
+ var issues = [];
+ function checkForCloseness(node1, node2, way) {
+ var issue = getWayIssueIfAny(node1, node2, way);
+ if (issue)
+ issues.push(issue);
+ }
+ for (var i3 = 0; i3 < parentWays.length; i3++) {
+ var parentWay = parentWays[i3];
+ 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);
+ }
}
- }
- if (nodeID) {
- focusCenter = graph.entity(nodeID).loc;
}
}
+ return issues;
}
- if (focusCenter) {
- const setZoom = Math.max(context.map().zoom(), 19);
- context.map().unobscuredCenterZoomEase(focusCenter, setZoom);
+ 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;
}
- if (selectID) {
- window.setTimeout(() => {
- context.enter(modeSelect(context, [selectID]));
- dispatch14.call("focusedIssue", this, issue);
- }, 250);
+ 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);
+ }
}
- };
- 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;
+ function getWayIssueIfAny(node1, node2, way) {
+ if (node1.id === node2.id || node1.hasInterestingTags() && node2.hasInterestingTags()) {
+ return null;
}
- 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;
+ 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);
}
- });
- };
- 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 = (keys2) => {
- _disabledRules = {};
- keys2.forEach((k2) => _disabledRules[k2] = 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;
- dispatch14.call("validated");
- return Promise.resolve();
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/crossing_ways.js
+ var import_lodash3 = __toESM(require_lodash());
+ function validationCrossingWays(context) {
+ var type2 = "crossing_ways";
+ function getFeatureWithFeatureTypeTagsForWay(way, graph) {
+ if (getFeatureType(way, graph) === null) {
+ var parentRels = graph.parentRelations(way);
+ for (var i3 = 0; i3 < parentRels.length; i3++) {
+ var rel = parentRels[i3];
+ if (getFeatureType(rel, graph) !== null) {
+ return rel;
+ }
+ }
}
- if (_headPromise) {
- _headIsCurrent = false;
- return _headPromise;
+ 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;
}
- _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) {
- dispatch14.call("validated");
- return Promise.resolve();
+ 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;
}
- _headPromise = validateEntitiesAsync(entityIDs, _headCache).then(() => updateResolvedIssues(entityIDs)).then(() => dispatch14.call("validated")).catch(() => {
- }).then(() => {
- _headPromise = null;
- if (!_headIsCurrent) {
- validator.validate();
- }
- });
- return _headPromise;
+ return false;
+ }
+ var highwaysDisallowingFords = {
+ motorway: true,
+ motorway_link: true,
+ trunk: true,
+ trunk_link: true,
+ primary: true,
+ primary_link: true,
+ secondary: true,
+ secondary_link: true
};
- 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;
+ function tagsForConnectionNodeIfAllowed(entity1, entity2, graph, lessLikelyTags) {
+ 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;
+ var pathFeature = entity1IsPath ? entity1 : entity2;
+ if (roadFeature.tags.highway === "track") {
+ return {};
+ }
+ if (!lessLikelyTags && roadFeature.tags.highway === "service" && pathFeature.tags.highway === "footway" && pathFeature.tags.footway === "sidewalk") {
+ return {};
+ }
+ if (["marked", "unmarked", "traffic_signals", "uncontrolled"].indexOf(pathFeature.tags.crossing) !== -1) {
+ return bothLines ? { highway: "crossing", crossing: pathFeature.tags.crossing } : {};
+ }
+ return bothLines ? { highway: "crossing" } : {};
+ }
+ return {};
}
- detected = detected.filter(applySeverityOverrides);
- result.issues = result.issues.concat(detected);
- function applySeverityOverrides(issue) {
- const type2 = issue.type;
- const subtype = issue.subtype || "";
- let i3;
- for (i3 = 0; i3 < _errorOverrides.length; i3++) {
- if (_errorOverrides[i3].type.test(type2) && _errorOverrides[i3].subtype.test(subtype)) {
- issue.severity = "error";
- return true;
+ 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" };
}
}
- for (i3 = 0; i3 < _warningOverrides.length; i3++) {
- if (_warningOverrides[i3].type.test(type2) && _warningOverrides[i3].subtype.test(subtype)) {
- issue.severity = "warning";
- return true;
+ 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" } : {};
}
- for (i3 = 0; i3 < _disableOverrides.length; i3++) {
- if (_disableOverrides[i3].type.test(type2) && _disableOverrides[i3].subtype.test(subtype)) {
- return false;
+ }
+ }
+ 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 i3, j2;
+ var extent;
+ var n1, n22, nA, nB, nAId, nBId;
+ var segment1, segment2;
+ var oneOnly;
+ var segmentInfos, segment2Info, way2, taggedFeature2, way2FeatureType;
+ var way1Nodes = graph.childNodes(way1);
+ var comparedWays = {};
+ for (i3 = 0; i3 < way1Nodes.length - 1; i3++) {
+ n1 = way1Nodes[i3];
+ n22 = way1Nodes[i3 + 1];
+ extent = geoExtent([
+ [
+ Math.min(n1.loc[0], n22.loc[0]),
+ Math.min(n1.loc[1], n22.loc[1])
+ ],
+ [
+ Math.max(n1.loc[0], n22.loc[0]),
+ Math.max(n1.loc[1], n22.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 === n22.id || nBId === n1.id || nBId === n22.id) {
+ continue;
+ }
+ nA = graph.hasEntity(nAId);
+ if (!nA)
+ continue;
+ nB = graph.hasEntity(nBId);
+ if (!nB)
+ continue;
+ segment1 = [n1.loc, n22.loc];
+ segment2 = [nA.loc, nB.loc];
+ var point2 = geoLineIntersection(segment1, segment2);
+ if (point2) {
+ edgeCrossInfos.push({
+ wayInfos: [
+ {
+ way: way1,
+ featureType: way1FeatureType,
+ edge: [n1.id, n22.id]
+ },
+ {
+ way: way2,
+ featureType: way2FeatureType,
+ edge: [nA.id, nB.id]
+ }
+ ],
+ crossPoint: point2
+ });
+ if (oneOnly) {
+ checkedSingleCrossingWays[way2.id] = true;
+ break;
}
}
- return true;
}
}
+ return edgeCrossInfos;
}
- 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);
+ 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);
+ }
}
- 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)
- dispatch14.call("validated");
- }).then(() => processQueue(cache));
+ return array2;
+ }, []);
+ }
+ return [];
}
- 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);
+ 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));
}
- });
- 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);
+ return issues;
};
- 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];
- }
- });
+ 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;
});
- return result;
- };
- return cache;
- }
-
- // modules/core/uploader.js
- function coreUploader(context) {
- var dispatch14 = 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(d2) {
- _discardTags = d2;
- }).catch(function() {
- });
- var uploader = utilRebind({}, dispatch14, "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;
- dispatch14.call("saveStarted", this);
- }
- var history = context.history();
- _conflicts = [];
- _errors = [];
- _origChanges = history.changes(actionDiscardTags(history.difference(), _discardTags));
- if (!tryAgain) {
- history.perform(actionNoop());
+ 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 (!checkConflicts) {
- upload(changeset);
- } else {
- performFullConflictCheck(changeset);
+ if (connectionTags && (isCrossingIndoors || isCrossingTunnels || isCrossingBridges)) {
+ crossingTypeID += "_connectable";
}
- };
- 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 i3 = 0; i3 < summary.length; i3++) {
- var item = summary[i3];
- if (item.changeType === "modified") {
- _toCheck.push(item.entity.id);
+ 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));
+ let lessLikelyConnectionTags = tagsForConnectionNodeIfAllowed(entities[0], entities[1], graph, true);
+ if (lessLikelyConnectionTags && !(0, import_lodash3.isEqual)(connectionTags, lessLikelyConnectionTags)) {
+ fixes.push(makeConnectWaysFix(lessLikelyConnectionTags));
+ }
+ }
+ 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"));
}
- var _toLoad = withChildNodes(_toCheck, localGraph);
- var _loaded = {};
- var _toLoadCount = 0;
- var _toLoadTotal = _toLoad.length;
- if (_toCheck.length) {
- dispatch14.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 s2 = new Set(ids);
- ids.forEach(function(id2) {
- var entity = graph.entity(id2);
- if (entity.type !== "way")
+ }
+ 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;
- graph.childNodes(entity).forEach(function(child) {
- if (child.version !== void 0) {
- s2.add(child.id);
+ 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();
}
- });
- });
- return Array.from(s2);
- }
- 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 i4, id2;
- if (entity.type === "way") {
- for (i4 = 0; i4 < entity.nodes.length; i4++) {
- id2 = entity.nodes[i4];
- if (_loaded[id2] === void 0) {
- _loaded[id2] = false;
- loadMore.push(id2);
+ 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);
+ }
}
}
- } else if (entity.type === "relation" && entity.isMultipolygon()) {
- for (i4 = 0; i4 < entity.members.length; i4++) {
- id2 = entity.members[i4].id;
- if (_loaded[id2] === void 0) {
- _loaded[id2] = false;
- loadMore.push(id2);
- }
+ 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;
}
- });
- _toLoadCount += result.data.length;
- _toLoadTotal += loadMore.length;
- dispatch14.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);
+ 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 formatUser(d2) {
- return '' + escape_default(d2) + "";
- }
- function entityName(entity) {
- return utilDisplayName(entity) || utilDisplayType(entity.id) + " " + entity.id;
+ });
+ }
+ function makeConnectWaysFix(connectionTags) {
+ var fixTitleID = "connect_features";
+ var fixIcon = "iD-icon-crossing";
+ if (connectionTags.highway === "crossing") {
+ fixTitleID = "connect_using_crossing";
+ fixIcon = "temaki-pedestrian";
+ }
+ if (connectionTags.ford) {
+ fixTitleID = "connect_using_ford";
+ fixIcon = "roentgen-ford";
+ }
+ const fix = new validationIssueFix({
+ icon: fixIcon,
+ title: _t.append("issues.fix." + fixTitleID + ".title"),
+ onClick: function(context2) {
+ var loc = this.issue.loc;
+ var edges = this.issue.data.edges;
+ context2.perform(
+ function actionConnectCrossingWays(graph) {
+ var node = osmNode({ loc, tags: connectionTags });
+ graph = graph.replace(node);
+ var nodesToMerge = [node.id];
+ var mergeThresholdInMeters = 0.75;
+ edges.forEach(function(edge) {
+ var edgeNodes = [graph.entity(edge[0]), graph.entity(edge[1])];
+ var nearby = geoSphericalClosestNode(edgeNodes, loc);
+ if ((!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 sameVersions(local, remote) {
- if (local.version !== remote.version)
- return false;
- if (local.type === "way") {
- var children2 = utilArrayUnion(local.nodes, remote.nodes);
- for (var i4 = 0; i4 < children2.length; i4++) {
- var a2 = localGraph.hasEntity(children2[i4]);
- var b2 = remoteGraph.hasEntity(children2[i4]);
- if (a2 && b2 && a2.version !== b2.version)
- return false;
+ });
+ fix._connectionTags = connectionTags;
+ return fix;
+ }
+ 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;
}
}
- return true;
+ tags.layer = layer.toString();
+ context2.perform(
+ actionChangeTags(entity.id, tags),
+ _t("operations.change_tags.annotation")
+ );
}
- _toCheck.forEach(function(id2) {
- var local = localGraph.entity(id2);
- var remote = remoteGraph.entity(id2);
- if (sameVersions(local, remote))
- return;
- var merge2 = actionMergeRemoteChanges(id2, localGraph, remoteGraph, _discardTags, formatUser);
- history.replace(merge2);
- var mergeConflicts = merge2.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" });
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/behavior/draw_way.js
+ function behaviorDrawWay(context, wayID, mode, startGraph) {
+ const keybinding = utilKeybinding("drawWay");
+ var dispatch14 = 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);
}
- 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) {
- dispatch14.call("willAttemptUpload", this);
- osm.putChangeset(changeset, changes, uploadCallback);
- } else {
- didResultInNoChanges();
+ }
+ 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 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();
+ function allowsVertex(d2) {
+ return d2.geometry(context.graph()) === "vertex" || _mainPresetIndex.allowsVertex(d2, 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;
}
- } else {
- didResultInSuccess(changeset);
}
+ context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
+ _drawNode = context.entity(_drawNode.id);
+ checkGeometry(
+ true
+ /* includeDrawNode */
+ );
}
- function didResultInNoChanges() {
- dispatch14.call("resultNoChanges", this);
- endSave();
- context.flush();
+ 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 didResultInErrors() {
- context.history().pop();
- dispatch14.call("resultErrors", this, _errors);
- endSave();
+ 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 didResultInConflicts(changeset) {
- _conflicts.sort(function(a2, b2) {
- return b2.id.localeCompare(a2.id);
- });
- dispatch14.call("resultConflicts", this, changeset, _conflicts, _origChanges);
- endSave();
+ 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 didResultInSuccess(changeset) {
- context.history().clearSaved();
- dispatch14.call("resultSuccess", this, changeset);
- window.setTimeout(function() {
- endSave();
- context.flush();
- }, 2500);
+ function setActiveElements() {
+ if (!_drawNode)
+ return;
+ context.surface().selectAll("." + _drawNode.id).classed("active", true);
}
- function endSave() {
- _isSaving = false;
- dispatch14.call("saveEnded", this);
+ function resetToStartGraph() {
+ while (context.graph() !== startGraph) {
+ context.pop();
+ }
}
- uploader.cancelConflictResolution = function() {
- context.history().pop();
- };
- uploader.processResolvedConflicts = function(changeset) {
- var history = context.history();
- for (var i3 = 0; i3 < _conflicts.length; i3++) {
- if (_conflicts[i3].chosen === 1) {
- var entity = context.hasEntity(_conflicts[i3].id);
- if (entity && entity.type === "way") {
- var children2 = utilArrayUniq(entity.nodes);
- for (var j3 = 0; j3 < children2.length; j3++) {
- history.replace(actionRevert(children2[j3]));
- }
- }
- history.replace(actionRevert(_conflicts[i3].id));
- }
+ 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];
}
- uploader.save(changeset, true, false);
+ _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);
};
- uploader.reset = function() {
+ 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);
};
- return uploader;
- }
-
- // modules/renderer/background_source.js
- var import_lodash4 = __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));
+ function attemptAdd(d2, loc, doAdd) {
+ if (_drawNode) {
+ context.replace(actionMoveNode(_drawNode.id, loc), _annotation);
+ _drawNode = context.entity(_drawNode.id);
+ } else {
+ createDrawNode(loc);
+ }
+ checkGeometry(
+ true
+ /* includeDrawNode */
+ );
+ if (d2 && d2.properties && d2.properties.nope || context.surface().classed("nope")) {
+ if (!_pointerHasMoved) {
+ removeDrawNode();
+ }
+ dispatch14.call("rejectedSelfIntersection", this);
+ return;
+ }
+ context.pauseChangeDispatch();
+ doAdd();
+ _didResolveTempEdit = true;
+ context.resumeChangeDispatch();
+ context.enter(mode);
}
- clear() {
- this.pendingHandles.forEach((timeoutHandle) => {
- clearTimeout(timeoutHandle);
+ drawWay.add = function(loc, d2) {
+ attemptAdd(d2, loc, function() {
});
- 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(s2) {
- if (!s2)
- return null;
- var options2 = { day: "numeric", month: "short", year: "numeric" };
- var d2 = new Date(s2);
- if (isNaN(d2.getTime()))
- return null;
- return d2.toLocaleDateString(_mainLocalizer.localeCode(), options2);
- }
- function vintageRange(vintage) {
- var s2;
- if (vintage.start || vintage.end) {
- s2 = vintage.start || "?";
- if (vintage.start !== vintage.end) {
- s2 += " - " + (vintage.end || "?");
+ };
+ drawWay.addWay = function(loc, edge, d2) {
+ attemptAdd(d2, loc, function() {
+ context.replace(
+ actionAddMidpoint({ loc, edge }, _drawNode),
+ _annotation
+ );
+ });
+ };
+ drawWay.addNode = function(node, d2) {
+ if (node.id === _headNodeID || // or the first node when drawing an area
+ _origWay.isClosed() && node.id === _origWay.first()) {
+ drawWay.finish();
+ return;
+ }
+ attemptAdd(d2, 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((w2) => w2.id !== wayID);
+ const secondLastNodesParents = historyGraph.parentWays(historyGraph.entity(secondLastNodeId)).filter((w2) => w2.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.".concat(featureType)))();
+ return;
+ }
+ if (!secondLastNodesParents.some((n3) => n3.id === lastNodesParents[0].id)) {
+ context.ui().flash.duration(4e3).iconName("#iD-icon-no").label(_t.append("operations.follow.error.intersection_of_different_ways.".concat(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"))();
}
}
- return s2;
- }
- 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;
+ keybinding.on(_t("operations.follow.key"), followMode);
+ select_default2(document).call(keybinding);
+ drawWay.finish = function() {
+ checkGeometry(
+ false
+ /* includeDrawNode */
+ );
+ if (context.surface().classed("nope")) {
+ dispatch14.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));
};
- source.name = function() {
- var id_safe = source.id.replace(/\./g, "");
- return _t("imagery." + id_safe + ".name", { default: (0, import_lodash4.escape)(_name) });
+ 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));
};
- source.label = function() {
- var id_safe = source.id.replace(/\./g, "");
- return _t.append("imagery." + id_safe + ".name", { default: (0, import_lodash4.escape)(_name) });
+ drawWay.nodeIndex = function(val) {
+ if (!arguments.length)
+ return _nodeIndex;
+ _nodeIndex = val;
+ return drawWay;
};
- source.hasDescription = function() {
- var id_safe = source.id.replace(/\./g, "");
- var descriptionText = _mainLocalizer.tInfo("imagery." + id_safe + ".description", { default: (0, import_lodash4.escape)(_description) }).text;
- return descriptionText !== "";
+ drawWay.activeID = function() {
+ if (!arguments.length)
+ return _drawNode && _drawNode.id;
+ return drawWay;
};
- source.description = function() {
- var id_safe = source.id.replace(/\./g, "");
- return _t.append("imagery." + id_safe + ".description", { default: (0, import_lodash4.escape)(_description) });
+ return utilRebind(drawWay, dispatch14, "on");
+ }
+
+ // modules/modes/draw_line.js
+ function modeDrawLine(context, wayID, startGraph, button, affix, continuing) {
+ var mode = {
+ button,
+ id: "draw-line"
};
- source.best = function() {
- return _best;
+ 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);
};
- source.area = function() {
- if (!data.polygon)
- return Number.MAX_VALUE;
- var area = area_default({ type: "MultiPolygon", coordinates: [data.polygon] });
- return isNaN(area) ? 0 : area;
+ mode.exit = function() {
+ context.uninstall(behavior);
};
- source.imageryUsed = function() {
- return _name || source.id;
+ mode.selectedIDs = function() {
+ return [wayID];
};
- source.template = function(val) {
- if (!arguments.length)
- return _template;
- if (source.id === "custom" || source.id === "Bing") {
- _template = val;
- }
- return source;
+ mode.activeID = function() {
+ return behavior && behavior.activeID() || [];
};
- 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(x2, y2, z2) {
- var zoomSize = Math.pow(2, z2);
- var lon = x2 / zoomSize * Math.PI * 2 - Math.PI;
- var lat = Math.atan(Math.sinh(Math.PI * (1 - 2 * y2 / 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]
- };
+ 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);
}
- };
- 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;
+ 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 operation2 = operationDelete(context2, [id2]);
+ if (!operation2.disabled()) {
+ operation2();
}
- case "w":
- return minXmaxY.x;
- case "s":
- return maxXminY.y;
- case "n":
- return maxXminY.x;
- case "e":
- return minXmaxY.y;
- default:
- return token;
+ }
+ }));
+ } 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 i3 in childNodes) {
+ var vertex = childNodes[i3];
+ 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;
});
- } 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 u2 = "";
- for (var zoom = coord2[2]; zoom > 0; zoom--) {
- var b2 = 0;
- var mask = 1 << zoom - 1;
- if ((coord2[0] & mask) !== 0)
- b2++;
- if ((coord2[1] & mask) !== 0)
- b2 += 2;
- u2 += b2.toString();
+ }
+ 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)
+ );
}
- return u2;
});
}
- result = result.replace(/\{switch:([^}]+)\}/, function(s2, r2) {
- var subdomains = r2.split(",");
- return subdomains[(coord2[0] + coord2[1]) % subdomains.length];
- });
- return result;
- };
- source.validZoom = function(z2) {
- return source.zoomExtent[0] <= z2 && (source.overzoom || source.zoomExtent[1] > z2);
- };
- 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;
+ validation.type = type2;
+ return validation;
}
- rendererBackgroundSource.Bing = function(data, dispatch14) {
- 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`;
+
+ // 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);
}
- 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]])
- };
- })
- };
- });
- dispatch14.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;
+ 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(s2) {
+ return s2.trim();
+ }).filter(function(x2) {
+ return !isValidEmail(x2);
});
- }).map(function(provider) {
- return provider.attribution;
- }).join(", ");
+ 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;
};
- 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] = {};
+ 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 [];
}
- if (cache[tileID] && cache[tileID].metadata) {
- return callback(null, cache[tileID].metadata);
+ 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"));
}
- inflight[tileID] = true;
- if (metadataLastZoom !== tileCoord[2]) {
- metadataLastZoom = tileCoord[2];
- taskQueue.clear();
+ };
+ 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;
}
- taskQueue.enqueue(() => {
- json_default(url2).then(function(result) {
- delete inflight[tileID];
- if (!result) {
- throw new Error("Unknown Error");
+ 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;
}
- 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 z2 = 20;
- var dummyUrl = esri.url([1, 2, 3]);
- var x2 = Math.floor((center[0] + 180) / 360 * Math.pow(2, z2));
- var y2 = 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, z2));
- var tilemapUrl = dummyUrl.replace(/tile\/[0-9]+\/[0-9]+\/[0-9]+\?blankTile=false/, "tilemap") + "/" + z2 + "/" + y2 + "/" + x2 + "/8/8";
- json_default(tilemapUrl).then(function(tilemap) {
- if (!tilemap) {
- throw new Error("Unknown Error");
}
- var hasTiles = true;
- for (var i3 = 0; i3 < tilemap.data.length; i3++) {
- if (!tilemap.data[i3]) {
- hasTiles = false;
- break;
+ 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;
}
}
- 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] = {};
+ return false;
}
- if (cache[tileID] && cache[tileID].metadata) {
- return callback(null, cache[tileID].metadata);
+ 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;
+ });
}
- inflight[tileID] = true;
- json_default(url).then(function(result) {
- delete inflight[tileID];
- result = result.features.map((f3) => f3.attributes).filter((a2) => a2.MinMapLevel <= zoom && a2.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);
+ 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 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";
+ var placement = isFirst ? "start" : "end", messageID = wayType + ".", referenceID = wayType + ".";
+ if (wayType === "waterway") {
+ messageID += "connected." + placement;
+ referenceID += "connected";
+ } else {
+ messageID += placement;
+ referenceID += placement;
}
- if (isFinite(metadata.accuracy)) {
- metadata.accuracy += " m";
+ 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"));
+ };
}
- 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);
+ function continueDrawing(way, vertex, context) {
+ var map2 = context.map();
+ if (!context.editable() || !map2.trimmedExtent().contains(vertex.loc)) {
+ map2.zoomToEase(vertex);
}
- 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;
+ context.enter(
+ modeDrawLine(context, way.id, context.graph(), "line", way.affix(vertex.id), true)
+ );
}
- feat.properties = properties || {};
- feat.geometry = geom;
- return feat;
+ validation.type = type2;
+ return validation;
}
- 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 j3 = 0; j3 < ring[ring.length - 1].length; j3++) {
- if (ring[ring.length - 1][j3] !== ring[0][j3]) {
- throw new Error("First and last Position are not equivalent.");
- }
+
+ // 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.".concat(id2)));
+ };
}
- }
- 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);
+ validation.type = type2;
+ return validation;
}
- // node_modules/@turf/invariant/dist/es/index.js
- function getGeom(geojson) {
- if (geojson.type === "Feature") {
- return geojson.geometry;
- }
- return geojson;
+ // 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 i3 = 0; i3 < rules.length; i3++) {
+ var rule = rules[i3];
+ rule.findIssues(entity, graph, issues);
+ }
+ return issues;
+ };
+ validation.type = type2;
+ return validation;
}
- // 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 = [], i3, codeB, lastCode;
- var a2;
- var b2;
- if (!result)
- result = [];
- for (i3 = 1; i3 < len; i3++) {
- a2 = points[i3 - 1];
- b2 = points[i3];
- codeB = lastCode = bitCode(b2, bbox2);
- while (true) {
- if (!(codeA | codeB)) {
- part.push(a2);
- if (codeB !== lastCode) {
- part.push(b2);
- if (i3 < len - 1) {
- result.push(part);
- part = [];
- }
- } else if (i3 === len - 1) {
- part.push(b2);
- }
- break;
- } else if (codeA & codeB) {
- break;
- } else if (codeA) {
- a2 = intersect(a2, b2, codeA, bbox2);
- codeA = bitCode(a2, bbox2);
- } else {
- b2 = intersect(a2, b2, codeB, bbox2);
- codeB = bitCode(b2, bbox2);
- }
+ // 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;
}
- codeA = lastCode;
- }
- if (part.length)
- result.push(part);
- return result;
- }
- function polygonclip(points, bbox2) {
- var result, edge, prev, prevInside, i3, p2, inside;
- for (edge = 1; edge <= 8; edge *= 2) {
- result = [];
- prev = points[points.length - 1];
- prevInside = !(bitCode(prev, bbox2) & edge);
- for (i3 = 0; i3 < points.length; i3++) {
- p2 = points[i3];
- inside = !(bitCode(p2, bbox2) & edge);
- if (inside !== prevInside)
- result.push(intersect(prev, p2, edge, bbox2));
- if (inside)
- result.push(p2);
- prev = p2;
- prevInside = inside;
+ var asLine = _mainPresetIndex.matchTags(tagSuggestingArea, "line");
+ var asArea = _mainPresetIndex.matchTags(tagSuggestingArea, "area");
+ if (asLine && asArea && (0, import_fast_deep_equal4.default)(asLine.tags, asArea.tags)) {
+ return null;
}
- points = result;
- if (!points.length)
- break;
+ return tagSuggestingArea;
}
- return result;
- }
- function intersect(a2, b2, edge, bbox2) {
- return edge & 8 ? [a2[0] + (b2[0] - a2[0]) * (bbox2[3] - a2[1]) / (b2[1] - a2[1]), bbox2[3]] : edge & 4 ? [a2[0] + (b2[0] - a2[0]) * (bbox2[1] - a2[1]) / (b2[1] - a2[1]), bbox2[1]] : edge & 2 ? [bbox2[2], a2[1] + (b2[1] - a2[1]) * (bbox2[2] - a2[0]) / (b2[0] - a2[0])] : edge & 1 ? [bbox2[0], a2[1] + (b2[1] - a2[1]) * (bbox2[0] - a2[0]) / (b2[0] - a2[0])] : null;
- }
- function bitCode(p2, bbox2) {
- var code = 0;
- if (p2[0] < bbox2[0])
- code |= 1;
- else if (p2[0] > bbox2[2])
- code |= 2;
- if (p2[1] < bbox2[1])
- code |= 4;
- else if (p2[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);
+ 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")
+ );
+ };
}
- 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");
+ 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 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]);
+ 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 (clipped.length >= 4) {
- outRings.push(clipped);
+ if (Object.keys(presetAsLine.tags).length === 0) {
+ validAsLine = false;
}
}
- }
- return outRings;
- }
-
- // node_modules/@turf/meta/dist/es/index.js
- function coordEach(geojson, callback, excludeWrapCoord) {
- if (geojson === null)
- return;
- var j3, k2, l2, 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 (j3 = 0; j3 < coords.length; j3++) {
- if (callback(
- coords[j3],
- coordIndex,
- featureIndex,
- multiFeatureIndex,
- geometryIndex
- ) === false)
- return false;
- coordIndex++;
- if (geomType === "MultiPoint")
- multiFeatureIndex++;
- }
- if (geomType === "LineString")
- multiFeatureIndex++;
- break;
- case "Polygon":
- case "MultiLineString":
- for (j3 = 0; j3 < coords.length; j3++) {
- for (k2 = 0; k2 < coords[j3].length - wrapShrink; k2++) {
- if (callback(
- coords[j3][k2],
- coordIndex,
- featureIndex,
- multiFeatureIndex,
- geometryIndex
- ) === false)
- return false;
- coordIndex++;
- }
- if (geomType === "MultiLineString")
- multiFeatureIndex++;
- if (geomType === "Polygon")
- geometryIndex++;
- }
- if (geomType === "Polygon")
- multiFeatureIndex++;
- break;
- case "MultiPolygon":
- for (j3 = 0; j3 < coords.length; j3++) {
- geometryIndex = 0;
- for (k2 = 0; k2 < coords[j3].length; k2++) {
- for (l2 = 0; l2 < coords[j3][k2].length - wrapShrink; l2++) {
- if (callback(
- coords[j3][k2][l2],
- coordIndex,
- featureIndex,
- multiFeatureIndex,
- geometryIndex
- ) === false)
- return false;
- coordIndex++;
- }
- geometryIndex++;
+ 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];
}
- multiFeatureIndex++;
+ context2.perform(
+ actionChangeTags(entityId, tags),
+ _t("issues.fix.remove_tag.annotation")
+ );
}
- break;
- case "GeometryCollection":
- for (j3 = 0; j3 < geometry.geometries.length; j3++)
- if (coordEach(geometry.geometries[j3], callback, excludeWrapCoord) === false)
- return false;
- break;
- default:
- throw new Error("Unknown Geometry Type");
+ }));
+ 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"));
}
}
- }
-
- // 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];
+ 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
+ });
}
- if (result[3] < coord2[1]) {
- result[3] = coord2[1];
+ 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 result;
- }
- bbox["default"] = bbox;
- var es_default = bbox;
-
- // modules/renderer/background.js
- var import_which_polygon3 = __toESM(require_which_polygon());
-
- // modules/renderer/tile_layer.js
- function rendererTileLayer(context) {
- var transformProp = utilPrefixCSSProperty("Transform");
- var tiler9 = utilTiler();
- var _tileSize = 256;
- var _projection;
- var _cache4 = {};
- var _tileOrigin;
- var _zoom;
- var _source;
- function tileSizeAtZoom(d2, z2) {
- var EPSILON = 2e-3;
- return _tileSize * Math.pow(2, z2 - d2[2]) / _tileSize + EPSILON;
+ 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 atZoom(t2, distance) {
- var power = Math.pow(2, distance);
+ 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 [
- Math.floor(t2[0] * power),
- Math.floor(t2[1] * power),
- t2[2] + distance
+ new validationIssueFix({
+ icon: "iD-icon-line",
+ title: _t.append("issues.fix.convert_to_line.title"),
+ onClick: convertOnClick
+ })
];
}
- function lookUp(d2) {
- for (var up = -1; up > -d2[2]; up--) {
- var tile = atZoom(d2, up);
- if (_cache4[_source.url(tile)] !== false) {
- return tile;
- }
+ 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 uniqueBy(a2, n3) {
- var o2 = [];
- var seen = {};
- for (var i3 = 0; i3 < a2.length; i3++) {
- if (seen[a2[i3][n3]] === void 0) {
- o2.push(a2[i3]);
- seen[a2[i3][n3]] = true;
- }
+ 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 i3 in sequences) {
+ var sequence = sequences[i3];
+ 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 o2;
- }
- function addSource(d2) {
- d2.push(_source.url(d2));
- return d2;
- }
- 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];
+ 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 translate = [
- _projection.translate()[0] + pixelOffset[0],
- _projection.translate()[1] + pixelOffset[1]
- ];
- tiler9.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)) {
- tiler9.skipNullIsland(!!_source.overlay);
- tiler9().forEach(function(d2) {
- addSource(d2);
- if (d2[3] === "")
- return;
- if (typeof d2[3] !== "string")
+ 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;
- requests.push(d2);
- if (_cache4[d2[3]] === false && lookUp(d2)) {
- requests.push(addSource(lookUp(d2)));
+ var member = relation.memberById(entity.id);
+ if (member && isMissingRole(member)) {
+ issues.push(makeIssue(entity, relation, member));
}
});
- requests = uniqueBy(requests, 3).filter(function(r2) {
- return _cache4[r2[3]] !== false;
+ } 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));
+ }
});
}
- function load(d3_event, d2) {
- _cache4[d2[3]] = true;
- select_default2(this).on("error", null).on("load", null).classed("tile-loaded", true);
- render(selection2);
- }
- function error(d3_event, d2) {
- _cache4[d2[3]] = false;
- select_default2(this).on("error", null).on("load", null).remove();
- render(selection2);
- }
- function imageTransform(d2) {
- var ts = _tileSize * Math.pow(2, _zoom - d2[2]);
- var scale = tileSizeAtZoom(d2, _zoom);
- return "translate(" + (d2[0] * ts - _tileOrigin[0]) + "px," + (d2[1] * ts - _tileOrigin[1]) + "px) scale(" + scale + "," + scale + ")";
- }
- function tileCenter(d2) {
- var ts = _tileSize * Math.pow(2, _zoom - d2[2]);
- return [
- d2[0] * ts - _tileOrigin[0] + ts / 2,
- d2[1] * ts - _tileOrigin[1] + ts / 2
- ];
- }
- function debugTransform(d2) {
- var coord2 = tileCenter(d2);
- return "translate(" + coord2[0] + "px," + coord2[1] + "px)";
- }
- var dims = tiler9.size();
- var mapCenter = [dims[0] / 2, dims[1] / 2];
- var minDist = Math.max(dims[0], dims[1]);
- var nearCenter;
- requests.forEach(function(d2) {
- var c2 = tileCenter(d2);
- var dist = geoVecLength(c2, mapCenter);
- if (dist < minDist) {
- minDist = dist;
- nearCenter = d2;
+ 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
+ })
+ );
+ }
+ })
+ ];
}
});
- var image = selection2.selectAll("img").data(requests, function(d2) {
- return d2[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(d2) {
- return d2[3];
- }).on("error", error).on("load", load).merge(image).style(transformProp, imageTransform).classed("tile-debug", showDebug).classed("tile-removing", false).classed("tile-center", function(d2) {
- return d2 === nearCenter;
- });
- var debug2 = selection2.selectAll(".tile-label-debug").data(showDebug ? requests : [], function(d2) {
- return d2[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(d2) {
- return d2[2] + " / " + d2[0] + " / " + d2[1];
- });
- debug2.selectAll(".tile-label-debug-vintage").each(function(d2) {
- var span = select_default2(this);
- var center = context.projection.invert(tileCenter(d2));
- _source.getMetadata(center, d2, 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"));
- }
- });
- });
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.missing_role.multipolygon.reference"));
}
}
- background.projection = function(val) {
- if (!arguments.length)
- return _projection;
- _projection = val;
- return background;
- };
- background.dimensions = function(val) {
- if (!arguments.length)
- return tiler9.size();
- tiler9.size(val);
- return background;
- };
- background.source = function(val) {
- if (!arguments.length)
- return _source;
- _source = val;
- _tileSize = _source.tileSize;
- _cache4 = {};
- tiler9.tileSize(_source.tileSize).zoomExtent(_source.zoomExtent);
- return background;
- };
- return background;
+ 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/renderer/background.js
- var _imageryIndex = null;
- function rendererBackground(context) {
- const dispatch14 = 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_polygon3.default)({ type: "FeatureCollection", features });
- _imageryIndex.backgrounds = sources.map((source) => {
- if (source.type === "bing") {
- return rendererBackgroundSource.Bing(source, dispatch14);
- } else if (/^EsriWorldImagery/.test(source.id)) {
- return rendererBackgroundSource.Esri(source);
- } else {
- return rendererBackgroundSource(source);
- }
+ // 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(k2) {
+ if (k2 === "area" || !osmIsInterestingTag(k2))
+ return false;
+ return !onlyAttributeKeys.some(function(attributeKey) {
+ return k2 === attributeKey || k2.indexOf(attributeKey + ":") === 0;
});
- _imageryIndex.backgrounds.unshift(rendererBackgroundSource.None());
- let template = corePreferences("background-custom-template") || "";
- const custom = rendererBackgroundSource.Custom(template);
- _imageryIndex.backgrounds.unshift(custom);
- return _imageryIndex;
});
+ if (entity.type === "relation" && entityDescriptiveKeys.length === 1 && entity.tags.type === "multipolygon") {
+ return osmOldMultipolygonOuterMemberOfRelation(entity, graph);
+ }
+ return entityDescriptiveKeys.length > 0;
}
- 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);
+ 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";
}
}
- const sources = background.sources(context.map().extent());
- const wasValid = _isValid;
- _isValid = !!sources.filter((d2) => d2 === 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 (!subtype && isUnknownRoad(entity)) {
+ subtype = "highway_classification";
}
- if (_sharpness < 1) {
- const blur = number_default(0.5, 5)(1 - _sharpness);
- baseFilter += ` blur(${blur}px)`;
+ 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 operation2 = operationDelete(context2, [id2]);
+ var disabledReasonID = operation2.disabled();
+ if (!disabledReasonID) {
+ deleteOnClick = function(context3) {
+ var id3 = this.issue.entityIds[0];
+ var operation3 = operationDelete(context3, [id3]);
+ if (!operation3.disabled()) {
+ operation3();
+ }
+ };
+ }
+ 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"));
}
- 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})`;
+ };
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/validations/mutually_exclusive_tags.js
+ function validationMutuallyExclusiveTags() {
+ const type2 = "mutually_exclusive_tags";
+ const tagKeyPairs = osmMutuallyExclusiveTagPairs;
+ const validation = function checkMutuallyExclusiveTags(entity) {
+ let pairsFounds = tagKeyPairs.filter((pair3) => {
+ return pair3[0] in entity.tags && pair3[1] in entity.tags;
+ }).filter((pair3) => {
+ return !(pair3[0].match(/^(addr:)?no[a-z]/) && entity.tags[pair3[0]] === "no" || pair3[1].match(/^(addr:)?no[a-z]/) && entity.tags[pair3[1]] === "no");
+ });
+ Object.keys(entity.tags).forEach((key) => {
+ let negative_key = "not:" + key;
+ if (negative_key in entity.tags && entity.tags[negative_key].split(";").includes(entity.tags[key])) {
+ pairsFounds.push([negative_key, key, "same_value"]);
+ }
+ if (key.match(/^name:[a-z]+/)) {
+ negative_key = "not:name";
+ if (negative_key in entity.tags && entity.tags[negative_key].split(";").includes(entity.tags[key])) {
+ pairsFounds.push([negative_key, key, "same_value"]);
+ }
+ }
+ });
+ let issues = pairsFounds.map((pair3) => {
+ const subtype = pair3[2] || "default";
+ return new validationIssue({
+ type: type2,
+ subtype,
+ severity: "warning",
+ message: function(context) {
+ let entity2 = context.hasEntity(this.entityIds[0]);
+ return entity2 ? _t.append("issues.".concat(type2, ".").concat(subtype, ".message"), {
+ feature: utilDisplayLabel(entity2, context.graph()),
+ tag1: pair3[0],
+ tag2: pair3[1]
+ }) : "";
+ },
+ reference: (selection2) => showReference(selection2, pair3, subtype),
+ entityIds: [entity.id],
+ dynamicFixes: () => pair3.slice(0, 2).map((tagToRemove) => createIssueFix(tagToRemove))
+ });
+ });
+ function createIssueFix(tagToRemove) {
+ return new validationIssueFix({
+ icon: "iD-operation-delete",
+ title: _t.append("issues.fix.remove_named_tag.title", { tag: tagToRemove }),
+ onClick: function(context) {
+ const entityId = this.issue.entityIds[0];
+ const entity2 = context.entity(entityId);
+ let tags = Object.assign({}, entity2.tags);
+ delete tags[tagToRemove];
+ context.perform(
+ actionChangeTags(entityId, tags),
+ _t("issues.fix.remove_named_tag.annotation", { tag: tagToRemove })
+ );
+ }
+ });
}
- 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, (d2) => d2.source().name());
- overlays.exit().remove();
- overlays.enter().insert("div", ".layer-data").attr("class", "layer layer-overlay").merge(overlays).each((layer, i3, nodes) => select_default2(nodes[i3]).call(layer));
- }
- background.updateImagery = function() {
- let currSource = baseLayer.source();
- if (context.inIntro() || !currSource)
- return;
- let o2 = _overlayLayers.filter((d2) => !d2.source().isLocatorOverlay() && !d2.source().isHidden()).map((d2) => d2.source().id).join(",");
- const meters = geoOffsetToMeters(currSource.offset());
- const EPSILON = 0.01;
- const x2 = +meters[0].toFixed(2);
- const y2 = +meters[1].toFixed(2);
- let hash = utilStringQs(window.location.hash);
- let id2 = currSource.id;
- if (id2 === "custom") {
- id2 = `custom:${currSource.template()}`;
+ function showReference(selection2, pair3, subtype) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.".concat(type2, ".").concat(subtype, ".reference"), { tag1: pair3[0], tag2: pair3[1] }));
}
- if (id2) {
- hash.background = id2;
- } else {
- delete hash.background;
+ return issues;
+ };
+ 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((d2) => _dataDeprecated = d2).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 (o2) {
- hash.overlays = o2;
- } else {
- delete hash.overlays;
+ if (_dataDeprecated) {
+ const deprecatedTags = entity.deprecatedTags(_dataDeprecated);
+ if (deprecatedTags.length) {
+ deprecatedTags.forEach((tag2) => {
+ graph = actionUpgradeTags(entity.id, tag2.old, tag2.replace)(graph);
+ });
+ entity = graph.entity(entity.id);
+ }
}
- if (Math.abs(x2) > EPSILON || Math.abs(y2) > EPSILON) {
- hash.offset = `${x2},${y2}`;
- } else {
- delete hash.offset;
+ let newTags = Object.assign({}, entity.tags);
+ if (preset.tags !== preset.addTags) {
+ Object.keys(preset.addTags).forEach((k2) => {
+ if (!newTags[k2]) {
+ if (preset.addTags[k2] === "*") {
+ newTags[k2] = "yes";
+ } else {
+ newTags[k2] = preset.addTags[k2];
+ }
+ }
+ });
}
- if (!window.mocha) {
- window.location.replace("#" + utilQsString(hash, true));
+ 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 imageryUsed = [];
- let photoOverlaysUsed = [];
- const currUsed = currSource.imageryUsed();
- if (currUsed && _isValid) {
- imageryUsed.push(currUsed);
+ let issues = [];
+ issues.provisional = _waitingForDeprecated || waitingForNsi;
+ const tagDiff = utilTagDiff(oldTags, newTags);
+ if (!tagDiff.length)
+ return issues;
+ const isOnlyAddingTags = tagDiff.every((d2) => d2.type === "+");
+ let prefix = "";
+ if (nsiResult) {
+ prefix = "noncanonical_brand.";
+ } else if (subtype === "deprecated_tags" && isOnlyAddingTags) {
+ subtype = "incomplete_tags";
+ prefix = "incomplete.";
}
- _overlayLayers.filter((d2) => !d2.source().isLocatorOverlay() && !d2.source().isHidden()).forEach((d2) => imageryUsed.push(d2.source().imageryUsed()));
- const dataLayer = context.layers().layer("data");
- if (dataLayer && dataLayer.enabled() && dataLayer.hasData()) {
- imageryUsed.push(dataLayer.getSrc());
+ 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);
}
- const photoOverlayLayers = {
- streetside: "Bing Streetside",
- mapillary: "Mapillary Images",
- "mapillary-map-features": "Mapillary Map Features",
- "mapillary-signs": "Mapillary Signs",
- kartaview: "KartaView Images",
- vegbilder: "Norwegian Road Administration Images",
- mapilio: "Mapilio Images"
- };
- for (let layerID in photoOverlayLayers) {
- const layer = context.layers().layer(layerID);
- if (layer && layer.enabled()) {
- photoOverlaysUsed.push(layerID);
- imageryUsed.push(photoOverlayLayers[layerID]);
+ 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:".concat(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);
}
- 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((d2) => visible[d2.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()));
+ function showMessage(context) {
+ const currEntity = context.hasEntity(entity.id);
+ if (!currEntity)
+ return "";
+ let messageID = "issues.outdated_tags.".concat(prefix, "message");
+ if (subtype === "noncanonical_brand" && isOnlyAddingTags) {
+ messageID += "_incomplete";
+ }
+ return _t.append(messageID, {
+ feature: utilDisplayLabel(
+ currEntity,
+ context.graph(),
+ true
+ /* verbose */
+ )
});
- _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(d2) {
- if (!arguments.length)
- return baseLayer.source();
- const osm = context.connection();
- if (!osm)
- return background;
- const blocklists = osm.imageryBlocklists();
- const template = d2.template();
- let fail = false;
- let tested = 0;
- let regex;
- for (let i3 = 0; i3 < blocklists.length; i3++) {
- regex = blocklists[i3];
- fail = regex.test(template);
- tested++;
- if (fail)
- break;
+ 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.".concat(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", (d2) => {
+ let klass = d2.type === "+" ? "add" : "remove";
+ return "tagDiff-cell tagDiff-cell-".concat(klass);
+ }).html((d2) => d2.display);
}
- if (!tested) {
- regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
- fail = regex.test(template);
+ }
+ 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 [];
}
- baseLayer.source(!fail ? d2 : background.findSource("none"));
- dispatch14.call("change");
- background.updateImagery();
- return background;
- };
- background.findSource = (id2) => {
- if (!id2 || !_imageryIndex)
- return null;
- return _imageryIndex.backgrounds.find((d2) => d2.id && d2.id === id2);
- };
- background.bing = () => {
- background.baseLayerSource(background.findSource("Bing"));
- };
- background.showsLayer = (d2) => {
- const currSource = baseLayer.source();
- if (!d2 || !currSource)
- return false;
- return d2.id === currSource.id || _overlayLayers.some((layer) => d2.id === layer.source().id);
- };
- background.overlayLayerSources = () => {
- return _overlayLayers.map((layer) => layer.source());
- };
- background.toggleOverlayLayer = (d2) => {
- let layer;
- for (let i3 = 0; i3 < _overlayLayers.length; i3++) {
- layer = _overlayLayers[i3];
- if (layer.source() === d2) {
- _overlayLayers.splice(i3, 1);
- dispatch14.call("change");
- background.updateImagery();
- 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);
}
- layer = rendererTileLayer(context).source(d2).projection(context.projection).dimensions(
- baseLayer.dimensions()
- );
- _overlayLayers.push(layer);
- dispatch14.call("change");
- background.updateImagery();
- };
- background.nudge = (d2, zoom) => {
- const currSource = baseLayer.source();
- if (currSource) {
- currSource.nudge(d2, zoom);
- dispatch14.call("change");
- background.updateImagery();
- }
- return background;
- };
- background.offset = function(d2) {
- const currSource = baseLayer.source();
- if (!arguments.length) {
- return currSource && currSource.offset() || [0, 0];
+ 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 */
+ ) }
+ );
}
- if (currSource) {
- currSource.offset(d2);
- dispatch14.call("change");
- background.updateImagery();
+ function showReference(selection2) {
+ selection2.selectAll(".issue-reference").data([0]).enter().append("div").attr("class", "issue-reference").call(_t.append("issues.old_multipolygon.reference"));
}
- return background;
- };
- background.brightness = function(d2) {
- if (!arguments.length)
- return _brightness;
- _brightness = d2;
- if (context.mode())
- dispatch14.call("change");
- return background;
- };
- background.contrast = function(d2) {
- if (!arguments.length)
- return _contrast;
- _contrast = d2;
- if (context.mode())
- dispatch14.call("change");
- return background;
+ }
+ let validation = function checkOutdatedTags(entity, graph) {
+ let issues = oldMultipolygonIssues(entity, graph);
+ if (!issues.length)
+ issues = oldTagIssues(entity, graph);
+ return issues;
};
- background.saturation = function(d2) {
- if (!arguments.length)
- return _saturation;
- _saturation = d2;
- if (context.mode())
- dispatch14.call("change");
- return background;
+ 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
};
- background.sharpness = function(d2) {
- if (!arguments.length)
- return _sharpness;
- _sharpness = d2;
- if (context.mode())
- dispatch14.call("change");
- return background;
+ var publicKeys = {
+ amenity: true,
+ craft: true,
+ historic: true,
+ leisure: true,
+ office: true,
+ shop: true,
+ tourism: true
};
- let _loadPromise;
- background.ensureLoaded = () => {
- if (_loadPromise)
- return _loadPromise;
- return _loadPromise = ensureImageryIndex();
+ var personalTags = {
+ "contact:email": true,
+ "contact:fax": true,
+ "contact:phone": true,
+ email: true,
+ fax: true,
+ phone: true
};
- 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((d2) => d2.id !== "none" && d2.id !== "custom");
- const first = validBackgrounds.length && validBackgrounds[0];
- const isLastUsedValid = !!validBackgrounds.find((d2) => d2.id && d2.id === lastUsedBackground);
- let best;
- if (!requestedBackground && extent) {
- const viewArea = extent.area();
- best = validBackgrounds.find((s2) => {
- if (!s2.best() || s2.overlay)
- return false;
- let bbox2 = es_default(bboxClip(
- { type: "MultiPolygon", coordinates: [s2.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")
- );
+ var validation = function checkPrivateData(entity) {
+ var tags = entity.tags;
+ if (!tags.building || !privateBuildingValues[tags.building])
+ return [];
+ var keepTags = {};
+ for (var k2 in tags) {
+ if (publicKeys[k2])
+ return [];
+ if (!personalTags[k2]) {
+ keepTags[k2] = tags[k2];
}
- const locator = imageryIndex.backgrounds.find((d2) => d2.overlay && d2.default);
- if (locator) {
- background.toggleOverlayLayer(locator);
+ }
+ 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"));
+ }
+ })
+ ];
}
- const overlays = (hash.overlays || "").split(",");
- overlays.forEach((overlay) => {
- overlay = background.findSource(overlay);
- if (overlay) {
- background.toggleOverlayLayer(overlay);
+ })];
+ 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;
}
});
- 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((n3) => !isNaN(n3) && n3);
- if (offset.length === 2) {
- background.offset(geoMetersToOffset(offset));
- }
- }
- }).catch((err) => {
- console.error(err);
- });
+ 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(d2) {
+ var klass = d2.type === "+" ? "add" : "remove";
+ return "tagDiff-cell tagDiff-cell-" + klass;
+ }).html(function(d2) {
+ return d2.display;
+ });
+ }
};
- return utilRebind(background, dispatch14, "on");
+ validation.type = type2;
+ return validation;
}
- // modules/renderer/features.js
- function rendererFeatures(context) {
- var dispatch14 = dispatch_default("change", "redraw");
- var features = utilRebind({}, dispatch14, "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;
+ // 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);
}
- window.location.replace("#" + utilQsString(hash, true));
- corePreferences("disabled-features", disabled.join(","));
}
- _hidden = features.hidden();
- dispatch14.call("change");
- dispatch14.call("redraw");
+ return false;
}
- function defineRule(k2, filter2, max3) {
- var isEnabled = true;
- _keys.push(k2);
- _rules[k2] = {
- 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;
+ function nameMatchesRawTag(lowercaseName, tags) {
+ for (let i3 = 0; i3 < keysToTestForGenericValues.length; i3++) {
+ let key = keysToTestForGenericValues[i3];
+ let val = tags[key];
+ if (val) {
+ val = val.toLowerCase();
+ if (key === lowercaseName || val === lowercaseName || key.replace(/\_/g, " ") === lowercaseName || val.replace(/\_/g, " ") === lowercaseName) {
+ return true;
+ }
}
- };
- }
- 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 i3 = 0; i3 < strings.length; i3++) {
- var s2 = strings[i3];
- if (osmLifecyclePrefixes[s2] || osmLifecyclePrefixes[tags[s2]])
- 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(k2) {
- if (!arguments.length) {
- return _keys.filter(function(k3) {
- return _rules[k3].enabled;
- });
+ }
+ 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: "".concat(nameKey, "=").concat(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"));
}
- return _rules[k2] && _rules[k2].enabled;
- };
- features.disabled = function(k2) {
- if (!arguments.length) {
- return _keys.filter(function(k3) {
- return !_rules[k3].enabled;
- });
+ }
+ let validation = function checkGenericName(entity) {
+ const tags = entity.tags;
+ const hasWikidata = !!tags.wikidata || !!tags["brand:wikidata"] || !!tags["operator:wikidata"];
+ if (hasWikidata)
+ return [];
+ let issues = [];
+ for (let key in tags) {
+ const m2 = key.match(/^name(?:(?::)([a-zA-Z_-]+))?$/);
+ if (!m2)
+ continue;
+ const langCode = m2.length >= 2 ? m2[1] : null;
+ const value = tags[key];
+ if (isGenericName(value, tags)) {
+ issues.provisional = _waitingForNsi;
+ issues.push(makeGenericNameIssue(entity.id, key, value, langCode));
+ }
}
- return _rules[k2] && !_rules[k2].enabled;
+ return issues;
};
- features.hidden = function(k2) {
- if (!arguments.length) {
- return _keys.filter(function(k3) {
- return _rules[k3].hidden();
+ 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 _rules[k2] && _rules[k2].hidden();
- };
- features.autoHidden = function(k2) {
- if (!arguments.length) {
- return _keys.filter(function(k3) {
- return _rules[k3].autoHidden();
- });
+ 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"));
}
- return _rules[k2] && _rules[k2].autoHidden();
};
- features.enable = function(k2) {
- if (_rules[k2] && !_rules[k2].enabled) {
- _rules[k2].enable();
- update();
+ validation.type = type2;
+ return validation;
+ }
+
+ // modules/core/validator.js
+ function coreValidator(context) {
+ let dispatch14 = dispatch_default("validated", "focusedIssue");
+ let validator = utilRebind({}, dispatch14, "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(str) {
+ const escaped = str.replace(/[-\/\\^$+?.()|[\]{}]/g, "\\$&").replace(/\*/g, ".*");
+ return new RegExp("^" + escaped + "$");
}
- };
- features.enableAll = function() {
- var didEnable = false;
- for (var k2 in _rules) {
- if (!_rules[k2].enabled) {
- didEnable = true;
- _rules[k2].enable();
- }
+ }
+ 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((k2) => _disabledRules[k2] = true);
}
- if (didEnable)
- update();
};
- features.disable = function(k2) {
- if (_rules[k2] && _rules[k2].enabled) {
- _rules[k2].disable();
- update();
- }
+ 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);
};
- features.disableAll = function() {
- var didDisable = false;
- for (var k2 in _rules) {
- if (_rules[k2].enabled) {
- didDisable = true;
- _rules[k2].disable();
- }
- }
- if (didDisable)
- update();
+ validator.resetIgnoredIssues = () => {
+ _ignoredIssueIDs.clear();
+ dispatch14.call("validated");
};
- features.toggle = function(k2) {
- if (_rules[k2]) {
- (function(f3) {
- return f3.enabled ? f3.disable() : f3.enable();
- })(_rules[k2]);
- update();
- }
+ validator.revalidateUnsquare = () => {
+ revalidateUnsquare(_headCache);
+ revalidateUnsquare(_baseCache);
+ dispatch14.call("validated");
};
- features.resetStats = function() {
- for (var i3 = 0; i3 < _keys.length; i3++) {
- _rules[_keys[i3]].count = 0;
+ 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);
+ });
}
- dispatch14.call("change");
- };
- features.gatherStats = function(d2, resolver, dimensions) {
- var needsRedraw = false;
- var types = utilArrayGroupBy(d2, "type");
- var entities = [].concat(types.relation || [], types.way || [], types.node || []);
- var currHidden, geometry, matches, i3, j3;
- for (i3 = 0; i3 < _keys.length; i3++) {
- _rules[_keys[i3]].count = 0;
+ if (opts.what === "all") {
+ Object.values(_baseCache.issuesByIssueID).forEach((issue) => {
+ if (!filter2(issue))
+ return;
+ seen.add(issue.id);
+ results.push(issue);
+ });
}
- _cullFactor = dimensions[0] * dimensions[1] / 1e6;
- for (i3 = 0; i3 < entities.length; i3++) {
- geometry = entities[i3].geometry(resolver);
- matches = Object.keys(features.getMatches(entities[i3], resolver, geometry));
- for (j3 = 0; j3 < matches.length; j3++) {
- _rules[matches[j3]].count++;
+ 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;
}
- currHidden = features.hidden();
- if (currHidden !== _hidden) {
- _hidden = currHidden;
- needsRedraw = true;
- dispatch14.call("change");
- }
- return needsRedraw;
- };
- features.stats = function() {
- for (var i3 = 0; i3 < _keys.length; i3++) {
- _stats[_keys[i3]] = _rules[_keys[i3]].count;
- }
- return _stats;
- };
- features.clear = function(d2) {
- for (var i3 = 0; i3 < d2.length; i3++) {
- features.clearEntity(d2[i3]);
- }
- };
- 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 = {};
+ validator.getResolvedIssues = () => {
+ return Array.from(_resolvedIssueIDs).map((issueID) => _baseCache.issuesByIssueID[issueID]).filter(Boolean);
};
- 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] = {};
+ validator.focusIssue = (issue) => {
+ const graph = context.graph();
+ let selectID;
+ let focusCenter;
+ const issueExtent = issue.extent(graph);
+ if (issueExtent) {
+ focusCenter = issueExtent.center();
}
- if (!_cache4[ent].matches) {
- var matches = {};
- var hasMatch = false;
- for (var i3 = 0; i3 < _keys.length; i3++) {
- if (_keys[i3] === "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 (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 (_rules[_keys[i3]].filter(entity.tags, geometry)) {
- matches[_keys[i3]] = hasMatch = true;
+ if (nodeID) {
+ focusCenter = graph.entity(nodeID).loc;
}
}
- _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 (focusCenter) {
+ const setZoom = Math.max(context.map().zoom(), 19);
+ context.map().unobscuredCenterZoomEase(focusCenter, setZoom);
}
- if (!_cache4[ent].parents) {
- var parents = [];
- if (geometry === "vertex") {
- parents = resolver.parentWays(entity);
- } else {
- parents = resolver.parentRelations(entity);
- }
- _cache4[ent].parents = parents;
+ if (selectID) {
+ window.setTimeout(() => {
+ context.enter(modeSelect(context, [selectID]));
+ dispatch14.call("focusedIssue", this, issue);
+ }, 250);
}
- 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;
+ validator.getIssuesBySeverity = (options2) => {
+ let groups = utilArrayGroupBy(validator.getIssues(options2), "severity");
+ groups.error = groups.error || [];
+ groups.warning = groups.warning || [];
+ return groups;
};
- 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(k2) {
- return features.hidden(k2);
+ 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;
+ }
});
};
- 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 i3 = 0; i3 < parents.length; i3++) {
- if (!features.isHidden(parents[i3], resolver, parents[i3].geometry(resolver))) {
- return false;
- }
- }
- return true;
+ validator.getEntityIssues = (entityID, options2) => {
+ return validator.getSharedEntityIssues([entityID], options2);
};
- 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 = [];
+ validator.getRuleKeys = () => {
+ return Object.keys(_rules);
+ };
+ validator.isRuleEnabled = (key) => {
+ return !_disabledRules[key];
+ };
+ validator.toggleRule = (key) => {
+ if (_disabledRules[key]) {
+ delete _disabledRules[key];
} else {
- childNodes = entity.nodes ? resolver.childNodes(entity) : [];
- connections = features.getParents(entity, resolver, entity.geometry(resolver));
+ _disabledRules[key] = true;
}
- connections = childNodes.reduce(function(result, e3) {
- return resolver.isShared(e3) ? utilArrayUnion(result, resolver.parentWays(e3)) : result;
- }, connections);
- return connections.some(function(e3) {
- return features.isHidden(e3, resolver, e3.geometry(resolver));
- });
+ corePreferences("validate-disabledRules", Object.keys(_disabledRules).join(","));
+ validator.validate();
};
- 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);
+ validator.disableRules = (keys2) => {
+ _disabledRules = {};
+ keys2.forEach((k2) => _disabledRules[k2] = true);
+ corePreferences("validate-disabledRules", Object.keys(_disabledRules).join(","));
+ validator.validate();
};
- features.filter = function(d2, resolver) {
- if (!_hidden.length)
- return d2;
- var result = [];
- for (var i3 = 0; i3 < d2.length; i3++) {
- var entity = d2[i3];
- if (!features.isHidden(entity, resolver, entity.geometry(resolver))) {
- result.push(entity);
- }
+ 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;
+ dispatch14.call("validated");
+ return Promise.resolve();
}
- return result;
+ 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) {
+ dispatch14.call("validated");
+ return Promise.resolve();
+ }
+ _headPromise = validateEntitiesAsync(entityIDs, _headCache).then(() => updateResolvedIssues(entityIDs)).then(() => dispatch14.call("validated")).catch(() => {
+ }).then(() => {
+ _headPromise = null;
+ if (!_headIsCurrent) {
+ validator.validate();
+ }
+ });
+ return _headPromise;
};
- features.forceVisible = function(entityIDs) {
- if (!arguments.length)
- return Object.keys(_forceVisible);
- _forceVisible = {};
- for (var i3 = 0; i3 < entityIDs.length; i3++) {
- _forceVisible[entityIDs[i3]] = true;
- var entity = context.hasEntity(entityIDs[i3]);
- if (entity && entity.type === "relation") {
- for (var j3 in entity.members) {
- _forceVisible[entity.members[j3].id] = true;
+ 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 i3;
+ for (i3 = 0; i3 < _errorOverrides.length; i3++) {
+ if (_errorOverrides[i3].type.test(type2) && _errorOverrides[i3].subtype.test(subtype)) {
+ issue.severity = "error";
+ return true;
+ }
+ }
+ for (i3 = 0; i3 < _warningOverrides.length; i3++) {
+ if (_warningOverrides[i3].type.test(type2) && _warningOverrides[i3].subtype.test(subtype)) {
+ issue.severity = "warning";
+ return true;
+ }
+ }
+ for (i3 = 0; i3 < _disableOverrides.length; i3++) {
+ if (_disableOverrides[i3].type.test(type2) && _disableOverrides[i3].subtype.test(subtype)) {
+ return false;
+ }
}
+ return 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);
- }
+ }
+ 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)
+ dispatch14.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)
};
- 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 i3 = 0; i3 < entities.length; i3++) {
- var geometry = entities[i3].geometry(graph);
- features.getMatches(entities[i3], graph, geometry);
+ cache.cacheIssue = (issue) => {
+ (issue.entityIds || []).forEach((entityID) => {
+ if (!cache.issuesByEntityID[entityID]) {
+ cache.issuesByEntityID[entityID] = /* @__PURE__ */ new Set();
}
+ cache.issuesByEntityID[entityID].add(issue.id);
});
- _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 i3, j3, nodes, isClosed, ix1, ix2, ix3, ix4, max3;
- for (i3 = 0; i3 < parents.length; i3++) {
- nodes = parents[i3].nodes;
- isClosed = parents[i3].isClosed();
- for (j3 = 0; j3 < nodes.length; j3++) {
- if (nodes[j3] === node.id) {
- ix1 = j3 - 2;
- ix2 = j3 - 1;
- ix3 = j3 + 1;
- ix4 = j3 + 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;
+ cache.issuesByIssueID[issue.id] = issue;
+ };
+ cache.uncacheIssue = (issue) => {
+ (issue.entityIds || []).forEach((entityID) => {
+ if (cache.issuesByEntityID[entityID]) {
+ cache.issuesByEntityID[entityID].delete(issue.id);
}
- }
- }
- return 1;
- }
- function svgMarkerSegments(projection2, graph, dt2, shouldReverse, bothDirections) {
- return function(entity) {
- var i3 = 0;
- var offset = dt2;
- var segments = [];
- var clip = identity_default2().clipExtent(projection2.clipExtent()).stream;
- var coordinates = graph.childNodes(entity).map(function(n3) {
- return n3.loc;
});
- var a2, b2;
- if (shouldReverse(entity)) {
- coordinates.reverse();
- }
- stream_default({
- type: "LineString",
- coordinates
- }, projection2.stream(clip({
- lineStart: function() {
- },
- lineEnd: function() {
- a2 = null;
- },
- point: function(x2, y2) {
- b2 = [x2, y2];
- if (a2) {
- var span = geoVecLength(a2, b2) - offset;
- if (span >= 0) {
- var heading = geoVecAngle(a2, b2);
- var dx = dt2 * Math.cos(heading);
- var dy = dt2 * Math.sin(heading);
- var p2 = [
- a2[0] + offset * Math.cos(heading),
- a2[1] + offset * Math.sin(heading)
- ];
- var coord2 = [a2, p2];
- for (span -= dt2; span >= 0; span -= dt2) {
- p2 = geoVecAdd(p2, [dx, dy]);
- coord2.push(p2);
- }
- coord2.push(b2);
- var segment = "";
- var j3;
- for (j3 = 0; j3 < coord2.length; j3++) {
- segment += (j3 === 0 ? "M" : "L") + coord2[j3][0] + "," + coord2[j3][1];
- }
- segments.push({ id: entity.id, index: i3++, d: segment });
- if (bothDirections(entity)) {
- segment = "";
- for (j3 = coord2.length - 1; j3 >= 0; j3--) {
- segment += (j3 === coord2.length - 1 ? "M" : "L") + coord2[j3][0] + "," + coord2[j3][1];
- }
- segments.push({ id: entity.id, index: i3++, d: segment });
- }
- }
- offset = -span;
- }
- a2 = b2;
- }
- })));
- return segments;
+ delete cache.issuesByIssueID[issue.id];
};
- }
- 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));
- }
+ cache.cacheIssues = (issues) => {
+ issues.forEach(cache.cacheIssue);
};
- svgpath.geojson = function(d2) {
- if (d2.__featurehash__ !== void 0) {
- if (d2.__featurehash__ in cache) {
- return cache[d2.__featurehash__];
- } else {
- return cache[d2.__featurehash__] = path(d2);
- }
- } else {
- return path(d2);
- }
+ cache.uncacheIssues = (issues) => {
+ issues.forEach(cache.uncacheIssue);
};
- return svgpath;
- }
- function svgPointTransform(projection2) {
- var svgpoint = function(entity) {
- var pt2 = projection2(entity.loc);
- return "translate(" + pt2[0] + "," + pt2[1] + ")";
+ cache.uncacheIssuesOfType = (type2) => {
+ const issuesOfType = Object.values(cache.issuesByIssueID).filter((issue) => issue.type === type2);
+ cache.uncacheIssues(issuesOfType);
};
- svgpoint.geojson = function(d2) {
- return svgpoint(d2.properties.entity);
+ 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);
};
- 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);
+ 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 tags;
+ return result;
};
+ return cache;
}
- 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 i3 = 0; i3 < way.nodes.length; i3++) {
- node = graph.entity(way.nodes[i3]);
- 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, i3);
- } else if (start2.type === 0 && end.type === 0) {
- pushActive(start2, end, i3);
- } else {
- pushPassive(start2, end, i3);
- }
- }
- start2 = end;
+
+ // modules/core/uploader.js
+ function coreUploader(context) {
+ var dispatch14 = 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(d2) {
+ _discardTags = d2;
+ }).catch(function() {
+ });
+ var uploader = utilRebind({}, dispatch14, "on");
+ uploader.isSaving = function() {
+ return _isSaving;
+ };
+ uploader.save = function(changeset, tryAgain, checkConflicts) {
+ if (_isSaving && !tryAgain) {
+ return;
}
- 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]
+ var osm = context.connection();
+ if (!osm)
+ return;
+ if (!osm.authenticated()) {
+ osm.authenticate(function(err) {
+ if (!err) {
+ uploader.save(changeset, tryAgain, checkConflicts);
}
});
+ return;
}
- 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]
- }
- });
+ if (!_isSaving) {
+ _isSaving = true;
+ dispatch14.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);
}
- }
- }
-
- // 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 t2 = _tags(entity);
- var computed = tagClasses.getClassesString(t2, value);
- if (computed !== value) {
- select_default2(this).attr("class", computed);
- }
- });
};
- tagClasses.getClassesString = function(t2, value) {
- var primary, status;
- var i3, j3, k2, v2;
- var overrideGeometry;
- if (/\bstroke\b/.test(value)) {
- if (!!t2.barrier && t2.barrier !== "no") {
- overrideGeometry = "line";
+ 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 i3 = 0; i3 < summary.length; i3++) {
+ var item = summary[i3];
+ if (item.changeType === "modified") {
+ _toCheck.push(item.entity.id);
}
}
- 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 (i3 = 0; i3 < primaries.length; i3++) {
- k2 = primaries[i3];
- v2 = t2[k2];
- if (!v2 || v2 === "no")
- continue;
- if (k2 === "piste:type") {
- k2 = "piste";
- } else if (k2 === "building:part") {
- k2 = "building_part";
- }
- primary = k2;
- if (statuses.indexOf(v2) !== -1) {
- status = v2;
- classes.push("tag-" + k2);
- } else {
- classes.push("tag-" + k2);
- classes.push("tag-" + k2 + "-" + v2);
- }
- break;
+ var _toLoad = withChildNodes(_toCheck, localGraph);
+ var _loaded = {};
+ var _toLoadCount = 0;
+ var _toLoadTotal = _toLoad.length;
+ if (_toCheck.length) {
+ dispatch14.call("progressChanged", this, _toLoadCount, _toLoadTotal);
+ _toLoad.forEach(function(id2) {
+ _loaded[id2] = false;
+ });
+ osm.loadMultiple(_toLoad, loaded);
+ } else {
+ upload(changeset);
}
- if (!primary) {
- for (i3 = 0; i3 < statuses.length; i3++) {
- for (j3 = 0; j3 < primaries.length; j3++) {
- k2 = statuses[i3] + ":" + primaries[j3];
- v2 = t2[k2];
- if (!v2 || v2 === "no")
- continue;
- status = statuses[i3];
- break;
+ return;
+ function withChildNodes(ids, graph) {
+ var s2 = 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) {
+ s2.add(child.id);
+ }
+ });
+ });
+ return Array.from(s2);
+ }
+ 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 i4, id2;
+ if (entity.type === "way") {
+ for (i4 = 0; i4 < entity.nodes.length; i4++) {
+ id2 = entity.nodes[i4];
+ if (_loaded[id2] === void 0) {
+ _loaded[id2] = false;
+ loadMore.push(id2);
+ }
+ }
+ } else if (entity.type === "relation" && entity.isMultipolygon()) {
+ for (i4 = 0; i4 < entity.members.length; i4++) {
+ id2 = entity.members[i4].id;
+ if (_loaded[id2] === void 0) {
+ _loaded[id2] = false;
+ loadMore.push(id2);
+ }
+ }
+ }
+ });
+ _toLoadCount += result.data.length;
+ _toLoadTotal += loadMore.length;
+ dispatch14.call("progressChanged", this, _toLoadCount, _toLoadTotal);
+ if (loadMore.length) {
+ _toLoad.push.apply(_toLoad, loadMore);
+ osm.loadMultiple(loadMore, loaded);
+ }
+ if (!_toLoad.length) {
+ detectConflicts();
+ upload(changeset);
}
}
}
- if (!status) {
- for (i3 = 0; i3 < statuses.length; i3++) {
- k2 = statuses[i3];
- v2 = t2[k2];
- if (!v2 || v2 === "no")
- continue;
- if (v2 === "yes") {
- status = k2;
- } else if (primary && primary === v2) {
- status = k2;
- } else if (!primary && primaries.indexOf(v2) !== -1) {
- status = k2;
- primary = v2;
- classes.push("tag-" + v2);
+ function detectConflicts() {
+ function choice(id2, text, action) {
+ return {
+ id: id2,
+ text,
+ action: function() {
+ history.replace(action);
+ }
+ };
+ }
+ function formatUser(d2) {
+ return '' + escape_default(d2) + "";
+ }
+ 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 i4 = 0; i4 < children2.length; i4++) {
+ var a2 = localGraph.hasEntity(children2[i4]);
+ var b2 = remoteGraph.hasEntity(children2[i4]);
+ if (a2 && b2 && a2.version !== b2.version)
+ return false;
+ }
}
- if (status)
- break;
+ return true;
}
+ _toCheck.forEach(function(id2) {
+ var local = localGraph.entity(id2);
+ var remote = remoteGraph.entity(id2);
+ if (sameVersions(local, remote))
+ return;
+ var merge2 = actionMergeRemoteChanges(id2, localGraph, remoteGraph, _discardTags, formatUser);
+ history.replace(merge2);
+ var mergeConflicts = merge2.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)
+ ]
+ });
+ });
}
- if (status) {
- classes.push("tag-status");
- classes.push("tag-status-" + status);
+ }
+ function upload(changeset) {
+ var osm = context.connection();
+ if (!osm) {
+ _errors.push({ msg: "No OSM Service" });
}
- for (i3 = 0; i3 < secondaries.length; i3++) {
- k2 = secondaries[i3];
- v2 = t2[k2];
- if (!v2 || v2 === "no" || k2 === primary)
- continue;
- classes.push("tag-" + k2);
- classes.push("tag-" + k2 + "-" + v2);
+ 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) {
+ dispatch14.call("willAttemptUpload", this);
+ osm.putChangeset(changeset, changes, uploadCallback);
+ } else {
+ didResultInNoChanges();
+ }
}
- if (primary === "highway" && !osmPathHighwayTagValues[t2.highway] || primary === "aeroway") {
- var surface = t2.highway === "track" ? "unpaved" : "paved";
- for (k2 in t2) {
- v2 = t2[k2];
- if (k2 in osmPavedTags) {
- surface = osmPavedTags[k2][v2] ? "paved" : "unpaved";
- }
- if (k2 in osmSemipavedTags && !!osmSemipavedTags[k2][v2]) {
- surface = "semipaved";
- }
+ }
+ 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();
}
- classes.push("tag-" + surface);
+ } else {
+ didResultInSuccess(changeset);
}
- var qid = t2.wikidata || t2["flag:wikidata"] || t2["brand:wikidata"] || t2["network:wikidata"] || t2["operator:wikidata"];
- if (qid) {
- classes.push("tag-wikidata");
+ }
+ function didResultInNoChanges() {
+ dispatch14.call("resultNoChanges", this);
+ endSave();
+ context.flush();
+ }
+ function didResultInErrors() {
+ context.history().pop();
+ dispatch14.call("resultErrors", this, _errors);
+ endSave();
+ }
+ function didResultInConflicts(changeset) {
+ _conflicts.sort(function(a2, b2) {
+ return b2.id.localeCompare(a2.id);
+ });
+ dispatch14.call("resultConflicts", this, changeset, _conflicts, _origChanges);
+ endSave();
+ }
+ function didResultInSuccess(changeset) {
+ context.history().clearSaved();
+ dispatch14.call("resultSuccess", this, changeset);
+ window.setTimeout(function() {
+ endSave();
+ context.flush();
+ }, 2500);
+ }
+ function endSave() {
+ _isSaving = false;
+ dispatch14.call("saveEnded", this);
+ }
+ uploader.cancelConflictResolution = function() {
+ context.history().pop();
+ };
+ uploader.processResolvedConflicts = function(changeset) {
+ var history = context.history();
+ for (var i3 = 0; i3 < _conflicts.length; i3++) {
+ if (_conflicts[i3].chosen === 1) {
+ var entity = context.hasEntity(_conflicts[i3].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[i3].id));
+ }
}
- return classes.filter((klass) => /^[-_a-z0-9]+$/.test(klass)).join(" ").trim();
+ uploader.save(changeset, true, false);
};
- tagClasses.tags = function(val) {
- if (!arguments.length)
- return _tags;
- _tags = val;
- return tagClasses;
+ uploader.reset = function() {
};
- return tagClasses;
+ return uploader;
}
- // 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"
+ // modules/renderer/background_source.js
+ var import_lodash4 = __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;
}
};
- function svgTagPattern(tags) {
- if (tags.building && tags.building !== "no") {
+
+ // modules/renderer/background_source.js
+ var isRetina = window.devicePixelRatio && window.devicePixelRatio >= 2;
+ 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(s2) {
+ if (!s2)
+ return null;
+ var options2 = { day: "numeric", month: "short", year: "numeric" };
+ var d2 = new Date(s2);
+ if (isNaN(d2.getTime()))
return null;
+ return d2.toLocaleDateString(_mainLocalizer.localeCode(), options2);
+ }
+ function vintageRange(vintage) {
+ var s2;
+ if (vintage.start || vintage.end) {
+ s2 = vintage.start || "?";
+ if (vintage.start !== vintage.end) {
+ s2 += " - " + (vintage.end || "?");
+ }
}
- 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;
+ return s2;
+ }
+ 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, "");
+ return _t("imagery." + id_safe + ".name", { default: (0, import_lodash4.escape)(_name) });
+ };
+ source.label = function() {
+ var id_safe = source.id.replace(/\./g, "");
+ return _t.append("imagery." + id_safe + ".name", { default: (0, import_lodash4.escape)(_name) });
+ };
+ source.hasDescription = function() {
+ var id_safe = source.id.replace(/\./g, "");
+ var descriptionText = _mainLocalizer.tInfo("imagery." + id_safe + ".description", { default: (0, import_lodash4.escape)(_description) }).text;
+ return descriptionText !== "";
+ };
+ source.description = function() {
+ var id_safe = source.id.replace(/\./g, "");
+ return _t.append("imagery." + id_safe + ".description", { default: (0, import_lodash4.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(x2, y2, z2) {
+ var zoomSize = Math.pow(2, z2);
+ var lon = x2 / zoomSize * Math.PI * 2 - Math.PI;
+ var lat = Math.atan(Math.sinh(Math.PI * (1 - 2 * y2 / 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]
+ };
}
- for (var ruleKey in rules) {
- var rule = rules[ruleKey];
- var pass = true;
- for (var criterion in rule) {
- if (criterion !== "pattern") {
- var v2 = tags[criterion];
- if (!v2 || v2 !== rule[criterion]) {
- pass = false;
- break;
- }
+ };
+ 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;
}
- }
- if (pass) {
- return "pattern-" + rule.pattern;
- }
+ 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 u2 = "";
+ for (var zoom = coord2[2]; zoom > 0; zoom--) {
+ var b2 = 0;
+ var mask = 1 << zoom - 1;
+ if ((coord2[0] & mask) !== 0)
+ b2++;
+ if ((coord2[1] & mask) !== 0)
+ b2 += 2;
+ u2 += b2.toString();
+ }
+ return u2;
+ });
}
- }
- return null;
+ result = result.replace(/\{switch:([^}]+)\}/, function(s2, r2) {
+ var subdomains = r2.split(",");
+ return subdomains[(coord2[0] + coord2[1]) % subdomains.length];
+ });
+ return result;
+ };
+ source.validZoom = function(z2) {
+ return source.zoomExtent[0] <= z2 && (source.overzoom || source.zoomExtent[1] > z2);
+ };
+ 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;
}
-
- // modules/svg/areas.js
- function svgAreas(projection2, context) {
- function getPatternStyle(tags) {
- var imageID = svgTagPattern(tags);
- if (imageID) {
- return 'url("#ideditor-' + imageID + '")';
+ rendererBackgroundSource.Bing = function(data, dispatch14) {
+ 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:".concat(subDomainNumbers, "}")).replace("{quadkey}", "{u}");
+ if (!new URLSearchParams(template).has(strictParam)) {
+ template += "&".concat(strictParam, "=z");
}
- 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(d2) {
- return filter2(d2.properties.entity);
- }).data(targetData, function key(d2) {
- return d2.id;
+ 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]])
+ };
+ })
+ };
});
- targets.exit().remove();
- var segmentWasEdited = function(d2) {
- var wayID = d2.properties.entity.id;
- if (!base.entities[wayID] || !(0, import_fast_deep_equal5.default)(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
- return false;
- }
- return d2.properties.nodes.some(function(n3) {
- return !base.entities[n3.id] || !(0, import_fast_deep_equal5.default)(graph.entities[n3.id].loc, base.entities[n3.id].loc);
+ dispatch14.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);
});
- };
- targets.enter().append("path").merge(targets).attr("d", getPath).attr("class", function(d2) {
- return "way area target target-allowed " + targetClass + d2.id;
- }).classed("segment-edited", segmentWasEdited);
- var nopeData = data.nopes.filter(getPath);
- var nopes = selection2.selectAll(".area.target-nope").filter(function(d2) {
- return filter2(d2.properties.entity);
- }).data(nopeData, function key(d2) {
- return d2.id;
});
- nopes.exit().remove();
- nopes.enter().append("path").merge(nopes).attr("d", getPath).attr("class", function(d2) {
- return "way area target target-nope " + nopeClass + d2.id;
- }).classed("segment-edited", segmentWasEdited);
+ };
+ 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";
}
- function drawAreas(selection2, graph, entities, filter2) {
- var path = svgPath(projection2, graph, true);
- var areas = {};
- var multipolygon;
- var base = context.history().base();
- for (var i3 = 0; i3 < entities.length; i3++) {
- var entity = entities[i3];
- 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 esri = rendererBackgroundSource(data);
+ var cache = {};
+ var inflight = {};
+ var _prevCenter;
+ esri.fetchTilemap = function(center) {
+ if (_prevCenter && geoSphericalDistance(center, _prevCenter) < 5e3)
+ return;
+ _prevCenter = center;
+ var z2 = 20;
+ var dummyUrl = esri.url([1, 2, 3]);
+ var x2 = Math.floor((center[0] + 180) / 360 * Math.pow(2, z2));
+ var y2 = 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, z2));
+ var tilemapUrl = dummyUrl.replace(/tile\/[0-9]+\/[0-9]+\/[0-9]+\?blankTile=false/, "tilemap") + "/" + z2 + "/" + y2 + "/" + x2 + "/8/8";
+ json_default(tilemapUrl).then(function(tilemap) {
+ if (!tilemap) {
+ throw new Error("Unknown Error");
}
- }
- var fills = Object.values(areas).filter(function hasPath(a2) {
- return path(a2.entity);
- });
- fills.sort(function areaSort(a2, b2) {
- return b2.area - a2.area;
- });
- fills = fills.map(function(a2) {
- return a2.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(d2) {
- return "areagroup area-" + d2;
- }).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))];
+ var hasTiles = true;
+ for (var i3 = 0; i3 < tilemap.data.length; i3++) {
+ if (!tilemap.data[i3]) {
+ hasTiles = false;
+ break;
+ }
}
+ esri.zoomExtent[1] = hasTiles ? 22 : 19;
+ }).catch(function() {
+ });
+ };
+ esri.getMetadata = function(center, tileCoord, callback) {
+ if (esri.id !== "EsriWorldImagery") {
+ return callback(null, {});
}
- 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);
+ 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((a2) => a2.MinMapLevel <= zoom && a2.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);
}
- }).classed("added", function(d2) {
- return !base.entities[d2.id];
- }).classed("geometry-edited", function(d2) {
- return graph.entities[d2.id] && base.entities[d2.id] && !(0, import_fast_deep_equal5.default)(graph.entities[d2.id].nodes, base.entities[d2.id].nodes);
- }).classed("retagged", function(d2) {
- return graph.entities[d2.id] && base.entities[d2.id] && !(0, import_fast_deep_equal5.default)(graph.entities[d2.id].tags, base.entities[d2.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 n3 = node.getElementsByTagName(tagName);
- const result = n3.length ? n3[0] : null;
- if (result && callback)
- callback(result);
- return result;
- }
- function get3(node, tagName, callback) {
- const properties = {};
- if (!node)
- return properties;
- const n3 = node.getElementsByTagName(tagName);
- const result = n3.length ? n3[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 (callback)
- callback(val);
- return val;
- }
- function getMulti(node, propertyNames) {
- const properties = {};
- for (const property of propertyNames) {
- val1(node, property, (val) => {
- properties[property] = val;
+ 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);
});
- }
- 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)]);
+ function clean2(val) {
+ return String(val).trim() || unknown;
}
- }
- 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 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;
- }
- 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 i3 = 0; i3 < pts.length; i3++) {
- const c2 = coordPair$1(pts[i3]);
- if (!c2) {
- continue;
- }
- line.push(c2.coordinates);
- if (c2.time)
- times.push(c2.time);
- for (const [name, val] of c2.extendedValues) {
- const plural = name === "heart" ? name : name.replace("gpxtpx:", "") + "s";
- if (!extendedValues[plural]) {
- extendedValues[plural] = Array(pts.length).fill(null);
- }
- extendedValues[plural][i3] = val;
+ 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", "Signature"].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 = {};
}
- 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
- }
- };
+ 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 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);
- }
+ function polygon(coordinates, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
}
- 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 = {};
+ 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.");
}
- const props = properties.coordinateProperties;
- const entries = Object.entries(line.extendedValues);
- for (let i3 = 0; i3 < entries.length; i3++) {
- const [name, val] = entries[i3];
- if (multi) {
- if (!props[name]) {
- props[name] = extractedLines.map((line2) => new Array(line2.line.length).fill(null));
- }
- props[name][i3] = val;
- } else {
- props[name] = val;
+ 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.");
}
}
}
- 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 pair3 = coordPair$1(node);
- if (!pair3)
- return null;
- return {
- type: "Feature",
- properties,
- geometry: {
- type: "Point",
- coordinates: pair3.coordinates
- }
+ var geom = {
+ type: "Polygon",
+ coordinates
};
+ return feature2(geom, properties, options2);
}
- 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;
+ function lineString(coordinates, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
}
- for (const waypoint of $(node, "wpt")) {
- const point2 = getPoint(waypoint);
- if (point2)
- yield point2;
+ if (coordinates.length < 2) {
+ throw new Error("coordinates must be an array of two or more positions");
}
- }
- function gpx(node) {
- return {
- type: "FeatureCollection",
- features: Array.from(gpxGen(node))
+ var geom = {
+ type: "LineString",
+ coordinates
};
+ return feature2(geom, properties, options2);
}
- function fixColor(v2, prefix) {
- const properties = {};
- const colorProp = prefix == "stroke" || prefix === "fill" ? prefix : prefix + "-color";
- if (v2[0] === "#") {
- v2 = v2.substring(1);
- }
- if (v2.length === 6 || v2.length === 3) {
- properties[colorProp] = "#" + v2;
- } else if (v2.length === 8) {
- properties[prefix + "-opacity"] = parseInt(v2.substring(0, 2), 16) / 255;
- properties[colorProp] = "#" + v2.substring(6, 8) + v2.substring(4, 6) + v2.substring(2, 4);
+ function multiLineString(coordinates, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
}
- 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 extractIconHref(node) {
- return get3(node, "Icon", (icon2, properties) => {
- val1(icon2, "href", (href) => {
- properties.icon = href;
- });
- return properties;
- });
- }
- 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 {};
- }), extractIconHref(iconStyle));
- });
- }
- 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 };
- }));
- });
+ var geom = {
+ type: "MultiLineString",
+ coordinates
+ };
+ return feature2(geom, properties, options2);
}
- function extractStyle(node) {
- return Object.assign({}, extractPoly(node), extractLine(node), extractLabel(node), extractIcon(node));
+ function multiPolygon(coordinates, properties, options2) {
+ if (options2 === void 0) {
+ options2 = {};
+ }
+ var geom = {
+ type: "MultiPolygon",
+ coordinates
+ };
+ return feature2(geom, properties, options2);
}
- var toNumber2 = (x2) => Number(x2);
- var typeConverters = {
- string: (x2) => x2,
- int: toNumber2,
- uint: toNumber2,
- short: toNumber2,
- ushort: toNumber2,
- float: toNumber2,
- double: toNumber2,
- bool: (x2) => Boolean(x2)
- };
- function extractExtendedData(node, schema) {
- 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")) {
- const name = simpleData.getAttribute("name") || "";
- const typeConverter = schema[name] || typeConverters.string;
- properties[name] = typeConverter(nodeVal(simpleData));
- }
- return properties;
- });
+
+ // node_modules/@turf/invariant/dist/es/index.js
+ function getGeom(geojson) {
+ if (geojson.type === "Feature") {
+ return geojson.geometry;
+ }
+ return geojson;
}
- function getMaybeHTMLDescription(node) {
- const descriptionNode = get1(node, "description");
- for (const c2 of Array.from(descriptionNode?.childNodes || [])) {
- if (c2.nodeType === 4) {
- return {
- description: {
- "@type": "html",
- value: nodeVal(c2)
+
+ // 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 = [], i3, codeB, lastCode;
+ var a2;
+ var b2;
+ if (!result)
+ result = [];
+ for (i3 = 1; i3 < len; i3++) {
+ a2 = points[i3 - 1];
+ b2 = points[i3];
+ codeB = lastCode = bitCode(b2, bbox2);
+ while (true) {
+ if (!(codeA | codeB)) {
+ part.push(a2);
+ if (codeB !== lastCode) {
+ part.push(b2);
+ if (i3 < len - 1) {
+ result.push(part);
+ part = [];
+ }
+ } else if (i3 === len - 1) {
+ part.push(b2);
}
- };
+ break;
+ } else if (codeA & codeB) {
+ break;
+ } else if (codeA) {
+ a2 = intersect(a2, b2, codeA, bbox2);
+ codeA = bitCode(a2, bbox2);
+ } else {
+ b2 = intersect(a2, b2, codeB, bbox2);
+ codeB = bitCode(b2, bbox2);
+ }
}
+ codeA = lastCode;
}
- return {};
- }
- 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")) };
- });
+ if (part.length)
+ result.push(part);
+ return result;
}
- function extractCascadedStyle(node, styleMap) {
- return val1(node, "styleUrl", (styleUrl) => {
- styleUrl = normalizeId(styleUrl);
- if (styleMap[styleUrl]) {
- return Object.assign({ styleUrl }, styleMap[styleUrl]);
+ function polygonclip(points, bbox2) {
+ var result, edge, prev, prevInside, i3, p2, inside;
+ for (edge = 1; edge <= 8; edge *= 2) {
+ result = [];
+ prev = points[points.length - 1];
+ prevInside = !(bitCode(prev, bbox2) & edge);
+ for (i3 = 0; i3 < points.length; i3++) {
+ p2 = points[i3];
+ inside = !(bitCode(p2, bbox2) & edge);
+ if (inside !== prevInside)
+ result.push(intersect(prev, p2, edge, bbox2));
+ if (inside)
+ result.push(p2);
+ prev = p2;
+ prevInside = inside;
}
- return { styleUrl };
- });
+ points = result;
+ if (!points.length)
+ break;
+ }
+ return result;
}
- 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 intersect(a2, b2, edge, bbox2) {
+ return edge & 8 ? [a2[0] + (b2[0] - a2[0]) * (bbox2[3] - a2[1]) / (b2[1] - a2[1]), bbox2[3]] : edge & 4 ? [a2[0] + (b2[0] - a2[0]) * (bbox2[1] - a2[1]) / (b2[1] - a2[1]), bbox2[1]] : edge & 2 ? [bbox2[2], a2[1] + (b2[1] - a2[1]) * (bbox2[2] - a2[0]) / (b2[0] - a2[0])] : edge & 1 ? [bbox2[0], a2[1] + (b2[1] - a2[1]) * (bbox2[0] - a2[0]) / (b2[0] - a2[0])] : null;
}
- function coord(value) {
- return value.replace(trimSpace, "").split(splitSpace).map(coord1).filter((coord2) => {
- return coord2.length >= 2;
- });
+ function bitCode(p2, bbox2) {
+ var code = 0;
+ if (p2[0] < bbox2[0])
+ code |= 1;
+ else if (p2[0] > bbox2[2])
+ code |= 2;
+ if (p2[1] < bbox2[1])
+ code |= 4;
+ else if (p2[1] > bbox2[3])
+ code |= 8;
+ return code;
}
- 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;
+
+ // 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");
}
- 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 i3 = 0; i3 < Math.max(first.length, last.length); i3++) {
- if (first[i3] !== last[i3]) {
- equal = false;
- break;
+ 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);
+ }
}
}
- if (!equal) {
- return ring.concat([ring[0]]);
- }
- return ring;
- }
- function getCoordinates(node) {
- return nodeVal(get1(node, "coordinates"));
+ return outRings;
}
- function getGeometry(node) {
- let geometries = [];
- let coordTimes = [];
- for (let i3 = 0; i3 < node.childNodes.length; i3++) {
- const child = node.childNodes.item(i3);
- if (isElement(child)) {
- switch (child.tagName) {
- case "MultiGeometry":
- case "MultiTrack":
- case "gx:MultiTrack": {
- const childGeometries = getGeometry(child);
- geometries = geometries.concat(childGeometries.geometries);
- coordTimes = coordTimes.concat(childGeometries.coordTimes);
+
+ // node_modules/@turf/meta/dist/es/index.js
+ function coordEach(geojson, callback, excludeWrapCoord) {
+ if (geojson === null)
+ return;
+ var j2, k2, l2, 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": {
- const coordinates = coord1(getCoordinates(child));
- if (coordinates.length >= 2) {
- geometries.push({
- type: "Point",
- coordinates
- });
- }
+ case "Point":
+ if (callback(
+ coords,
+ coordIndex,
+ featureIndex,
+ multiFeatureIndex,
+ geometryIndex
+ ) === false)
+ return false;
+ coordIndex++;
+ multiFeatureIndex++;
break;
- }
- case "LinearRing":
- case "LineString": {
- const coordinates = coord(getCoordinates(child));
- if (coordinates.length >= 2) {
- geometries.push({
- type: "LineString",
- coordinates
- });
+ 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": {
- const coords = [];
- for (const linearRing of $(child, "LinearRing")) {
- const ring = fixRing(coord(getCoordinates(linearRing)));
- if (ring.length >= 4) {
- coords.push(ring);
+ case "Polygon":
+ case "MultiLineString":
+ for (j2 = 0; j2 < coords.length; j2++) {
+ for (k2 = 0; k2 < coords[j2].length - wrapShrink; k2++) {
+ if (callback(
+ coords[j2][k2],
+ coordIndex,
+ featureIndex,
+ multiFeatureIndex,
+ geometryIndex
+ ) === false)
+ return false;
+ coordIndex++;
}
+ if (geomType === "MultiLineString")
+ multiFeatureIndex++;
+ if (geomType === "Polygon")
+ geometryIndex++;
}
- if (coords.length) {
- geometries.push({
- type: "Polygon",
- coordinates: coords
- });
+ if (geomType === "Polygon")
+ multiFeatureIndex++;
+ break;
+ case "MultiPolygon":
+ for (j2 = 0; j2 < coords.length; j2++) {
+ geometryIndex = 0;
+ for (k2 = 0; k2 < coords[j2].length; k2++) {
+ for (l2 = 0; l2 < coords[j2][k2].length - wrapShrink; l2++) {
+ if (callback(
+ coords[j2][k2][l2],
+ coordIndex,
+ featureIndex,
+ multiFeatureIndex,
+ geometryIndex
+ ) === false)
+ return false;
+ coordIndex++;
+ }
+ geometryIndex++;
+ }
+ multiFeatureIndex++;
}
break;
- }
- case "Track":
- case "gx:Track": {
- const gx = gxCoords(child);
- if (!gx)
- break;
- const { times, geometry } = gx;
- geometries.push(geometry);
- if (times.length)
- coordTimes.push(times);
+ 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");
}
}
}
- return {
- geometries,
- coordTimes
- };
}
- function geometryListToGeometry(geometries) {
- return geometries.length === 0 ? null : geometries.length === 1 ? geometries[0] : {
- type: "GeometryCollection",
- geometries
- };
+
+ // 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;
}
- function getPlacemark(node, styleMap, schema, options2) {
- const { coordTimes, geometries } = getGeometry(node);
- const geometry = geometryListToGeometry(geometries);
- if (!geometry && options2.skipNullGeometry) {
- return null;
+ bbox["default"] = bbox;
+ var es_default = bbox;
+
+ // modules/renderer/background.js
+ var import_which_polygon3 = __toESM(require_which_polygon());
+
+ // modules/renderer/tile_layer.js
+ function rendererTileLayer(context) {
+ var transformProp = utilPrefixCSSProperty("Transform");
+ var tiler9 = utilTiler();
+ var _tileSize = 256;
+ var _projection;
+ var _cache5 = {};
+ var _tileOrigin;
+ var _zoom;
+ var _source;
+ function tileSizeAtZoom(d2, z2) {
+ var EPSILON = 2e-3;
+ return _tileSize * Math.pow(2, z2 - d2[2]) / _tileSize + EPSILON;
}
- const feature3 = {
- type: "Feature",
- geometry,
- properties: Object.assign(getMulti(node, [
- "name",
- "address",
- "visibility",
- "open",
- "phoneNumber",
- "description"
- ]), getMaybeHTMLDescription(node), extractCascadedStyle(node, styleMap), extractStyle(node), extractExtendedData(node, schema), 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";
+ function atZoom(t2, distance) {
+ var power = Math.pow(2, distance);
+ return [
+ Math.floor(t2[0] * power),
+ Math.floor(t2[1] * power),
+ t2[2] + distance
+ ];
}
- const id2 = node.getAttribute("id");
- if (id2 !== null && id2 !== "")
- feature3.id = id2;
- return feature3;
- }
- function getGroundOverlayBox(node) {
- const latLonQuad = get1(node, "gx:LatLonQuad");
- if (latLonQuad) {
- const ring = fixRing(coord(getCoordinates(node)));
- return {
- geometry: {
- type: "Polygon",
- coordinates: [ring]
+ function lookUp(d2) {
+ for (var up = -1; up > -d2[2]; up--) {
+ var tile = atZoom(d2, up);
+ if (_cache5[_source.url(tile)] !== false) {
+ return tile;
}
- };
+ }
}
- return getLatLonBox(node);
- }
- var DEGREES_TO_RADIANS = Math.PI / 180;
- function rotateBox(bbox2, coordinates, rotation) {
- const center = [(bbox2[0] + bbox2[2]) / 2, (bbox2[1] + bbox2[3]) / 2];
- return [
- coordinates[0].map((coordinate) => {
- const dy = coordinate[1] - center[1];
- const dx = coordinate[0] - center[0];
- const distance = Math.sqrt(Math.pow(dy, 2) + Math.pow(dx, 2));
- const angle2 = Math.atan2(dy, dx) + rotation * DEGREES_TO_RADIANS;
- return [
- center[0] + Math.cos(angle2) * distance,
- center[1] + Math.sin(angle2) * distance
- ];
- })
- ];
- }
- function getLatLonBox(node) {
- const latLonBox = get1(node, "LatLonBox");
- if (latLonBox) {
- const north = num1(latLonBox, "north");
- const west = num1(latLonBox, "west");
- const east = num1(latLonBox, "east");
- const south = num1(latLonBox, "south");
- const rotation = num1(latLonBox, "rotation");
- if (typeof north === "number" && typeof south === "number" && typeof west === "number" && typeof east === "number") {
- const bbox2 = [west, south, east, north];
- let coordinates = [
- [
- [west, north],
- [east, north],
- [east, south],
- [west, south],
- [west, north]
- // top left (again)
- ]
- ];
- if (typeof rotation === "number") {
- coordinates = rotateBox(bbox2, coordinates, rotation);
+ function uniqueBy(a2, n3) {
+ var o2 = [];
+ var seen = {};
+ for (var i3 = 0; i3 < a2.length; i3++) {
+ if (seen[a2[i3][n3]] === void 0) {
+ o2.push(a2[i3]);
+ seen[a2[i3][n3]] = true;
}
- return {
- bbox: bbox2,
- geometry: {
- type: "Polygon",
- coordinates
- }
- };
}
+ return o2;
}
- return null;
- }
- function getGroundOverlay(node, styleMap, schema, options2) {
- const box = getGroundOverlayBox(node);
- const geometry = box?.geometry || null;
- if (!geometry && options2.skipNullGeometry) {
- return null;
- }
- const feature3 = {
- type: "Feature",
- geometry,
- properties: Object.assign(
- /**
- * Related to
- * https://gist.github.com/tmcw/037a1cb6660d74a392e9da7446540f46
- */
- { "@geometry-type": "groundoverlay" },
- getMulti(node, [
- "name",
- "address",
- "visibility",
- "open",
- "phoneNumber",
- "description"
- ]),
- getMaybeHTMLDescription(node),
- extractCascadedStyle(node, styleMap),
- extractStyle(node),
- extractIconHref(node),
- extractExtendedData(node, schema),
- extractTimeSpan(node),
- extractTimeStamp(node)
- )
- };
- if (box?.bbox) {
- feature3.bbox = box.bbox;
- }
- 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");
+ function addSource(d2) {
+ d2.push(_source.url(d2));
+ return d2;
}
- return normalizeId(id2 || "");
- }
- function buildStyleMap(node) {
- const styleMap = {};
- for (const style of $(node, "Style")) {
- styleMap[getStyleId(style)] = extractStyle(style);
+ 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]
+ ];
+ tiler9.scale(_projection.scale() * 2 * Math.PI).translate(translate);
+ _tileOrigin = [
+ _projection.scale() * Math.PI - translate[0],
+ _projection.scale() * Math.PI - translate[1]
+ ];
+ render(selection2);
}
- 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];
+ function render(selection2) {
+ if (!_source)
+ return;
+ var requests = [];
+ var showDebug = context.getDebug("tile") && !_source.overlay;
+ if (_source.validZoom(_zoom)) {
+ tiler9.skipNullIsland(!!_source.overlay);
+ tiler9().forEach(function(d2) {
+ addSource(d2);
+ if (d2[3] === "")
+ return;
+ if (typeof d2[3] !== "string")
+ return;
+ requests.push(d2);
+ if (_cache5[d2[3]] === false && lookUp(d2)) {
+ requests.push(addSource(lookUp(d2)));
+ }
+ });
+ requests = uniqueBy(requests, 3).filter(function(r2) {
+ return _cache5[r2[3]] !== false;
+ });
+ }
+ function load(d3_event, d2) {
+ _cache5[d2[3]] = true;
+ select_default2(this).on("error", null).on("load", null).classed("tile-loaded", true);
+ render(selection2);
+ }
+ function error(d3_event, d2) {
+ _cache5[d2[3]] = false;
+ select_default2(this).on("error", null).on("load", null).remove();
+ render(selection2);
+ }
+ function imageTransform(d2) {
+ var ts = _tileSize * Math.pow(2, _zoom - d2[2]);
+ var scale = tileSizeAtZoom(d2, _zoom);
+ return "translate(" + (d2[0] * ts - _tileOrigin[0]) + "px," + (d2[1] * ts - _tileOrigin[1]) + "px) scale(" + scale + "," + scale + ")";
+ }
+ function tileCenter(d2) {
+ var ts = _tileSize * Math.pow(2, _zoom - d2[2]);
+ return [
+ d2[0] * ts - _tileOrigin[0] + ts / 2,
+ d2[1] * ts - _tileOrigin[1] + ts / 2
+ ];
+ }
+ function debugTransform(d2) {
+ var coord2 = tileCenter(d2);
+ return "translate(" + coord2[0] + "px," + coord2[1] + "px)";
+ }
+ var dims = tiler9.size();
+ var mapCenter = [dims[0] / 2, dims[1] / 2];
+ var minDist = Math.max(dims[0], dims[1]);
+ var nearCenter;
+ requests.forEach(function(d2) {
+ var c2 = tileCenter(d2);
+ var dist = geoVecLength(c2, mapCenter);
+ if (dist < minDist) {
+ minDist = dist;
+ nearCenter = d2;
}
});
+ var image = selection2.selectAll("img").data(requests, function(d2) {
+ return d2[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(d2) {
+ return d2[3];
+ }).on("error", error).on("load", load).merge(image).style(transformProp, imageTransform).classed("tile-debug", showDebug).classed("tile-removing", false).classed("tile-center", function(d2) {
+ return d2 === nearCenter;
+ });
+ var debug2 = selection2.selectAll(".tile-label-debug").data(showDebug ? requests : [], function(d2) {
+ return d2[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(d2) {
+ return d2[2] + " / " + d2[0] + " / " + d2[1];
+ });
+ debug2.selectAll(".tile-label-debug-vintage").each(function(d2) {
+ var span = select_default2(this);
+ var center = context.projection.invert(tileCenter(d2));
+ _source.getMetadata(center, d2, 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"));
+ }
+ });
+ });
+ }
}
- return styleMap;
- }
- function buildSchema(node) {
- const schema = {};
- for (const field of $(node, "SimpleField")) {
- schema[field.getAttribute("name") || ""] = typeConverters[field.getAttribute("type") || ""] || typeConverters["string"];
- }
- return schema;
- }
- function* kmlGen(node, options2 = {
- skipNullGeometry: false
- }) {
- const styleMap = buildStyleMap(node);
- const schema = buildSchema(node);
- for (const placemark of $(node, "Placemark")) {
- const feature3 = getPlacemark(placemark, styleMap, schema, options2);
- if (feature3)
- yield feature3;
- }
- for (const groundOverlay of $(node, "GroundOverlay")) {
- const feature3 = getGroundOverlay(groundOverlay, styleMap, schema, options2);
- if (feature3)
- yield feature3;
- }
- }
- function kml(node, options2 = {
- skipNullGeometry: false
- }) {
- return {
- type: "FeatureCollection",
- features: Array.from(kmlGen(node, options2))
+ background.projection = function(val) {
+ if (!arguments.length)
+ return _projection;
+ _projection = val;
+ return background;
+ };
+ background.dimensions = function(val) {
+ if (!arguments.length)
+ return tiler9.size();
+ tiler9.size(val);
+ return background;
+ };
+ background.source = function(val) {
+ if (!arguments.length)
+ return _source;
+ _source = val;
+ _tileSize = _source.tileSize;
+ _cache5 = {};
+ tiler9.tileSize(_source.tileSize).zoomExtent(_source.zoomExtent);
+ return background;
};
+ return background;
}
- // modules/svg/data.js
- var _initialized = false;
- var _enabled = false;
- var _geojson;
- function svgData(projection2, context, dispatch14) {
- var throttledRedraw = throttle_default(function() {
- dispatch14.call("change");
- }, 1e3);
- var _showLabels = true;
- var detected = utilDetect();
- var layer = select_default2(null);
- var _vtService;
- var _fileList;
- var _template;
- var _src;
- const supportedFormats = [
- ".gpx",
- ".kml",
- ".geojson",
- ".json"
- ];
- 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;
- var f3 = d3_event.dataTransfer.files[0];
- var extension = getExtension(f3.name);
- if (!supportedFormats.includes(extension))
- 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() {
- dispatch14.call("change");
+ // modules/renderer/background.js
+ var _imageryIndex = null;
+ function rendererBackground(context) {
+ const dispatch14 = 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_polygon3.default)({ type: "FeatureCollection", features });
+ _imageryIndex.backgrounds = sources.map((source) => {
+ if (source.type === "bing") {
+ return rendererBackgroundSource.Bing(source, dispatch14);
+ } 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 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 i3 = 0; i3 < gj.features.length; i3++) {
- ensureFeatureID(gj.features[i3]);
+ 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);
}
- } else {
- ensureFeatureID(gj);
}
- return gj;
+ const sources = background.sources(context.map().extent());
+ const wasValid = _isValid;
+ _isValid = !!sources.filter((d2) => d2 === currSource).length;
+ if (wasValid !== _isValid) {
+ background.updateImagery();
+ }
+ let baseFilter = "";
+ if (_brightness !== 1) {
+ baseFilter += " brightness(".concat(_brightness, ")");
+ }
+ if (_contrast !== 1) {
+ baseFilter += " contrast(".concat(_contrast, ")");
+ }
+ if (_saturation !== 1) {
+ baseFilter += " saturate(".concat(_saturation, ")");
+ }
+ if (_sharpness < 1) {
+ const blur = number_default(0.5, 5)(1 - _sharpness);
+ baseFilter += " blur(".concat(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(".concat(contrast, ")");
+ let brightness = number_default(1, 0.85)(_sharpness - 1);
+ maskFilter += " brightness(".concat(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, (d2) => d2.source().name());
+ overlays.exit().remove();
+ overlays.enter().insert("div", ".layer-data").attr("class", "layer layer-overlay").merge(overlays).each((layer, i3, nodes) => select_default2(nodes[i3]).call(layer));
}
- function ensureFeatureID(feature3) {
- if (!feature3)
+ background.updateImagery = function() {
+ let currSource = baseLayer.source();
+ if (context.inIntro() || !currSource)
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;
+ let o2 = _overlayLayers.filter((d2) => !d2.source().isLocatorOverlay() && !d2.source().isHidden()).map((d2) => d2.source().id).join(",");
+ const meters = geoOffsetToMeters(currSource.offset());
+ const EPSILON = 0.01;
+ const x2 = +meters[0].toFixed(2);
+ const y2 = +meters[1].toFixed(2);
+ let hash = utilStringQs(window.location.hash);
+ let id2 = currSource.id;
+ if (id2 === "custom") {
+ id2 = "custom:".concat(currSource.template());
+ }
+ if (id2) {
+ hash.background = id2;
} else {
- return [gj];
+ delete hash.background;
}
- }
- function featureKey(d2) {
- return d2.__featurehash__;
- }
- function isPolygon(d2) {
- return d2.geometry.type === "Polygon" || d2.geometry.type === "MultiPolygon";
- }
- function clipPathID(d2) {
- return "ideditor-data-" + d2.__featurehash__ + "-clippath";
- }
- function featureClasses(d2) {
- return [
- "data" + d2.__featurehash__,
- d2.geometry.type,
- isPolygon(d2) ? "area" : "",
- d2.__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);
+ if (o2) {
+ hash.overlays = o2;
} else {
- geoData = getFeatures(_geojson);
+ delete hash.overlays;
}
- 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(d2) {
- return "datagroup datagroup-" + d2;
- }).merge(datagroups);
- var pathData = {
- fill: polygonData,
- shadow: geoData,
- stroke: geoData
+ if (Math.abs(x2) > EPSILON || Math.abs(y2) > EPSILON) {
+ hash.offset = "".concat(x2, ",").concat(y2);
+ } 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((d2) => !d2.source().isLocatorOverlay() && !d2.source().isHidden()).forEach((d2) => imageryUsed.push(d2.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",
+ vegbilder: "Norwegian Road Administration Images",
+ mapilio: "Mapilio Images"
};
- var paths = datagroups.selectAll("path").data(function(layer2) {
- return pathData[layer2];
- }, featureKey);
- paths.exit().remove();
- paths = paths.enter().append("path").attr("class", function(d2) {
- var datagroup = this.parentNode.__data__;
- return "pathdata " + datagroup + " " + featureClasses(d2);
- }).attr("clip-path", function(d2) {
- var datagroup = this.parentNode.__data__;
- return datagroup === "fill" ? "url(#" + clipPathID(d2) + ")" : null;
- }).merge(paths).attr("d", function(d2) {
- var datagroup = this.parentNode.__data__;
- return datagroup === "fill" ? getAreaPath(d2) : getPath(d2);
- });
- 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(d2) {
- return _showLabels && d2.properties && (d2.properties.desc || d2.properties.name);
- });
- var labels = selection3.selectAll("text." + textClass).data(labelData, featureKey);
- labels.exit().remove();
- labels = labels.enter().append("text").attr("class", function(d2) {
- return textClass + " " + featureClasses(d2);
- }).merge(labels).text(function(d2) {
- return d2.properties.desc || d2.properties.name;
- }).attr("x", function(d2) {
- var centroid = labelPath.centroid(d2);
- return centroid[0] + 11;
- }).attr("y", function(d2) {
- var centroid = labelPath.centroid(d2);
- return centroid[1];
+ 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((d2) => visible[d2.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));
}
- }
- function getExtension(fileName) {
- if (!fileName)
+ 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;
- var re3 = /\.(gpx|kml|(geo)?json|png)$/i;
- var match = fileName.toLowerCase().match(re3);
- 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);
+ baseLayer.dimensions(val);
+ _overlayLayers.forEach((layer) => layer.dimensions(val));
+ };
+ background.baseLayerSource = function(d2) {
+ if (!arguments.length)
+ return baseLayer.source();
+ const osm = context.connection();
+ if (!osm)
+ return background;
+ const blocklists = osm.imageryBlocklists();
+ const template = d2.template();
+ let fail = false;
+ let tested = 0;
+ let regex;
+ for (let i3 = 0; i3 < blocklists.length; i3++) {
+ regex = blocklists[i3];
+ fail = regex.test(template);
+ tested++;
+ if (fail)
+ break;
+ }
+ if (!tested) {
+ regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
+ fail = regex.test(template);
+ }
+ baseLayer.source(!fail ? d2 : background.findSource("none"));
+ dispatch14.call("change");
+ background.updateImagery();
+ return background;
+ };
+ background.findSource = (id2) => {
+ if (!id2 || !_imageryIndex)
+ return null;
+ return _imageryIndex.backgrounds.find((d2) => d2.id && d2.id === id2);
+ };
+ background.bing = () => {
+ background.baseLayerSource(background.findSource("Bing"));
+ };
+ background.showsLayer = (d2) => {
+ const currSource = baseLayer.source();
+ if (!d2 || !currSource)
+ return false;
+ return d2.id === currSource.id || _overlayLayers.some((layer) => d2.id === layer.source().id);
+ };
+ background.overlayLayerSources = () => {
+ return _overlayLayers.map((layer) => layer.source());
+ };
+ background.toggleOverlayLayer = (d2) => {
+ let layer;
+ for (let i3 = 0; i3 < _overlayLayers.length; i3++) {
+ layer = _overlayLayers[i3];
+ if (layer.source() === d2) {
+ _overlayLayers.splice(i3, 1);
+ dispatch14.call("change");
+ background.updateImagery();
+ return;
+ }
+ }
+ layer = rendererTileLayer(context).source(d2).projection(context.projection).dimensions(
+ baseLayer.dimensions()
+ );
+ _overlayLayers.push(layer);
+ dispatch14.call("change");
+ background.updateImagery();
+ };
+ background.nudge = (d2, zoom) => {
+ const currSource = baseLayer.source();
+ if (currSource) {
+ currSource.nudge(d2, zoom);
+ dispatch14.call("change");
+ background.updateImagery();
+ }
+ return background;
+ };
+ background.offset = function(d2) {
+ const currSource = baseLayer.source();
+ if (!arguments.length) {
+ return currSource && currSource.offset() || [0, 0];
+ }
+ if (currSource) {
+ currSource.offset(d2);
+ dispatch14.call("change");
+ background.updateImagery();
+ }
+ return background;
+ };
+ background.brightness = function(d2) {
+ if (!arguments.length)
+ return _brightness;
+ _brightness = d2;
+ if (context.mode())
+ dispatch14.call("change");
+ return background;
+ };
+ background.contrast = function(d2) {
+ if (!arguments.length)
+ return _contrast;
+ _contrast = d2;
+ if (context.mode())
+ dispatch14.call("change");
+ return background;
+ };
+ background.saturation = function(d2) {
+ if (!arguments.length)
+ return _saturation;
+ _saturation = d2;
+ if (context.mode())
+ dispatch14.call("change");
+ return background;
+ };
+ background.sharpness = function(d2) {
+ if (!arguments.length)
+ return _sharpness;
+ _sharpness = d2;
+ if (context.mode())
+ dispatch14.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((d2) => d2.id !== "none" && d2.id !== "custom");
+ const first = validBackgrounds.length && validBackgrounds[0];
+ const isLastUsedValid = !!validBackgrounds.find((d2) => d2.id && d2.id === lastUsedBackground);
+ let best;
+ if (!requestedBackground && extent) {
+ const viewArea = extent.area();
+ best = validBackgrounds.find((s2) => {
+ if (!s2.best() || s2.overlay)
+ return false;
+ let bbox2 = es_default(bboxClip(
+ { type: "MultiPolygon", coordinates: [s2.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((d2) => d2.overlay && d2.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((n3) => !isNaN(n3) && n3);
+ if (offset.length === 2) {
+ background.offset(geoMetersToOffset(offset));
+ }
+ }
+ }).catch((err) => {
+ console.error(err);
+ });
+ };
+ return utilRebind(background, dispatch14, "on");
+ }
+
+ // modules/renderer/features.js
+ function rendererFeatures(context) {
+ var dispatch14 = dispatch_default("change", "redraw");
+ var features = utilRebind({}, dispatch14, "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 _cache5 = {};
+ 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();
+ dispatch14.call("change");
+ dispatch14.call("redraw");
}
- 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;
+ function defineRule(k2, filter2, max3) {
+ var isEnabled = true;
+ _keys.push(k2);
+ _rules[k2] = {
+ 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;
}
- gj = gj || {};
- if (Object.keys(gj).length) {
- _geojson = ensureIDs(gj);
- _src = extension + " data file";
- this.fitZoom();
+ var strings = Object.keys(tags);
+ for (var i3 = 0; i3 < strings.length; i3++) {
+ var s2 = strings[i3];
+ if (osmLifecyclePrefixes[s2] || osmLifecyclePrefixes[tags[s2]])
+ return true;
}
- dispatch14.call("change");
- return this;
+ return false;
+ });
+ defineRule("others", function isOther(tags, geometry) {
+ return geometry === "line" || geometry === "area";
+ });
+ features.features = function() {
+ return _rules;
};
- drawData.showLabels = function(val) {
- if (!arguments.length)
- return _showLabels;
- _showLabels = val;
- return this;
+ features.keys = function() {
+ return _keys;
};
- drawData.enabled = function(val) {
- if (!arguments.length)
- return _enabled;
- _enabled = val;
- if (_enabled) {
- showLayer();
- } else {
- hideLayer();
+ features.enabled = function(k2) {
+ if (!arguments.length) {
+ return _keys.filter(function(k3) {
+ return _rules[k3].enabled;
+ });
}
- dispatch14.call("change");
- return this;
- };
- drawData.hasData = function() {
- var gj = _geojson || {};
- return !!(_template || Object.keys(gj).length);
+ return _rules[k2] && _rules[k2].enabled;
};
- 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 i3 = 0; i3 < blocklists.length; i3++) {
- regex = blocklists[i3];
- fail = regex.test(val);
- tested++;
- if (fail)
- break;
- }
- if (!tested) {
- regex = /.*\.google(apis)?\..*\/(vt|kh)[\?\/].*([xyz]=.*){3}.*/;
- fail = regex.test(val);
- }
+ features.disabled = function(k2) {
+ if (!arguments.length) {
+ return _keys.filter(function(k3) {
+ return !_rules[k3].enabled;
+ });
}
- _template = val;
- _fileList = null;
- _geojson = null;
- _src = src || "vectortile:" + val.split(/[?#]/)[0];
- dispatch14.call("change");
- return this;
+ return _rules[k2] && !_rules[k2].enabled;
};
- 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";
+ features.hidden = function(k2) {
+ if (!arguments.length) {
+ return _keys.filter(function(k3) {
+ return _rules[k3].hidden();
+ });
}
- dispatch14.call("change");
- return this;
- };
- drawData.fileList = function(fileList) {
- if (!arguments.length)
- return _fileList;
- _template = null;
- _geojson = null;
- _src = null;
- _fileList = fileList;
- if (!fileList || !fileList.length)
- return this;
- var f3 = fileList[0];
- var extension = getExtension(f3.name);
- var reader = new FileReader();
- reader.onload = function() {
- return function(e3) {
- drawData.setFile(extension, e3.target.result);
- };
- }(f3);
- reader.readAsText(f3);
- return this;
+ return _rules[k2] && _rules[k2].hidden();
};
- 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() {
+ features.autoHidden = function(k2) {
+ if (!arguments.length) {
+ return _keys.filter(function(k3) {
+ return _rules[k3].autoHidden();
});
- } else {
- drawData.template(url);
}
- return this;
+ return _rules[k2] && _rules[k2].autoHidden();
};
- drawData.getSrc = function() {
- return _src || "";
+ features.enable = function(k2) {
+ if (_rules[k2] && !_rules[k2].enabled) {
+ _rules[k2].enable();
+ update();
+ }
};
- 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 c2 = geom.coordinates;
- switch (geom.type) {
- case "Point":
- c2 = [c2];
- case "MultiPoint":
- case "LineString":
- break;
- case "MultiPolygon":
- c2 = utilArrayFlatten(c2);
- case "Polygon":
- case "MultiLineString":
- c2 = utilArrayFlatten(c2);
- break;
+ features.enableAll = function() {
+ var didEnable = false;
+ for (var k2 in _rules) {
+ if (!_rules[k2].enabled) {
+ didEnable = true;
+ _rules[k2].enable();
}
- return utilArrayUnion(coords2, c2);
- }, []);
- if (!geoPolygonIntersectsPolygon(viewport, coords, true)) {
- var extent = geoExtent(bounds_default({ type: "LineString", coordinates: coords }));
- map2.centerZoom(extent.center(), map2.trimmedExtentZoom(extent));
}
- return this;
+ if (didEnable)
+ update();
};
- 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, (d2) => d2.label);
- legendItems.exit().remove();
- legendItems.enter().append("span").attr("class", (d2) => `debug-legend-item ${d2.class}`).text((d2) => d2.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((d2) => {
- const hits = showImagery && d2.query.bbox(extent.rectangle(), true) || [];
- const features = hits.map((d4) => d4.features[d4.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]
- ]]
- }
- };
- });
+ features.disable = function(k2) {
+ if (_rules[k2] && _rules[k2].enabled) {
+ _rules[k2].disable();
+ update();
}
- 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;
+ };
+ features.disableAll = function() {
+ var didDisable = false;
+ for (var k2 in _rules) {
+ if (_rules[k2].enabled) {
+ didDisable = true;
+ _rules[k2].disable();
+ }
}
+ if (didDisable)
+ update();
};
- return drawDebug;
- }
-
- // modules/svg/defs.js
- function svgDefs(context) {
- var _defsSelection = select_default2(null);
- var _spritesheetIds = [
- "iD-sprite",
- "maki-sprite",
- "temaki-sprite",
- "fa-sprite",
- "roentgen-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);
+ features.toggle = function(k2) {
+ if (_rules[k2]) {
+ (function(f2) {
+ return f2.enabled ? f2.disable() : f2.enable();
+ })(_rules[k2]);
+ update();
}
- 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(d2) {
- return "ideditor-pattern-" + d2[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(d2) {
- return "pattern-color-" + d2[0];
- });
- patterns2.append("image").attr("x", 0).attr("y", 0).attr("width", 32).attr("height", 32).attr("xlink:href", function(d2) {
- return context.imagePath("pattern/" + d2[1] + ".png");
- });
- _defsSelection.selectAll("clipPath").data([12, 18, 20, 32, 45]).enter().append("clipPath").attr("id", function(d2) {
- return "ideditor-clip-square-" + d2;
- }).append("rect").attr("x", 0).attr("y", 0).attr("width", function(d2) {
- return d2;
- }).attr("height", function(d2) {
- return d2;
- });
- 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(d2) {
- return "spritesheet spritesheet-" + d2;
- }).each(function(d2) {
- var url = context.imagePath(d2 + ".svg");
- var node = select_default2(this).node();
- svg(url).then(function(svg2) {
- node.appendChild(
- select_default2(svg2.documentElement).attr("id", "ideditor-" + d2).node()
- );
- if (overrideColors && d2 !== "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, dispatch14) {
- const throttledRedraw = throttle_default(() => dispatch14.call("change"), 1e3);
- const minZoom4 = 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;
+ };
+ features.resetStats = function() {
+ for (var i3 = 0; i3 < _keys.length; i3++) {
+ _rules[_keys[i3]].count = 0;
}
- return _qaService;
- }
- function editOn() {
- if (!layerVisible) {
- layerVisible = true;
- drawLayer.style("display", "block");
+ dispatch14.call("change");
+ };
+ features.gatherStats = function(d2, resolver, dimensions) {
+ var needsRedraw = false;
+ var types = utilArrayGroupBy(d2, "type");
+ var entities = [].concat(types.relation || [], types.way || [], types.node || []);
+ var currHidden, geometry, matches, i3, j2;
+ for (i3 = 0; i3 < _keys.length; i3++) {
+ _rules[_keys[i3]].count = 0;
}
- }
- function editOff() {
- if (layerVisible) {
- layerVisible = false;
- drawLayer.style("display", "none");
- drawLayer.selectAll(".qaItem.keepRight").remove();
- touchLayer.selectAll(".qaItem.keepRight").remove();
+ _cullFactor = dimensions[0] * dimensions[1] / 1e6;
+ for (i3 = 0; i3 < entities.length; i3++) {
+ geometry = entities[i3].geometry(resolver);
+ matches = Object.keys(features.getMatches(entities[i3], resolver, geometry));
+ for (j2 = 0; j2 < matches.length; j2++) {
+ _rules[matches[j2]].count++;
+ }
}
- }
- function layerOn() {
- editOn();
- drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch14.call("change"));
- }
- function layerOff() {
- throttledRedraw.cancel();
- drawLayer.interrupt();
- touchLayer.selectAll(".qaItem.keepRight").remove();
- drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
- editOff();
+ currHidden = features.hidden();
+ if (currHidden !== _hidden) {
+ _hidden = currHidden;
+ needsRedraw = true;
dispatch14.call("change");
+ }
+ return needsRedraw;
+ };
+ features.stats = function() {
+ for (var i3 = 0; i3 < _keys.length; i3++) {
+ _stats[_keys[i3]] = _rules[_keys[i3]].count;
+ }
+ return _stats;
+ };
+ features.clear = function(d2) {
+ for (var i3 = 0; i3 < d2.length; i3++) {
+ features.clearEntity(d2[i3]);
+ }
+ };
+ features.clearEntity = function(entity) {
+ delete _cache5[osmEntity.key(entity)];
+ };
+ features.reset = function() {
+ Array.from(_deferred2).forEach(function(handle) {
+ window.cancelIdleCallback(handle);
+ _deferred2.delete(handle);
});
+ _cache5 = {};
+ };
+ function relationShouldBeChecked(relation) {
+ return relation.tags.type === "boundary";
}
- 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, (d2) => d2.id);
- markers.exit().remove();
- const markersEnter = markers.enter().append("g").attr("class", (d2) => `qaItem ${d2.service} itemId-${d2.id} itemType-${d2.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", (d2) => d2.id === selectedID).attr("transform", getTransform);
- if (touchLayer.empty())
- return;
- const fillClass = context.getDebug("target") ? "pink " : "nocolor ";
- const targets = touchLayer.selectAll(".qaItem.keepRight").data(data, (d2) => d2.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", (d2) => `qaItem ${d2.service} target ${fillClass} itemId-${d2.id}`).attr("transform", getTransform);
- function sortY(a2, b2) {
- return a2.id === selectedID ? 1 : b2.id === selectedID ? -1 : a2.severity === "error" && b2.severity !== "error" ? 1 : b2.severity === "error" && a2.severity !== "error" ? -1 : b2.loc[1] - a2.loc[1];
+ features.getMatches = function(entity, resolver, geometry) {
+ if (geometry === "vertex" || geometry === "relation" && !relationShouldBeChecked(entity))
+ return {};
+ var ent = osmEntity.key(entity);
+ if (!_cache5[ent]) {
+ _cache5[ent] = {};
}
- }
- function drawKeepRight(selection2) {
- const service = getService();
- const surface = context.surface();
- if (surface && !surface.empty()) {
- touchLayer = surface.selectAll(".data-layer.touch .layer-touch.markers");
+ if (!_cache5[ent].matches) {
+ var matches = {};
+ var hasMatch = false;
+ for (var i3 = 0; i3 < _keys.length; i3++) {
+ if (_keys[i3] === "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 (_cache5[pkey] && _cache5[pkey].matches) {
+ matches = Object.assign({}, _cache5[pkey].matches);
+ continue;
+ }
+ }
+ }
+ }
+ if (_rules[_keys[i3]].filter(entity.tags, geometry)) {
+ matches[_keys[i3]] = hasMatch = true;
+ }
+ }
+ _cache5[ent].matches = matches;
}
- 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() >= minZoom4) {
- editOn();
- service.loadIssues(projection2);
- updateMarkers();
+ return _cache5[ent].matches;
+ };
+ features.getParents = function(entity, resolver, geometry) {
+ if (geometry === "point")
+ return [];
+ var ent = osmEntity.key(entity);
+ if (!_cache5[ent]) {
+ _cache5[ent] = {};
+ }
+ if (!_cache5[ent].parents) {
+ var parents = [];
+ if (geometry === "vertex") {
+ parents = resolver.parentWays(entity);
} else {
- editOff();
+ parents = resolver.parentRelations(entity);
}
+ _cache5[ent].parents = parents;
}
- }
- drawKeepRight.enabled = function(val) {
- if (!arguments.length)
- return _layerEnabled;
- _layerEnabled = val;
- if (_layerEnabled) {
- layerOn();
+ return _cache5[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(k2) {
+ return features.hidden(k2);
+ });
+ };
+ 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 i3 = 0; i3 < parents.length; i3++) {
+ if (!features.isHidden(parents[i3], resolver, parents[i3].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 {
- layerOff();
- if (context.selectedErrorID()) {
- context.enter(modeBrowse(context));
+ childNodes = entity.nodes ? resolver.childNodes(entity) : [];
+ connections = features.getParents(entity, resolver, entity.geometry(resolver));
+ }
+ connections = childNodes.reduce(function(result, e3) {
+ return resolver.isShared(e3) ? utilArrayUnion(result, resolver.parentWays(e3)) : result;
+ }, connections);
+ return connections.some(function(e3) {
+ return features.isHidden(e3, resolver, e3.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(d2, resolver) {
+ if (!_hidden.length)
+ return d2;
+ var result = [];
+ for (var i3 = 0; i3 < d2.length; i3++) {
+ var entity = d2[i3];
+ if (!features.isHidden(entity, resolver, entity.geometry(resolver))) {
+ result.push(entity);
}
}
- dispatch14.call("change");
- return this;
+ return result;
};
- drawKeepRight.supported = () => !!getService();
- return drawKeepRight;
+ features.forceVisible = function(entityIDs) {
+ if (!arguments.length)
+ return Object.keys(_forceVisible);
+ _forceVisible = {};
+ for (var i3 = 0; i3 < entityIDs.length; i3++) {
+ _forceVisible[entityIDs[i3]] = true;
+ var entity = context.hasEntity(entityIDs[i3]);
+ 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 i3 = 0; i3 < entities.length; i3++) {
+ var geometry = entities[i3].geometry(graph);
+ features.getMatches(entities[i3], graph, geometry);
+ }
+ });
+ _deferred2.add(handle);
+ });
+ return features;
}
- // 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(d2) {
- return svgPointTransform(projection2)(d2);
- }
- 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();
+ // 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 i3, j2, nodes, isClosed, ix1, ix2, ix3, ix4, max3;
+ for (i3 = 0; i3 < parents.length; i3++) {
+ nodes = parents[i3].nodes;
+ isClosed = parents[i3].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;
+ }
}
}
- drawLocation.enabled = function(position, enabled) {
- if (!arguments.length)
- return svgGeolocate.enabled;
- _position = position;
- svgGeolocate.enabled = enabled;
- if (svgGeolocate.enabled) {
- showLayer();
- layerOn();
- } else {
- hideLayer();
+ return 1;
+ }
+ function svgMarkerSegments(projection2, graph, dt2, shouldReverse, bothDirections) {
+ return function(entity) {
+ var i3 = 0;
+ var offset = dt2;
+ var segments = [];
+ var clip = identity_default2().clipExtent(projection2.clipExtent()).stream;
+ var coordinates = graph.childNodes(entity).map(function(n3) {
+ return n3.loc;
+ });
+ var a2, b2;
+ if (shouldReverse(entity)) {
+ coordinates.reverse();
}
- return this;
+ stream_default({
+ type: "LineString",
+ coordinates
+ }, projection2.stream(clip({
+ lineStart: function() {
+ },
+ lineEnd: function() {
+ a2 = null;
+ },
+ point: function(x2, y2) {
+ b2 = [x2, y2];
+ if (a2) {
+ var span = geoVecLength(a2, b2) - offset;
+ if (span >= 0) {
+ var heading2 = geoVecAngle(a2, b2);
+ var dx = dt2 * Math.cos(heading2);
+ var dy = dt2 * Math.sin(heading2);
+ var p2 = [
+ a2[0] + offset * Math.cos(heading2),
+ a2[1] + offset * Math.sin(heading2)
+ ];
+ var coord2 = [a2, p2];
+ for (span -= dt2; span >= 0; span -= dt2) {
+ p2 = geoVecAdd(p2, [dx, dy]);
+ coord2.push(p2);
+ }
+ coord2.push(b2);
+ 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: i3++, 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: i3++, d: segment });
+ }
+ }
+ offset = -span;
+ }
+ a2 = b2;
+ }
+ })));
+ return segments;
};
- init2();
- return drawLocation;
}
-
- // modules/svg/labels.js
- var import_rbush6 = __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_rbush6.default();
- var _rskipped = new import_rbush6.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 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]
];
- function shouldSkipIcon(preset) {
- var noIcons = ["building", "landuse", "natural"];
- return noIcons.some(function(s2) {
- return preset.id.indexOf(s2) >= 0;
- });
- }
- function get4(array2, prop) {
- return function(d2, i3) {
- return array2[i3][prop];
- };
- }
- function textWidth(text2, size, elem) {
- var c2 = _textWidthCache[size];
- if (!c2)
- c2 = _textWidthCache[size] = {};
- if (c2[text2]) {
- return c2[text2];
- } else if (elem) {
- c2[text2] = elem.getComputedTextLength();
- return c2[text2];
+ 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 {
- var str2 = encodeURIComponent(text2).match(/%[CDEFcdef]/g);
- if (str2 === null) {
- return size / 3 * 2 * text2.length;
+ return cache[entity.id] = path(entity.asGeoJSON(graph));
+ }
+ };
+ svgpath.geojson = function(d2) {
+ if (d2.__featurehash__ !== void 0) {
+ if (d2.__featurehash__ in cache) {
+ return cache[d2.__featurehash__];
} else {
- return size / 3 * (2 * text2.length + str2.length);
+ return cache[d2.__featurehash__] = path(d2);
+ }
+ } else {
+ return path(d2);
+ }
+ };
+ return svgpath;
+ }
+ function svgPointTransform(projection2) {
+ var svgpoint = function(entity) {
+ var pt2 = projection2(entity.loc);
+ return "translate(" + pt2[0] + "," + pt2[1] + ")";
+ };
+ svgpoint.geojson = function(d2) {
+ return svgpoint(d2.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);
}
- }
- }
- 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(d2) {
- return "ideditor-labelpath-" + d2.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(d2, i3) {
- return classes + " " + labels[i3].classes + " " + d2.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(d2) {
- return "#ideditor-labelpath-" + d2.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(d2, i3) {
- return classes + " " + labels[i3].classes + " " + d2.id;
- }).merge(texts).attr("x", get4(labels, "x")).attr("y", get4(labels, "y")).style("text-anchor", get4(labels, "textAnchor")).text(utilDisplayName).each(function(d2, i3) {
- textWidth(utilDisplayName(d2), labels[i3].height, this);
});
+ return tags;
+ };
+ }
+ function svgSegmentWay(way, graph, activeID) {
+ if (activeID === void 0) {
+ return graph.transient(way, "waySegments", getWaySegments);
+ } else {
+ return getWaySegments();
}
- function drawAreaLabels(selection2, entities, filter2, classes, labels) {
- entities = entities.filter(hasText);
- labels = labels.filter(hasText);
- drawPointLabels(selection2, entities, filter2, classes, labels);
- function hasText(d2, i3) {
- return labels[i3].hasOwnProperty("x") && labels[i3].hasOwnProperty("y");
+ function getWaySegments() {
+ var isActiveWay = way.nodes.indexOf(activeID) !== -1;
+ var features = { passive: [], active: [] };
+ var start2 = {};
+ var end = {};
+ var node, type2;
+ for (var i3 = 0; i3 < way.nodes.length; i3++) {
+ node = graph.entity(way.nodes[i3]);
+ 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, i3);
+ } else if (start2.type === 0 && end.type === 0) {
+ pushActive(start2, end, i3);
+ } else {
+ pushPassive(start2, end, i3);
+ }
+ }
+ start2 = end;
}
- }
- 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(d2) {
- var preset = _mainPresetIndex.match(d2, 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(d2) {
- return { type: "Polygon", coordinates: [[
- [d2.minX, d2.minY],
- [d2.maxX, d2.minY],
- [d2.maxX, d2.maxY],
- [d2.minX, d2.maxY],
- [d2.minX, d2.minY]
- ]] };
+ 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]
+ }
});
}
- 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 i3, j3, k2, entity, geometry;
- for (i3 = 0; i3 < labelStack.length; i3++) {
- labelable.push([]);
- }
- if (fullRedraw) {
- _rdrawn.clear();
- _rskipped.clear();
- _entitybboxes = {};
- } else {
- for (i3 = 0; i3 < entities.length; i3++) {
- entity = entities[i3];
- var toRemove = [].concat(_entitybboxes[entity.id] || []).concat(_entitybboxes[entity.id + "I"] || []);
- for (j3 = 0; j3 < toRemove.length; j3++) {
- _rdrawn.remove(toRemove[j3]);
- _rskipped.remove(toRemove[j3]);
+ 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]
}
- }
+ });
}
- for (i3 = 0; i3 < entities.length; i3++) {
- entity = entities[i3];
- 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");
+ }
+ }
+
+ // 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",
+ "roller_coaster",
+ "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;
}
- if (geometry === "vertex") {
- geometry = "point";
+ var t2 = _tags(entity);
+ var computed = tagClasses.getClassesString(t2, value);
+ if (computed !== value) {
+ select_default2(this).attr("class", computed);
}
- var preset = geometry === "area" && _mainPresetIndex.match(entity, graph);
- var icon2 = preset && !shouldSkipIcon(preset) && preset.icon;
- if (!icon2 && !utilDisplayName(entity))
+ });
+ };
+ tagClasses.getClassesString = function(t2, value) {
+ var primary, status;
+ var i3, j2, k2, v2;
+ var overrideGeometry;
+ if (/\bstroke\b/.test(value)) {
+ if (!!t2.barrier && t2.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 (i3 = 0; i3 < primaries.length; i3++) {
+ k2 = primaries[i3];
+ v2 = t2[k2];
+ if (!v2 || v2 === "no")
continue;
- for (k2 = 0; k2 < labelStack.length; k2++) {
- var matchGeom = labelStack[k2][0];
- var matchKey = labelStack[k2][1];
- var matchVal = labelStack[k2][2];
- var hasVal = entity.tags[matchKey];
- if (geometry === matchGeom && hasVal && (matchVal === "*" || matchVal === hasVal)) {
- labelable[k2].push(entity);
- break;
- }
+ if (k2 === "piste:type") {
+ k2 = "piste";
+ } else if (k2 === "building:part") {
+ k2 = "building_part";
+ }
+ primary = k2;
+ if (statuses.indexOf(v2) !== -1) {
+ status = v2;
+ classes.push("tag-" + k2);
+ } else {
+ classes.push("tag-" + k2);
+ classes.push("tag-" + k2 + "-" + v2);
}
+ break;
}
- var positions = {
- point: [],
- line: [],
- area: []
- };
- var labelled = {
- point: [],
- line: [],
- area: []
- };
- for (k2 = 0; k2 < labelable.length; k2++) {
- var fontSize = labelStack[k2][3];
- for (i3 = 0; i3 < labelable[k2].length; i3++) {
- entity = labelable[k2][i3];
- geometry = entity.geometry(graph);
- var getName = geometry === "line" ? utilDisplayNameForPath : utilDisplayName;
- var name = getName(entity);
- var width = name && textWidth(name, fontSize);
- var p2 = null;
- if (geometry === "point" || geometry === "vertex") {
- if (wireframe)
- continue;
- var renderAs = renderNodeAs[entity.id];
- if (renderAs === "vertex" && zoom < 17)
+ if (!primary) {
+ for (i3 = 0; i3 < statuses.length; i3++) {
+ for (j2 = 0; j2 < primaries.length; j2++) {
+ k2 = statuses[i3] + ":" + primaries[j2];
+ v2 = t2[k2];
+ if (!v2 || v2 === "no")
continue;
- p2 = getPointLabel(entity, width, fontSize, renderAs);
- } else if (geometry === "line") {
- p2 = getLineLabel(entity, width, fontSize);
- } else if (geometry === "area") {
- p2 = getAreaLabel(entity, width, fontSize);
+ status = statuses[i3];
+ break;
}
- if (p2) {
- if (geometry === "vertex") {
- geometry = "point";
- }
- p2.classes = geometry + " tag-" + labelStack[k2][1];
- positions[geometry].push(p2);
- labelled[geometry].push(entity);
+ }
+ }
+ if (!status) {
+ for (i3 = 0; i3 < statuses.length; i3++) {
+ k2 = statuses[i3];
+ v2 = t2[k2];
+ if (!v2 || v2 === "no")
+ continue;
+ if (v2 === "yes") {
+ status = k2;
+ } else if (primary && primary === v2) {
+ status = k2;
+ } else if (!primary && primaries.indexOf(v2) !== -1) {
+ status = k2;
+ primary = v2;
+ classes.push("tag-" + v2);
}
+ if (status)
+ break;
}
}
- 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;
- });
+ if (status) {
+ classes.push("tag-status");
+ classes.push("tag-status-" + status);
}
- function getPointLabel(entity2, width2, height, geometry2) {
- var y2 = geometry2 === "point" ? -12 : 0;
- var pointOffsets = {
- ltr: [15, y2, "start"],
- rtl: [-15, y2, "end"]
- };
- var textDirection = _mainLocalizer.textDirection();
- var coord3 = projection2(entity2.loc);
- var textPadding = 2;
- var offset = pointOffsets[textDirection];
- var p3 = {
- height,
- width: width2,
- x: coord3[0] + offset[0],
- y: coord3[1] + offset[1],
- textAnchor: offset[2]
- };
- var bbox3;
- if (textDirection === "rtl") {
- bbox3 = {
- minX: p3.x - width2 - textPadding,
- minY: p3.y - height / 2 - textPadding,
- maxX: p3.x + textPadding,
- maxY: p3.y + height / 2 + textPadding
- };
- } else {
- bbox3 = {
- minX: p3.x - textPadding,
- minY: p3.y - height / 2 - textPadding,
- maxX: p3.x + width2 + textPadding,
- maxY: p3.y + height / 2 + textPadding
- };
- }
- if (tryInsert([bbox3], entity2.id, true)) {
- return p3;
+ for (i3 = 0; i3 < secondaries.length; i3++) {
+ k2 = secondaries[i3];
+ v2 = t2[k2];
+ if (!v2 || v2 === "no" || k2 === primary)
+ continue;
+ classes.push("tag-" + k2);
+ classes.push("tag-" + k2 + "-" + v2);
+ }
+ if (primary === "highway" && !osmPathHighwayTagValues[t2.highway] || primary === "aeroway") {
+ var surface = t2.highway === "track" ? "unpaved" : "paved";
+ for (k2 in t2) {
+ v2 = t2[k2];
+ if (k2 in osmPavedTags) {
+ surface = osmPavedTags[k2][v2] ? "paved" : "unpaved";
+ }
+ if (k2 in osmSemipavedTags && !!osmSemipavedTags[k2][v2]) {
+ surface = "semipaved";
+ }
}
+ classes.push("tag-" + surface);
}
- 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 i4 = 0; i4 < lineOffsets.length; i4++) {
- var offset = lineOffsets[i4];
- 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)) {
+ var qid = t2.wikidata || t2["flag:wikidata"] || t2["brand:wikidata"] || t2["network:wikidata"] || t2["operator:wikidata"];
+ if (qid) {
+ classes.push("tag-wikidata");
+ }
+ return classes.filter((klass) => /^[-_a-z0-9]+$/.test(klass)).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 tag2 in patterns) {
+ var entityValue = tags[tag2];
+ if (!entityValue)
+ continue;
+ if (typeof patterns[tag2] === "string") {
+ return "pattern-" + patterns[tag2];
+ } else {
+ var values = patterns[tag2];
+ for (var value in values) {
+ if (entityValue !== value)
continue;
+ var rules = values[value];
+ if (typeof rules === "string") {
+ return "pattern-" + rules;
}
- var isReverse = reverse(sub);
- if (isReverse) {
- sub = sub.reverse();
- }
- var bboxes = [];
- var boxsize = (height + 2) / 2;
- for (var j4 = 0; j4 < sub.length - 1; j4++) {
- var a2 = sub[j4];
- var b2 = sub[j4 + 1];
- var num = Math.max(1, Math.floor(geoVecLength(a2, b2) / boxsize / 2));
- for (var box = 0; box < num; box++) {
- var p3 = geoVecInterp(a2, b2, box / num);
- var x05 = p3[0] - boxsize - padding;
- var y05 = p3[1] - boxsize - padding;
- var x12 = p3[0] + boxsize + padding;
- var y12 = p3[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(p4) {
- var angle2 = Math.atan2(p4[1][1] - p4[0][1], p4[1][0] - p4[0][0]);
- return !(p4[0][0] < p4[p4.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 i5 = 0; i5 < points2.length - 1; i5++) {
- var a3 = points2[i5];
- var b3 = points2[i5 + 1];
- var current = geoVecLength(a3, b3);
- var portion;
- if (!start3 && sofar + current >= from) {
- portion = (from - sofar) / current;
- start3 = [
- a3[0] + portion * (b3[0] - a3[0]),
- a3[1] + portion * (b3[1] - a3[1])
- ];
- i0 = i5 + 1;
+ for (var ruleKey in rules) {
+ var rule = rules[ruleKey];
+ var pass = true;
+ for (var criterion in rule) {
+ if (criterion !== "pattern") {
+ var v2 = tags[criterion];
+ if (!v2 || v2 !== rule[criterion]) {
+ pass = false;
+ break;
+ }
+ }
}
- if (!end && sofar + current >= to) {
- portion = (to - sofar) / current;
- end = [
- a3[0] + portion * (b3[0] - a3[0]),
- a3[1] + portion * (b3[1] - a3[1])
- ];
- i1 = i5 + 1;
+ if (pass) {
+ return "pattern-" + rule.pattern;
}
- 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 p3 = {};
- if (picon) {
- if (addIcon()) {
- addLabel(iconSize + padding);
- return p3;
- }
- } else {
- if (addLabel(0)) {
- return p3;
- }
+ }
+ 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(d2) {
+ return filter2(d2.properties.entity);
+ }).data(targetData, function key(d2) {
+ return d2.id;
+ });
+ targets.exit().remove();
+ var segmentWasEdited = function(d2) {
+ var wayID = d2.properties.entity.id;
+ if (!base.entities[wayID] || !(0, import_fast_deep_equal5.default)(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
+ return false;
}
- 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
+ return d2.properties.nodes.some(function(n3) {
+ return !base.entities[n3.id] || !(0, import_fast_deep_equal5.default)(graph.entities[n3.id].loc, base.entities[n3.id].loc);
+ });
+ };
+ targets.enter().append("path").merge(targets).attr("d", getPath).attr("class", function(d2) {
+ return "way area target target-allowed " + targetClass + d2.id;
+ }).classed("segment-edited", segmentWasEdited);
+ var nopeData = data.nopes.filter(getPath);
+ var nopes = selection2.selectAll(".area.target-nope").filter(function(d2) {
+ return filter2(d2.properties.entity);
+ }).data(nopeData, function key(d2) {
+ return d2.id;
+ });
+ nopes.exit().remove();
+ nopes.enter().append("path").merge(nopes).attr("d", getPath).attr("class", function(d2) {
+ return "way area target target-nope " + nopeClass + d2.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 i3 = 0; i3 < entities.length; i3++) {
+ var entity = entities[i3];
+ 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))
};
- if (tryInsert([bbox3], entity2.id + "I", true)) {
- p3.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)) {
- p3.x = labelX;
- p3.y = labelY;
- p3.textAnchor = "middle";
- p3.height = height;
- return true;
- }
- }
- return false;
+ }
+ var fills = Object.values(areas).filter(function hasPath(a2) {
+ return path(a2.entity);
+ });
+ fills.sort(function areaSort(a2, b2) {
+ return b2.area - a2.area;
+ });
+ fills = fills.map(function(a2) {
+ return a2.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(d2) {
+ return "areagroup area-" + d2;
+ }).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))];
}
}
- function doInsert(bbox3, id2) {
- bbox3.id = id2;
- var oldbox = _entitybboxes[id2];
- if (oldbox) {
- _rdrawn.remove(oldbox);
+ 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);
}
- _entitybboxes[id2] = bbox3;
- _rdrawn.insert(bbox3);
+ }).classed("added", function(d2) {
+ return !base.entities[d2.id];
+ }).classed("geometry-edited", function(d2) {
+ return graph.entities[d2.id] && base.entities[d2.id] && !(0, import_fast_deep_equal5.default)(graph.entities[d2.id].nodes, base.entities[d2.id].nodes);
+ }).classed("retagged", function(d2) {
+ return graph.entities[d2.id] && base.entities[d2.id] && !(0, import_fast_deep_equal5.default)(graph.entities[d2.id].tags, base.entities[d2.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 : "#".concat(id2);
+ }
+ function $ns(element, tagName, ns) {
+ return Array.from(element.getElementsByTagNameNS(ns, tagName));
+ }
+ function nodeVal(node) {
+ node == null ? void 0 : node.normalize();
+ return node && node.textContent || "";
+ }
+ function get1(node, tagName, callback) {
+ const n3 = node.getElementsByTagName(tagName);
+ const result = n3.length ? n3[0] : null;
+ if (result && callback)
+ callback(result);
+ return result;
+ }
+ function get3(node, tagName, callback) {
+ const properties = {};
+ if (!node)
+ return properties;
+ const n3 = node.getElementsByTagName(tagName);
+ const result = n3.length ? n3[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 (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 == null ? void 0 : node.nodeType) === 1;
+ }
+ function getLineStyle(node) {
+ return get3(node, "line", (lineStyle) => {
+ const val = Object.assign({}, val1(lineStyle, "color", (color2) => {
+ return { stroke: "#".concat(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)]);
}
- function tryInsert(bboxes, id2, saveSkipped) {
- var skipped = false;
- for (var i4 = 0; i4 < bboxes.length; i4++) {
- var bbox3 = bboxes[i4];
- 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;
- }
+ }
+ 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) {
+ var _a2;
+ 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 (((_a2 = child.parentNode) == null ? void 0 : _a2.parentNode) === node) {
+ properties[child.tagName.replace(":", "_")] = nodeVal(child);
+ }
+ }
+ const links = $(node, "link");
+ if (links.length) {
+ properties.links = links.map((link3) => Object.assign({ href: link3.getAttribute("href") }, getMulti(link3, ["text", "type"])));
+ }
+ return properties;
+ }
+ function getPoints$1(node, pointname) {
+ const pts = $(node, pointname);
+ const line = [];
+ const times = [];
+ const extendedValues = {};
+ for (let i3 = 0; i3 < pts.length; i3++) {
+ const c2 = coordPair$1(pts[i3]);
+ if (!c2) {
+ continue;
+ }
+ line.push(c2.coordinates);
+ if (c2.time)
+ times.push(c2.time);
+ for (const [name, val] of c2.extendedValues) {
+ const plural = name === "heart" ? name : name.replace("gpxtpx:", "") + "s";
+ if (!extendedValues[plural]) {
+ extendedValues[plural] = Array(pts.length).fill(null);
}
- _entitybboxes[id2] = bboxes;
- if (skipped) {
- if (saveSkipped) {
- _rskipped.load(bboxes);
+ extendedValues[plural][i3] = 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 i3 = 0; i3 < entries.length; i3++) {
+ const [name, val] = entries[i3];
+ if (multi) {
+ if (!props[name]) {
+ props[name] = extractedLines.map((line2) => new Array(line2.line.length).fill(null));
}
+ props[name][i3] = val;
} else {
- _rdrawn.load(bboxes);
+ props[name] = val;
}
- return !skipped;
}
- var layer = selection2.selectAll(".layer-osm.labels");
- layer.selectAll(".labels-group").data(["halo", "label", "debug"]).enter().append("g").attr("class", function(d2) {
- return "labels-group " + d2;
- });
- 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 i3 = 0; i3 < selectedIDs.length; i3++) {
- var entity = graph.hasEntity(selectedIDs[i3]);
- if (entity && entity.type === "node") {
- ids.push(selectedIDs[i3]);
- }
+ return {
+ type: "Feature",
+ properties,
+ geometry: multi ? {
+ type: "MultiLineString",
+ coordinates: track
+ } : {
+ type: "LineString",
+ coordinates: track[0]
}
- 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]
- ]]
- }] : [];
+ };
+ }
+ function getPoint(node) {
+ const properties = Object.assign(extractProperties(node), getMulti(node, ["sym"]));
+ const pair3 = coordPair$1(node);
+ if (!pair3)
+ return null;
+ return {
+ type: "Feature",
+ properties,
+ geometry: {
+ type: "Point",
+ coordinates: pair3.coordinates
}
- 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);
+ }
+ 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))
};
- return drawLabels;
}
-
- // node_modules/exifr/dist/full.esm.mjs
- var e = "undefined" != typeof self ? self : global;
- var t = "undefined" != typeof navigator;
- var i2 = t && "undefined" == typeof HTMLImageElement;
- var n2 = !("undefined" == typeof global || "undefined" == typeof process || !process.versions || !process.versions.node);
- var s = e.Buffer;
- var r = e.BigInt;
- var a = !!s;
- var o = (e3) => e3;
- function l(e3, t2 = o) {
- if (n2)
- try {
- return "function" == typeof __require ? Promise.resolve(t2(__require(e3))) : import(
- /* webpackIgnore: true */
- e3
- ).then(t2);
- } catch (t3) {
- console.warn(`Couldn't load ${e3}`);
- }
+ function fixColor(v2, prefix) {
+ const properties = {};
+ const colorProp = prefix == "stroke" || prefix === "fill" ? prefix : prefix + "-color";
+ if (v2[0] === "#") {
+ v2 = v2.substring(1);
+ }
+ if (v2.length === 6 || v2.length === 3) {
+ properties[colorProp] = "#" + v2;
+ } else if (v2.length === 8) {
+ properties[prefix + "-opacity"] = parseInt(v2.substring(0, 2), 16) / 255;
+ properties[colorProp] = "#" + v2.substring(6, 8) + v2.substring(4, 6) + v2.substring(2, 4);
+ }
+ return properties;
}
- var h = e.fetch;
- var u = (e3) => h = e3;
- if (!e.fetch) {
- const e3 = l("http", (e4) => e4), t2 = l("https", (e4) => e4), i3 = (n3, { headers: s2 } = {}) => new Promise(async (r2, a2) => {
- let { port: o2, hostname: l2, pathname: h2, protocol: u2, search: c2 } = new URL(n3);
- const f3 = { method: "GET", hostname: l2, path: encodeURI(h2) + c2, headers: s2 };
- "" !== o2 && (f3.port = Number(o2));
- const d2 = ("https:" === u2 ? await t2 : await e3).request(f3, (e4) => {
- if (301 === e4.statusCode || 302 === e4.statusCode) {
- let t3 = new URL(e4.headers.location, n3).toString();
- return i3(t3, { headers: s2 }).then(r2).catch(a2);
- }
- r2({ status: e4.statusCode, arrayBuffer: () => new Promise((t3) => {
- let i4 = [];
- e4.on("data", (e6) => i4.push(e6)), e4.on("end", () => t3(Buffer.concat(i4)));
- }) });
+ 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 extractIconHref(node) {
+ return get3(node, "Icon", (icon2, properties) => {
+ val1(icon2, "href", (href) => {
+ properties.icon = href;
});
- d2.on("error", a2), d2.end();
+ return properties;
});
- u(i3);
}
- function c(e3, t2, i3) {
- return t2 in e3 ? Object.defineProperty(e3, t2, { value: i3, enumerable: true, configurable: true, writable: true }) : e3[t2] = i3, e3;
+ 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 {};
+ }), extractIconHref(iconStyle));
+ });
}
- var f2 = (e3) => p(e3) ? void 0 : e3;
- var d = (e3) => void 0 !== e3;
- function p(e3) {
- return void 0 === e3 || (e3 instanceof Map ? 0 === e3.size : 0 === Object.values(e3).filter(d).length);
+ function extractLabel(node) {
+ return get3(node, "LabelStyle", (labelStyle) => {
+ return Object.assign(getColor(labelStyle, "label"), numericProperty(labelStyle, "scale", "label-scale"));
+ });
}
- function g2(e3) {
- let t2 = new Error(e3);
- throw delete t2.stack, t2;
+ function extractLine(node) {
+ return get3(node, "LineStyle", (lineStyle) => {
+ return Object.assign(getColor(lineStyle, "stroke"), numericProperty(lineStyle, "width", "stroke-width"));
+ });
}
- function m(e3) {
- return "" === (e3 = function(e4) {
- for (; e4.endsWith("\0"); )
- e4 = e4.slice(0, -1);
- return e4;
- }(e3).trim()) ? void 0 : e3;
+ 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 S(e3) {
- let t2 = function(e4) {
- let t3 = 0;
- return e4.ifd0.enabled && (t3 += 1024), e4.exif.enabled && (t3 += 2048), e4.makerNote && (t3 += 2048), e4.userComment && (t3 += 1024), e4.gps.enabled && (t3 += 512), e4.interop.enabled && (t3 += 100), e4.ifd1.enabled && (t3 += 1024), t3 + 2048;
- }(e3);
- return e3.jfif.enabled && (t2 += 50), e3.xmp.enabled && (t2 += 2e4), e3.iptc.enabled && (t2 += 14e3), e3.icc.enabled && (t2 += 6e3), t2;
+ function extractStyle(node) {
+ return Object.assign({}, extractPoly(node), extractLine(node), extractLabel(node), extractIcon(node));
}
- var C = (e3) => String.fromCharCode.apply(null, e3);
- var y = "undefined" != typeof TextDecoder ? new TextDecoder("utf-8") : void 0;
- function b(e3) {
- return y ? y.decode(e3) : a ? Buffer.from(e3).toString("utf8") : decodeURIComponent(escape(C(e3)));
+ var toNumber2 = (x2) => Number(x2);
+ var typeConverters = {
+ string: (x2) => x2,
+ int: toNumber2,
+ uint: toNumber2,
+ short: toNumber2,
+ ushort: toNumber2,
+ float: toNumber2,
+ double: toNumber2,
+ bool: (x2) => Boolean(x2)
+ };
+ function extractExtendedData(node, schema) {
+ 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")) {
+ const name = simpleData.getAttribute("name") || "";
+ const typeConverter = schema[name] || typeConverters.string;
+ properties[name] = typeConverter(nodeVal(simpleData));
+ }
+ return properties;
+ });
}
- var I = class _I {
- static from(e3, t2) {
- return e3 instanceof this && e3.le === t2 ? e3 : new _I(e3, void 0, void 0, t2);
- }
- constructor(e3, t2 = 0, i3, n3) {
- if ("boolean" == typeof n3 && (this.le = n3), Array.isArray(e3) && (e3 = new Uint8Array(e3)), 0 === e3)
- this.byteOffset = 0, this.byteLength = 0;
- else if (e3 instanceof ArrayBuffer) {
- void 0 === i3 && (i3 = e3.byteLength - t2);
- let n4 = new DataView(e3, t2, i3);
- this._swapDataView(n4);
- } else if (e3 instanceof Uint8Array || e3 instanceof DataView || e3 instanceof _I) {
- void 0 === i3 && (i3 = e3.byteLength - t2), (t2 += e3.byteOffset) + i3 > e3.byteOffset + e3.byteLength && g2("Creating view outside of available memory in ArrayBuffer");
- let n4 = new DataView(e3.buffer, t2, i3);
- this._swapDataView(n4);
- } else if ("number" == typeof e3) {
- let t3 = new DataView(new ArrayBuffer(e3));
- this._swapDataView(t3);
- } else
- g2("Invalid input argument for BufferView: " + e3);
- }
- _swapArrayBuffer(e3) {
- this._swapDataView(new DataView(e3));
- }
- _swapBuffer(e3) {
- this._swapDataView(new DataView(e3.buffer, e3.byteOffset, e3.byteLength));
- }
- _swapDataView(e3) {
- this.dataView = e3, this.buffer = e3.buffer, this.byteOffset = e3.byteOffset, this.byteLength = e3.byteLength;
- }
- _lengthToEnd(e3) {
- return this.byteLength - e3;
- }
- set(e3, t2, i3 = _I) {
- return e3 instanceof DataView || e3 instanceof _I ? e3 = new Uint8Array(e3.buffer, e3.byteOffset, e3.byteLength) : e3 instanceof ArrayBuffer && (e3 = new Uint8Array(e3)), e3 instanceof Uint8Array || g2("BufferView.set(): Invalid data argument."), this.toUint8().set(e3, t2), new i3(this, t2, e3.byteLength);
- }
- subarray(e3, t2) {
- return t2 = t2 || this._lengthToEnd(e3), new _I(this, e3, t2);
- }
- toUint8() {
- return new Uint8Array(this.buffer, this.byteOffset, this.byteLength);
- }
- getUint8Array(e3, t2) {
- return new Uint8Array(this.buffer, this.byteOffset + e3, t2);
- }
- getString(e3 = 0, t2 = this.byteLength) {
- return b(this.getUint8Array(e3, t2));
- }
- getLatin1String(e3 = 0, t2 = this.byteLength) {
- let i3 = this.getUint8Array(e3, t2);
- return C(i3);
- }
- getUnicodeString(e3 = 0, t2 = this.byteLength) {
- const i3 = [];
- for (let n3 = 0; n3 < t2 && e3 + n3 < this.byteLength; n3 += 2)
- i3.push(this.getUint16(e3 + n3));
- return C(i3);
- }
- getInt8(e3) {
- return this.dataView.getInt8(e3);
- }
- getUint8(e3) {
- return this.dataView.getUint8(e3);
- }
- getInt16(e3, t2 = this.le) {
- return this.dataView.getInt16(e3, t2);
- }
- getInt32(e3, t2 = this.le) {
- return this.dataView.getInt32(e3, t2);
- }
- getUint16(e3, t2 = this.le) {
- return this.dataView.getUint16(e3, t2);
- }
- getUint32(e3, t2 = this.le) {
- return this.dataView.getUint32(e3, t2);
- }
- getFloat32(e3, t2 = this.le) {
- return this.dataView.getFloat32(e3, t2);
- }
- getFloat64(e3, t2 = this.le) {
- return this.dataView.getFloat64(e3, t2);
+ function getMaybeHTMLDescription(node) {
+ const descriptionNode = get1(node, "description");
+ for (const c2 of Array.from((descriptionNode == null ? void 0 : descriptionNode.childNodes) || [])) {
+ if (c2.nodeType === 4) {
+ return {
+ description: {
+ "@type": "html",
+ value: nodeVal(c2)
+ }
+ };
+ }
}
- getFloat(e3, t2 = this.le) {
- return this.dataView.getFloat32(e3, t2);
+ return {};
+ }
+ 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 };
+ });
+ }
+ 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", "*");
}
- getDouble(e3, t2 = this.le) {
- return this.dataView.getFloat64(e3, t2);
+ const coordinates = elems.map((elem) => {
+ return nodeVal(elem).split(" ").map(parseFloat);
+ });
+ if (coordinates.length === 0) {
+ return null;
}
- getUintBytes(e3, t2, i3) {
- switch (t2) {
- case 1:
- return this.getUint8(e3, i3);
- case 2:
- return this.getUint16(e3, i3);
- case 4:
- return this.getUint32(e3, i3);
- case 8:
- return this.getUint64 && this.getUint64(e3, i3);
+ 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 i3 = 0; i3 < Math.max(first.length, last.length); i3++) {
+ if (first[i3] !== last[i3]) {
+ equal = false;
+ break;
}
}
- getUint(e3, t2, i3) {
- switch (t2) {
- case 8:
- return this.getUint8(e3, i3);
- case 16:
- return this.getUint16(e3, i3);
- case 32:
- return this.getUint32(e3, i3);
- case 64:
- return this.getUint64 && this.getUint64(e3, i3);
+ if (!equal) {
+ return ring.concat([ring[0]]);
+ }
+ return ring;
+ }
+ function getCoordinates(node) {
+ return nodeVal(get1(node, "coordinates"));
+ }
+ function getGeometry(node) {
+ let geometries = [];
+ let coordTimes = [];
+ for (let i3 = 0; i3 < node.childNodes.length; i3++) {
+ const child = node.childNodes.item(i3);
+ if (isElement(child)) {
+ switch (child.tagName) {
+ case "MultiGeometry":
+ case "MultiTrack":
+ case "gx:MultiTrack": {
+ const childGeometries = getGeometry(child);
+ geometries = geometries.concat(childGeometries.geometries);
+ coordTimes = coordTimes.concat(childGeometries.coordTimes);
+ break;
+ }
+ case "Point": {
+ const coordinates = coord1(getCoordinates(child));
+ if (coordinates.length >= 2) {
+ geometries.push({
+ type: "Point",
+ coordinates
+ });
+ }
+ break;
+ }
+ case "LinearRing":
+ case "LineString": {
+ const coordinates = coord(getCoordinates(child));
+ if (coordinates.length >= 2) {
+ geometries.push({
+ type: "LineString",
+ coordinates
+ });
+ }
+ break;
+ }
+ case "Polygon": {
+ const coords = [];
+ for (const linearRing of $(child, "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(child);
+ if (!gx)
+ break;
+ const { times, geometry } = gx;
+ geometries.push(geometry);
+ if (times.length)
+ coordTimes.push(times);
+ break;
+ }
+ }
}
}
- toString(e3) {
- return this.dataView.toString(e3, this.constructor.name);
- }
- ensureChunk() {
- }
- };
- function P(e3, t2) {
- g2(`${e3} '${t2}' was not loaded, try using full build of exifr.`);
+ return {
+ geometries,
+ coordTimes
+ };
}
- var k = class extends Map {
- constructor(e3) {
- super(), this.kind = e3;
- }
- get(e3, t2) {
- return this.has(e3) || P(this.kind, e3), t2 && (e3 in t2 || function(e4, t3) {
- g2(`Unknown ${e4} '${t3}'.`);
- }(this.kind, e3), t2[e3].enabled || P(this.kind, e3)), super.get(e3);
+ function geometryListToGeometry(geometries) {
+ return geometries.length === 0 ? null : geometries.length === 1 ? geometries[0] : {
+ type: "GeometryCollection",
+ geometries
+ };
+ }
+ function getPlacemark(node, styleMap, schema, options2) {
+ var _a2;
+ const { coordTimes, geometries } = getGeometry(node);
+ const geometry = geometryListToGeometry(geometries);
+ if (!geometry && options2.skipNullGeometry) {
+ return null;
}
- keyList() {
- return Array.from(this.keys());
+ const feature3 = {
+ type: "Feature",
+ geometry,
+ properties: Object.assign(getMulti(node, [
+ "name",
+ "address",
+ "visibility",
+ "open",
+ "phoneNumber",
+ "description"
+ ]), getMaybeHTMLDescription(node), extractCascadedStyle(node, styleMap), extractStyle(node), extractExtendedData(node, schema), extractTimeSpan(node), extractTimeStamp(node), coordTimes.length ? {
+ coordinateProperties: {
+ times: coordTimes.length === 1 ? coordTimes[0] : coordTimes
+ }
+ } : {})
+ };
+ if (((_a2 = feature3.properties) == null ? void 0 : _a2.visibility) !== void 0) {
+ feature3.properties.visibility = feature3.properties.visibility !== "0";
}
- };
- var w = new k("file parser");
- var T = new k("segment parser");
- var A = new k("file reader");
- function D(e3, n3) {
- return "string" == typeof e3 ? O(e3, n3) : t && !i2 && e3 instanceof HTMLImageElement ? O(e3.src, n3) : e3 instanceof Uint8Array || e3 instanceof ArrayBuffer || e3 instanceof DataView ? new I(e3) : t && e3 instanceof Blob ? x(e3, n3, "blob", R) : void g2("Invalid input argument");
- }
- function O(e3, i3) {
- return (s2 = e3).startsWith("data:") || s2.length > 1e4 ? v(e3, i3, "base64") : n2 && e3.includes("://") ? x(e3, i3, "url", M) : n2 ? v(e3, i3, "fs") : t ? x(e3, i3, "url", M) : void g2("Invalid input argument");
- var s2;
- }
- async function x(e3, t2, i3, n3) {
- return A.has(i3) ? v(e3, t2, i3) : n3 ? async function(e4, t3) {
- let i4 = await t3(e4);
- return new I(i4);
- }(e3, n3) : void g2(`Parser ${i3} is not loaded`);
- }
- async function v(e3, t2, i3) {
- let n3 = new (A.get(i3))(e3, t2);
- return await n3.read(), n3;
+ const id2 = node.getAttribute("id");
+ if (id2 !== null && id2 !== "")
+ feature3.id = id2;
+ return feature3;
}
- var M = (e3) => h(e3).then((e4) => e4.arrayBuffer());
- var R = (e3) => new Promise((t2, i3) => {
- let n3 = new FileReader();
- n3.onloadend = () => t2(n3.result || new ArrayBuffer()), n3.onerror = i3, n3.readAsArrayBuffer(e3);
- });
- var L = class extends Map {
- get tagKeys() {
- return this.allKeys || (this.allKeys = Array.from(this.keys())), this.allKeys;
- }
- get tagValues() {
- return this.allValues || (this.allValues = Array.from(this.values())), this.allValues;
+ function getGroundOverlayBox(node) {
+ const latLonQuad = get1(node, "gx:LatLonQuad");
+ if (latLonQuad) {
+ const ring = fixRing(coord(getCoordinates(node)));
+ return {
+ geometry: {
+ type: "Polygon",
+ coordinates: [ring]
+ }
+ };
}
- };
- function U(e3, t2, i3) {
- let n3 = new L();
- for (let [e4, t3] of i3)
- n3.set(e4, t3);
- if (Array.isArray(t2))
- for (let i4 of t2)
- e3.set(i4, n3);
- else
- e3.set(t2, n3);
- return n3;
+ return getLatLonBox(node);
}
- function F(e3, t2, i3) {
- let n3, s2 = e3.get(t2);
- for (n3 of i3)
- s2.set(n3[0], n3[1]);
+ var DEGREES_TO_RADIANS = Math.PI / 180;
+ function rotateBox(bbox2, coordinates, rotation) {
+ const center = [(bbox2[0] + bbox2[2]) / 2, (bbox2[1] + bbox2[3]) / 2];
+ return [
+ coordinates[0].map((coordinate) => {
+ const dy = coordinate[1] - center[1];
+ const dx = coordinate[0] - center[0];
+ const distance = Math.sqrt(Math.pow(dy, 2) + Math.pow(dx, 2));
+ const angle2 = Math.atan2(dy, dx) + rotation * DEGREES_TO_RADIANS;
+ return [
+ center[0] + Math.cos(angle2) * distance,
+ center[1] + Math.sin(angle2) * distance
+ ];
+ })
+ ];
}
- var E = /* @__PURE__ */ new Map();
- var B = /* @__PURE__ */ new Map();
- var N = /* @__PURE__ */ new Map();
- var G = ["chunked", "firstChunkSize", "firstChunkSizeNode", "firstChunkSizeBrowser", "chunkSize", "chunkLimit"];
- var V = ["jfif", "xmp", "icc", "iptc", "ihdr"];
- var z = ["tiff", ...V];
- var H = ["ifd0", "ifd1", "exif", "gps", "interop"];
- var j2 = [...z, ...H];
- var W = ["makerNote", "userComment"];
- var K = ["translateKeys", "translateValues", "reviveValues", "multiSegment"];
- var X = [...K, "sanitize", "mergeOutput", "silentErrors"];
- var _ = class {
- get translate() {
- return this.translateKeys || this.translateValues || this.reviveValues;
- }
- };
- var Y = class extends _ {
- get needed() {
- return this.enabled || this.deps.size > 0;
- }
- constructor(e3, t2, i3, n3) {
- if (super(), c(this, "enabled", false), c(this, "skip", /* @__PURE__ */ new Set()), c(this, "pick", /* @__PURE__ */ new Set()), c(this, "deps", /* @__PURE__ */ new Set()), c(this, "translateKeys", false), c(this, "translateValues", false), c(this, "reviveValues", false), this.key = e3, this.enabled = t2, this.parse = this.enabled, this.applyInheritables(n3), this.canBeFiltered = H.includes(e3), this.canBeFiltered && (this.dict = E.get(e3)), void 0 !== i3)
- if (Array.isArray(i3))
- this.parse = this.enabled = true, this.canBeFiltered && i3.length > 0 && this.translateTagSet(i3, this.pick);
- else if ("object" == typeof i3) {
- if (this.enabled = true, this.parse = false !== i3.parse, this.canBeFiltered) {
- let { pick: e4, skip: t3 } = i3;
- e4 && e4.length > 0 && this.translateTagSet(e4, this.pick), t3 && t3.length > 0 && this.translateTagSet(t3, this.skip);
+ function getLatLonBox(node) {
+ const latLonBox = get1(node, "LatLonBox");
+ if (latLonBox) {
+ const north = num1(latLonBox, "north");
+ const west = num1(latLonBox, "west");
+ const east = num1(latLonBox, "east");
+ const south = num1(latLonBox, "south");
+ const rotation = num1(latLonBox, "rotation");
+ if (typeof north === "number" && typeof south === "number" && typeof west === "number" && typeof east === "number") {
+ const bbox2 = [west, south, east, north];
+ let coordinates = [
+ [
+ [west, north],
+ [east, north],
+ [east, south],
+ [west, south],
+ [west, north]
+ // top left (again)
+ ]
+ ];
+ if (typeof rotation === "number") {
+ coordinates = rotateBox(bbox2, coordinates, rotation);
+ }
+ return {
+ bbox: bbox2,
+ geometry: {
+ type: "Polygon",
+ coordinates
}
- this.applyInheritables(i3);
- } else
- true === i3 || false === i3 ? this.parse = this.enabled = i3 : g2(`Invalid options argument: ${i3}`);
- }
- applyInheritables(e3) {
- let t2, i3;
- for (t2 of K)
- i3 = e3[t2], void 0 !== i3 && (this[t2] = i3);
- }
- translateTagSet(e3, t2) {
- if (this.dict) {
- let i3, n3, { tagKeys: s2, tagValues: r2 } = this.dict;
- for (i3 of e3)
- "string" == typeof i3 ? (n3 = r2.indexOf(i3), -1 === n3 && (n3 = s2.indexOf(Number(i3))), -1 !== n3 && t2.add(Number(s2[n3]))) : t2.add(i3);
- } else
- for (let i3 of e3)
- t2.add(i3);
- }
- finalizeFilters() {
- !this.enabled && this.deps.size > 0 ? (this.enabled = true, ee(this.pick, this.deps)) : this.enabled && this.pick.size > 0 && ee(this.pick, this.deps);
- }
- };
- var $2 = { jfif: false, tiff: true, xmp: false, icc: false, iptc: false, ifd0: true, ifd1: false, exif: true, gps: true, interop: false, ihdr: void 0, makerNote: false, userComment: false, multiSegment: false, skip: [], pick: [], translateKeys: true, translateValues: true, reviveValues: true, sanitize: true, mergeOutput: true, silentErrors: true, chunked: true, firstChunkSize: void 0, firstChunkSizeNode: 512, firstChunkSizeBrowser: 65536, chunkSize: 65536, chunkLimit: 5 };
- var J = /* @__PURE__ */ new Map();
- var q = class extends _ {
- static useCached(e3) {
- let t2 = J.get(e3);
- return void 0 !== t2 || (t2 = new this(e3), J.set(e3, t2)), t2;
- }
- constructor(e3) {
- super(), true === e3 ? this.setupFromTrue() : void 0 === e3 ? this.setupFromUndefined() : Array.isArray(e3) ? this.setupFromArray(e3) : "object" == typeof e3 ? this.setupFromObject(e3) : g2(`Invalid options argument ${e3}`), void 0 === this.firstChunkSize && (this.firstChunkSize = t ? this.firstChunkSizeBrowser : this.firstChunkSizeNode), this.mergeOutput && (this.ifd1.enabled = false), this.filterNestedSegmentTags(), this.traverseTiffDependencyTree(), this.checkLoadedPlugins();
- }
- setupFromUndefined() {
- let e3;
- for (e3 of G)
- this[e3] = $2[e3];
- for (e3 of X)
- this[e3] = $2[e3];
- for (e3 of W)
- this[e3] = $2[e3];
- for (e3 of j2)
- this[e3] = new Y(e3, $2[e3], void 0, this);
- }
- setupFromTrue() {
- let e3;
- for (e3 of G)
- this[e3] = $2[e3];
- for (e3 of X)
- this[e3] = $2[e3];
- for (e3 of W)
- this[e3] = true;
- for (e3 of j2)
- this[e3] = new Y(e3, true, void 0, this);
- }
- setupFromArray(e3) {
- let t2;
- for (t2 of G)
- this[t2] = $2[t2];
- for (t2 of X)
- this[t2] = $2[t2];
- for (t2 of W)
- this[t2] = $2[t2];
- for (t2 of j2)
- this[t2] = new Y(t2, false, void 0, this);
- this.setupGlobalFilters(e3, void 0, H);
- }
- setupFromObject(e3) {
- let t2;
- for (t2 of (H.ifd0 = H.ifd0 || H.image, H.ifd1 = H.ifd1 || H.thumbnail, Object.assign(this, e3), G))
- this[t2] = Z(e3[t2], $2[t2]);
- for (t2 of X)
- this[t2] = Z(e3[t2], $2[t2]);
- for (t2 of W)
- this[t2] = Z(e3[t2], $2[t2]);
- for (t2 of z)
- this[t2] = new Y(t2, $2[t2], e3[t2], this);
- for (t2 of H)
- this[t2] = new Y(t2, $2[t2], e3[t2], this.tiff);
- this.setupGlobalFilters(e3.pick, e3.skip, H, j2), true === e3.tiff ? this.batchEnableWithBool(H, true) : false === e3.tiff ? this.batchEnableWithUserValue(H, e3) : Array.isArray(e3.tiff) ? this.setupGlobalFilters(e3.tiff, void 0, H) : "object" == typeof e3.tiff && this.setupGlobalFilters(e3.tiff.pick, e3.tiff.skip, H);
- }
- batchEnableWithBool(e3, t2) {
- for (let i3 of e3)
- this[i3].enabled = t2;
- }
- batchEnableWithUserValue(e3, t2) {
- for (let i3 of e3) {
- let e4 = t2[i3];
- this[i3].enabled = false !== e4 && void 0 !== e4;
- }
- }
- setupGlobalFilters(e3, t2, i3, n3 = i3) {
- if (e3 && e3.length) {
- for (let e4 of n3)
- this[e4].enabled = false;
- let t3 = Q(e3, i3);
- for (let [e4, i4] of t3)
- ee(this[e4].pick, i4), this[e4].enabled = true;
- } else if (t2 && t2.length) {
- let e4 = Q(t2, i3);
- for (let [t3, i4] of e4)
- ee(this[t3].skip, i4);
+ };
}
}
- filterNestedSegmentTags() {
- let { ifd0: e3, exif: t2, xmp: i3, iptc: n3, icc: s2 } = this;
- this.makerNote ? t2.deps.add(37500) : t2.skip.add(37500), this.userComment ? t2.deps.add(37510) : t2.skip.add(37510), i3.enabled || e3.skip.add(700), n3.enabled || e3.skip.add(33723), s2.enabled || e3.skip.add(34675);
- }
- traverseTiffDependencyTree() {
- let { ifd0: e3, exif: t2, gps: i3, interop: n3 } = this;
- n3.needed && (t2.deps.add(40965), e3.deps.add(40965)), t2.needed && e3.deps.add(34665), i3.needed && e3.deps.add(34853), this.tiff.enabled = H.some((e4) => true === this[e4].enabled) || this.makerNote || this.userComment;
- for (let e4 of H)
- this[e4].finalizeFilters();
- }
- get onlyTiff() {
- return !V.map((e3) => this[e3].enabled).some((e3) => true === e3) && this.tiff.enabled;
+ return null;
+ }
+ function getGroundOverlay(node, styleMap, schema, options2) {
+ var _a2;
+ const box = getGroundOverlayBox(node);
+ const geometry = (box == null ? void 0 : box.geometry) || null;
+ if (!geometry && options2.skipNullGeometry) {
+ return null;
}
- checkLoadedPlugins() {
- for (let e3 of z)
- this[e3].enabled && !T.has(e3) && P("segment parser", e3);
+ const feature3 = {
+ type: "Feature",
+ geometry,
+ properties: Object.assign(
+ /**
+ * Related to
+ * https://gist.github.com/tmcw/037a1cb6660d74a392e9da7446540f46
+ */
+ { "@geometry-type": "groundoverlay" },
+ getMulti(node, [
+ "name",
+ "address",
+ "visibility",
+ "open",
+ "phoneNumber",
+ "description"
+ ]),
+ getMaybeHTMLDescription(node),
+ extractCascadedStyle(node, styleMap),
+ extractStyle(node),
+ extractIconHref(node),
+ extractExtendedData(node, schema),
+ extractTimeSpan(node),
+ extractTimeStamp(node)
+ )
+ };
+ if (box == null ? void 0 : box.bbox) {
+ feature3.bbox = box.bbox;
}
- };
- function Q(e3, t2) {
- let i3, n3, s2, r2, a2 = [];
- for (s2 of t2) {
- for (r2 of (i3 = E.get(s2), n3 = [], i3))
- (e3.includes(r2[0]) || e3.includes(r2[1])) && n3.push(r2[0]);
- n3.length && a2.push([s2, n3]);
+ if (((_a2 = feature3.properties) == null ? void 0 : _a2.visibility) !== void 0) {
+ feature3.properties.visibility = feature3.properties.visibility !== "0";
}
- return a2;
- }
- function Z(e3, t2) {
- return void 0 !== e3 ? e3 : void 0 !== t2 ? t2 : void 0;
- }
- function ee(e3, t2) {
- for (let i3 of t2)
- e3.add(i3);
+ const id2 = node.getAttribute("id");
+ if (id2 !== null && id2 !== "")
+ feature3.id = id2;
+ return feature3;
}
- c(q, "default", $2);
- var te = class {
- constructor(e3) {
- c(this, "parsers", {}), c(this, "output", {}), c(this, "errors", []), c(this, "pushToErrors", (e4) => this.errors.push(e4)), this.options = q.useCached(e3);
- }
- async read(e3) {
- this.file = await D(e3, this.options);
- }
- setup() {
- if (this.fileParser)
- return;
- let { file: e3 } = this, t2 = e3.getUint16(0);
- for (let [i3, n3] of w)
- if (n3.canHandle(e3, t2))
- return this.fileParser = new n3(this.options, this.file, this.parsers), e3[i3] = true;
- this.file.close && this.file.close(), g2("Unknown file format");
+ 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");
}
- async parse() {
- let { output: e3, errors: t2 } = this;
- return this.setup(), this.options.silentErrors ? (await this.executeParsers().catch(this.pushToErrors), t2.push(...this.fileParser.errors)) : await this.executeParsers(), this.file.close && this.file.close(), this.options.silentErrors && t2.length > 0 && (e3.errors = t2), f2(e3);
+ return normalizeId(id2 || "");
+ }
+ function buildStyleMap(node) {
+ const styleMap = {};
+ for (const style of $(node, "Style")) {
+ styleMap[getStyleId(style)] = extractStyle(style);
}
- async executeParsers() {
- let { output: e3 } = this;
- await this.fileParser.parse();
- let t2 = Object.values(this.parsers).map(async (t3) => {
- let i3 = await t3.parse();
- t3.assignToOutput(e3, i3);
+ 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];
+ }
});
- this.options.silentErrors && (t2 = t2.map((e4) => e4.catch(this.pushToErrors))), await Promise.all(t2);
}
- async extractThumbnail() {
- this.setup();
- let { options: e3, file: t2 } = this, i3 = T.get("tiff", e3);
- var n3;
- if (t2.tiff ? n3 = { start: 0, type: "tiff" } : t2.jpeg && (n3 = await this.fileParser.getOrFindSegment("tiff")), void 0 === n3)
- return;
- let s2 = await this.fileParser.ensureSegmentChunk(n3), r2 = this.parsers.tiff = new i3(s2, e3, t2), a2 = await r2.extractThumbnail();
- return t2.close && t2.close(), a2;
+ return styleMap;
+ }
+ function buildSchema(node) {
+ const schema = {};
+ for (const field of $(node, "SimpleField")) {
+ schema[field.getAttribute("name") || ""] = typeConverters[field.getAttribute("type") || ""] || typeConverters["string"];
}
- };
- async function ie(e3, t2) {
- let i3 = new te(t2);
- return await i3.read(e3), i3.parse();
+ return schema;
}
- var ne = Object.freeze({ __proto__: null, parse: ie, Exifr: te, fileParsers: w, segmentParsers: T, fileReaders: A, tagKeys: E, tagValues: B, tagRevivers: N, createDictionary: U, extendDictionary: F, fetchUrlAsArrayBuffer: M, readBlobAsArrayBuffer: R, chunkedProps: G, otherSegments: V, segments: z, tiffBlocks: H, segmentsAndBlocks: j2, tiffExtractables: W, inheritables: K, allFormatters: X, Options: q });
- var se = class {
- constructor(e3, t2, i3) {
- c(this, "errors", []), c(this, "ensureSegmentChunk", async (e4) => {
- let t3 = e4.start, i4 = e4.size || 65536;
- if (this.file.chunked)
- if (this.file.available(t3, i4))
- e4.chunk = this.file.subarray(t3, i4);
- else
- try {
- e4.chunk = await this.file.readChunk(t3, i4);
- } catch (t4) {
- g2(`Couldn't read segment: ${JSON.stringify(e4)}. ${t4.message}`);
- }
- else
- this.file.byteLength > t3 + i4 ? e4.chunk = this.file.subarray(t3, i4) : void 0 === e4.size ? e4.chunk = this.file.subarray(t3) : g2("Segment unreachable: " + JSON.stringify(e4));
- return e4.chunk;
- }), this.extendOptions && this.extendOptions(e3), this.options = e3, this.file = t2, this.parsers = i3;
+ function* kmlGen(node, options2 = {
+ skipNullGeometry: false
+ }) {
+ const styleMap = buildStyleMap(node);
+ const schema = buildSchema(node);
+ for (const placemark of $(node, "Placemark")) {
+ const feature3 = getPlacemark(placemark, styleMap, schema, options2);
+ if (feature3)
+ yield feature3;
}
- injectSegment(e3, t2) {
- this.options[e3].enabled && this.createParser(e3, t2);
+ for (const groundOverlay of $(node, "GroundOverlay")) {
+ const feature3 = getGroundOverlay(groundOverlay, styleMap, schema, options2);
+ if (feature3)
+ yield feature3;
}
- createParser(e3, t2) {
- let i3 = new (T.get(e3))(t2, this.options, this.file);
- return this.parsers[e3] = i3;
+ }
+ function kml(node, options2 = {
+ skipNullGeometry: false
+ }) {
+ return {
+ type: "FeatureCollection",
+ features: Array.from(kmlGen(node, options2))
+ };
+ }
+
+ // modules/svg/data.js
+ var _initialized = false;
+ var _enabled = false;
+ var _geojson;
+ function svgData(projection2, context, dispatch14) {
+ var throttledRedraw = throttle_default(function() {
+ dispatch14.call("change");
+ }, 1e3);
+ var _showLabels = true;
+ var detected = utilDetect();
+ var layer = select_default2(null);
+ var _vtService;
+ var _fileList;
+ var _template;
+ var _src;
+ const supportedFormats = [
+ ".gpx",
+ ".kml",
+ ".geojson",
+ ".json"
+ ];
+ 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;
+ var f2 = d3_event.dataTransfer.files[0];
+ var extension = getExtension(f2.name);
+ if (!supportedFormats.includes(extension))
+ return;
+ drawData.fileList(d3_event.dataTransfer.files);
+ }).on("dragenter.svgData", over).on("dragexit.svgData", over).on("dragover.svgData", over);
+ _initialized = true;
}
- createParsers(e3) {
- for (let t2 of e3) {
- let { type: e4, chunk: i3 } = t2, n3 = this.options[e4];
- if (n3 && n3.enabled) {
- let t3 = this.parsers[e4];
- t3 && t3.append || t3 || this.createParser(e4, i3);
- }
+ function getService() {
+ if (services.vectorTile && !_vtService) {
+ _vtService = services.vectorTile;
+ _vtService.event.on("loadedData", throttledRedraw);
+ } else if (!services.vectorTile && _vtService) {
+ _vtService = null;
}
+ return _vtService;
}
- async readSegments(e3) {
- let t2 = e3.map(this.ensureSegmentChunk);
- await Promise.all(t2);
+ function showLayer() {
+ layerOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
+ dispatch14.call("change");
+ });
}
- };
- var re2 = class {
- static findPosition(e3, t2) {
- let i3 = e3.getUint16(t2 + 2) + 2, n3 = "function" == typeof this.headerLength ? this.headerLength(e3, t2, i3) : this.headerLength, s2 = t2 + n3, r2 = i3 - n3;
- return { offset: t2, length: i3, headerLength: n3, start: s2, size: r2, end: s2 + r2 };
+ function hideLayer() {
+ throttledRedraw.cancel();
+ layer.transition().duration(250).style("opacity", 0).on("end", layerOff);
}
- static parse(e3, t2 = {}) {
- return new this(e3, new q({ [this.type]: t2 }), e3).parse();
+ function layerOn() {
+ layer.style("display", "block");
}
- normalizeInput(e3) {
- return e3 instanceof I ? e3 : new I(e3);
+ function layerOff() {
+ layer.selectAll(".viewfield-group").remove();
+ layer.style("display", "none");
}
- constructor(e3, t2 = {}, i3) {
- c(this, "errors", []), c(this, "raw", /* @__PURE__ */ new Map()), c(this, "handleError", (e4) => {
- if (!this.options.silentErrors)
- throw e4;
- this.errors.push(e4.message);
- }), this.chunk = this.normalizeInput(e3), this.file = i3, this.type = this.constructor.type, this.globalOptions = this.options = t2, this.localOptions = t2[this.type], this.canTranslate = this.localOptions && this.localOptions.translate;
+ function ensureIDs(gj) {
+ if (!gj)
+ return null;
+ if (gj.type === "FeatureCollection") {
+ for (var i3 = 0; i3 < gj.features.length; i3++) {
+ ensureFeatureID(gj.features[i3]);
+ }
+ } else {
+ ensureFeatureID(gj);
+ }
+ return gj;
}
- translate() {
- this.canTranslate && (this.translated = this.translateBlock(this.raw, this.type));
+ function ensureFeatureID(feature3) {
+ if (!feature3)
+ return;
+ feature3.__featurehash__ = utilHashcode((0, import_fast_json_stable_stringify.default)(feature3));
+ return feature3;
}
- get output() {
- return this.translated ? this.translated : this.raw ? Object.fromEntries(this.raw) : void 0;
+ function getFeatures(gj) {
+ if (!gj)
+ return [];
+ if (gj.type === "FeatureCollection") {
+ return gj.features;
+ } else {
+ return [gj];
+ }
}
- translateBlock(e3, t2) {
- let i3 = N.get(t2), n3 = B.get(t2), s2 = E.get(t2), r2 = this.options[t2], a2 = r2.reviveValues && !!i3, o2 = r2.translateValues && !!n3, l2 = r2.translateKeys && !!s2, h2 = {};
- for (let [t3, r3] of e3)
- a2 && i3.has(t3) ? r3 = i3.get(t3)(r3) : o2 && n3.has(t3) && (r3 = this.translateValue(r3, n3.get(t3))), l2 && s2.has(t3) && (t3 = s2.get(t3) || t3), h2[t3] = r3;
- return h2;
+ function featureKey(d2) {
+ return d2.__featurehash__;
}
- translateValue(e3, t2) {
- return t2[e3] || t2.DEFAULT || e3;
+ function isPolygon(d2) {
+ return d2.geometry.type === "Polygon" || d2.geometry.type === "MultiPolygon";
}
- assignToOutput(e3, t2) {
- this.assignObjectToOutput(e3, this.constructor.type, t2);
+ function clipPathID(d2) {
+ return "ideditor-data-" + d2.__featurehash__ + "-clippath";
}
- assignObjectToOutput(e3, t2, i3) {
- if (this.globalOptions.mergeOutput)
- return Object.assign(e3, i3);
- e3[t2] ? Object.assign(e3[t2], i3) : e3[t2] = i3;
+ function featureClasses(d2) {
+ return [
+ "data" + d2.__featurehash__,
+ d2.geometry.type,
+ isPolygon(d2) ? "area" : "",
+ d2.__layerID__ || ""
+ ].filter(Boolean).join(" ");
}
- };
- c(re2, "headerLength", 4), c(re2, "type", void 0), c(re2, "multiSegment", false), c(re2, "canHandle", () => false);
- function ae(e3) {
- return 192 === e3 || 194 === e3 || 196 === e3 || 219 === e3 || 221 === e3 || 218 === e3 || 254 === e3;
- }
- function oe(e3) {
- return e3 >= 224 && e3 <= 239;
- }
- function le(e3, t2, i3) {
- for (let [n3, s2] of T)
- if (s2.canHandle(e3, t2, i3))
- return n3;
- }
- var he = class extends se {
- constructor(...e3) {
- super(...e3), c(this, "appSegments", []), c(this, "jpegSegments", []), c(this, "unknownSegments", []);
+ 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(d2) {
+ return "datagroup datagroup-" + d2;
+ }).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(d2) {
+ var datagroup = this.parentNode.__data__;
+ return "pathdata " + datagroup + " " + featureClasses(d2);
+ }).attr("clip-path", function(d2) {
+ var datagroup = this.parentNode.__data__;
+ return datagroup === "fill" ? "url(#" + clipPathID(d2) + ")" : null;
+ }).merge(paths).attr("d", function(d2) {
+ var datagroup = this.parentNode.__data__;
+ return datagroup === "fill" ? getAreaPath(d2) : getPath(d2);
+ });
+ 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(d2) {
+ return _showLabels && d2.properties && (d2.properties.desc || d2.properties.name);
+ });
+ var labels = selection3.selectAll("text." + textClass).data(labelData, featureKey);
+ labels.exit().remove();
+ labels = labels.enter().append("text").attr("class", function(d2) {
+ return textClass + " " + featureClasses(d2);
+ }).merge(labels).text(function(d2) {
+ return d2.properties.desc || d2.properties.name;
+ }).attr("x", function(d2) {
+ var centroid = labelPath.centroid(d2);
+ return centroid[0] + 11;
+ }).attr("y", function(d2) {
+ var centroid = labelPath.centroid(d2);
+ return centroid[1];
+ });
+ }
}
- static canHandle(e3, t2) {
- return 65496 === t2;
+ function getExtension(fileName) {
+ if (!fileName)
+ return;
+ var re3 = /\.(gpx|kml|(geo)?json|png)$/i;
+ var match = fileName.toLowerCase().match(re3);
+ return match && match.length && match[0];
}
- async parse() {
- await this.findAppSegments(), await this.readSegments(this.appSegments), this.mergeMultiSegments(), this.createParsers(this.mergedAppSegments || this.appSegments);
+ function xmlToDom(textdata) {
+ return new DOMParser().parseFromString(textdata, "text/xml");
}
- setupSegmentFinderArgs(e3) {
- true === e3 ? (this.findAll = true, this.wanted = new Set(T.keyList())) : (e3 = void 0 === e3 ? T.keyList().filter((e4) => this.options[e4].enabled) : e3.filter((e4) => this.options[e4].enabled && T.has(e4)), this.findAll = false, this.remaining = new Set(e3), this.wanted = new Set(e3)), this.unfinishedMultiSegment = false;
+ 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);
+ }
+ }
}
- async findAppSegments(e3 = 0, t2) {
- this.setupSegmentFinderArgs(t2);
- let { file: i3, findAll: n3, wanted: s2, remaining: r2 } = this;
- if (!n3 && this.file.chunked && (n3 = Array.from(s2).some((e4) => {
- let t3 = T.get(e4), i4 = this.options[e4];
- return t3.multiSegment && i4.multiSegment;
- }), n3 && await this.file.readWhole()), e3 = this.findAppSegmentsInRange(e3, i3.byteLength), !this.options.onlyTiff && i3.chunked) {
- let t3 = false;
- for (; r2.size > 0 && !t3 && (i3.canReadNextChunk || this.unfinishedMultiSegment); ) {
- let { nextChunkOffset: n4 } = i3, s3 = this.appSegments.some((e4) => !this.file.available(e4.offset || e4.start, e4.length || e4.size));
- if (t3 = e3 > n4 && !s3 ? !await i3.readNextChunk(e3) : !await i3.readNextChunk(n4), void 0 === (e3 = this.findAppSegmentsInRange(e3, i3.byteLength)))
- return;
+ 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();
+ }
+ dispatch14.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();
+ }
+ dispatch14.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 i3 = 0; i3 < blocklists.length; i3++) {
+ regex = blocklists[i3];
+ 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];
+ dispatch14.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";
+ }
+ dispatch14.call("change");
+ return this;
+ };
+ drawData.fileList = function(fileList) {
+ if (!arguments.length)
+ return _fileList;
+ _template = null;
+ _geojson = null;
+ _src = null;
+ _fileList = fileList;
+ if (!fileList || !fileList.length)
+ return this;
+ var f2 = fileList[0];
+ var extension = getExtension(f2.name);
+ var reader = new FileReader();
+ reader.onload = /* @__PURE__ */ function() {
+ return function(e3) {
+ drawData.setFile(extension, e3.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 c2 = geom.coordinates;
+ switch (geom.type) {
+ case "Point":
+ c2 = [c2];
+ case "MultiPoint":
+ case "LineString":
+ break;
+ case "MultiPolygon":
+ c2 = utilArrayFlatten(c2);
+ case "Polygon":
+ case "MultiLineString":
+ c2 = utilArrayFlatten(c2);
+ break;
}
+ return utilArrayUnion(coords2, c2);
+ }, []);
+ 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, (d2) => d2.label);
+ legendItems.exit().remove();
+ legendItems.enter().append("span").attr("class", (d2) => "debug-legend-item ".concat(d2.class)).text((d2) => d2.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((d2) => {
+ const hits = showImagery && d2.query.bbox(extent.rectangle(), true) || [];
+ const features = hits.map((d4) => d4.features[d4.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",
+ "roentgen-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(d2) {
+ return "ideditor-pattern-" + d2[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(d2) {
+ return "pattern-color-" + d2[0];
+ });
+ patterns2.append("image").attr("x", 0).attr("y", 0).attr("width", 32).attr("height", 32).attr("xlink:href", function(d2) {
+ return context.imagePath("pattern/" + d2[1] + ".png");
+ });
+ _defsSelection.selectAll("clipPath").data([12, 18, 20, 32, 45]).enter().append("clipPath").attr("id", function(d2) {
+ return "ideditor-clip-square-" + d2;
+ }).append("rect").attr("x", 0).attr("y", 0).attr("width", function(d2) {
+ return d2;
+ }).attr("height", function(d2) {
+ return d2;
+ });
+ addSprites(_spritesheetIds, true);
}
- findAppSegmentsInRange(e3, t2) {
- t2 -= 2;
- let i3, n3, s2, r2, a2, o2, { file: l2, findAll: h2, wanted: u2, remaining: c2, options: f3 } = this;
- for (; e3 < t2; e3++)
- if (255 === l2.getUint8(e3)) {
- if (i3 = l2.getUint8(e3 + 1), oe(i3)) {
- if (n3 = l2.getUint16(e3 + 2), s2 = le(l2, e3, n3), s2 && u2.has(s2) && (r2 = T.get(s2), a2 = r2.findPosition(l2, e3), o2 = f3[s2], a2.type = s2, this.appSegments.push(a2), !h2 && (r2.multiSegment && o2.multiSegment ? (this.unfinishedMultiSegment = a2.chunkNumber < a2.chunkCount, this.unfinishedMultiSegment || c2.delete(s2)) : c2.delete(s2), 0 === c2.size)))
- break;
- f3.recordUnknownSegments && (a2 = re2.findPosition(l2, e3), a2.marker = i3, this.unknownSegments.push(a2)), e3 += n3 + 1;
- } else if (ae(i3)) {
- if (n3 = l2.getUint16(e3 + 2), 218 === i3 && false !== f3.stopAfterSos)
- return;
- f3.recordJpegSegments && this.jpegSegments.push({ offset: e3, length: n3, marker: i3 }), e3 += n3 + 1;
+ function addSprites(ids, overrideColors) {
+ _spritesheetIds = utilArrayUniq(_spritesheetIds.concat(ids));
+ var spritesheets = _defsSelection.selectAll(".spritesheet").data(_spritesheetIds);
+ spritesheets.enter().append("g").attr("class", function(d2) {
+ return "spritesheet spritesheet-" + d2;
+ }).each(function(d2) {
+ var url = context.imagePath(d2 + ".svg");
+ var node = select_default2(this).node();
+ svg(url).then(function(svg2) {
+ node.appendChild(
+ select_default2(svg2.documentElement).attr("id", "ideditor-" + d2).node()
+ );
+ if (overrideColors && d2 !== "iD-sprite") {
+ select_default2(node).selectAll("path").attr("fill", "currentColor");
}
- }
- return e3;
- }
- mergeMultiSegments() {
- if (!this.appSegments.some((e4) => e4.multiSegment))
- return;
- let e3 = function(e4, t2) {
- let i3, n3, s2, r2 = /* @__PURE__ */ new Map();
- for (let a2 = 0; a2 < e4.length; a2++)
- i3 = e4[a2], n3 = i3[t2], r2.has(n3) ? s2 = r2.get(n3) : r2.set(n3, s2 = []), s2.push(i3);
- return Array.from(r2);
- }(this.appSegments, "type");
- this.mergedAppSegments = e3.map(([e4, t2]) => {
- let i3 = T.get(e4, this.options);
- if (i3.handleMultiSegments) {
- return { type: e4, chunk: i3.handleMultiSegments(t2) };
- }
- return t2[0];
+ }).catch(function() {
+ });
});
+ spritesheets.exit().remove();
}
- getSegment(e3) {
- return this.appSegments.find((t2) => t2.type === e3);
- }
- async getOrFindSegment(e3) {
- let t2 = this.getSegment(e3);
- return void 0 === t2 && (await this.findAppSegments(0, [e3]), t2 = this.getSegment(e3)), t2;
- }
- };
- c(he, "type", "jpeg"), w.set("jpeg", he);
- var ue = [void 0, 1, 1, 2, 4, 8, 1, 1, 2, 4, 8, 4, 8, 4];
- var ce = class extends re2 {
- parseHeader() {
- var e3 = this.chunk.getUint16();
- 18761 === e3 ? this.le = true : 19789 === e3 && (this.le = false), this.chunk.le = this.le, this.headerParsed = true;
- }
- parseTags(e3, t2, i3 = /* @__PURE__ */ new Map()) {
- let { pick: n3, skip: s2 } = this.options[t2];
- n3 = new Set(n3);
- let r2 = n3.size > 0, a2 = 0 === s2.size, o2 = this.chunk.getUint16(e3);
- e3 += 2;
- for (let l2 = 0; l2 < o2; l2++) {
- let o3 = this.chunk.getUint16(e3);
- if (r2) {
- if (n3.has(o3) && (i3.set(o3, this.parseTag(e3, o3, t2)), n3.delete(o3), 0 === n3.size))
- break;
- } else
- !a2 && s2.has(o3) || i3.set(o3, this.parseTag(e3, o3, t2));
- e3 += 12;
- }
- return i3;
+ drawDefs.addSprites = addSprites;
+ return drawDefs;
+ }
+
+ // modules/svg/keepRight.js
+ var _layerEnabled = false;
+ var _qaService;
+ function svgKeepRight(projection2, context, dispatch14) {
+ const throttledRedraw = throttle_default(() => dispatch14.call("change"), 1e3);
+ const minZoom4 = 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");
}
- parseTag(e3, t2, i3) {
- let { chunk: n3 } = this, s2 = n3.getUint16(e3 + 2), r2 = n3.getUint32(e3 + 4), a2 = ue[s2];
- if (a2 * r2 <= 4 ? e3 += 8 : e3 = n3.getUint32(e3 + 8), (s2 < 1 || s2 > 13) && g2(`Invalid TIFF value type. block: ${i3.toUpperCase()}, tag: ${t2.toString(16)}, type: ${s2}, offset ${e3}`), e3 > n3.byteLength && g2(`Invalid TIFF value offset. block: ${i3.toUpperCase()}, tag: ${t2.toString(16)}, type: ${s2}, offset ${e3} is outside of chunk size ${n3.byteLength}`), 1 === s2)
- return n3.getUint8Array(e3, r2);
- if (2 === s2)
- return m(n3.getString(e3, r2));
- if (7 === s2)
- return n3.getUint8Array(e3, r2);
- if (1 === r2)
- return this.parseTagValue(s2, e3);
- {
- let t3 = new (function(e4) {
- switch (e4) {
- case 1:
- return Uint8Array;
- case 3:
- return Uint16Array;
- case 4:
- return Uint32Array;
- case 5:
- return Array;
- case 6:
- return Int8Array;
- case 8:
- return Int16Array;
- case 9:
- return Int32Array;
- case 10:
- return Array;
- case 11:
- return Float32Array;
- case 12:
- return Float64Array;
- default:
- return Array;
- }
- }(s2))(r2), i4 = a2;
- for (let n4 = 0; n4 < r2; n4++)
- t3[n4] = this.parseTagValue(s2, e3), e3 += i4;
- return t3;
+ function getService() {
+ if (services.keepRight && !_qaService) {
+ _qaService = services.keepRight;
+ _qaService.on("loaded", throttledRedraw);
+ } else if (!services.keepRight && _qaService) {
+ _qaService = null;
}
+ return _qaService;
}
- parseTagValue(e3, t2) {
- let { chunk: i3 } = this;
- switch (e3) {
- case 1:
- return i3.getUint8(t2);
- case 3:
- return i3.getUint16(t2);
- case 4:
- return i3.getUint32(t2);
- case 5:
- return i3.getUint32(t2) / i3.getUint32(t2 + 4);
- case 6:
- return i3.getInt8(t2);
- case 8:
- return i3.getInt16(t2);
- case 9:
- return i3.getInt32(t2);
- case 10:
- return i3.getInt32(t2) / i3.getInt32(t2 + 4);
- case 11:
- return i3.getFloat(t2);
- case 12:
- return i3.getDouble(t2);
- case 13:
- return i3.getUint32(t2);
- default:
- g2(`Invalid tiff type ${e3}`);
+ function editOn() {
+ if (!layerVisible) {
+ layerVisible = true;
+ drawLayer.style("display", "block");
}
}
- };
- var fe = class extends ce {
- static canHandle(e3, t2) {
- return 225 === e3.getUint8(t2 + 1) && 1165519206 === e3.getUint32(t2 + 4) && 0 === e3.getUint16(t2 + 8);
- }
- async parse() {
- this.parseHeader();
- let { options: e3 } = this;
- return e3.ifd0.enabled && await this.parseIfd0Block(), e3.exif.enabled && await this.safeParse("parseExifBlock"), e3.gps.enabled && await this.safeParse("parseGpsBlock"), e3.interop.enabled && await this.safeParse("parseInteropBlock"), e3.ifd1.enabled && await this.safeParse("parseThumbnailBlock"), this.createOutput();
- }
- safeParse(e3) {
- let t2 = this[e3]();
- return void 0 !== t2.catch && (t2 = t2.catch(this.handleError)), t2;
- }
- findIfd0Offset() {
- void 0 === this.ifd0Offset && (this.ifd0Offset = this.chunk.getUint32(4));
- }
- findIfd1Offset() {
- if (void 0 === this.ifd1Offset) {
- this.findIfd0Offset();
- let e3 = this.chunk.getUint16(this.ifd0Offset), t2 = this.ifd0Offset + 2 + 12 * e3;
- this.ifd1Offset = this.chunk.getUint32(t2);
+ function editOff() {
+ if (layerVisible) {
+ layerVisible = false;
+ drawLayer.style("display", "none");
+ drawLayer.selectAll(".qaItem.keepRight").remove();
+ touchLayer.selectAll(".qaItem.keepRight").remove();
}
}
- parseBlock(e3, t2) {
- let i3 = /* @__PURE__ */ new Map();
- return this[t2] = i3, this.parseTags(e3, t2, i3), i3;
- }
- async parseIfd0Block() {
- if (this.ifd0)
- return;
- let { file: e3 } = this;
- this.findIfd0Offset(), this.ifd0Offset < 8 && g2("Malformed EXIF data"), !e3.chunked && this.ifd0Offset > e3.byteLength && g2(`IFD0 offset points to outside of file.
-this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tiff && await e3.ensureChunk(this.ifd0Offset, S(this.options));
- let t2 = this.parseBlock(this.ifd0Offset, "ifd0");
- return 0 !== t2.size ? (this.exifOffset = t2.get(34665), this.interopOffset = t2.get(40965), this.gpsOffset = t2.get(34853), this.xmp = t2.get(700), this.iptc = t2.get(33723), this.icc = t2.get(34675), this.options.sanitize && (t2.delete(34665), t2.delete(40965), t2.delete(34853), t2.delete(700), t2.delete(33723), t2.delete(34675)), t2) : void 0;
- }
- async parseExifBlock() {
- if (this.exif)
- return;
- if (this.ifd0 || await this.parseIfd0Block(), void 0 === this.exifOffset)
- return;
- this.file.tiff && await this.file.ensureChunk(this.exifOffset, S(this.options));
- let e3 = this.parseBlock(this.exifOffset, "exif");
- return this.interopOffset || (this.interopOffset = e3.get(40965)), this.makerNote = e3.get(37500), this.userComment = e3.get(37510), this.options.sanitize && (e3.delete(40965), e3.delete(37500), e3.delete(37510)), this.unpack(e3, 41728), this.unpack(e3, 41729), e3;
+ function layerOn() {
+ editOn();
+ drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch14.call("change"));
}
- unpack(e3, t2) {
- let i3 = e3.get(t2);
- i3 && 1 === i3.length && e3.set(t2, i3[0]);
+ function layerOff() {
+ throttledRedraw.cancel();
+ drawLayer.interrupt();
+ touchLayer.selectAll(".qaItem.keepRight").remove();
+ drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
+ editOff();
+ dispatch14.call("change");
+ });
}
- async parseGpsBlock() {
- if (this.gps)
- return;
- if (this.ifd0 || await this.parseIfd0Block(), void 0 === this.gpsOffset)
+ function updateMarkers() {
+ if (!layerVisible || !_layerEnabled)
return;
- let e3 = this.parseBlock(this.gpsOffset, "gps");
- return e3 && e3.has(2) && e3.has(4) && (e3.set("latitude", de(...e3.get(2), e3.get(1))), e3.set("longitude", de(...e3.get(4), e3.get(3)))), e3;
- }
- async parseInteropBlock() {
- if (!this.interop && (this.ifd0 || await this.parseIfd0Block(), void 0 !== this.interopOffset || this.exif || await this.parseExifBlock(), void 0 !== this.interopOffset))
- return this.parseBlock(this.interopOffset, "interop");
- }
- async parseThumbnailBlock(e3 = false) {
- if (!this.ifd1 && !this.ifd1Parsed && (!this.options.mergeOutput || e3))
- return this.findIfd1Offset(), this.ifd1Offset > 0 && (this.parseBlock(this.ifd1Offset, "ifd1"), this.ifd1Parsed = true), this.ifd1;
- }
- async extractThumbnail() {
- if (this.headerParsed || this.parseHeader(), this.ifd1Parsed || await this.parseThumbnailBlock(true), void 0 === this.ifd1)
+ 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, (d2) => d2.id);
+ markers.exit().remove();
+ const markersEnter = markers.enter().append("g").attr("class", (d2) => "qaItem ".concat(d2.service, " itemId-").concat(d2.id, " itemType-").concat(d2.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", (d2) => d2.id === selectedID).attr("transform", getTransform);
+ if (touchLayer.empty())
return;
- let e3 = this.ifd1.get(513), t2 = this.ifd1.get(514);
- return this.chunk.getUint8Array(e3, t2);
- }
- get image() {
- return this.ifd0;
- }
- get thumbnail() {
- return this.ifd1;
- }
- createOutput() {
- let e3, t2, i3, n3 = {};
- for (t2 of H)
- if (e3 = this[t2], !p(e3))
- if (i3 = this.canTranslate ? this.translateBlock(e3, t2) : Object.fromEntries(e3), this.options.mergeOutput) {
- if ("ifd1" === t2)
- continue;
- Object.assign(n3, i3);
- } else
- n3[t2] = i3;
- return this.makerNote && (n3.makerNote = this.makerNote), this.userComment && (n3.userComment = this.userComment), n3;
- }
- assignToOutput(e3, t2) {
- if (this.globalOptions.mergeOutput)
- Object.assign(e3, t2);
- else
- for (let [i3, n3] of Object.entries(t2))
- this.assignObjectToOutput(e3, i3, n3);
- }
- };
- function de(e3, t2, i3, n3) {
- var s2 = e3 + t2 / 60 + i3 / 3600;
- return "S" !== n3 && "W" !== n3 || (s2 *= -1), s2;
- }
- c(fe, "type", "tiff"), c(fe, "headerLength", 10), T.set("tiff", fe);
- var pe = Object.freeze({ __proto__: null, default: ne, Exifr: te, fileParsers: w, segmentParsers: T, fileReaders: A, tagKeys: E, tagValues: B, tagRevivers: N, createDictionary: U, extendDictionary: F, fetchUrlAsArrayBuffer: M, readBlobAsArrayBuffer: R, chunkedProps: G, otherSegments: V, segments: z, tiffBlocks: H, segmentsAndBlocks: j2, tiffExtractables: W, inheritables: K, allFormatters: X, Options: q, parse: ie });
- var ge = { ifd0: false, ifd1: false, exif: false, gps: false, interop: false, sanitize: false, reviveValues: true, translateKeys: false, translateValues: false, mergeOutput: false };
- var me = Object.assign({}, ge, { firstChunkSize: 4e4, gps: [1, 2, 3, 4] });
- async function Se(e3) {
- let t2 = new te(me);
- await t2.read(e3);
- let i3 = await t2.parse();
- if (i3 && i3.gps) {
- let { latitude: e4, longitude: t3 } = i3.gps;
- return { latitude: e4, longitude: t3 };
- }
- }
- var Ce = Object.assign({}, ge, { tiff: false, ifd1: true, mergeOutput: false });
- async function ye(e3) {
- let t2 = new te(Ce);
- await t2.read(e3);
- let i3 = await t2.extractThumbnail();
- return i3 && a ? s.from(i3) : i3;
- }
- async function be(e3) {
- let t2 = await this.thumbnail(e3);
- if (void 0 !== t2) {
- let e4 = new Blob([t2]);
- return URL.createObjectURL(e4);
- }
- }
- var Ie = Object.assign({}, ge, { firstChunkSize: 4e4, ifd0: [274] });
- async function Pe(e3) {
- let t2 = new te(Ie);
- await t2.read(e3);
- let i3 = await t2.parse();
- if (i3 && i3.ifd0)
- return i3.ifd0[274];
- }
- var ke = Object.freeze({ 1: { dimensionSwapped: false, scaleX: 1, scaleY: 1, deg: 0, rad: 0 }, 2: { dimensionSwapped: false, scaleX: -1, scaleY: 1, deg: 0, rad: 0 }, 3: { dimensionSwapped: false, scaleX: 1, scaleY: 1, deg: 180, rad: 180 * Math.PI / 180 }, 4: { dimensionSwapped: false, scaleX: -1, scaleY: 1, deg: 180, rad: 180 * Math.PI / 180 }, 5: { dimensionSwapped: true, scaleX: 1, scaleY: -1, deg: 90, rad: 90 * Math.PI / 180 }, 6: { dimensionSwapped: true, scaleX: 1, scaleY: 1, deg: 90, rad: 90 * Math.PI / 180 }, 7: { dimensionSwapped: true, scaleX: 1, scaleY: -1, deg: 270, rad: 270 * Math.PI / 180 }, 8: { dimensionSwapped: true, scaleX: 1, scaleY: 1, deg: 270, rad: 270 * Math.PI / 180 } });
- var we = true;
- var Te = true;
- if ("object" == typeof navigator) {
- let e3 = navigator.userAgent;
- if (e3.includes("iPad") || e3.includes("iPhone")) {
- let t2 = e3.match(/OS (\d+)_(\d+)/);
- if (t2) {
- let [, e4, i3] = t2, n3 = Number(e4) + 0.1 * Number(i3);
- we = n3 < 13.4, Te = false;
+ const fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ const targets = touchLayer.selectAll(".qaItem.keepRight").data(data, (d2) => d2.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", (d2) => "qaItem ".concat(d2.service, " target ").concat(fillClass, " itemId-").concat(d2.id)).attr("transform", getTransform);
+ function sortY(a2, b2) {
+ return a2.id === selectedID ? 1 : b2.id === selectedID ? -1 : a2.severity === "error" && b2.severity !== "error" ? 1 : b2.severity === "error" && a2.severity !== "error" ? -1 : b2.loc[1] - a2.loc[1];
}
- } else if (e3.includes("OS X 10")) {
- let [, t2] = e3.match(/OS X 10[_.](\d+)/);
- we = Te = Number(t2) < 15;
}
- if (e3.includes("Chrome/")) {
- let [, t2] = e3.match(/Chrome\/(\d+)/);
- we = Te = Number(t2) < 81;
- } else if (e3.includes("Firefox/")) {
- let [, t2] = e3.match(/Firefox\/(\d+)/);
- we = Te = Number(t2) < 77;
- }
- }
- async function Ae(e3) {
- let t2 = await Pe(e3);
- return Object.assign({ canvas: we, css: Te }, ke[t2]);
- }
- var De = class extends I {
- constructor(...e3) {
- super(...e3), c(this, "ranges", new Oe()), 0 !== this.byteLength && this.ranges.add(0, this.byteLength);
+ 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() >= minZoom4) {
+ editOn();
+ service.loadIssues(projection2);
+ updateMarkers();
+ } else {
+ editOff();
+ }
+ }
}
- _tryExtend(e3, t2, i3) {
- if (0 === e3 && 0 === this.byteLength && i3) {
- let e4 = new DataView(i3.buffer || i3, i3.byteOffset, i3.byteLength);
- this._swapDataView(e4);
+ drawKeepRight.enabled = function(val) {
+ if (!arguments.length)
+ return _layerEnabled;
+ _layerEnabled = val;
+ if (_layerEnabled) {
+ layerOn();
} else {
- let i4 = e3 + t2;
- if (i4 > this.byteLength) {
- let { dataView: e4 } = this._extend(i4);
- this._swapDataView(e4);
+ layerOff();
+ if (context.selectedErrorID()) {
+ context.enter(modeBrowse(context));
}
}
+ dispatch14.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;
}
- _extend(e3) {
- let t2;
- t2 = a ? s.allocUnsafe(e3) : new Uint8Array(e3);
- let i3 = new DataView(t2.buffer, t2.byteOffset, t2.byteLength);
- return t2.set(new Uint8Array(this.buffer, this.byteOffset, this.byteLength), 0), { uintView: t2, dataView: i3 };
- }
- subarray(e3, t2, i3 = false) {
- return t2 = t2 || this._lengthToEnd(e3), i3 && this._tryExtend(e3, t2), this.ranges.add(e3, t2), super.subarray(e3, t2);
- }
- set(e3, t2, i3 = false) {
- i3 && this._tryExtend(t2, e3.byteLength, e3);
- let n3 = super.set(e3, t2);
- return this.ranges.add(t2, n3.byteLength), n3;
- }
- async ensureChunk(e3, t2) {
- this.chunked && (this.ranges.available(e3, t2) || await this.readChunk(e3, t2));
- }
- available(e3, t2) {
- return this.ranges.available(e3, t2);
- }
- };
- var Oe = class {
- constructor() {
- c(this, "list", []);
- }
- get length() {
- return this.list.length;
- }
- add(e3, t2, i3 = 0) {
- let n3 = e3 + t2, s2 = this.list.filter((t3) => xe(e3, t3.offset, n3) || xe(e3, t3.end, n3));
- if (s2.length > 0) {
- e3 = Math.min(e3, ...s2.map((e4) => e4.offset)), n3 = Math.max(n3, ...s2.map((e4) => e4.end)), t2 = n3 - e3;
- let i4 = s2.shift();
- i4.offset = e3, i4.length = t2, i4.end = n3, this.list = this.list.filter((e4) => !s2.includes(e4));
- } else
- this.list.push({ offset: e3, length: t2, end: n3 });
- }
- available(e3, t2) {
- let i3 = e3 + t2;
- return this.list.some((t3) => t3.offset <= e3 && i3 <= t3.end);
+ function showLayer() {
+ layer.style("display", "block");
}
- };
- function xe(e3, t2, i3) {
- return e3 <= t2 && t2 <= i3;
- }
- var ve = class extends De {
- constructor(e3, t2) {
- super(0), c(this, "chunksRead", 0), this.input = e3, this.options = t2;
+ function hideLayer() {
+ layer.transition().duration(250).style("opacity", 0);
}
- async readWhole() {
- this.chunked = false, await this.readChunk(this.nextChunkOffset);
+ function layerOn() {
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1);
}
- async readChunked() {
- this.chunked = true, await this.readChunk(0, this.options.firstChunkSize);
+ function layerOff() {
+ layer.style("display", "none");
}
- async readNextChunk(e3 = this.nextChunkOffset) {
- if (this.fullyRead)
- return this.chunksRead++, false;
- let t2 = this.options.chunkSize, i3 = await this.readChunk(e3, t2);
- return !!i3 && i3.byteLength === t2;
+ function transform2(d2) {
+ return svgPointTransform(projection2)(d2);
}
- async readChunk(e3, t2) {
- if (this.chunksRead++, 0 !== (t2 = this.safeWrapAddress(e3, t2)))
- return this._readChunk(e3, t2);
+ 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();
}
- safeWrapAddress(e3, t2) {
- return void 0 !== this.size && e3 + t2 > this.size ? Math.max(0, this.size - e3) : t2;
+ 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));
}
- get nextChunkOffset() {
- if (0 !== this.ranges.list.length)
- return this.ranges.list[0].length;
+ 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();
+ }
}
- get canReadNextChunk() {
- return this.chunksRead < this.options.chunkLimit;
+ 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_rbush6 = __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_rbush6.default();
+ var _rskipped = new import_rbush6.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(s2) {
+ return preset.id.indexOf(s2) >= 0;
+ });
}
- get fullyRead() {
- return void 0 !== this.size && this.nextChunkOffset === this.size;
+ function get4(array2, prop) {
+ return function(d2, i3) {
+ return array2[i3][prop];
+ };
}
- read() {
- return this.options.chunked ? this.readChunked() : this.readWhole();
+ function textWidth(text, size, elem) {
+ var c2 = _textWidthCache[size];
+ if (!c2)
+ c2 = _textWidthCache[size] = {};
+ if (c2[text]) {
+ return c2[text];
+ } else if (elem) {
+ c2[text] = elem.getComputedTextLength();
+ return c2[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);
+ }
+ }
}
- close() {
+ 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(d2) {
+ return "ideditor-labelpath-" + d2.id;
+ }).attr("class", classes).merge(paths).attr("d", get4(labels, "lineString"));
}
- };
- A.set("blob", class extends ve {
- async readWhole() {
- this.chunked = false;
- let e3 = await R(this.input);
- this._swapArrayBuffer(e3);
+ 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(d2, i3) {
+ return classes + " " + labels[i3].classes + " " + d2.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(d2) {
+ return "#ideditor-labelpath-" + d2.id;
+ }).text(utilDisplayNameForPath);
}
- readChunked() {
- return this.chunked = true, this.size = this.input.size, super.readChunked();
+ 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(d2, i3) {
+ return classes + " " + labels[i3].classes + " " + d2.id;
+ }).merge(texts).attr("x", get4(labels, "x")).attr("y", get4(labels, "y")).style("text-anchor", get4(labels, "textAnchor")).text(utilDisplayName).each(function(d2, i3) {
+ textWidth(utilDisplayName(d2), labels[i3].height, this);
+ });
}
- async _readChunk(e3, t2) {
- let i3 = t2 ? e3 + t2 : void 0, n3 = this.input.slice(e3, i3), s2 = await R(n3);
- return this.set(s2, e3, true);
+ function drawAreaLabels(selection2, entities, filter2, classes, labels) {
+ entities = entities.filter(hasText);
+ labels = labels.filter(hasText);
+ drawPointLabels(selection2, entities, filter2, classes, labels);
+ function hasText(d2, i3) {
+ return labels[i3].hasOwnProperty("x") && labels[i3].hasOwnProperty("y");
+ }
}
- });
- var Me = Object.freeze({ __proto__: null, default: pe, Exifr: te, fileParsers: w, segmentParsers: T, fileReaders: A, tagKeys: E, tagValues: B, tagRevivers: N, createDictionary: U, extendDictionary: F, fetchUrlAsArrayBuffer: M, readBlobAsArrayBuffer: R, chunkedProps: G, otherSegments: V, segments: z, tiffBlocks: H, segmentsAndBlocks: j2, tiffExtractables: W, inheritables: K, allFormatters: X, Options: q, parse: ie, gpsOnlyOptions: me, gps: Se, thumbnailOnlyOptions: Ce, thumbnail: ye, thumbnailUrl: be, orientationOnlyOptions: Ie, orientation: Pe, rotations: ke, get rotateCanvas() {
- return we;
- }, get rotateCss() {
- return Te;
- }, rotation: Ae });
- A.set("url", class extends ve {
- async readWhole() {
- this.chunked = false;
- let e3 = await M(this.input);
- e3 instanceof ArrayBuffer ? this._swapArrayBuffer(e3) : e3 instanceof Uint8Array && this._swapBuffer(e3);
+ 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(d2) {
+ var preset = _mainPresetIndex.match(d2, context.graph());
+ var picon = preset && preset.icon;
+ return picon ? "#" + picon : "";
+ });
}
- async _readChunk(e3, t2) {
- let i3 = t2 ? e3 + t2 - 1 : void 0, n3 = this.options.httpHeaders || {};
- (e3 || i3) && (n3.range = `bytes=${[e3, i3].join("-")}`);
- let s2 = await h(this.input, { headers: n3 }), r2 = await s2.arrayBuffer(), a2 = r2.byteLength;
- if (416 !== s2.status)
- return a2 !== t2 && (this.size = e3 + a2), this.set(r2, e3, true);
+ 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(d2) {
+ return { type: "Polygon", coordinates: [[
+ [d2.minX, d2.minY],
+ [d2.maxX, d2.minY],
+ [d2.maxX, d2.maxY],
+ [d2.minX, d2.maxY],
+ [d2.minX, d2.minY]
+ ]] };
+ });
+ }
+ var boxes = selection2.selectAll("." + which).data(gj);
+ boxes.exit().remove();
+ boxes.enter().append("path").attr("class", classes).merge(boxes).attr("d", path_default());
}
- });
- I.prototype.getUint64 = function(e3) {
- let t2 = this.getUint32(e3), i3 = this.getUint32(e3 + 4);
- return t2 < 1048575 ? t2 << 32 | i3 : void 0 !== typeof r ? (console.warn("Using BigInt because of type 64uint but JS can only handle 53b numbers."), r(t2) << r(32) | r(i3)) : void g2("Trying to read 64b value but JS can only handle 53b numbers.");
- };
- var Re = class extends se {
- parseBoxes(e3 = 0) {
- let t2 = [];
- for (; e3 < this.file.byteLength - 4; ) {
- let i3 = this.parseBoxHead(e3);
- if (t2.push(i3), 0 === i3.length)
- break;
- e3 += i3.length;
+ 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 i3, j2, k2, entity, geometry;
+ for (i3 = 0; i3 < labelStack.length; i3++) {
+ labelable.push([]);
+ }
+ if (fullRedraw) {
+ _rdrawn.clear();
+ _rskipped.clear();
+ _entitybboxes = {};
+ } else {
+ for (i3 = 0; i3 < entities.length; i3++) {
+ entity = entities[i3];
+ 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 (i3 = 0; i3 < entities.length; i3++) {
+ entity = entities[i3];
+ 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 (k2 = 0; k2 < labelStack.length; k2++) {
+ var matchGeom = labelStack[k2][0];
+ var matchKey = labelStack[k2][1];
+ var matchVal = labelStack[k2][2];
+ var hasVal = entity.tags[matchKey];
+ if (geometry === matchGeom && hasVal && (matchVal === "*" || matchVal === hasVal)) {
+ labelable[k2].push(entity);
+ break;
+ }
+ }
+ }
+ var positions = {
+ point: [],
+ line: [],
+ area: []
+ };
+ var labelled = {
+ point: [],
+ line: [],
+ area: []
+ };
+ for (k2 = 0; k2 < labelable.length; k2++) {
+ var fontSize = labelStack[k2][3];
+ for (i3 = 0; i3 < labelable[k2].length; i3++) {
+ entity = labelable[k2][i3];
+ geometry = entity.geometry(graph);
+ var getName = geometry === "line" ? utilDisplayNameForPath : utilDisplayName;
+ var name = getName(entity);
+ var width = name && textWidth(name, fontSize);
+ var p2 = null;
+ if (geometry === "point" || geometry === "vertex") {
+ if (wireframe)
+ continue;
+ var renderAs = renderNodeAs[entity.id];
+ if (renderAs === "vertex" && zoom < 17)
+ continue;
+ p2 = getPointLabel(entity, width, fontSize, renderAs);
+ } else if (geometry === "line") {
+ p2 = getLineLabel(entity, width, fontSize);
+ } else if (geometry === "area") {
+ p2 = getAreaLabel(entity, width, fontSize);
+ }
+ if (p2) {
+ if (geometry === "vertex") {
+ geometry = "point";
+ }
+ p2.classes = geometry + " tag-" + labelStack[k2][1];
+ positions[geometry].push(p2);
+ 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 y2 = geometry2 === "point" ? -12 : 0;
+ var pointOffsets = {
+ ltr: [15, y2, "start"],
+ rtl: [-15, y2, "end"]
+ };
+ var textDirection = _mainLocalizer.textDirection();
+ var coord3 = projection2(entity2.loc);
+ var textPadding = 2;
+ var offset = pointOffsets[textDirection];
+ var p3 = {
+ height,
+ width: width2,
+ x: coord3[0] + offset[0],
+ y: coord3[1] + offset[1],
+ textAnchor: offset[2]
+ };
+ var bbox3;
+ if (textDirection === "rtl") {
+ bbox3 = {
+ minX: p3.x - width2 - textPadding,
+ minY: p3.y - height / 2 - textPadding,
+ maxX: p3.x + textPadding,
+ maxY: p3.y + height / 2 + textPadding
+ };
+ } else {
+ bbox3 = {
+ minX: p3.x - textPadding,
+ minY: p3.y - height / 2 - textPadding,
+ maxX: p3.x + width2 + textPadding,
+ maxY: p3.y + height / 2 + textPadding
+ };
+ }
+ if (tryInsert([bbox3], entity2.id, true)) {
+ return p3;
+ }
+ }
+ 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 length2 = geoPathLength(points);
+ if (length2 < 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 i4 = 0; i4 < lineOffsets.length; i4++) {
+ var offset = lineOffsets[i4];
+ var middle = offset / 100 * length2;
+ var start2 = middle - width2 / 2;
+ if (start2 < 0 || start2 + width2 > length2)
+ 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 a2 = sub[j3];
+ var b2 = sub[j3 + 1];
+ var num = Math.max(1, Math.floor(geoVecLength(a2, b2) / boxsize / 2));
+ for (var box = 0; box < num; box++) {
+ var p3 = geoVecInterp(a2, b2, box / num);
+ var x05 = p3[0] - boxsize - padding;
+ var y05 = p3[1] - boxsize - padding;
+ var x12 = p3[0] + boxsize + padding;
+ var y12 = p3[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(p4) {
+ var angle2 = Math.atan2(p4[1][1] - p4[0][1], p4[1][0] - p4[0][0]);
+ return !(p4[0][0] < p4[p4.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 i5 = 0; i5 < points2.length - 1; i5++) {
+ var a3 = points2[i5];
+ var b3 = points2[i5 + 1];
+ var current = geoVecLength(a3, b3);
+ var portion;
+ if (!start3 && sofar + current >= from) {
+ portion = (from - sofar) / current;
+ start3 = [
+ a3[0] + portion * (b3[0] - a3[0]),
+ a3[1] + portion * (b3[1] - a3[1])
+ ];
+ i0 = i5 + 1;
+ }
+ if (!end && sofar + current >= to) {
+ portion = (to - sofar) / current;
+ end = [
+ a3[0] + portion * (b3[0] - a3[0]),
+ a3[1] + portion * (b3[1] - a3[1])
+ ];
+ i1 = i5 + 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 p3 = {};
+ if (picon) {
+ if (addIcon()) {
+ addLabel(iconSize + padding);
+ return p3;
+ }
+ } else {
+ if (addLabel(0)) {
+ return p3;
+ }
+ }
+ 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)) {
+ p3.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)) {
+ p3.x = labelX;
+ p3.y = labelY;
+ p3.textAnchor = "middle";
+ p3.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 i4 = 0; i4 < bboxes.length; i4++) {
+ var bbox3 = bboxes[i4];
+ 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;
}
- return t2;
- }
- parseSubBoxes(e3) {
- e3.boxes = this.parseBoxes(e3.start);
- }
- findBox(e3, t2) {
- return void 0 === e3.boxes && this.parseSubBoxes(e3), e3.boxes.find((e4) => e4.kind === t2);
- }
- parseBoxHead(e3) {
- let t2 = this.file.getUint32(e3), i3 = this.file.getString(e3 + 4, 4), n3 = e3 + 8;
- return 1 === t2 && (t2 = this.file.getUint64(e3 + 8), n3 += 8), { offset: e3, length: t2, kind: i3, start: n3 };
- }
- parseBoxFullHead(e3) {
- if (void 0 !== e3.version)
- return;
- let t2 = this.file.getUint32(e3.start);
- e3.version = t2 >> 24, e3.start += 4;
- }
- };
- var Le = class extends Re {
- static canHandle(e3, t2) {
- if (0 !== t2)
- return false;
- let i3 = e3.getUint16(2);
- if (i3 > 50)
- return false;
- let n3 = 16, s2 = [];
- for (; n3 < i3; )
- s2.push(e3.getString(n3, 4)), n3 += 4;
- return s2.includes(this.type);
- }
- async parse() {
- let e3 = this.file.getUint32(0), t2 = this.parseBoxHead(e3);
- for (; "meta" !== t2.kind; )
- e3 += t2.length, await this.file.ensureChunk(e3, 16), t2 = this.parseBoxHead(e3);
- await this.file.ensureChunk(t2.offset, t2.length), this.parseBoxFullHead(t2), this.parseSubBoxes(t2), this.options.icc.enabled && await this.findIcc(t2), this.options.tiff.enabled && await this.findExif(t2);
- }
- async registerSegment(e3, t2, i3) {
- await this.file.ensureChunk(t2, i3);
- let n3 = this.file.subarray(t2, i3);
- this.createParser(e3, n3);
- }
- async findIcc(e3) {
- let t2 = this.findBox(e3, "iprp");
- if (void 0 === t2)
- return;
- let i3 = this.findBox(t2, "ipco");
- if (void 0 === i3)
- return;
- let n3 = this.findBox(i3, "colr");
- void 0 !== n3 && await this.registerSegment("icc", n3.offset + 12, n3.length);
- }
- async findExif(e3) {
- let t2 = this.findBox(e3, "iinf");
- if (void 0 === t2)
- return;
- let i3 = this.findBox(e3, "iloc");
- if (void 0 === i3)
- return;
- let n3 = this.findExifLocIdInIinf(t2), s2 = this.findExtentInIloc(i3, n3);
- if (void 0 === s2)
- return;
- let [r2, a2] = s2;
- await this.file.ensureChunk(r2, a2);
- let o2 = 4 + this.file.getUint32(r2);
- r2 += o2, a2 -= o2, await this.registerSegment("tiff", r2, a2);
+ var layer = selection2.selectAll(".layer-osm.labels");
+ layer.selectAll(".labels-group").data(["halo", "label", "debug"]).enter().append("g").attr("class", function(d2) {
+ return "labels-group " + d2;
+ });
+ 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);
}
- findExifLocIdInIinf(e3) {
- this.parseBoxFullHead(e3);
- let t2, i3, n3, s2, r2 = e3.start, a2 = this.file.getUint16(r2);
- for (r2 += 2; a2--; ) {
- if (t2 = this.parseBoxHead(r2), this.parseBoxFullHead(t2), i3 = t2.start, t2.version >= 2 && (n3 = 3 === t2.version ? 4 : 2, s2 = this.file.getString(i3 + n3 + 2, 4), "Exif" === s2))
- return this.file.getUintBytes(i3, n3);
- r2 += t2.length;
+ 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);
}
- }
- get8bits(e3) {
- let t2 = this.file.getUint8(e3);
- return [t2 >> 4, 15 & t2];
- }
- findExtentInIloc(e3, t2) {
- this.parseBoxFullHead(e3);
- let i3 = e3.start, [n3, s2] = this.get8bits(i3++), [r2, a2] = this.get8bits(i3++), o2 = 2 === e3.version ? 4 : 2, l2 = 1 === e3.version || 2 === e3.version ? 2 : 0, h2 = a2 + n3 + s2, u2 = 2 === e3.version ? 4 : 2, c2 = this.file.getUintBytes(i3, u2);
- for (i3 += u2; c2--; ) {
- let e4 = this.file.getUintBytes(i3, o2);
- i3 += o2 + l2 + 2 + r2;
- let u3 = this.file.getUint16(i3);
- if (i3 += 2, e4 === t2)
- return u3 > 1 && console.warn("ILOC box has more than one extent but we're only processing one\nPlease create an issue at https://github.com/MikeKovarik/exifr with this file"), [this.file.getUintBytes(i3 + a2, n3), this.file.getUintBytes(i3 + a2 + n3, s2)];
- i3 += u3 * h2;
+ for (var i3 = 0; i3 < selectedIDs.length; i3++) {
+ var entity = graph.hasEntity(selectedIDs[i3]);
+ if (entity && entity.type === "node") {
+ ids.push(selectedIDs[i3]);
+ }
+ }
+ 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 Ue = class extends Le {
- };
- c(Ue, "type", "heic");
- var Fe = class extends Le {
- };
- c(Fe, "type", "avif"), w.set("heic", Ue), w.set("avif", Fe), U(E, ["ifd0", "ifd1"], [[256, "ImageWidth"], [257, "ImageHeight"], [258, "BitsPerSample"], [259, "Compression"], [262, "PhotometricInterpretation"], [270, "ImageDescription"], [271, "Make"], [272, "Model"], [273, "StripOffsets"], [274, "Orientation"], [277, "SamplesPerPixel"], [278, "RowsPerStrip"], [279, "StripByteCounts"], [282, "XResolution"], [283, "YResolution"], [284, "PlanarConfiguration"], [296, "ResolutionUnit"], [301, "TransferFunction"], [305, "Software"], [306, "ModifyDate"], [315, "Artist"], [316, "HostComputer"], [317, "Predictor"], [318, "WhitePoint"], [319, "PrimaryChromaticities"], [513, "ThumbnailOffset"], [514, "ThumbnailLength"], [529, "YCbCrCoefficients"], [530, "YCbCrSubSampling"], [531, "YCbCrPositioning"], [532, "ReferenceBlackWhite"], [700, "ApplicationNotes"], [33432, "Copyright"], [33723, "IPTC"], [34665, "ExifIFD"], [34675, "ICC"], [34853, "GpsIFD"], [330, "SubIFD"], [40965, "InteropIFD"], [40091, "XPTitle"], [40092, "XPComment"], [40093, "XPAuthor"], [40094, "XPKeywords"], [40095, "XPSubject"]]), U(E, "exif", [[33434, "ExposureTime"], [33437, "FNumber"], [34850, "ExposureProgram"], [34852, "SpectralSensitivity"], [34855, "ISO"], [34858, "TimeZoneOffset"], [34859, "SelfTimerMode"], [34864, "SensitivityType"], [34865, "StandardOutputSensitivity"], [34866, "RecommendedExposureIndex"], [34867, "ISOSpeed"], [34868, "ISOSpeedLatitudeyyy"], [34869, "ISOSpeedLatitudezzz"], [36864, "ExifVersion"], [36867, "DateTimeOriginal"], [36868, "CreateDate"], [36873, "GooglePlusUploadCode"], [36880, "OffsetTime"], [36881, "OffsetTimeOriginal"], [36882, "OffsetTimeDigitized"], [37121, "ComponentsConfiguration"], [37122, "CompressedBitsPerPixel"], [37377, "ShutterSpeedValue"], [37378, "ApertureValue"], [37379, "BrightnessValue"], [37380, "ExposureCompensation"], [37381, "MaxApertureValue"], [37382, "SubjectDistance"], [37383, "MeteringMode"], [37384, "LightSource"], [37385, "Flash"], [37386, "FocalLength"], [37393, "ImageNumber"], [37394, "SecurityClassification"], [37395, "ImageHistory"], [37396, "SubjectArea"], [37500, "MakerNote"], [37510, "UserComment"], [37520, "SubSecTime"], [37521, "SubSecTimeOriginal"], [37522, "SubSecTimeDigitized"], [37888, "AmbientTemperature"], [37889, "Humidity"], [37890, "Pressure"], [37891, "WaterDepth"], [37892, "Acceleration"], [37893, "CameraElevationAngle"], [40960, "FlashpixVersion"], [40961, "ColorSpace"], [40962, "ExifImageWidth"], [40963, "ExifImageHeight"], [40964, "RelatedSoundFile"], [41483, "FlashEnergy"], [41486, "FocalPlaneXResolution"], [41487, "FocalPlaneYResolution"], [41488, "FocalPlaneResolutionUnit"], [41492, "SubjectLocation"], [41493, "ExposureIndex"], [41495, "SensingMethod"], [41728, "FileSource"], [41729, "SceneType"], [41730, "CFAPattern"], [41985, "CustomRendered"], [41986, "ExposureMode"], [41987, "WhiteBalance"], [41988, "DigitalZoomRatio"], [41989, "FocalLengthIn35mmFormat"], [41990, "SceneCaptureType"], [41991, "GainControl"], [41992, "Contrast"], [41993, "Saturation"], [41994, "Sharpness"], [41996, "SubjectDistanceRange"], [42016, "ImageUniqueID"], [42032, "OwnerName"], [42033, "SerialNumber"], [42034, "LensInfo"], [42035, "LensMake"], [42036, "LensModel"], [42037, "LensSerialNumber"], [42080, "CompositeImage"], [42081, "CompositeImageCount"], [42082, "CompositeImageExposureTimes"], [42240, "Gamma"], [59932, "Padding"], [59933, "OffsetSchema"], [65e3, "OwnerName"], [65001, "SerialNumber"], [65002, "Lens"], [65100, "RawFile"], [65101, "Converter"], [65102, "WhiteBalance"], [65105, "Exposure"], [65106, "Shadows"], [65107, "Brightness"], [65108, "Contrast"], [65109, "Saturation"], [65110, "Sharpness"], [65111, "Smoothness"], [65112, "MoireFilter"], [40965, "InteropIFD"]]), U(E, "gps", [[0, "GPSVersionID"], [1, "GPSLatitudeRef"], [2, "GPSLatitude"], [3, "GPSLongitudeRef"], [4, "GPSLongitude"], [5, "GPSAltitudeRef"], [6, "GPSAltitude"], [7, "GPSTimeStamp"], [8, "GPSSatellites"], [9, "GPSStatus"], [10, "GPSMeasureMode"], [11, "GPSDOP"], [12, "GPSSpeedRef"], [13, "GPSSpeed"], [14, "GPSTrackRef"], [15, "GPSTrack"], [16, "GPSImgDirectionRef"], [17, "GPSImgDirection"], [18, "GPSMapDatum"], [19, "GPSDestLatitudeRef"], [20, "GPSDestLatitude"], [21, "GPSDestLongitudeRef"], [22, "GPSDestLongitude"], [23, "GPSDestBearingRef"], [24, "GPSDestBearing"], [25, "GPSDestDistanceRef"], [26, "GPSDestDistance"], [27, "GPSProcessingMethod"], [28, "GPSAreaInformation"], [29, "GPSDateStamp"], [30, "GPSDifferential"], [31, "GPSHPositioningError"]]), U(B, ["ifd0", "ifd1"], [[274, { 1: "Horizontal (normal)", 2: "Mirror horizontal", 3: "Rotate 180", 4: "Mirror vertical", 5: "Mirror horizontal and rotate 270 CW", 6: "Rotate 90 CW", 7: "Mirror horizontal and rotate 90 CW", 8: "Rotate 270 CW" }], [296, { 1: "None", 2: "inches", 3: "cm" }]]);
- var Ee = U(B, "exif", [[34850, { 0: "Not defined", 1: "Manual", 2: "Normal program", 3: "Aperture priority", 4: "Shutter priority", 5: "Creative program", 6: "Action program", 7: "Portrait mode", 8: "Landscape mode" }], [37121, { 0: "-", 1: "Y", 2: "Cb", 3: "Cr", 4: "R", 5: "G", 6: "B" }], [37383, { 0: "Unknown", 1: "Average", 2: "CenterWeightedAverage", 3: "Spot", 4: "MultiSpot", 5: "Pattern", 6: "Partial", 255: "Other" }], [37384, { 0: "Unknown", 1: "Daylight", 2: "Fluorescent", 3: "Tungsten (incandescent light)", 4: "Flash", 9: "Fine weather", 10: "Cloudy weather", 11: "Shade", 12: "Daylight fluorescent (D 5700 - 7100K)", 13: "Day white fluorescent (N 4600 - 5400K)", 14: "Cool white fluorescent (W 3900 - 4500K)", 15: "White fluorescent (WW 3200 - 3700K)", 17: "Standard light A", 18: "Standard light B", 19: "Standard light C", 20: "D55", 21: "D65", 22: "D75", 23: "D50", 24: "ISO studio tungsten", 255: "Other" }], [37385, { 0: "Flash did not fire", 1: "Flash fired", 5: "Strobe return light not detected", 7: "Strobe return light detected", 9: "Flash fired, compulsory flash mode", 13: "Flash fired, compulsory flash mode, return light not detected", 15: "Flash fired, compulsory flash mode, return light detected", 16: "Flash did not fire, compulsory flash mode", 24: "Flash did not fire, auto mode", 25: "Flash fired, auto mode", 29: "Flash fired, auto mode, return light not detected", 31: "Flash fired, auto mode, return light detected", 32: "No flash function", 65: "Flash fired, red-eye reduction mode", 69: "Flash fired, red-eye reduction mode, return light not detected", 71: "Flash fired, red-eye reduction mode, return light detected", 73: "Flash fired, compulsory flash mode, red-eye reduction mode", 77: "Flash fired, compulsory flash mode, red-eye reduction mode, return light not detected", 79: "Flash fired, compulsory flash mode, red-eye reduction mode, return light detected", 89: "Flash fired, auto mode, red-eye reduction mode", 93: "Flash fired, auto mode, return light not detected, red-eye reduction mode", 95: "Flash fired, auto mode, return light detected, red-eye reduction mode" }], [41495, { 1: "Not defined", 2: "One-chip color area sensor", 3: "Two-chip color area sensor", 4: "Three-chip color area sensor", 5: "Color sequential area sensor", 7: "Trilinear sensor", 8: "Color sequential linear sensor" }], [41728, { 1: "Film Scanner", 2: "Reflection Print Scanner", 3: "Digital Camera" }], [41729, { 1: "Directly photographed" }], [41985, { 0: "Normal", 1: "Custom", 2: "HDR (no original saved)", 3: "HDR (original saved)", 4: "Original (for HDR)", 6: "Panorama", 7: "Portrait HDR", 8: "Portrait" }], [41986, { 0: "Auto", 1: "Manual", 2: "Auto bracket" }], [41987, { 0: "Auto", 1: "Manual" }], [41990, { 0: "Standard", 1: "Landscape", 2: "Portrait", 3: "Night", 4: "Other" }], [41991, { 0: "None", 1: "Low gain up", 2: "High gain up", 3: "Low gain down", 4: "High gain down" }], [41996, { 0: "Unknown", 1: "Macro", 2: "Close", 3: "Distant" }], [42080, { 0: "Unknown", 1: "Not a Composite Image", 2: "General Composite Image", 3: "Composite Image Captured While Shooting" }]]);
- var Be = { 1: "No absolute unit of measurement", 2: "Inch", 3: "Centimeter" };
- Ee.set(37392, Be), Ee.set(41488, Be);
- var Ne = { 0: "Normal", 1: "Low", 2: "High" };
- function Ge(e3) {
- return "object" == typeof e3 && void 0 !== e3.length ? e3[0] : e3;
+ 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;
}
- function Ve(e3) {
- let t2 = Array.from(e3).slice(1);
- return t2[1] > 15 && (t2 = t2.map((e4) => String.fromCharCode(e4))), "0" !== t2[2] && 0 !== t2[2] || t2.pop(), t2.join(".");
+
+ // node_modules/exifr/dist/full.esm.mjs
+ var e = "undefined" != typeof self ? self : global;
+ var t = "undefined" != typeof navigator;
+ var i2 = t && "undefined" == typeof HTMLImageElement;
+ var n2 = !("undefined" == typeof global || "undefined" == typeof process || !process.versions || !process.versions.node);
+ var s = e.Buffer;
+ var r = e.BigInt;
+ var a = !!s;
+ var o = (e3) => e3;
+ function l(e3, t2 = o) {
+ if (n2)
+ try {
+ return "function" == typeof __require ? Promise.resolve(t2(__require(e3))) : import(
+ /* webpackIgnore: true */
+ e3
+ ).then(t2);
+ } catch (t3) {
+ console.warn("Couldn't load ".concat(e3));
+ }
}
- function ze(e3) {
- if ("string" == typeof e3) {
- var [t2, i3, n3, s2, r2, a2] = e3.trim().split(/[-: ]/g).map(Number), o2 = new Date(t2, i3 - 1, n3);
- return Number.isNaN(s2) || Number.isNaN(r2) || Number.isNaN(a2) || (o2.setHours(s2), o2.setMinutes(r2), o2.setSeconds(a2)), Number.isNaN(+o2) ? e3 : o2;
- }
+ var h = e.fetch;
+ var u = (e3) => h = e3;
+ if (!e.fetch) {
+ const e3 = l("http", (e4) => e4), t2 = l("https", (e4) => e4), i3 = (n3, { headers: s2 } = {}) => new Promise(async (r2, a2) => {
+ let { port: o2, hostname: l2, pathname: h2, protocol: u2, search: c2 } = new URL(n3);
+ const f2 = { method: "GET", hostname: l2, path: encodeURI(h2) + c2, headers: s2 };
+ "" !== o2 && (f2.port = Number(o2));
+ const d2 = ("https:" === u2 ? await t2 : await e3).request(f2, (e4) => {
+ if (301 === e4.statusCode || 302 === e4.statusCode) {
+ let t3 = new URL(e4.headers.location, n3).toString();
+ return i3(t3, { headers: s2 }).then(r2).catch(a2);
+ }
+ r2({ status: e4.statusCode, arrayBuffer: () => new Promise((t3) => {
+ let i4 = [];
+ e4.on("data", (e6) => i4.push(e6)), e4.on("end", () => t3(Buffer.concat(i4)));
+ }) });
+ });
+ d2.on("error", a2), d2.end();
+ });
+ u(i3);
}
- function He(e3) {
- if ("string" == typeof e3)
- return e3;
- let t2 = [];
- if (0 === e3[1] && 0 === e3[e3.length - 1])
- for (let i3 = 0; i3 < e3.length; i3 += 2)
- t2.push(je(e3[i3 + 1], e3[i3]));
- else
- for (let i3 = 0; i3 < e3.length; i3 += 2)
- t2.push(je(e3[i3], e3[i3 + 1]));
- return m(String.fromCodePoint(...t2));
+ function c(e3, t2, i3) {
+ return t2 in e3 ? Object.defineProperty(e3, t2, { value: i3, enumerable: true, configurable: true, writable: true }) : e3[t2] = i3, e3;
}
- function je(e3, t2) {
- return e3 << 8 | t2;
+ var f = (e3) => p(e3) ? void 0 : e3;
+ var d = (e3) => void 0 !== e3;
+ function p(e3) {
+ return void 0 === e3 || (e3 instanceof Map ? 0 === e3.size : 0 === Object.values(e3).filter(d).length);
}
- Ee.set(41992, Ne), Ee.set(41993, Ne), Ee.set(41994, Ne), U(N, ["ifd0", "ifd1"], [[50827, function(e3) {
- return "string" != typeof e3 ? b(e3) : e3;
- }], [306, ze], [40091, He], [40092, He], [40093, He], [40094, He], [40095, He]]), U(N, "exif", [[40960, Ve], [36864, Ve], [36867, ze], [36868, ze], [40962, Ge], [40963, Ge]]), U(N, "gps", [[0, (e3) => Array.from(e3).join(".")], [7, (e3) => Array.from(e3).join(":")]]);
- var We = class extends re2 {
- static canHandle(e3, t2) {
- return 225 === e3.getUint8(t2 + 1) && 1752462448 === e3.getUint32(t2 + 4) && "http://ns.adobe.com/" === e3.getString(t2 + 4, "http://ns.adobe.com/".length);
- }
- static headerLength(e3, t2) {
- return "http://ns.adobe.com/xmp/extension/" === e3.getString(t2 + 4, "http://ns.adobe.com/xmp/extension/".length) ? 79 : 4 + "http://ns.adobe.com/xap/1.0/".length + 1;
- }
- static findPosition(e3, t2) {
- let i3 = super.findPosition(e3, t2);
- return i3.multiSegment = i3.extended = 79 === i3.headerLength, i3.multiSegment ? (i3.chunkCount = e3.getUint8(t2 + 72), i3.chunkNumber = e3.getUint8(t2 + 76), 0 !== e3.getUint8(t2 + 77) && i3.chunkNumber++) : (i3.chunkCount = 1 / 0, i3.chunkNumber = -1), i3;
- }
- static handleMultiSegments(e3) {
- return e3.map((e4) => e4.chunk.getString()).join("");
- }
- normalizeInput(e3) {
- return "string" == typeof e3 ? e3 : I.from(e3).getString();
- }
- parse(e3 = this.chunk) {
- if (!this.localOptions.parse)
- return e3;
- e3 = function(e4) {
- let t3 = {}, i4 = {};
- for (let e6 of Ze)
- t3[e6] = [], i4[e6] = 0;
- return e4.replace(et, (e6, n4, s2) => {
- if ("<" === n4) {
- let n5 = ++i4[s2];
- return t3[s2].push(n5), `${e6}#${n5}`;
- }
- return `${e6}#${t3[s2].pop()}`;
- });
- }(e3);
- let t2 = Xe.findAll(e3, "rdf", "Description");
- 0 === t2.length && t2.push(new Xe("rdf", "Description", void 0, e3));
- let i3, n3 = {};
- for (let e4 of t2)
- for (let t3 of e4.properties)
- i3 = Je(t3.ns, n3), _e(t3, i3);
- return function(e4) {
- let t3;
- for (let i4 in e4)
- t3 = e4[i4] = f2(e4[i4]), void 0 === t3 && delete e4[i4];
- return f2(e4);
- }(n3);
- }
- assignToOutput(e3, t2) {
- if (this.localOptions.parse)
- for (let [i3, n3] of Object.entries(t2))
- switch (i3) {
- case "tiff":
- this.assignObjectToOutput(e3, "ifd0", n3);
- break;
- case "exif":
- this.assignObjectToOutput(e3, "exif", n3);
- break;
- case "xmlns":
- break;
- default:
- this.assignObjectToOutput(e3, i3, n3);
- }
- else
- e3.xmp = t2;
- }
- };
- c(We, "type", "xmp"), c(We, "multiSegment", true), T.set("xmp", We);
- var Ke = class _Ke {
- static findAll(e3) {
- return qe(e3, /([a-zA-Z0-9-]+):([a-zA-Z0-9-]+)=("[^"]*"|'[^']*')/gm).map(_Ke.unpackMatch);
- }
- static unpackMatch(e3) {
- let t2 = e3[1], i3 = e3[2], n3 = e3[3].slice(1, -1);
- return n3 = Qe(n3), new _Ke(t2, i3, n3);
- }
- constructor(e3, t2, i3) {
- this.ns = e3, this.name = t2, this.value = i3;
- }
- serialize() {
- return this.value;
+ function g2(e3) {
+ let t2 = new Error(e3);
+ throw delete t2.stack, t2;
+ }
+ function m(e3) {
+ return "" === (e3 = function(e4) {
+ for (; e4.endsWith("\0"); )
+ e4 = e4.slice(0, -1);
+ return e4;
+ }(e3).trim()) ? void 0 : e3;
+ }
+ function S(e3) {
+ let t2 = function(e4) {
+ let t3 = 0;
+ return e4.ifd0.enabled && (t3 += 1024), e4.exif.enabled && (t3 += 2048), e4.makerNote && (t3 += 2048), e4.userComment && (t3 += 1024), e4.gps.enabled && (t3 += 512), e4.interop.enabled && (t3 += 100), e4.ifd1.enabled && (t3 += 1024), t3 + 2048;
+ }(e3);
+ return e3.jfif.enabled && (t2 += 50), e3.xmp.enabled && (t2 += 2e4), e3.iptc.enabled && (t2 += 14e3), e3.icc.enabled && (t2 += 6e3), t2;
+ }
+ var C = (e3) => String.fromCharCode.apply(null, e3);
+ var y = "undefined" != typeof TextDecoder ? new TextDecoder("utf-8") : void 0;
+ function b(e3) {
+ return y ? y.decode(e3) : a ? Buffer.from(e3).toString("utf8") : decodeURIComponent(escape(C(e3)));
+ }
+ var I = class _I {
+ static from(e3, t2) {
+ return e3 instanceof this && e3.le === t2 ? e3 : new _I(e3, void 0, void 0, t2);
}
- };
- var Xe = class _Xe {
- static findAll(e3, t2, i3) {
- if (void 0 !== t2 || void 0 !== i3) {
- t2 = t2 || "[\\w\\d-]+", i3 = i3 || "[\\w\\d-]+";
- var n3 = new RegExp(`<(${t2}):(${i3})(#\\d+)?((\\s+?[\\w\\d-:]+=("[^"]*"|'[^']*'))*\\s*)(\\/>|>([\\s\\S]*?)<\\/\\1:\\2\\3>)`, "gm");
+ constructor(e3, t2 = 0, i3, n3) {
+ if ("boolean" == typeof n3 && (this.le = n3), Array.isArray(e3) && (e3 = new Uint8Array(e3)), 0 === e3)
+ this.byteOffset = 0, this.byteLength = 0;
+ else if (e3 instanceof ArrayBuffer) {
+ void 0 === i3 && (i3 = e3.byteLength - t2);
+ let n4 = new DataView(e3, t2, i3);
+ this._swapDataView(n4);
+ } else if (e3 instanceof Uint8Array || e3 instanceof DataView || e3 instanceof _I) {
+ void 0 === i3 && (i3 = e3.byteLength - t2), (t2 += e3.byteOffset) + i3 > e3.byteOffset + e3.byteLength && g2("Creating view outside of available memory in ArrayBuffer");
+ let n4 = new DataView(e3.buffer, t2, i3);
+ this._swapDataView(n4);
+ } else if ("number" == typeof e3) {
+ let t3 = new DataView(new ArrayBuffer(e3));
+ this._swapDataView(t3);
} else
- n3 = /<([\w\d-]+):([\w\d-]+)(#\d+)?((\s+?[\w\d-:]+=("[^"]*"|'[^']*'))*\s*)(\/>|>([\s\S]*?)<\/\1:\2\3>)/gm;
- return qe(e3, n3).map(_Xe.unpackMatch);
- }
- static unpackMatch(e3) {
- let t2 = e3[1], i3 = e3[2], n3 = e3[4], s2 = e3[8];
- return new _Xe(t2, i3, n3, s2);
- }
- constructor(e3, t2, i3, n3) {
- this.ns = e3, this.name = t2, this.attrString = i3, this.innerXml = n3, this.attrs = Ke.findAll(i3), this.children = _Xe.findAll(n3), this.value = 0 === this.children.length ? Qe(n3) : void 0, this.properties = [...this.attrs, ...this.children];
- }
- get isPrimitive() {
- return void 0 !== this.value && 0 === this.attrs.length && 0 === this.children.length;
- }
- get isListContainer() {
- return 1 === this.children.length && this.children[0].isList;
- }
- get isList() {
- let { ns: e3, name: t2 } = this;
- return "rdf" === e3 && ("Seq" === t2 || "Bag" === t2 || "Alt" === t2);
- }
- get isListItem() {
- return "rdf" === this.ns && "li" === this.name;
+ g2("Invalid input argument for BufferView: " + e3);
}
- serialize() {
- if (0 === this.properties.length && void 0 === this.value)
- return;
- if (this.isPrimitive)
- return this.value;
- if (this.isListContainer)
- return this.children[0].serialize();
- if (this.isList)
- return $e(this.children.map(Ye));
- if (this.isListItem && 1 === this.children.length && 0 === this.attrs.length)
- return this.children[0].serialize();
- let e3 = {};
- for (let t2 of this.properties)
- _e(t2, e3);
- return void 0 !== this.value && (e3.value = this.value), f2(e3);
+ _swapArrayBuffer(e3) {
+ this._swapDataView(new DataView(e3));
}
- };
- function _e(e3, t2) {
- let i3 = e3.serialize();
- void 0 !== i3 && (t2[e3.name] = i3);
- }
- var Ye = (e3) => e3.serialize();
- var $e = (e3) => 1 === e3.length ? e3[0] : e3;
- var Je = (e3, t2) => t2[e3] ? t2[e3] : t2[e3] = {};
- function qe(e3, t2) {
- let i3, n3 = [];
- if (!e3)
- return n3;
- for (; null !== (i3 = t2.exec(e3)); )
- n3.push(i3);
- return n3;
- }
- function Qe(e3) {
- if (function(e4) {
- return null == e4 || "null" === e4 || "undefined" === e4 || "" === e4 || "" === e4.trim();
- }(e3))
- return;
- let t2 = Number(e3);
- if (!Number.isNaN(t2))
- return t2;
- let i3 = e3.toLowerCase();
- return "true" === i3 || "false" !== i3 && e3.trim();
- }
- var Ze = ["rdf:li", "rdf:Seq", "rdf:Bag", "rdf:Alt", "rdf:Description"];
- var et = new RegExp(`(<|\\/)(${Ze.join("|")})`, "g");
- var tt = Object.freeze({ __proto__: null, default: Me, Exifr: te, fileParsers: w, segmentParsers: T, fileReaders: A, tagKeys: E, tagValues: B, tagRevivers: N, createDictionary: U, extendDictionary: F, fetchUrlAsArrayBuffer: M, readBlobAsArrayBuffer: R, chunkedProps: G, otherSegments: V, segments: z, tiffBlocks: H, segmentsAndBlocks: j2, tiffExtractables: W, inheritables: K, allFormatters: X, Options: q, parse: ie, gpsOnlyOptions: me, gps: Se, thumbnailOnlyOptions: Ce, thumbnail: ye, thumbnailUrl: be, orientationOnlyOptions: Ie, orientation: Pe, rotations: ke, get rotateCanvas() {
- return we;
- }, get rotateCss() {
- return Te;
- }, rotation: Ae });
- var at = l("fs", (e3) => e3.promises);
- A.set("fs", class extends ve {
- async readWhole() {
- this.chunked = false, this.fs = await at;
- let e3 = await this.fs.readFile(this.input);
- this._swapBuffer(e3);
+ _swapBuffer(e3) {
+ this._swapDataView(new DataView(e3.buffer, e3.byteOffset, e3.byteLength));
}
- async readChunked() {
- this.chunked = true, this.fs = await at, await this.open(), await this.readChunk(0, this.options.firstChunkSize);
+ _swapDataView(e3) {
+ this.dataView = e3, this.buffer = e3.buffer, this.byteOffset = e3.byteOffset, this.byteLength = e3.byteLength;
}
- async open() {
- void 0 === this.fh && (this.fh = await this.fs.open(this.input, "r"), this.size = (await this.fh.stat(this.input)).size);
+ _lengthToEnd(e3) {
+ return this.byteLength - e3;
}
- async _readChunk(e3, t2) {
- void 0 === this.fh && await this.open(), e3 + t2 > this.size && (t2 = this.size - e3);
- var i3 = this.subarray(e3, t2, true);
- return await this.fh.read(i3.dataView, 0, t2, e3), i3;
+ set(e3, t2, i3 = _I) {
+ return e3 instanceof DataView || e3 instanceof _I ? e3 = new Uint8Array(e3.buffer, e3.byteOffset, e3.byteLength) : e3 instanceof ArrayBuffer && (e3 = new Uint8Array(e3)), e3 instanceof Uint8Array || g2("BufferView.set(): Invalid data argument."), this.toUint8().set(e3, t2), new i3(this, t2, e3.byteLength);
}
- async close() {
- if (this.fh) {
- let e3 = this.fh;
- this.fh = void 0, await e3.close();
- }
+ subarray(e3, t2) {
+ return t2 = t2 || this._lengthToEnd(e3), new _I(this, e3, t2);
}
- });
- A.set("base64", class extends ve {
- constructor(...e3) {
- super(...e3), this.input = this.input.replace(/^data:([^;]+);base64,/gim, ""), this.size = this.input.length / 4 * 3, this.input.endsWith("==") ? this.size -= 2 : this.input.endsWith("=") && (this.size -= 1);
+ toUint8() {
+ return new Uint8Array(this.buffer, this.byteOffset, this.byteLength);
}
- async _readChunk(e3, t2) {
- let i3, n3, r2 = this.input;
- void 0 === e3 ? (e3 = 0, i3 = 0, n3 = 0) : (i3 = 4 * Math.floor(e3 / 3), n3 = e3 - i3 / 4 * 3), void 0 === t2 && (t2 = this.size);
- let o2 = e3 + t2, l2 = i3 + 4 * Math.ceil(o2 / 3);
- r2 = r2.slice(i3, l2);
- let h2 = Math.min(t2, this.size - e3);
- if (a) {
- let t3 = s.from(r2, "base64").slice(n3, n3 + h2);
- return this.set(t3, e3, true);
- }
- {
- let t3 = this.subarray(e3, h2, true), i4 = atob(r2), s2 = t3.toUint8();
- for (let e4 = 0; e4 < h2; e4++)
- s2[e4] = i4.charCodeAt(n3 + e4);
- return t3;
- }
+ getUint8Array(e3, t2) {
+ return new Uint8Array(this.buffer, this.byteOffset + e3, t2);
}
- });
- var ot = class extends se {
- static canHandle(e3, t2) {
- return 18761 === t2 || 19789 === t2;
+ getString(e3 = 0, t2 = this.byteLength) {
+ return b(this.getUint8Array(e3, t2));
}
- extendOptions(e3) {
- let { ifd0: t2, xmp: i3, iptc: n3, icc: s2 } = e3;
- i3.enabled && t2.deps.add(700), n3.enabled && t2.deps.add(33723), s2.enabled && t2.deps.add(34675), t2.finalizeFilters();
+ getLatin1String(e3 = 0, t2 = this.byteLength) {
+ let i3 = this.getUint8Array(e3, t2);
+ return C(i3);
}
- async parse() {
- let { tiff: e3, xmp: t2, iptc: i3, icc: n3 } = this.options;
- if (e3.enabled || t2.enabled || i3.enabled || n3.enabled) {
- let e4 = Math.max(S(this.options), this.options.chunkSize);
- await this.file.ensureChunk(0, e4), this.createParser("tiff", this.file), this.parsers.tiff.parseHeader(), await this.parsers.tiff.parseIfd0Block(), this.adaptTiffPropAsSegment("xmp"), this.adaptTiffPropAsSegment("iptc"), this.adaptTiffPropAsSegment("icc");
- }
+ getUnicodeString(e3 = 0, t2 = this.byteLength) {
+ const i3 = [];
+ for (let n3 = 0; n3 < t2 && e3 + n3 < this.byteLength; n3 += 2)
+ i3.push(this.getUint16(e3 + n3));
+ return C(i3);
}
- adaptTiffPropAsSegment(e3) {
- if (this.parsers.tiff[e3]) {
- let t2 = this.parsers.tiff[e3];
- this.injectSegment(e3, t2);
- }
+ getInt8(e3) {
+ return this.dataView.getInt8(e3);
}
- };
- c(ot, "type", "tiff"), w.set("tiff", ot);
- var lt = l("zlib");
- var ht = ["ihdr", "iccp", "text", "itxt", "exif"];
- var ut = class extends se {
- constructor(...e3) {
- super(...e3), c(this, "catchError", (e4) => this.errors.push(e4)), c(this, "metaChunks", []), c(this, "unknownChunks", []);
+ getUint8(e3) {
+ return this.dataView.getUint8(e3);
}
- static canHandle(e3, t2) {
- return 35152 === t2 && 2303741511 === e3.getUint32(0) && 218765834 === e3.getUint32(4);
+ getInt16(e3, t2 = this.le) {
+ return this.dataView.getInt16(e3, t2);
}
- async parse() {
- let { file: e3 } = this;
- await this.findPngChunksInRange("\x89PNG\r\n\n".length, e3.byteLength), await this.readSegments(this.metaChunks), this.findIhdr(), this.parseTextChunks(), await this.findExif().catch(this.catchError), await this.findXmp().catch(this.catchError), await this.findIcc().catch(this.catchError);
+ getInt32(e3, t2 = this.le) {
+ return this.dataView.getInt32(e3, t2);
}
- async findPngChunksInRange(e3, t2) {
- let { file: i3 } = this;
- for (; e3 < t2; ) {
- let t3 = i3.getUint32(e3), n3 = i3.getUint32(e3 + 4), s2 = i3.getString(e3 + 4, 4).toLowerCase(), r2 = t3 + 4 + 4 + 4, a2 = { type: s2, offset: e3, length: r2, start: e3 + 4 + 4, size: t3, marker: n3 };
- ht.includes(s2) ? this.metaChunks.push(a2) : this.unknownChunks.push(a2), e3 += r2;
- }
+ getUint16(e3, t2 = this.le) {
+ return this.dataView.getUint16(e3, t2);
}
- parseTextChunks() {
- let e3 = this.metaChunks.filter((e4) => "text" === e4.type);
- for (let t2 of e3) {
- let [e4, i3] = this.file.getString(t2.start, t2.size).split("\0");
- this.injectKeyValToIhdr(e4, i3);
- }
+ getUint32(e3, t2 = this.le) {
+ return this.dataView.getUint32(e3, t2);
}
- injectKeyValToIhdr(e3, t2) {
- let i3 = this.parsers.ihdr;
- i3 && i3.raw.set(e3, t2);
+ getFloat32(e3, t2 = this.le) {
+ return this.dataView.getFloat32(e3, t2);
}
- findIhdr() {
- let e3 = this.metaChunks.find((e4) => "ihdr" === e4.type);
- e3 && false !== this.options.ihdr.enabled && this.createParser("ihdr", e3.chunk);
+ getFloat64(e3, t2 = this.le) {
+ return this.dataView.getFloat64(e3, t2);
}
- async findExif() {
- let e3 = this.metaChunks.find((e4) => "exif" === e4.type);
- e3 && this.injectSegment("tiff", e3.chunk);
+ getFloat(e3, t2 = this.le) {
+ return this.dataView.getFloat32(e3, t2);
}
- async findXmp() {
- let e3 = this.metaChunks.filter((e4) => "itxt" === e4.type);
- for (let t2 of e3) {
- "XML:com.adobe.xmp" === t2.chunk.getString(0, "XML:com.adobe.xmp".length) && this.injectSegment("xmp", t2.chunk);
+ getDouble(e3, t2 = this.le) {
+ return this.dataView.getFloat64(e3, t2);
+ }
+ getUintBytes(e3, t2, i3) {
+ switch (t2) {
+ case 1:
+ return this.getUint8(e3, i3);
+ case 2:
+ return this.getUint16(e3, i3);
+ case 4:
+ return this.getUint32(e3, i3);
+ case 8:
+ return this.getUint64 && this.getUint64(e3, i3);
}
}
- async findIcc() {
- let e3 = this.metaChunks.find((e4) => "iccp" === e4.type);
- if (!e3)
- return;
- let { chunk: t2 } = e3, i3 = t2.getUint8Array(0, 81), s2 = 0;
- for (; s2 < 80 && 0 !== i3[s2]; )
- s2++;
- let r2 = s2 + 2, a2 = t2.getString(0, s2);
- if (this.injectKeyValToIhdr("ProfileName", a2), n2) {
- let e4 = await lt, i4 = t2.getUint8Array(r2);
- i4 = e4.inflateSync(i4), this.injectSegment("icc", i4);
+ getUint(e3, t2, i3) {
+ switch (t2) {
+ case 8:
+ return this.getUint8(e3, i3);
+ case 16:
+ return this.getUint16(e3, i3);
+ case 32:
+ return this.getUint32(e3, i3);
+ case 64:
+ return this.getUint64 && this.getUint64(e3, i3);
}
}
+ toString(e3) {
+ return this.dataView.toString(e3, this.constructor.name);
+ }
+ ensureChunk() {
+ }
};
- c(ut, "type", "png"), w.set("png", ut), U(E, "interop", [[1, "InteropIndex"], [2, "InteropVersion"], [4096, "RelatedImageFileFormat"], [4097, "RelatedImageWidth"], [4098, "RelatedImageHeight"]]), F(E, "ifd0", [[11, "ProcessingSoftware"], [254, "SubfileType"], [255, "OldSubfileType"], [263, "Thresholding"], [264, "CellWidth"], [265, "CellLength"], [266, "FillOrder"], [269, "DocumentName"], [280, "MinSampleValue"], [281, "MaxSampleValue"], [285, "PageName"], [286, "XPosition"], [287, "YPosition"], [290, "GrayResponseUnit"], [297, "PageNumber"], [321, "HalftoneHints"], [322, "TileWidth"], [323, "TileLength"], [332, "InkSet"], [337, "TargetPrinter"], [18246, "Rating"], [18249, "RatingPercent"], [33550, "PixelScale"], [34264, "ModelTransform"], [34377, "PhotoshopSettings"], [50706, "DNGVersion"], [50707, "DNGBackwardVersion"], [50708, "UniqueCameraModel"], [50709, "LocalizedCameraModel"], [50736, "DNGLensInfo"], [50739, "ShadowScale"], [50740, "DNGPrivateData"], [33920, "IntergraphMatrix"], [33922, "ModelTiePoint"], [34118, "SEMInfo"], [34735, "GeoTiffDirectory"], [34736, "GeoTiffDoubleParams"], [34737, "GeoTiffAsciiParams"], [50341, "PrintIM"], [50721, "ColorMatrix1"], [50722, "ColorMatrix2"], [50723, "CameraCalibration1"], [50724, "CameraCalibration2"], [50725, "ReductionMatrix1"], [50726, "ReductionMatrix2"], [50727, "AnalogBalance"], [50728, "AsShotNeutral"], [50729, "AsShotWhiteXY"], [50730, "BaselineExposure"], [50731, "BaselineNoise"], [50732, "BaselineSharpness"], [50734, "LinearResponseLimit"], [50735, "CameraSerialNumber"], [50741, "MakerNoteSafety"], [50778, "CalibrationIlluminant1"], [50779, "CalibrationIlluminant2"], [50781, "RawDataUniqueID"], [50827, "OriginalRawFileName"], [50828, "OriginalRawFileData"], [50831, "AsShotICCProfile"], [50832, "AsShotPreProfileMatrix"], [50833, "CurrentICCProfile"], [50834, "CurrentPreProfileMatrix"], [50879, "ColorimetricReference"], [50885, "SRawType"], [50898, "PanasonicTitle"], [50899, "PanasonicTitle2"], [50931, "CameraCalibrationSig"], [50932, "ProfileCalibrationSig"], [50933, "ProfileIFD"], [50934, "AsShotProfileName"], [50936, "ProfileName"], [50937, "ProfileHueSatMapDims"], [50938, "ProfileHueSatMapData1"], [50939, "ProfileHueSatMapData2"], [50940, "ProfileToneCurve"], [50941, "ProfileEmbedPolicy"], [50942, "ProfileCopyright"], [50964, "ForwardMatrix1"], [50965, "ForwardMatrix2"], [50966, "PreviewApplicationName"], [50967, "PreviewApplicationVersion"], [50968, "PreviewSettingsName"], [50969, "PreviewSettingsDigest"], [50970, "PreviewColorSpace"], [50971, "PreviewDateTime"], [50972, "RawImageDigest"], [50973, "OriginalRawFileDigest"], [50981, "ProfileLookTableDims"], [50982, "ProfileLookTableData"], [51043, "TimeCodes"], [51044, "FrameRate"], [51058, "TStop"], [51081, "ReelName"], [51089, "OriginalDefaultFinalSize"], [51090, "OriginalBestQualitySize"], [51091, "OriginalDefaultCropSize"], [51105, "CameraLabel"], [51107, "ProfileHueSatMapEncoding"], [51108, "ProfileLookTableEncoding"], [51109, "BaselineExposureOffset"], [51110, "DefaultBlackRender"], [51111, "NewRawImageDigest"], [51112, "RawToPreviewGain"]]);
- var ct = [[273, "StripOffsets"], [279, "StripByteCounts"], [288, "FreeOffsets"], [289, "FreeByteCounts"], [291, "GrayResponseCurve"], [292, "T4Options"], [293, "T6Options"], [300, "ColorResponseUnit"], [320, "ColorMap"], [324, "TileOffsets"], [325, "TileByteCounts"], [326, "BadFaxLines"], [327, "CleanFaxData"], [328, "ConsecutiveBadFaxLines"], [330, "SubIFD"], [333, "InkNames"], [334, "NumberofInks"], [336, "DotRange"], [338, "ExtraSamples"], [339, "SampleFormat"], [340, "SMinSampleValue"], [341, "SMaxSampleValue"], [342, "TransferRange"], [343, "ClipPath"], [344, "XClipPathUnits"], [345, "YClipPathUnits"], [346, "Indexed"], [347, "JPEGTables"], [351, "OPIProxy"], [400, "GlobalParametersIFD"], [401, "ProfileType"], [402, "FaxProfile"], [403, "CodingMethods"], [404, "VersionYear"], [405, "ModeNumber"], [433, "Decode"], [434, "DefaultImageColor"], [435, "T82Options"], [437, "JPEGTables"], [512, "JPEGProc"], [515, "JPEGRestartInterval"], [517, "JPEGLosslessPredictors"], [518, "JPEGPointTransforms"], [519, "JPEGQTables"], [520, "JPEGDCTables"], [521, "JPEGACTables"], [559, "StripRowCounts"], [999, "USPTOMiscellaneous"], [18247, "XP_DIP_XML"], [18248, "StitchInfo"], [28672, "SonyRawFileType"], [28688, "SonyToneCurve"], [28721, "VignettingCorrection"], [28722, "VignettingCorrParams"], [28724, "ChromaticAberrationCorrection"], [28725, "ChromaticAberrationCorrParams"], [28726, "DistortionCorrection"], [28727, "DistortionCorrParams"], [29895, "SonyCropTopLeft"], [29896, "SonyCropSize"], [32781, "ImageID"], [32931, "WangTag1"], [32932, "WangAnnotation"], [32933, "WangTag3"], [32934, "WangTag4"], [32953, "ImageReferencePoints"], [32954, "RegionXformTackPoint"], [32955, "WarpQuadrilateral"], [32956, "AffineTransformMat"], [32995, "Matteing"], [32996, "DataType"], [32997, "ImageDepth"], [32998, "TileDepth"], [33300, "ImageFullWidth"], [33301, "ImageFullHeight"], [33302, "TextureFormat"], [33303, "WrapModes"], [33304, "FovCot"], [33305, "MatrixWorldToScreen"], [33306, "MatrixWorldToCamera"], [33405, "Model2"], [33421, "CFARepeatPatternDim"], [33422, "CFAPattern2"], [33423, "BatteryLevel"], [33424, "KodakIFD"], [33445, "MDFileTag"], [33446, "MDScalePixel"], [33447, "MDColorTable"], [33448, "MDLabName"], [33449, "MDSampleInfo"], [33450, "MDPrepDate"], [33451, "MDPrepTime"], [33452, "MDFileUnits"], [33589, "AdventScale"], [33590, "AdventRevision"], [33628, "UIC1Tag"], [33629, "UIC2Tag"], [33630, "UIC3Tag"], [33631, "UIC4Tag"], [33918, "IntergraphPacketData"], [33919, "IntergraphFlagRegisters"], [33921, "INGRReserved"], [34016, "Site"], [34017, "ColorSequence"], [34018, "IT8Header"], [34019, "RasterPadding"], [34020, "BitsPerRunLength"], [34021, "BitsPerExtendedRunLength"], [34022, "ColorTable"], [34023, "ImageColorIndicator"], [34024, "BackgroundColorIndicator"], [34025, "ImageColorValue"], [34026, "BackgroundColorValue"], [34027, "PixelIntensityRange"], [34028, "TransparencyIndicator"], [34029, "ColorCharacterization"], [34030, "HCUsage"], [34031, "TrapIndicator"], [34032, "CMYKEquivalent"], [34152, "AFCP_IPTC"], [34232, "PixelMagicJBIGOptions"], [34263, "JPLCartoIFD"], [34306, "WB_GRGBLevels"], [34310, "LeafData"], [34687, "TIFF_FXExtensions"], [34688, "MultiProfiles"], [34689, "SharedData"], [34690, "T88Options"], [34732, "ImageLayer"], [34750, "JBIGOptions"], [34856, "Opto-ElectricConvFactor"], [34857, "Interlace"], [34908, "FaxRecvParams"], [34909, "FaxSubAddress"], [34910, "FaxRecvTime"], [34929, "FedexEDR"], [34954, "LeafSubIFD"], [37387, "FlashEnergy"], [37388, "SpatialFrequencyResponse"], [37389, "Noise"], [37390, "FocalPlaneXResolution"], [37391, "FocalPlaneYResolution"], [37392, "FocalPlaneResolutionUnit"], [37397, "ExposureIndex"], [37398, "TIFF-EPStandardID"], [37399, "SensingMethod"], [37434, "CIP3DataFile"], [37435, "CIP3Sheet"], [37436, "CIP3Side"], [37439, "StoNits"], [37679, "MSDocumentText"], [37680, "MSPropertySetStorage"], [37681, "MSDocumentTextPosition"], [37724, "ImageSourceData"], [40965, "InteropIFD"], [40976, "SamsungRawPointersOffset"], [40977, "SamsungRawPointersLength"], [41217, "SamsungRawByteOrder"], [41218, "SamsungRawUnknown"], [41484, "SpatialFrequencyResponse"], [41485, "Noise"], [41489, "ImageNumber"], [41490, "SecurityClassification"], [41491, "ImageHistory"], [41494, "TIFF-EPStandardID"], [41995, "DeviceSettingDescription"], [42112, "GDALMetadata"], [42113, "GDALNoData"], [44992, "ExpandSoftware"], [44993, "ExpandLens"], [44994, "ExpandFilm"], [44995, "ExpandFilterLens"], [44996, "ExpandScanner"], [44997, "ExpandFlashLamp"], [46275, "HasselbladRawImage"], [48129, "PixelFormat"], [48130, "Transformation"], [48131, "Uncompressed"], [48132, "ImageType"], [48256, "ImageWidth"], [48257, "ImageHeight"], [48258, "WidthResolution"], [48259, "HeightResolution"], [48320, "ImageOffset"], [48321, "ImageByteCount"], [48322, "AlphaOffset"], [48323, "AlphaByteCount"], [48324, "ImageDataDiscard"], [48325, "AlphaDataDiscard"], [50215, "OceScanjobDesc"], [50216, "OceApplicationSelector"], [50217, "OceIDNumber"], [50218, "OceImageLogic"], [50255, "Annotations"], [50459, "HasselbladExif"], [50547, "OriginalFileName"], [50560, "USPTOOriginalContentType"], [50656, "CR2CFAPattern"], [50710, "CFAPlaneColor"], [50711, "CFALayout"], [50712, "LinearizationTable"], [50713, "BlackLevelRepeatDim"], [50714, "BlackLevel"], [50715, "BlackLevelDeltaH"], [50716, "BlackLevelDeltaV"], [50717, "WhiteLevel"], [50718, "DefaultScale"], [50719, "DefaultCropOrigin"], [50720, "DefaultCropSize"], [50733, "BayerGreenSplit"], [50737, "ChromaBlurRadius"], [50738, "AntiAliasStrength"], [50752, "RawImageSegmentation"], [50780, "BestQualityScale"], [50784, "AliasLayerMetadata"], [50829, "ActiveArea"], [50830, "MaskedAreas"], [50935, "NoiseReductionApplied"], [50974, "SubTileBlockSize"], [50975, "RowInterleaveFactor"], [51008, "OpcodeList1"], [51009, "OpcodeList2"], [51022, "OpcodeList3"], [51041, "NoiseProfile"], [51114, "CacheVersion"], [51125, "DefaultUserCrop"], [51157, "NikonNEFInfo"], [65024, "KdcIFD"]];
- F(E, "ifd0", ct), F(E, "exif", ct), U(B, "gps", [[23, { M: "Magnetic North", T: "True North" }], [25, { K: "Kilometers", M: "Miles", N: "Nautical Miles" }]]);
- var ft = class extends re2 {
- static canHandle(e3, t2) {
- return 224 === e3.getUint8(t2 + 1) && 1246120262 === e3.getUint32(t2 + 4) && 0 === e3.getUint8(t2 + 8);
+ function P(e3, t2) {
+ g2("".concat(e3, " '").concat(t2, "' was not loaded, try using full build of exifr."));
+ }
+ var k = class extends Map {
+ constructor(e3) {
+ super(), this.kind = e3;
}
- parse() {
- return this.parseTags(), this.translate(), this.output;
+ get(e3, t2) {
+ return this.has(e3) || P(this.kind, e3), t2 && (e3 in t2 || function(e4, t3) {
+ g2("Unknown ".concat(e4, " '").concat(t3, "'."));
+ }(this.kind, e3), t2[e3].enabled || P(this.kind, e3)), super.get(e3);
}
- parseTags() {
- this.raw = /* @__PURE__ */ new Map([[0, this.chunk.getUint16(0)], [2, this.chunk.getUint8(2)], [3, this.chunk.getUint16(3)], [5, this.chunk.getUint16(5)], [7, this.chunk.getUint8(7)], [8, this.chunk.getUint8(8)]]);
+ keyList() {
+ return Array.from(this.keys());
}
};
- c(ft, "type", "jfif"), c(ft, "headerLength", 9), T.set("jfif", ft), U(E, "jfif", [[0, "JFIFVersion"], [2, "ResolutionUnit"], [3, "XResolution"], [5, "YResolution"], [7, "ThumbnailWidth"], [8, "ThumbnailHeight"]]);
- var dt = class extends re2 {
- parse() {
- return this.parseTags(), this.translate(), this.output;
+ var w = new k("file parser");
+ var T = new k("segment parser");
+ var A = new k("file reader");
+ function D(e3, n3) {
+ return "string" == typeof e3 ? O(e3, n3) : t && !i2 && e3 instanceof HTMLImageElement ? O(e3.src, n3) : e3 instanceof Uint8Array || e3 instanceof ArrayBuffer || e3 instanceof DataView ? new I(e3) : t && e3 instanceof Blob ? x(e3, n3, "blob", R) : void g2("Invalid input argument");
+ }
+ function O(e3, i3) {
+ return (s2 = e3).startsWith("data:") || s2.length > 1e4 ? v(e3, i3, "base64") : n2 && e3.includes("://") ? x(e3, i3, "url", M) : n2 ? v(e3, i3, "fs") : t ? x(e3, i3, "url", M) : void g2("Invalid input argument");
+ var s2;
+ }
+ async function x(e3, t2, i3, n3) {
+ return A.has(i3) ? v(e3, t2, i3) : n3 ? async function(e4, t3) {
+ let i4 = await t3(e4);
+ return new I(i4);
+ }(e3, n3) : void g2("Parser ".concat(i3, " is not loaded"));
+ }
+ async function v(e3, t2, i3) {
+ let n3 = new (A.get(i3))(e3, t2);
+ return await n3.read(), n3;
+ }
+ var M = (e3) => h(e3).then((e4) => e4.arrayBuffer());
+ var R = (e3) => new Promise((t2, i3) => {
+ let n3 = new FileReader();
+ n3.onloadend = () => t2(n3.result || new ArrayBuffer()), n3.onerror = i3, n3.readAsArrayBuffer(e3);
+ });
+ var L = class extends Map {
+ get tagKeys() {
+ return this.allKeys || (this.allKeys = Array.from(this.keys())), this.allKeys;
}
- parseTags() {
- this.raw = new Map([[0, this.chunk.getUint32(0)], [4, this.chunk.getUint32(4)], [8, this.chunk.getUint8(8)], [9, this.chunk.getUint8(9)], [10, this.chunk.getUint8(10)], [11, this.chunk.getUint8(11)], [12, this.chunk.getUint8(12)], ...Array.from(this.raw)]);
+ get tagValues() {
+ return this.allValues || (this.allValues = Array.from(this.values())), this.allValues;
}
};
- c(dt, "type", "ihdr"), T.set("ihdr", dt), U(E, "ihdr", [[0, "ImageWidth"], [4, "ImageHeight"], [8, "BitDepth"], [9, "ColorType"], [10, "Compression"], [11, "Filter"], [12, "Interlace"]]), U(B, "ihdr", [[9, { 0: "Grayscale", 2: "RGB", 3: "Palette", 4: "Grayscale with Alpha", 6: "RGB with Alpha", DEFAULT: "Unknown" }], [10, { 0: "Deflate/Inflate", DEFAULT: "Unknown" }], [11, { 0: "Adaptive", DEFAULT: "Unknown" }], [12, { 0: "Noninterlaced", 1: "Adam7 Interlace", DEFAULT: "Unknown" }]]);
- var pt = class extends re2 {
- static canHandle(e3, t2) {
- return 226 === e3.getUint8(t2 + 1) && 1229144927 === e3.getUint32(t2 + 4);
+ function U(e3, t2, i3) {
+ let n3 = new L();
+ for (let [e4, t3] of i3)
+ n3.set(e4, t3);
+ if (Array.isArray(t2))
+ for (let i4 of t2)
+ e3.set(i4, n3);
+ else
+ e3.set(t2, n3);
+ return n3;
+ }
+ function F(e3, t2, i3) {
+ let n3, s2 = e3.get(t2);
+ for (n3 of i3)
+ s2.set(n3[0], n3[1]);
+ }
+ var E = /* @__PURE__ */ new Map();
+ var B = /* @__PURE__ */ new Map();
+ var N = /* @__PURE__ */ new Map();
+ var G = ["chunked", "firstChunkSize", "firstChunkSizeNode", "firstChunkSizeBrowser", "chunkSize", "chunkLimit"];
+ var V = ["jfif", "xmp", "icc", "iptc", "ihdr"];
+ var z = ["tiff", ...V];
+ var H = ["ifd0", "ifd1", "exif", "gps", "interop"];
+ var j = [...z, ...H];
+ var W = ["makerNote", "userComment"];
+ var K = ["translateKeys", "translateValues", "reviveValues", "multiSegment"];
+ var X = [...K, "sanitize", "mergeOutput", "silentErrors"];
+ var _ = class {
+ get translate() {
+ return this.translateKeys || this.translateValues || this.reviveValues;
}
- static findPosition(e3, t2) {
- let i3 = super.findPosition(e3, t2);
- return i3.chunkNumber = e3.getUint8(t2 + 16), i3.chunkCount = e3.getUint8(t2 + 17), i3.multiSegment = i3.chunkCount > 1, i3;
+ };
+ var Y = class extends _ {
+ get needed() {
+ return this.enabled || this.deps.size > 0;
}
- static handleMultiSegments(e3) {
- return function(e4) {
- let t2 = function(e6) {
- let t3 = e6[0].constructor, i3 = 0;
- for (let t4 of e6)
- i3 += t4.length;
- let n3 = new t3(i3), s2 = 0;
- for (let t4 of e6)
- n3.set(t4, s2), s2 += t4.length;
- return n3;
- }(e4.map((e6) => e6.chunk.toUint8()));
- return new I(t2);
- }(e3);
+ constructor(e3, t2, i3, n3) {
+ if (super(), c(this, "enabled", false), c(this, "skip", /* @__PURE__ */ new Set()), c(this, "pick", /* @__PURE__ */ new Set()), c(this, "deps", /* @__PURE__ */ new Set()), c(this, "translateKeys", false), c(this, "translateValues", false), c(this, "reviveValues", false), this.key = e3, this.enabled = t2, this.parse = this.enabled, this.applyInheritables(n3), this.canBeFiltered = H.includes(e3), this.canBeFiltered && (this.dict = E.get(e3)), void 0 !== i3)
+ if (Array.isArray(i3))
+ this.parse = this.enabled = true, this.canBeFiltered && i3.length > 0 && this.translateTagSet(i3, this.pick);
+ else if ("object" == typeof i3) {
+ if (this.enabled = true, this.parse = false !== i3.parse, this.canBeFiltered) {
+ let { pick: e4, skip: t3 } = i3;
+ e4 && e4.length > 0 && this.translateTagSet(e4, this.pick), t3 && t3.length > 0 && this.translateTagSet(t3, this.skip);
+ }
+ this.applyInheritables(i3);
+ } else
+ true === i3 || false === i3 ? this.parse = this.enabled = i3 : g2("Invalid options argument: ".concat(i3));
}
- parse() {
- return this.raw = /* @__PURE__ */ new Map(), this.parseHeader(), this.parseTags(), this.translate(), this.output;
+ applyInheritables(e3) {
+ let t2, i3;
+ for (t2 of K)
+ i3 = e3[t2], void 0 !== i3 && (this[t2] = i3);
}
- parseHeader() {
- let { raw: e3 } = this;
- this.chunk.byteLength < 84 && g2("ICC header is too short");
- for (let [t2, i3] of Object.entries(gt)) {
- t2 = parseInt(t2, 10);
- let n3 = i3(this.chunk, t2);
- "\0\0\0\0" !== n3 && e3.set(t2, n3);
- }
+ translateTagSet(e3, t2) {
+ if (this.dict) {
+ let i3, n3, { tagKeys: s2, tagValues: r2 } = this.dict;
+ for (i3 of e3)
+ "string" == typeof i3 ? (n3 = r2.indexOf(i3), -1 === n3 && (n3 = s2.indexOf(Number(i3))), -1 !== n3 && t2.add(Number(s2[n3]))) : t2.add(i3);
+ } else
+ for (let i3 of e3)
+ t2.add(i3);
}
- parseTags() {
- let e3, t2, i3, n3, s2, { raw: r2 } = this, a2 = this.chunk.getUint32(128), o2 = 132, l2 = this.chunk.byteLength;
- for (; a2--; ) {
- if (e3 = this.chunk.getString(o2, 4), t2 = this.chunk.getUint32(o2 + 4), i3 = this.chunk.getUint32(o2 + 8), n3 = this.chunk.getString(t2, 4), t2 + i3 > l2)
- return void console.warn("reached the end of the first ICC chunk. Enable options.tiff.multiSegment to read all ICC segments.");
- s2 = this.parseTag(n3, t2, i3), void 0 !== s2 && "\0\0\0\0" !== s2 && r2.set(e3, s2), o2 += 12;
- }
+ finalizeFilters() {
+ !this.enabled && this.deps.size > 0 ? (this.enabled = true, ee(this.pick, this.deps)) : this.enabled && this.pick.size > 0 && ee(this.pick, this.deps);
}
- parseTag(e3, t2, i3) {
- switch (e3) {
- case "desc":
- return this.parseDesc(t2);
- case "mluc":
- return this.parseMluc(t2);
- case "text":
- return this.parseText(t2, i3);
- case "sig ":
- return this.parseSig(t2);
- }
- if (!(t2 + i3 > this.chunk.byteLength))
- return this.chunk.getUint8Array(t2, i3);
+ };
+ var $2 = { jfif: false, tiff: true, xmp: false, icc: false, iptc: false, ifd0: true, ifd1: false, exif: true, gps: true, interop: false, ihdr: void 0, makerNote: false, userComment: false, multiSegment: false, skip: [], pick: [], translateKeys: true, translateValues: true, reviveValues: true, sanitize: true, mergeOutput: true, silentErrors: true, chunked: true, firstChunkSize: void 0, firstChunkSizeNode: 512, firstChunkSizeBrowser: 65536, chunkSize: 65536, chunkLimit: 5 };
+ var J = /* @__PURE__ */ new Map();
+ var q = class extends _ {
+ static useCached(e3) {
+ let t2 = J.get(e3);
+ return void 0 !== t2 || (t2 = new this(e3), J.set(e3, t2)), t2;
}
- parseDesc(e3) {
- let t2 = this.chunk.getUint32(e3 + 8) - 1;
- return m(this.chunk.getString(e3 + 12, t2));
+ constructor(e3) {
+ super(), true === e3 ? this.setupFromTrue() : void 0 === e3 ? this.setupFromUndefined() : Array.isArray(e3) ? this.setupFromArray(e3) : "object" == typeof e3 ? this.setupFromObject(e3) : g2("Invalid options argument ".concat(e3)), void 0 === this.firstChunkSize && (this.firstChunkSize = t ? this.firstChunkSizeBrowser : this.firstChunkSizeNode), this.mergeOutput && (this.ifd1.enabled = false), this.filterNestedSegmentTags(), this.traverseTiffDependencyTree(), this.checkLoadedPlugins();
}
- parseText(e3, t2) {
- return m(this.chunk.getString(e3 + 8, t2 - 8));
+ setupFromUndefined() {
+ let e3;
+ for (e3 of G)
+ this[e3] = $2[e3];
+ for (e3 of X)
+ this[e3] = $2[e3];
+ for (e3 of W)
+ this[e3] = $2[e3];
+ for (e3 of j)
+ this[e3] = new Y(e3, $2[e3], void 0, this);
}
- parseSig(e3) {
- return m(this.chunk.getString(e3 + 8, 4));
+ setupFromTrue() {
+ let e3;
+ for (e3 of G)
+ this[e3] = $2[e3];
+ for (e3 of X)
+ this[e3] = $2[e3];
+ for (e3 of W)
+ this[e3] = true;
+ for (e3 of j)
+ this[e3] = new Y(e3, true, void 0, this);
}
- parseMluc(e3) {
- let { chunk: t2 } = this, i3 = t2.getUint32(e3 + 8), n3 = t2.getUint32(e3 + 12), s2 = e3 + 16, r2 = [];
- for (let a2 = 0; a2 < i3; a2++) {
- let i4 = t2.getString(s2 + 0, 2), a3 = t2.getString(s2 + 2, 2), o2 = t2.getUint32(s2 + 4), l2 = t2.getUint32(s2 + 8) + e3, h2 = m(t2.getUnicodeString(l2, o2));
- r2.push({ lang: i4, country: a3, text: h2 }), s2 += n3;
- }
- return 1 === i3 ? r2[0].text : r2;
+ setupFromArray(e3) {
+ let t2;
+ for (t2 of G)
+ this[t2] = $2[t2];
+ for (t2 of X)
+ this[t2] = $2[t2];
+ for (t2 of W)
+ this[t2] = $2[t2];
+ for (t2 of j)
+ this[t2] = new Y(t2, false, void 0, this);
+ this.setupGlobalFilters(e3, void 0, H);
}
- translateValue(e3, t2) {
- return "string" == typeof e3 ? t2[e3] || t2[e3.toLowerCase()] || e3 : t2[e3] || e3;
+ setupFromObject(e3) {
+ let t2;
+ for (t2 of (H.ifd0 = H.ifd0 || H.image, H.ifd1 = H.ifd1 || H.thumbnail, Object.assign(this, e3), G))
+ this[t2] = Z(e3[t2], $2[t2]);
+ for (t2 of X)
+ this[t2] = Z(e3[t2], $2[t2]);
+ for (t2 of W)
+ this[t2] = Z(e3[t2], $2[t2]);
+ for (t2 of z)
+ this[t2] = new Y(t2, $2[t2], e3[t2], this);
+ for (t2 of H)
+ this[t2] = new Y(t2, $2[t2], e3[t2], this.tiff);
+ this.setupGlobalFilters(e3.pick, e3.skip, H, j), true === e3.tiff ? this.batchEnableWithBool(H, true) : false === e3.tiff ? this.batchEnableWithUserValue(H, e3) : Array.isArray(e3.tiff) ? this.setupGlobalFilters(e3.tiff, void 0, H) : "object" == typeof e3.tiff && this.setupGlobalFilters(e3.tiff.pick, e3.tiff.skip, H);
}
- };
- c(pt, "type", "icc"), c(pt, "multiSegment", true), c(pt, "headerLength", 18);
- var gt = { 4: mt, 8: function(e3, t2) {
- return [e3.getUint8(t2), e3.getUint8(t2 + 1) >> 4, e3.getUint8(t2 + 1) % 16].map((e4) => e4.toString(10)).join(".");
- }, 12: mt, 16: mt, 20: mt, 24: function(e3, t2) {
- const i3 = e3.getUint16(t2), n3 = e3.getUint16(t2 + 2) - 1, s2 = e3.getUint16(t2 + 4), r2 = e3.getUint16(t2 + 6), a2 = e3.getUint16(t2 + 8), o2 = e3.getUint16(t2 + 10);
- return new Date(Date.UTC(i3, n3, s2, r2, a2, o2));
- }, 36: mt, 40: mt, 48: mt, 52: mt, 64: (e3, t2) => e3.getUint32(t2), 80: mt };
- function mt(e3, t2) {
- return m(e3.getString(t2, 4));
- }
- T.set("icc", pt), U(E, "icc", [[4, "ProfileCMMType"], [8, "ProfileVersion"], [12, "ProfileClass"], [16, "ColorSpaceData"], [20, "ProfileConnectionSpace"], [24, "ProfileDateTime"], [36, "ProfileFileSignature"], [40, "PrimaryPlatform"], [44, "CMMFlags"], [48, "DeviceManufacturer"], [52, "DeviceModel"], [56, "DeviceAttributes"], [64, "RenderingIntent"], [68, "ConnectionSpaceIlluminant"], [80, "ProfileCreator"], [84, "ProfileID"], ["Header", "ProfileHeader"], ["MS00", "WCSProfiles"], ["bTRC", "BlueTRC"], ["bXYZ", "BlueMatrixColumn"], ["bfd", "UCRBG"], ["bkpt", "MediaBlackPoint"], ["calt", "CalibrationDateTime"], ["chad", "ChromaticAdaptation"], ["chrm", "Chromaticity"], ["ciis", "ColorimetricIntentImageState"], ["clot", "ColorantTableOut"], ["clro", "ColorantOrder"], ["clrt", "ColorantTable"], ["cprt", "ProfileCopyright"], ["crdi", "CRDInfo"], ["desc", "ProfileDescription"], ["devs", "DeviceSettings"], ["dmdd", "DeviceModelDesc"], ["dmnd", "DeviceMfgDesc"], ["dscm", "ProfileDescriptionML"], ["fpce", "FocalPlaneColorimetryEstimates"], ["gTRC", "GreenTRC"], ["gXYZ", "GreenMatrixColumn"], ["gamt", "Gamut"], ["kTRC", "GrayTRC"], ["lumi", "Luminance"], ["meas", "Measurement"], ["meta", "Metadata"], ["mmod", "MakeAndModel"], ["ncl2", "NamedColor2"], ["ncol", "NamedColor"], ["ndin", "NativeDisplayInfo"], ["pre0", "Preview0"], ["pre1", "Preview1"], ["pre2", "Preview2"], ["ps2i", "PS2RenderingIntent"], ["ps2s", "PostScript2CSA"], ["psd0", "PostScript2CRD0"], ["psd1", "PostScript2CRD1"], ["psd2", "PostScript2CRD2"], ["psd3", "PostScript2CRD3"], ["pseq", "ProfileSequenceDesc"], ["psid", "ProfileSequenceIdentifier"], ["psvm", "PS2CRDVMSize"], ["rTRC", "RedTRC"], ["rXYZ", "RedMatrixColumn"], ["resp", "OutputResponse"], ["rhoc", "ReflectionHardcopyOrigColorimetry"], ["rig0", "PerceptualRenderingIntentGamut"], ["rig2", "SaturationRenderingIntentGamut"], ["rpoc", "ReflectionPrintOutputColorimetry"], ["sape", "SceneAppearanceEstimates"], ["scoe", "SceneColorimetryEstimates"], ["scrd", "ScreeningDesc"], ["scrn", "Screening"], ["targ", "CharTarget"], ["tech", "Technology"], ["vcgt", "VideoCardGamma"], ["view", "ViewingConditions"], ["vued", "ViewingCondDesc"], ["wtpt", "MediaWhitePoint"]]);
- var St = { "4d2p": "Erdt Systems", AAMA: "Aamazing Technologies", ACER: "Acer", ACLT: "Acolyte Color Research", ACTI: "Actix Sytems", ADAR: "Adara Technology", ADBE: "Adobe", ADI: "ADI Systems", AGFA: "Agfa Graphics", ALMD: "Alps Electric", ALPS: "Alps Electric", ALWN: "Alwan Color Expertise", AMTI: "Amiable Technologies", AOC: "AOC International", APAG: "Apago", APPL: "Apple Computer", AST: "AST", "AT&T": "AT&T", BAEL: "BARBIERI electronic", BRCO: "Barco NV", BRKP: "Breakpoint", BROT: "Brother", BULL: "Bull", BUS: "Bus Computer Systems", "C-IT": "C-Itoh", CAMR: "Intel", CANO: "Canon", CARR: "Carroll Touch", CASI: "Casio", CBUS: "Colorbus PL", CEL: "Crossfield", CELx: "Crossfield", CGS: "CGS Publishing Technologies International", CHM: "Rochester Robotics", CIGL: "Colour Imaging Group, London", CITI: "Citizen", CL00: "Candela", CLIQ: "Color IQ", CMCO: "Chromaco", CMiX: "CHROMiX", COLO: "Colorgraphic Communications", COMP: "Compaq", COMp: "Compeq/Focus Technology", CONR: "Conrac Display Products", CORD: "Cordata Technologies", CPQ: "Compaq", CPRO: "ColorPro", CRN: "Cornerstone", CTX: "CTX International", CVIS: "ColorVision", CWC: "Fujitsu Laboratories", DARI: "Darius Technology", DATA: "Dataproducts", DCP: "Dry Creek Photo", DCRC: "Digital Contents Resource Center, Chung-Ang University", DELL: "Dell Computer", DIC: "Dainippon Ink and Chemicals", DICO: "Diconix", DIGI: "Digital", "DL&C": "Digital Light & Color", DPLG: "Doppelganger", DS: "Dainippon Screen", DSOL: "DOOSOL", DUPN: "DuPont", EPSO: "Epson", ESKO: "Esko-Graphics", ETRI: "Electronics and Telecommunications Research Institute", EVER: "Everex Systems", EXAC: "ExactCODE", Eizo: "Eizo", FALC: "Falco Data Products", FF: "Fuji Photo Film", FFEI: "FujiFilm Electronic Imaging", FNRD: "Fnord Software", FORA: "Fora", FORE: "Forefront Technology", FP: "Fujitsu", FPA: "WayTech Development", FUJI: "Fujitsu", FX: "Fuji Xerox", GCC: "GCC Technologies", GGSL: "Global Graphics Software", GMB: "Gretagmacbeth", GMG: "GMG", GOLD: "GoldStar Technology", GOOG: "Google", GPRT: "Giantprint", GTMB: "Gretagmacbeth", GVC: "WayTech Development", GW2K: "Sony", HCI: "HCI", HDM: "Heidelberger Druckmaschinen", HERM: "Hermes", HITA: "Hitachi America", HP: "Hewlett-Packard", HTC: "Hitachi", HiTi: "HiTi Digital", IBM: "IBM", IDNT: "Scitex", IEC: "Hewlett-Packard", IIYA: "Iiyama North America", IKEG: "Ikegami Electronics", IMAG: "Image Systems", IMI: "Ingram Micro", INTC: "Intel", INTL: "N/A (INTL)", INTR: "Intra Electronics", IOCO: "Iocomm International Technology", IPS: "InfoPrint Solutions Company", IRIS: "Scitex", ISL: "Ichikawa Soft Laboratory", ITNL: "N/A (ITNL)", IVM: "IVM", IWAT: "Iwatsu Electric", Idnt: "Scitex", Inca: "Inca Digital Printers", Iris: "Scitex", JPEG: "Joint Photographic Experts Group", JSFT: "Jetsoft Development", JVC: "JVC Information Products", KART: "Scitex", KFC: "KFC Computek Components", KLH: "KLH Computers", KMHD: "Konica Minolta", KNCA: "Konica", KODA: "Kodak", KYOC: "Kyocera", Kart: "Scitex", LCAG: "Leica", LCCD: "Leeds Colour", LDAK: "Left Dakota", LEAD: "Leading Technology", LEXM: "Lexmark International", LINK: "Link Computer", LINO: "Linotronic", LITE: "Lite-On", Leaf: "Leaf", Lino: "Linotronic", MAGC: "Mag Computronic", MAGI: "MAG Innovision", MANN: "Mannesmann", MICN: "Micron Technology", MICR: "Microtek", MICV: "Microvitec", MINO: "Minolta", MITS: "Mitsubishi Electronics America", MITs: "Mitsuba", MNLT: "Minolta", MODG: "Modgraph", MONI: "Monitronix", MONS: "Monaco Systems", MORS: "Morse Technology", MOTI: "Motive Systems", MSFT: "Microsoft", MUTO: "MUTOH INDUSTRIES", Mits: "Mitsubishi Electric", NANA: "NANAO", NEC: "NEC", NEXP: "NexPress Solutions", NISS: "Nissei Sangyo America", NKON: "Nikon", NONE: "none", OCE: "Oce Technologies", OCEC: "OceColor", OKI: "Oki", OKID: "Okidata", OKIP: "Okidata", OLIV: "Olivetti", OLYM: "Olympus", ONYX: "Onyx Graphics", OPTI: "Optiquest", PACK: "Packard Bell", PANA: "Matsushita Electric Industrial", PANT: "Pantone", PBN: "Packard Bell", PFU: "PFU", PHIL: "Philips Consumer Electronics", PNTX: "HOYA", POne: "Phase One A/S", PREM: "Premier Computer Innovations", PRIN: "Princeton Graphic Systems", PRIP: "Princeton Publishing Labs", QLUX: "Hong Kong", QMS: "QMS", QPCD: "QPcard AB", QUAD: "QuadLaser", QUME: "Qume", RADI: "Radius", RDDx: "Integrated Color Solutions", RDG: "Roland DG", REDM: "REDMS Group", RELI: "Relisys", RGMS: "Rolf Gierling Multitools", RICO: "Ricoh", RNLD: "Edmund Ronald", ROYA: "Royal", RPC: "Ricoh Printing Systems", RTL: "Royal Information Electronics", SAMP: "Sampo", SAMS: "Samsung", SANT: "Jaime Santana Pomares", SCIT: "Scitex", SCRN: "Dainippon Screen", SDP: "Scitex", SEC: "Samsung", SEIK: "Seiko Instruments", SEIk: "Seikosha", SGUY: "ScanGuy.com", SHAR: "Sharp Laboratories", SICC: "International Color Consortium", SONY: "Sony", SPCL: "SpectraCal", STAR: "Star", STC: "Sampo Technology", Scit: "Scitex", Sdp: "Scitex", Sony: "Sony", TALO: "Talon Technology", TAND: "Tandy", TATU: "Tatung", TAXA: "TAXAN America", TDS: "Tokyo Denshi Sekei", TECO: "TECO Information Systems", TEGR: "Tegra", TEKT: "Tektronix", TI: "Texas Instruments", TMKR: "TypeMaker", TOSB: "Toshiba", TOSH: "Toshiba", TOTK: "TOTOKU ELECTRIC", TRIU: "Triumph", TSBT: "Toshiba", TTX: "TTX Computer Products", TVM: "TVM Professional Monitor", TW: "TW Casper", ULSX: "Ulead Systems", UNIS: "Unisys", UTZF: "Utz Fehlau & Sohn", VARI: "Varityper", VIEW: "Viewsonic", VISL: "Visual communication", VIVO: "Vivo Mobile Communication", WANG: "Wang", WLBR: "Wilbur Imaging", WTG2: "Ware To Go", WYSE: "WYSE Technology", XERX: "Xerox", XRIT: "X-Rite", ZRAN: "Zoran", Zebr: "Zebra Technologies", appl: "Apple Computer", bICC: "basICColor", berg: "bergdesign", ceyd: "Integrated Color Solutions", clsp: "MacDermid ColorSpan", ds: "Dainippon Screen", dupn: "DuPont", ffei: "FujiFilm Electronic Imaging", flux: "FluxData", iris: "Scitex", kart: "Scitex", lcms: "Little CMS", lino: "Linotronic", none: "none", ob4d: "Erdt Systems", obic: "Medigraph", quby: "Qubyx Sarl", scit: "Scitex", scrn: "Dainippon Screen", sdp: "Scitex", siwi: "SIWI GRAFIKA", yxym: "YxyMaster" };
- var Ct = { scnr: "Scanner", mntr: "Monitor", prtr: "Printer", link: "Device Link", abst: "Abstract", spac: "Color Space Conversion Profile", nmcl: "Named Color", cenc: "ColorEncodingSpace profile", mid: "MultiplexIdentification profile", mlnk: "MultiplexLink profile", mvis: "MultiplexVisualization profile", nkpf: "Nikon Input Device Profile (NON-STANDARD!)" };
- U(B, "icc", [[4, St], [12, Ct], [40, Object.assign({}, St, Ct)], [48, St], [80, St], [64, { 0: "Perceptual", 1: "Relative Colorimetric", 2: "Saturation", 3: "Absolute Colorimetric" }], ["tech", { amd: "Active Matrix Display", crt: "Cathode Ray Tube Display", kpcd: "Photo CD", pmd: "Passive Matrix Display", dcam: "Digital Camera", dcpj: "Digital Cinema Projector", dmpc: "Digital Motion Picture Camera", dsub: "Dye Sublimation Printer", epho: "Electrophotographic Printer", esta: "Electrostatic Printer", flex: "Flexography", fprn: "Film Writer", fscn: "Film Scanner", grav: "Gravure", ijet: "Ink Jet Printer", imgs: "Photo Image Setter", mpfr: "Motion Picture Film Recorder", mpfs: "Motion Picture Film Scanner", offs: "Offset Lithography", pjtv: "Projection Television", rpho: "Photographic Paper Printer", rscn: "Reflective Scanner", silk: "Silkscreen", twax: "Thermal Wax Printer", vidc: "Video Camera", vidm: "Video Monitor" }]]);
- var yt = class extends re2 {
- static canHandle(e3, t2, i3) {
- return 237 === e3.getUint8(t2 + 1) && "Photoshop" === e3.getString(t2 + 4, 9) && void 0 !== this.containsIptc8bim(e3, t2, i3);
+ batchEnableWithBool(e3, t2) {
+ for (let i3 of e3)
+ this[i3].enabled = t2;
}
- static headerLength(e3, t2, i3) {
- let n3, s2 = this.containsIptc8bim(e3, t2, i3);
- if (void 0 !== s2)
- return n3 = e3.getUint8(t2 + s2 + 7), n3 % 2 != 0 && (n3 += 1), 0 === n3 && (n3 = 4), s2 + 8 + n3;
+ batchEnableWithUserValue(e3, t2) {
+ for (let i3 of e3) {
+ let e4 = t2[i3];
+ this[i3].enabled = false !== e4 && void 0 !== e4;
+ }
}
- static containsIptc8bim(e3, t2, i3) {
- for (let n3 = 0; n3 < i3; n3++)
- if (this.isIptcSegmentHead(e3, t2 + n3))
- return n3;
+ setupGlobalFilters(e3, t2, i3, n3 = i3) {
+ if (e3 && e3.length) {
+ for (let e4 of n3)
+ this[e4].enabled = false;
+ let t3 = Q(e3, i3);
+ for (let [e4, i4] of t3)
+ ee(this[e4].pick, i4), this[e4].enabled = true;
+ } else if (t2 && t2.length) {
+ let e4 = Q(t2, i3);
+ for (let [t3, i4] of e4)
+ ee(this[t3].skip, i4);
+ }
}
- static isIptcSegmentHead(e3, t2) {
- return 56 === e3.getUint8(t2) && 943868237 === e3.getUint32(t2) && 1028 === e3.getUint16(t2 + 4);
+ filterNestedSegmentTags() {
+ let { ifd0: e3, exif: t2, xmp: i3, iptc: n3, icc: s2 } = this;
+ this.makerNote ? t2.deps.add(37500) : t2.skip.add(37500), this.userComment ? t2.deps.add(37510) : t2.skip.add(37510), i3.enabled || e3.skip.add(700), n3.enabled || e3.skip.add(33723), s2.enabled || e3.skip.add(34675);
}
- parse() {
- let { raw: e3 } = this, t2 = this.chunk.byteLength - 1, i3 = false;
- for (let n3 = 0; n3 < t2; n3++)
- if (28 === this.chunk.getUint8(n3) && 2 === this.chunk.getUint8(n3 + 1)) {
- i3 = true;
- let t3 = this.chunk.getUint16(n3 + 3), s2 = this.chunk.getUint8(n3 + 2), r2 = this.chunk.getLatin1String(n3 + 5, t3);
- e3.set(s2, this.pluralizeValue(e3.get(s2), r2)), n3 += 4 + t3;
- } else if (i3)
- break;
- return this.translate(), this.output;
+ traverseTiffDependencyTree() {
+ let { ifd0: e3, exif: t2, gps: i3, interop: n3 } = this;
+ n3.needed && (t2.deps.add(40965), e3.deps.add(40965)), t2.needed && e3.deps.add(34665), i3.needed && e3.deps.add(34853), this.tiff.enabled = H.some((e4) => true === this[e4].enabled) || this.makerNote || this.userComment;
+ for (let e4 of H)
+ this[e4].finalizeFilters();
}
- pluralizeValue(e3, t2) {
- return void 0 !== e3 ? e3 instanceof Array ? (e3.push(t2), e3) : [e3, t2] : t2;
+ get onlyTiff() {
+ return !V.map((e3) => this[e3].enabled).some((e3) => true === e3) && this.tiff.enabled;
+ }
+ checkLoadedPlugins() {
+ for (let e3 of z)
+ this[e3].enabled && !T.has(e3) && P("segment parser", e3);
}
};
- c(yt, "type", "iptc"), c(yt, "translateValues", false), c(yt, "reviveValues", false), T.set("iptc", yt), U(E, "iptc", [[0, "ApplicationRecordVersion"], [3, "ObjectTypeReference"], [4, "ObjectAttributeReference"], [5, "ObjectName"], [7, "EditStatus"], [8, "EditorialUpdate"], [10, "Urgency"], [12, "SubjectReference"], [15, "Category"], [20, "SupplementalCategories"], [22, "FixtureIdentifier"], [25, "Keywords"], [26, "ContentLocationCode"], [27, "ContentLocationName"], [30, "ReleaseDate"], [35, "ReleaseTime"], [37, "ExpirationDate"], [38, "ExpirationTime"], [40, "SpecialInstructions"], [42, "ActionAdvised"], [45, "ReferenceService"], [47, "ReferenceDate"], [50, "ReferenceNumber"], [55, "DateCreated"], [60, "TimeCreated"], [62, "DigitalCreationDate"], [63, "DigitalCreationTime"], [65, "OriginatingProgram"], [70, "ProgramVersion"], [75, "ObjectCycle"], [80, "Byline"], [85, "BylineTitle"], [90, "City"], [92, "Sublocation"], [95, "State"], [100, "CountryCode"], [101, "Country"], [103, "OriginalTransmissionReference"], [105, "Headline"], [110, "Credit"], [115, "Source"], [116, "CopyrightNotice"], [118, "Contact"], [120, "Caption"], [121, "LocalCaption"], [122, "Writer"], [125, "RasterizedCaption"], [130, "ImageType"], [131, "ImageOrientation"], [135, "LanguageIdentifier"], [150, "AudioType"], [151, "AudioSamplingRate"], [152, "AudioSamplingResolution"], [153, "AudioDuration"], [154, "AudioOutcue"], [184, "JobID"], [185, "MasterDocumentID"], [186, "ShortDocumentID"], [187, "UniqueDocumentID"], [188, "OwnerID"], [200, "ObjectPreviewFileFormat"], [201, "ObjectPreviewFileVersion"], [202, "ObjectPreviewData"], [221, "Prefs"], [225, "ClassifyState"], [228, "SimilarityIndex"], [230, "DocumentNotes"], [231, "DocumentHistory"], [232, "ExifCameraInfo"], [255, "CatalogSets"]]), U(B, "iptc", [[10, { 0: "0 (reserved)", 1: "1 (most urgent)", 2: "2", 3: "3", 4: "4", 5: "5 (normal urgency)", 6: "6", 7: "7", 8: "8 (least urgent)", 9: "9 (user-defined priority)" }], [75, { a: "Morning", b: "Both Morning and Evening", p: "Evening" }], [131, { L: "Landscape", P: "Portrait", S: "Square" }]]);
- var full_esm_default = tt;
-
- // modules/services/plane_photo.js
- var dispatch6 = dispatch_default("viewerChanged");
- var _photo;
- var _wrapper;
- var imgZoom;
- var _widthOverflow;
- function zoomPan(d3_event) {
- let t2 = d3_event.transform;
- _photo.call(utilSetTransform, t2.x, t2.y, t2.k);
+ function Q(e3, t2) {
+ let i3, n3, s2, r2, a2 = [];
+ for (s2 of t2) {
+ for (r2 of (i3 = E.get(s2), n3 = [], i3))
+ (e3.includes(r2[0]) || e3.includes(r2[1])) && n3.push(r2[0]);
+ n3.length && a2.push([s2, n3]);
+ }
+ return a2;
}
- function zoomBeahvior() {
- const { width: wrapperWidth, height: wrapperHeight } = _wrapper.node().getBoundingClientRect();
- const { naturalHeight, naturalWidth } = _photo.node();
- const intrinsicRatio = naturalWidth / naturalHeight;
- _widthOverflow = wrapperHeight * intrinsicRatio - wrapperWidth;
- return zoom_default2().extent([[0, 0], [wrapperWidth, wrapperHeight]]).translateExtent([[0, 0], [wrapperWidth + _widthOverflow, wrapperHeight]]).scaleExtent([1, 15]).on("zoom", zoomPan);
+ function Z(e3, t2) {
+ return void 0 !== e3 ? e3 : void 0 !== t2 ? t2 : void 0;
}
- function loadImage(selection2, path) {
- return new Promise((resolve) => {
- selection2.attr("src", path);
- selection2.on("load", () => {
- resolve(selection2);
- });
- });
+ function ee(e3, t2) {
+ for (let i3 of t2)
+ e3.add(i3);
}
- var plane_photo_default = {
- init: async function(context, selection2) {
- this.event = utilRebind(this, dispatch6, "on");
- _wrapper = selection2.append("div").attr("class", "photo-frame plane-frame").classed("hide", true);
- _photo = _wrapper.append("img").attr("class", "plane-photo");
- context.ui().photoviewer.on("resize.plane", () => {
- imgZoom = zoomBeahvior();
- _wrapper.call(imgZoom);
- });
- await Promise.resolve();
- return this;
- },
- showPhotoFrame: function(context) {
- const isHidden = context.selectAll(".photo-frame.plane-frame.hide").size();
- if (isHidden) {
- context.selectAll(".photo-frame:not(.plane-frame)").classed("hide", true);
- context.selectAll(".photo-frame.plane-frame").classed("hide", false);
- }
- return this;
- },
- hidePhotoFrame: function(context) {
- context.select("photo-frame.plane-frame").classed("hide", false);
- return this;
- },
- selectPhoto: function(data, keepOrientation) {
- dispatch6.call("viewerChanged");
- loadImage(_photo, "");
- loadImage(_photo, data.image_path).then(() => {
- if (!keepOrientation) {
- imgZoom = zoomBeahvior();
- _wrapper.call(imgZoom);
- _wrapper.call(imgZoom.transform, identity2.translate(-_widthOverflow / 2, 0));
- }
+ c(q, "default", $2);
+ var te = class {
+ constructor(e3) {
+ c(this, "parsers", {}), c(this, "output", {}), c(this, "errors", []), c(this, "pushToErrors", (e4) => this.errors.push(e4)), this.options = q.useCached(e3);
+ }
+ async read(e3) {
+ this.file = await D(e3, this.options);
+ }
+ setup() {
+ if (this.fileParser)
+ return;
+ let { file: e3 } = this, t2 = e3.getUint16(0);
+ for (let [i3, n3] of w)
+ if (n3.canHandle(e3, t2))
+ return this.fileParser = new n3(this.options, this.file, this.parsers), e3[i3] = true;
+ this.file.close && this.file.close(), g2("Unknown file format");
+ }
+ async parse() {
+ let { output: e3, errors: t2 } = this;
+ return this.setup(), this.options.silentErrors ? (await this.executeParsers().catch(this.pushToErrors), t2.push(...this.fileParser.errors)) : await this.executeParsers(), this.file.close && this.file.close(), this.options.silentErrors && t2.length > 0 && (e3.errors = t2), f(e3);
+ }
+ async executeParsers() {
+ let { output: e3 } = this;
+ await this.fileParser.parse();
+ let t2 = Object.values(this.parsers).map(async (t3) => {
+ let i3 = await t3.parse();
+ t3.assignToOutput(e3, i3);
});
- return this;
- },
- getYaw: function() {
- return 0;
+ this.options.silentErrors && (t2 = t2.map((e4) => e4.catch(this.pushToErrors))), await Promise.all(t2);
+ }
+ async extractThumbnail() {
+ this.setup();
+ let { options: e3, file: t2 } = this, i3 = T.get("tiff", e3);
+ var n3;
+ if (t2.tiff ? n3 = { start: 0, type: "tiff" } : t2.jpeg && (n3 = await this.fileParser.getOrFindSegment("tiff")), void 0 === n3)
+ return;
+ let s2 = await this.fileParser.ensureSegmentChunk(n3), r2 = this.parsers.tiff = new i3(s2, e3, t2), a2 = await r2.extractThumbnail();
+ return t2.close && t2.close(), a2;
}
};
-
- // modules/svg/local_photos.js
- var _initialized2 = false;
- var _enabled2 = false;
- var minViewfieldZoom = 16;
- function svgLocalPhotos(projection2, context, dispatch14) {
- const detected = utilDetect();
- let layer = select_default2(null);
- let _fileList;
- let _photos = [];
- let _idAutoinc = 0;
- let _photoFrame;
- function init2() {
- if (_initialized2)
- return;
- _enabled2 = true;
- function over(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- d3_event.dataTransfer.dropEffect = "copy";
- }
- context.container().attr("dropzone", "copy").on("drop.svgLocalPhotos", function(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- if (!detected.filedrop)
- return;
- drawPhotos.fileList(d3_event.dataTransfer.files, (loaded) => {
- if (loaded.length > 0) {
- drawPhotos.fitZoom(false);
- }
- });
- }).on("dragenter.svgLocalPhotos", over).on("dragexit.svgLocalPhotos", over).on("dragover.svgLocalPhotos", over);
- _initialized2 = true;
+ async function ie(e3, t2) {
+ let i3 = new te(t2);
+ return await i3.read(e3), i3.parse();
+ }
+ var ne = Object.freeze({ __proto__: null, parse: ie, Exifr: te, fileParsers: w, segmentParsers: T, fileReaders: A, tagKeys: E, tagValues: B, tagRevivers: N, createDictionary: U, extendDictionary: F, fetchUrlAsArrayBuffer: M, readBlobAsArrayBuffer: R, chunkedProps: G, otherSegments: V, segments: z, tiffBlocks: H, segmentsAndBlocks: j, tiffExtractables: W, inheritables: K, allFormatters: X, Options: q });
+ var se = class {
+ constructor(e3, t2, i3) {
+ c(this, "errors", []), c(this, "ensureSegmentChunk", async (e4) => {
+ let t3 = e4.start, i4 = e4.size || 65536;
+ if (this.file.chunked)
+ if (this.file.available(t3, i4))
+ e4.chunk = this.file.subarray(t3, i4);
+ else
+ try {
+ e4.chunk = await this.file.readChunk(t3, i4);
+ } catch (t4) {
+ g2("Couldn't read segment: ".concat(JSON.stringify(e4), ". ").concat(t4.message));
+ }
+ else
+ this.file.byteLength > t3 + i4 ? e4.chunk = this.file.subarray(t3, i4) : void 0 === e4.size ? e4.chunk = this.file.subarray(t3) : g2("Segment unreachable: " + JSON.stringify(e4));
+ return e4.chunk;
+ }), this.extendOptions && this.extendOptions(e3), this.options = e3, this.file = t2, this.parsers = i3;
}
- function ensureViewerLoaded(context2) {
- if (_photoFrame) {
- return Promise.resolve(_photoFrame);
- }
- const viewer = context2.container().select(".photoviewer").selectAll(".local-photos-wrapper").data([0]);
- const viewerEnter = viewer.enter().append("div").attr("class", "photo-wrapper local-photos-wrapper").classed("hide", true);
- viewerEnter.append("div").attr("class", "photo-attribution fillD");
- return plane_photo_default.init(context2, viewerEnter).then((planePhotoFrame) => {
- _photoFrame = planePhotoFrame;
- });
+ injectSegment(e3, t2) {
+ this.options[e3].enabled && this.createParser(e3, t2);
}
- function click(d3_event, image, zoomTo) {
- ensureViewerLoaded(context).then(() => {
- const viewer = context.container().select(".photoviewer").datum(image).classed("hide", false);
- const viewerWrap = viewer.select(".local-photos-wrapper").classed("hide", false);
- const attribution = viewerWrap.selectAll(".photo-attribution").text("");
- if (image.name) {
- attribution.append("span").classed("filename", true).text(image.name);
+ createParser(e3, t2) {
+ let i3 = new (T.get(e3))(t2, this.options, this.file);
+ return this.parsers[e3] = i3;
+ }
+ createParsers(e3) {
+ for (let t2 of e3) {
+ let { type: e4, chunk: i3 } = t2, n3 = this.options[e4];
+ if (n3 && n3.enabled) {
+ let t3 = this.parsers[e4];
+ t3 && t3.append || t3 || this.createParser(e4, i3);
}
- _photoFrame.selectPhoto({ image_path: "" });
- image.getSrc().then((src) => {
- _photoFrame.selectPhoto({ image_path: src }).showPhotoFrame(viewerWrap);
- setStyles();
- });
- });
- if (zoomTo) {
- context.map().centerEase(image.loc);
}
}
- function transform2(d2) {
- var svgpoint = projection2(d2.loc);
- return "translate(" + svgpoint[0] + "," + svgpoint[1] + ")";
+ async readSegments(e3) {
+ let t2 = e3.map(this.ensureSegmentChunk);
+ await Promise.all(t2);
}
- function setStyles(hovered) {
- const viewer = context.container().select(".photoviewer");
- const selected = viewer.empty() ? void 0 : viewer.datum();
- context.container().selectAll(".layer-local-photos .viewfield-group").classed("hovered", (d2) => d2.id === hovered?.id).classed("highlighted", (d2) => d2.id === hovered?.id || d2.id === selected?.id).classed("currentView", (d2) => d2.id === selected?.id);
+ };
+ var re2 = class {
+ static findPosition(e3, t2) {
+ let i3 = e3.getUint16(t2 + 2) + 2, n3 = "function" == typeof this.headerLength ? this.headerLength(e3, t2, i3) : this.headerLength, s2 = t2 + n3, r2 = i3 - n3;
+ return { offset: t2, length: i3, headerLength: n3, start: s2, size: r2, end: s2 + r2 };
}
- function display_markers(imageList) {
- imageList = imageList.filter((image) => isArray_default(image.loc) && isNumber_default(image.loc[0]) && isNumber_default(image.loc[1]));
- const groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(imageList, function(d2) {
- return d2.id;
- });
- groups.exit().remove();
- const groupsEnter = groups.enter().append("g").attr("class", "viewfield-group").on("mouseenter", (d3_event, d2) => setStyles(d2)).on("mouseleave", () => setStyles(null)).on("click", click);
- 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 showViewfields = context.map().zoom() >= minViewfieldZoom;
- const viewfields = markers.selectAll(".viewfield").data(showViewfields ? [0] : []);
- viewfields.exit().remove();
- viewfields.enter().insert("path", "circle").attr("class", "viewfield").attr("transform", function() {
- const d2 = this.parentNode.__data__;
- return `rotate(${Math.round(d2.direction ?? 0)},0,0),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").style("visibility", function() {
- const d2 = this.parentNode.__data__;
- return isNumber_default(d2.direction) ? "visible" : "hidden";
- });
+ static parse(e3, t2 = {}) {
+ return new this(e3, new q({ [this.type]: t2 }), e3).parse();
}
- function drawPhotos(selection2) {
- layer = selection2.selectAll(".layer-local-photos").data(_photos ? [0] : []);
- layer.exit().remove();
- const layerEnter = layer.enter().append("g").attr("class", "layer-local-photos");
- layerEnter.append("g").attr("class", "markers");
- layer = layerEnter.merge(layer);
- if (_photos && _photos.length !== 0) {
- display_markers(_photos);
- }
+ normalizeInput(e3) {
+ return e3 instanceof I ? e3 : new I(e3);
}
- function readFileAsDataURL(file) {
- return new Promise((resolve, reject) => {
- const reader = new FileReader();
- reader.onload = () => resolve(reader.result);
- reader.onerror = (error) => reject(error);
- reader.readAsDataURL(file);
- });
+ constructor(e3, t2 = {}, i3) {
+ c(this, "errors", []), c(this, "raw", /* @__PURE__ */ new Map()), c(this, "handleError", (e4) => {
+ if (!this.options.silentErrors)
+ throw e4;
+ this.errors.push(e4.message);
+ }), this.chunk = this.normalizeInput(e3), this.file = i3, this.type = this.constructor.type, this.globalOptions = this.options = t2, this.localOptions = t2[this.type], this.canTranslate = this.localOptions && this.localOptions.translate;
}
- async function readmultifiles(files, callback) {
- const loaded = [];
- for (const file of files) {
- try {
- const exifData = await full_esm_default.parse(file);
- const photo = {
- id: _idAutoinc++,
- name: file.name,
- getSrc: () => readFileAsDataURL(file),
- file,
- loc: [exifData.longitude, exifData.latitude],
- direction: exifData.GPSImgDirection
- };
- loaded.push(photo);
- const sameName = _photos.filter((i3) => i3.name === photo.name);
- if (sameName.length === 0) {
- _photos.push(photo);
- } else {
- const thisContent = await photo.getSrc();
- const sameNameContent = await Promise.allSettled(sameName.map((i3) => i3.getSrc()));
- if (!sameNameContent.some((i3) => i3.value === thisContent)) {
- _photos.push(photo);
- }
- }
- } catch (err) {
- }
- }
- if (typeof callback === "function")
- callback(loaded);
- dispatch14.call("change");
+ translate() {
+ this.canTranslate && (this.translated = this.translateBlock(this.raw, this.type));
}
- drawPhotos.setFiles = function(fileList, callback) {
- readmultifiles(Array.from(fileList), callback);
- return this;
- };
- drawPhotos.fileList = function(fileList, callback) {
- if (!arguments.length)
- return _fileList;
- _fileList = fileList;
- if (!fileList || !fileList.length)
- return this;
- drawPhotos.setFiles(_fileList, callback);
- return this;
- };
- drawPhotos.getPhotos = function() {
- return _photos;
- };
- drawPhotos.removePhoto = function(id2) {
- _photos = _photos.filter((i3) => i3.id !== id2);
- dispatch14.call("change");
- return _photos;
- };
- drawPhotos.openPhoto = click;
- drawPhotos.fitZoom = function(force) {
- const coords = _photos.map((image) => image.loc).filter((l2) => isArray_default(l2) && isNumber_default(l2[0]) && isNumber_default(l2[1]));
- if (coords.length === 0)
- return;
- const extent = coords.map((l2) => geoExtent(l2, l2)).reduce((a2, b2) => a2.extend(b2));
- const map2 = context.map();
- var viewport = map2.trimmedExtent().polygon();
- if (force !== false || !geoPolygonIntersectsPolygon(viewport, coords, true)) {
- map2.centerZoom(extent.center(), Math.min(18, map2.trimmedExtentZoom(extent)));
- }
- };
- function showLayer() {
- layer.style("display", "block");
- layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
- dispatch14.call("change");
- });
+ get output() {
+ return this.translated ? this.translated : this.raw ? Object.fromEntries(this.raw) : void 0;
}
- function hideLayer() {
- layer.transition().duration(250).style("opacity", 0).on("end", () => {
- layer.selectAll(".viewfield-group").remove();
- layer.style("display", "none");
- });
+ translateBlock(e3, t2) {
+ let i3 = N.get(t2), n3 = B.get(t2), s2 = E.get(t2), r2 = this.options[t2], a2 = r2.reviveValues && !!i3, o2 = r2.translateValues && !!n3, l2 = r2.translateKeys && !!s2, h2 = {};
+ for (let [t3, r3] of e3)
+ a2 && i3.has(t3) ? r3 = i3.get(t3)(r3) : o2 && n3.has(t3) && (r3 = this.translateValue(r3, n3.get(t3))), l2 && s2.has(t3) && (t3 = s2.get(t3) || t3), h2[t3] = r3;
+ return h2;
}
- drawPhotos.enabled = function(val) {
- if (!arguments.length)
- return _enabled2;
- _enabled2 = val;
- if (_enabled2) {
- showLayer();
- } else {
- hideLayer();
- }
- dispatch14.call("change");
- return this;
- };
- drawPhotos.hasData = function() {
- return isArray_default(_photos) && _photos.length > 0;
- };
- init2();
- return drawPhotos;
+ translateValue(e3, t2) {
+ return t2[e3] || t2.DEFAULT || e3;
+ }
+ assignToOutput(e3, t2) {
+ this.assignObjectToOutput(e3, this.constructor.type, t2);
+ }
+ assignObjectToOutput(e3, t2, i3) {
+ if (this.globalOptions.mergeOutput)
+ return Object.assign(e3, i3);
+ e3[t2] ? Object.assign(e3[t2], i3) : e3[t2] = i3;
+ }
+ };
+ c(re2, "headerLength", 4), c(re2, "type", void 0), c(re2, "multiSegment", false), c(re2, "canHandle", () => false);
+ function ae(e3) {
+ return 192 === e3 || 194 === e3 || 196 === e3 || 219 === e3 || 221 === e3 || 218 === e3 || 254 === e3;
}
-
- // modules/svg/improveOSM.js
- var _layerEnabled2 = false;
- var _qaService2;
- function svgImproveOSM(projection2, context, dispatch14) {
- const throttledRedraw = throttle_default(() => dispatch14.call("change"), 1e3);
- const minZoom4 = 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 oe(e3) {
+ return e3 >= 224 && e3 <= 239;
+ }
+ function le(e3, t2, i3) {
+ for (let [n3, s2] of T)
+ if (s2.canHandle(e3, t2, i3))
+ return n3;
+ }
+ var he = class extends se {
+ constructor(...e3) {
+ super(...e3), c(this, "appSegments", []), c(this, "jpegSegments", []), c(this, "unknownSegments", []);
}
- function getService() {
- if (services.improveOSM && !_qaService2) {
- _qaService2 = services.improveOSM;
- _qaService2.on("loaded", throttledRedraw);
- } else if (!services.improveOSM && _qaService2) {
- _qaService2 = null;
- }
- return _qaService2;
+ static canHandle(e3, t2) {
+ return 65496 === t2;
}
- function editOn() {
- if (!layerVisible) {
- layerVisible = true;
- drawLayer.style("display", "block");
- }
+ async parse() {
+ await this.findAppSegments(), await this.readSegments(this.appSegments), this.mergeMultiSegments(), this.createParsers(this.mergedAppSegments || this.appSegments);
}
- function editOff() {
- if (layerVisible) {
- layerVisible = false;
- drawLayer.style("display", "none");
- drawLayer.selectAll(".qaItem.improveOSM").remove();
- touchLayer.selectAll(".qaItem.improveOSM").remove();
+ setupSegmentFinderArgs(e3) {
+ true === e3 ? (this.findAll = true, this.wanted = new Set(T.keyList())) : (e3 = void 0 === e3 ? T.keyList().filter((e4) => this.options[e4].enabled) : e3.filter((e4) => this.options[e4].enabled && T.has(e4)), this.findAll = false, this.remaining = new Set(e3), this.wanted = new Set(e3)), this.unfinishedMultiSegment = false;
+ }
+ async findAppSegments(e3 = 0, t2) {
+ this.setupSegmentFinderArgs(t2);
+ let { file: i3, findAll: n3, wanted: s2, remaining: r2 } = this;
+ if (!n3 && this.file.chunked && (n3 = Array.from(s2).some((e4) => {
+ let t3 = T.get(e4), i4 = this.options[e4];
+ return t3.multiSegment && i4.multiSegment;
+ }), n3 && await this.file.readWhole()), e3 = this.findAppSegmentsInRange(e3, i3.byteLength), !this.options.onlyTiff && i3.chunked) {
+ let t3 = false;
+ for (; r2.size > 0 && !t3 && (i3.canReadNextChunk || this.unfinishedMultiSegment); ) {
+ let { nextChunkOffset: n4 } = i3, s3 = this.appSegments.some((e4) => !this.file.available(e4.offset || e4.start, e4.length || e4.size));
+ if (t3 = e3 > n4 && !s3 ? !await i3.readNextChunk(e3) : !await i3.readNextChunk(n4), void 0 === (e3 = this.findAppSegmentsInRange(e3, i3.byteLength)))
+ return;
+ }
}
}
- function layerOn() {
- editOn();
- drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch14.call("change"));
+ findAppSegmentsInRange(e3, t2) {
+ t2 -= 2;
+ let i3, n3, s2, r2, a2, o2, { file: l2, findAll: h2, wanted: u2, remaining: c2, options: f2 } = this;
+ for (; e3 < t2; e3++)
+ if (255 === l2.getUint8(e3)) {
+ if (i3 = l2.getUint8(e3 + 1), oe(i3)) {
+ if (n3 = l2.getUint16(e3 + 2), s2 = le(l2, e3, n3), s2 && u2.has(s2) && (r2 = T.get(s2), a2 = r2.findPosition(l2, e3), o2 = f2[s2], a2.type = s2, this.appSegments.push(a2), !h2 && (r2.multiSegment && o2.multiSegment ? (this.unfinishedMultiSegment = a2.chunkNumber < a2.chunkCount, this.unfinishedMultiSegment || c2.delete(s2)) : c2.delete(s2), 0 === c2.size)))
+ break;
+ f2.recordUnknownSegments && (a2 = re2.findPosition(l2, e3), a2.marker = i3, this.unknownSegments.push(a2)), e3 += n3 + 1;
+ } else if (ae(i3)) {
+ if (n3 = l2.getUint16(e3 + 2), 218 === i3 && false !== f2.stopAfterSos)
+ return;
+ f2.recordJpegSegments && this.jpegSegments.push({ offset: e3, length: n3, marker: i3 }), e3 += n3 + 1;
+ }
+ }
+ return e3;
}
- function layerOff() {
- throttledRedraw.cancel();
- drawLayer.interrupt();
- touchLayer.selectAll(".qaItem.improveOSM").remove();
- drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
- editOff();
- dispatch14.call("change");
+ mergeMultiSegments() {
+ if (!this.appSegments.some((e4) => e4.multiSegment))
+ return;
+ let e3 = function(e4, t2) {
+ let i3, n3, s2, r2 = /* @__PURE__ */ new Map();
+ for (let a2 = 0; a2 < e4.length; a2++)
+ i3 = e4[a2], n3 = i3[t2], r2.has(n3) ? s2 = r2.get(n3) : r2.set(n3, s2 = []), s2.push(i3);
+ return Array.from(r2);
+ }(this.appSegments, "type");
+ this.mergedAppSegments = e3.map(([e4, t2]) => {
+ let i3 = T.get(e4, this.options);
+ if (i3.handleMultiSegments) {
+ return { type: e4, chunk: i3.handleMultiSegments(t2) };
+ }
+ return t2[0];
});
}
- 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, (d2) => d2.id);
- markers.exit().remove();
- const markersEnter = markers.enter().append("g").attr("class", (d2) => `qaItem ${d2.service} itemId-${d2.id} itemType-${d2.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", (d2) => d2.icon ? "#" + d2.icon : "");
- markers.merge(markersEnter).sort(sortY).classed("selected", (d2) => d2.id === selectedID).attr("transform", getTransform);
- if (touchLayer.empty())
- return;
- const fillClass = context.getDebug("target") ? "pink " : "nocolor ";
- const targets = touchLayer.selectAll(".qaItem.improveOSM").data(data, (d2) => d2.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", (d2) => `qaItem ${d2.service} target ${fillClass} itemId-${d2.id}`).attr("transform", getTransform);
- function sortY(a2, b2) {
- return a2.id === selectedID ? 1 : b2.id === selectedID ? -1 : b2.loc[1] - a2.loc[1];
- }
+ getSegment(e3) {
+ return this.appSegments.find((t2) => t2.type === e3);
}
- 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() >= minZoom4) {
- editOn();
- service.loadIssues(projection2);
- updateMarkers();
- } else {
- editOff();
- }
+ async getOrFindSegment(e3) {
+ let t2 = this.getSegment(e3);
+ return void 0 === t2 && (await this.findAppSegments(0, [e3]), t2 = this.getSegment(e3)), t2;
+ }
+ };
+ c(he, "type", "jpeg"), w.set("jpeg", he);
+ var ue = [void 0, 1, 1, 2, 4, 8, 1, 1, 2, 4, 8, 4, 8, 4];
+ var ce = class extends re2 {
+ parseHeader() {
+ var e3 = this.chunk.getUint16();
+ 18761 === e3 ? this.le = true : 19789 === e3 && (this.le = false), this.chunk.le = this.le, this.headerParsed = true;
+ }
+ parseTags(e3, t2, i3 = /* @__PURE__ */ new Map()) {
+ let { pick: n3, skip: s2 } = this.options[t2];
+ n3 = new Set(n3);
+ let r2 = n3.size > 0, a2 = 0 === s2.size, o2 = this.chunk.getUint16(e3);
+ e3 += 2;
+ for (let l2 = 0; l2 < o2; l2++) {
+ let o3 = this.chunk.getUint16(e3);
+ if (r2) {
+ if (n3.has(o3) && (i3.set(o3, this.parseTag(e3, o3, t2)), n3.delete(o3), 0 === n3.size))
+ break;
+ } else
+ !a2 && s2.has(o3) || i3.set(o3, this.parseTag(e3, o3, t2));
+ e3 += 12;
}
+ return i3;
}
- drawImproveOSM.enabled = function(val) {
- if (!arguments.length)
- return _layerEnabled2;
- _layerEnabled2 = val;
- if (_layerEnabled2) {
- layerOn();
- } else {
- layerOff();
- if (context.selectedErrorID()) {
- context.enter(modeBrowse(context));
- }
+ parseTag(e3, t2, i3) {
+ let { chunk: n3 } = this, s2 = n3.getUint16(e3 + 2), r2 = n3.getUint32(e3 + 4), a2 = ue[s2];
+ if (a2 * r2 <= 4 ? e3 += 8 : e3 = n3.getUint32(e3 + 8), (s2 < 1 || s2 > 13) && g2("Invalid TIFF value type. block: ".concat(i3.toUpperCase(), ", tag: ").concat(t2.toString(16), ", type: ").concat(s2, ", offset ").concat(e3)), e3 > n3.byteLength && g2("Invalid TIFF value offset. block: ".concat(i3.toUpperCase(), ", tag: ").concat(t2.toString(16), ", type: ").concat(s2, ", offset ").concat(e3, " is outside of chunk size ").concat(n3.byteLength)), 1 === s2)
+ return n3.getUint8Array(e3, r2);
+ if (2 === s2)
+ return m(n3.getString(e3, r2));
+ if (7 === s2)
+ return n3.getUint8Array(e3, r2);
+ if (1 === r2)
+ return this.parseTagValue(s2, e3);
+ {
+ let t3 = new (function(e4) {
+ switch (e4) {
+ case 1:
+ return Uint8Array;
+ case 3:
+ return Uint16Array;
+ case 4:
+ return Uint32Array;
+ case 5:
+ return Array;
+ case 6:
+ return Int8Array;
+ case 8:
+ return Int16Array;
+ case 9:
+ return Int32Array;
+ case 10:
+ return Array;
+ case 11:
+ return Float32Array;
+ case 12:
+ return Float64Array;
+ default:
+ return Array;
+ }
+ }(s2))(r2), i4 = a2;
+ for (let n4 = 0; n4 < r2; n4++)
+ t3[n4] = this.parseTagValue(s2, e3), e3 += i4;
+ return t3;
}
- dispatch14.call("change");
- return this;
- };
- drawImproveOSM.supported = () => !!getService();
- return drawImproveOSM;
- }
-
- // modules/svg/osmose.js
- var _layerEnabled3 = false;
- var _qaService3;
- function svgOsmose(projection2, context, dispatch14) {
- const throttledRedraw = throttle_default(() => dispatch14.call("change"), 1e3);
- const minZoom4 = 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;
+ parseTagValue(e3, t2) {
+ let { chunk: i3 } = this;
+ switch (e3) {
+ case 1:
+ return i3.getUint8(t2);
+ case 3:
+ return i3.getUint16(t2);
+ case 4:
+ return i3.getUint32(t2);
+ case 5:
+ return i3.getUint32(t2) / i3.getUint32(t2 + 4);
+ case 6:
+ return i3.getInt8(t2);
+ case 8:
+ return i3.getInt16(t2);
+ case 9:
+ return i3.getInt32(t2);
+ case 10:
+ return i3.getInt32(t2) / i3.getInt32(t2 + 4);
+ case 11:
+ return i3.getFloat(t2);
+ case 12:
+ return i3.getDouble(t2);
+ case 13:
+ return i3.getUint32(t2);
+ default:
+ g2("Invalid tiff type ".concat(e3));
}
- return _qaService3;
}
- function editOn() {
- if (!layerVisible) {
- layerVisible = true;
- drawLayer.style("display", "block");
- }
+ };
+ var fe = class extends ce {
+ static canHandle(e3, t2) {
+ return 225 === e3.getUint8(t2 + 1) && 1165519206 === e3.getUint32(t2 + 4) && 0 === e3.getUint16(t2 + 8);
}
- function editOff() {
- if (layerVisible) {
- layerVisible = false;
- drawLayer.style("display", "none");
- drawLayer.selectAll(".qaItem.osmose").remove();
- touchLayer.selectAll(".qaItem.osmose").remove();
+ async parse() {
+ this.parseHeader();
+ let { options: e3 } = this;
+ return e3.ifd0.enabled && await this.parseIfd0Block(), e3.exif.enabled && await this.safeParse("parseExifBlock"), e3.gps.enabled && await this.safeParse("parseGpsBlock"), e3.interop.enabled && await this.safeParse("parseInteropBlock"), e3.ifd1.enabled && await this.safeParse("parseThumbnailBlock"), this.createOutput();
+ }
+ safeParse(e3) {
+ let t2 = this[e3]();
+ return void 0 !== t2.catch && (t2 = t2.catch(this.handleError)), t2;
+ }
+ findIfd0Offset() {
+ void 0 === this.ifd0Offset && (this.ifd0Offset = this.chunk.getUint32(4));
+ }
+ findIfd1Offset() {
+ if (void 0 === this.ifd1Offset) {
+ this.findIfd0Offset();
+ let e3 = this.chunk.getUint16(this.ifd0Offset), t2 = this.ifd0Offset + 2 + 12 * e3;
+ this.ifd1Offset = this.chunk.getUint32(t2);
}
}
- function layerOn() {
- editOn();
- drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch14.call("change"));
+ parseBlock(e3, t2) {
+ let i3 = /* @__PURE__ */ new Map();
+ return this[t2] = i3, this.parseTags(e3, t2, i3), i3;
}
- function layerOff() {
- throttledRedraw.cancel();
- drawLayer.interrupt();
- touchLayer.selectAll(".qaItem.osmose").remove();
- drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
- editOff();
- dispatch14.call("change");
- });
+ async parseIfd0Block() {
+ if (this.ifd0)
+ return;
+ let { file: e3 } = this;
+ this.findIfd0Offset(), this.ifd0Offset < 8 && g2("Malformed EXIF data"), !e3.chunked && this.ifd0Offset > e3.byteLength && g2("IFD0 offset points to outside of file.\nthis.ifd0Offset: ".concat(this.ifd0Offset, ", file.byteLength: ").concat(e3.byteLength)), e3.tiff && await e3.ensureChunk(this.ifd0Offset, S(this.options));
+ let t2 = this.parseBlock(this.ifd0Offset, "ifd0");
+ return 0 !== t2.size ? (this.exifOffset = t2.get(34665), this.interopOffset = t2.get(40965), this.gpsOffset = t2.get(34853), this.xmp = t2.get(700), this.iptc = t2.get(33723), this.icc = t2.get(34675), this.options.sanitize && (t2.delete(34665), t2.delete(40965), t2.delete(34853), t2.delete(700), t2.delete(33723), t2.delete(34675)), t2) : void 0;
}
- function updateMarkers() {
- if (!layerVisible || !_layerEnabled3)
+ async parseExifBlock() {
+ if (this.exif)
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, (d2) => d2.id);
- markers.exit().remove();
- const markersEnter = markers.enter().append("g").attr("class", (d2) => `qaItem ${d2.service} itemId-${d2.id} itemType-${d2.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", (d2) => service.getColor(d2.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", (d2) => d2.icon ? "#" + d2.icon : "");
- markers.merge(markersEnter).sort(sortY).classed("selected", (d2) => d2.id === selectedID).attr("transform", getTransform);
- if (touchLayer.empty())
+ if (this.ifd0 || await this.parseIfd0Block(), void 0 === this.exifOffset)
return;
- const fillClass = context.getDebug("target") ? "pink" : "nocolor";
- const targets = touchLayer.selectAll(".qaItem.osmose").data(data, (d2) => d2.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", (d2) => `qaItem ${d2.service} target ${fillClass} itemId-${d2.id}`).attr("transform", getTransform);
- function sortY(a2, b2) {
- return a2.id === selectedID ? 1 : b2.id === selectedID ? -1 : b2.loc[1] - a2.loc[1];
- }
+ this.file.tiff && await this.file.ensureChunk(this.exifOffset, S(this.options));
+ let e3 = this.parseBlock(this.exifOffset, "exif");
+ return this.interopOffset || (this.interopOffset = e3.get(40965)), this.makerNote = e3.get(37500), this.userComment = e3.get(37510), this.options.sanitize && (e3.delete(40965), e3.delete(37500), e3.delete(37510)), this.unpack(e3, 41728), this.unpack(e3, 41729), e3;
}
- 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() >= minZoom4) {
- editOn();
- service.loadIssues(projection2);
- updateMarkers();
- } else {
- editOff();
- }
- }
+ unpack(e3, t2) {
+ let i3 = e3.get(t2);
+ i3 && 1 === i3.length && e3.set(t2, i3[0]);
}
- 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));
- }
- }
- dispatch14.call("change");
- return this;
- };
- drawOsmose.supported = () => !!getService();
- return drawOsmose;
- }
-
- // modules/svg/streetside.js
- function svgStreetside(projection2, context, dispatch14) {
- var throttledRedraw = throttle_default(function() {
- dispatch14.call("change");
- }, 1e3);
- var minZoom4 = 14;
- var minMarkerZoom = 16;
- var minViewfieldZoom2 = 18;
- var layer = select_default2(null);
- var _viewerYaw = 0;
- var _selectedSequence = null;
- var _streetside;
- function init2() {
- if (svgStreetside.initialized)
+ async parseGpsBlock() {
+ if (this.gps)
return;
- svgStreetside.enabled = false;
- svgStreetside.initialized = true;
+ if (this.ifd0 || await this.parseIfd0Block(), void 0 === this.gpsOffset)
+ return;
+ let e3 = this.parseBlock(this.gpsOffset, "gps");
+ return e3 && e3.has(2) && e3.has(4) && (e3.set("latitude", de(...e3.get(2), e3.get(1))), e3.set("longitude", de(...e3.get(4), e3.get(3)))), e3;
}
- 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;
+ async parseInteropBlock() {
+ if (!this.interop && (this.ifd0 || await this.parseIfd0Block(), void 0 !== this.interopOffset || this.exif || await this.parseExifBlock(), void 0 !== this.interopOffset))
+ return this.parseBlock(this.interopOffset, "interop");
}
- function showLayer() {
- var service = getService();
- if (!service)
+ async parseThumbnailBlock(e3 = false) {
+ if (!this.ifd1 && !this.ifd1Parsed && (!this.options.mergeOutput || e3))
+ return this.findIfd1Offset(), this.ifd1Offset > 0 && (this.parseBlock(this.ifd1Offset, "ifd1"), this.ifd1Parsed = true), this.ifd1;
+ }
+ async extractThumbnail() {
+ if (this.headerParsed || this.parseHeader(), this.ifd1Parsed || await this.parseThumbnailBlock(true), void 0 === this.ifd1)
return;
- editOn();
- layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
- dispatch14.call("change");
- });
+ let e3 = this.ifd1.get(513), t2 = this.ifd1.get(514);
+ return this.chunk.getUint8Array(e3, t2);
}
- function hideLayer() {
- throttledRedraw.cancel();
- layer.transition().duration(250).style("opacity", 0).on("end", editOff);
+ get image() {
+ return this.ifd0;
}
- function editOn() {
- layer.style("display", "block");
+ get thumbnail() {
+ return this.ifd1;
}
- function editOff() {
- layer.selectAll(".viewfield-group").remove();
- layer.style("display", "none");
+ createOutput() {
+ let e3, t2, i3, n3 = {};
+ for (t2 of H)
+ if (e3 = this[t2], !p(e3))
+ if (i3 = this.canTranslate ? this.translateBlock(e3, t2) : Object.fromEntries(e3), this.options.mergeOutput) {
+ if ("ifd1" === t2)
+ continue;
+ Object.assign(n3, i3);
+ } else
+ n3[t2] = i3;
+ return this.makerNote && (n3.makerNote = this.makerNote), this.userComment && (n3.userComment = this.userComment), n3;
}
- function click(d3_event, d2) {
- var service = getService();
- if (!service)
- return;
- if (d2.sequenceKey !== _selectedSequence) {
- _viewerYaw = 0;
- }
- _selectedSequence = d2.sequenceKey;
- service.ensureViewerLoaded(context).then(function() {
- service.selectImage(context, d2.key).yaw(_viewerYaw).showViewer(context);
- });
- context.map().centerEase(d2.loc);
+ assignToOutput(e3, t2) {
+ if (this.globalOptions.mergeOutput)
+ Object.assign(e3, t2);
+ else
+ for (let [i3, n3] of Object.entries(t2))
+ this.assignObjectToOutput(e3, i3, n3);
}
- function mouseover(d3_event, d2) {
- var service = getService();
- if (service)
- service.setStyles(context, d2);
+ };
+ function de(e3, t2, i3, n3) {
+ var s2 = e3 + t2 / 60 + i3 / 3600;
+ return "S" !== n3 && "W" !== n3 || (s2 *= -1), s2;
+ }
+ c(fe, "type", "tiff"), c(fe, "headerLength", 10), T.set("tiff", fe);
+ var pe = Object.freeze({ __proto__: null, default: ne, Exifr: te, fileParsers: w, segmentParsers: T, fileReaders: A, tagKeys: E, tagValues: B, tagRevivers: N, createDictionary: U, extendDictionary: F, fetchUrlAsArrayBuffer: M, readBlobAsArrayBuffer: R, chunkedProps: G, otherSegments: V, segments: z, tiffBlocks: H, segmentsAndBlocks: j, tiffExtractables: W, inheritables: K, allFormatters: X, Options: q, parse: ie });
+ var ge = { ifd0: false, ifd1: false, exif: false, gps: false, interop: false, sanitize: false, reviveValues: true, translateKeys: false, translateValues: false, mergeOutput: false };
+ var me = Object.assign({}, ge, { firstChunkSize: 4e4, gps: [1, 2, 3, 4] });
+ async function Se(e3) {
+ let t2 = new te(me);
+ await t2.read(e3);
+ let i3 = await t2.parse();
+ if (i3 && i3.gps) {
+ let { latitude: e4, longitude: t3 } = i3.gps;
+ return { latitude: e4, longitude: t3 };
}
- function mouseout() {
- var service = getService();
- if (service)
- service.setStyles(context, null);
+ }
+ var Ce = Object.assign({}, ge, { tiff: false, ifd1: true, mergeOutput: false });
+ async function ye(e3) {
+ let t2 = new te(Ce);
+ await t2.read(e3);
+ let i3 = await t2.extractThumbnail();
+ return i3 && a ? s.from(i3) : i3;
+ }
+ async function be(e3) {
+ let t2 = await this.thumbnail(e3);
+ if (void 0 !== t2) {
+ let e4 = new Blob([t2]);
+ return URL.createObjectURL(e4);
}
- function transform2(d2) {
- var t2 = svgPointTransform(projection2)(d2);
- var rot = d2.ca + _viewerYaw;
- if (rot) {
- t2 += " rotate(" + Math.floor(rot) + ",0,0)";
+ }
+ var Ie = Object.assign({}, ge, { firstChunkSize: 4e4, ifd0: [274] });
+ async function Pe(e3) {
+ let t2 = new te(Ie);
+ await t2.read(e3);
+ let i3 = await t2.parse();
+ if (i3 && i3.ifd0)
+ return i3.ifd0[274];
+ }
+ var ke = Object.freeze({ 1: { dimensionSwapped: false, scaleX: 1, scaleY: 1, deg: 0, rad: 0 }, 2: { dimensionSwapped: false, scaleX: -1, scaleY: 1, deg: 0, rad: 0 }, 3: { dimensionSwapped: false, scaleX: 1, scaleY: 1, deg: 180, rad: 180 * Math.PI / 180 }, 4: { dimensionSwapped: false, scaleX: -1, scaleY: 1, deg: 180, rad: 180 * Math.PI / 180 }, 5: { dimensionSwapped: true, scaleX: 1, scaleY: -1, deg: 90, rad: 90 * Math.PI / 180 }, 6: { dimensionSwapped: true, scaleX: 1, scaleY: 1, deg: 90, rad: 90 * Math.PI / 180 }, 7: { dimensionSwapped: true, scaleX: 1, scaleY: -1, deg: 270, rad: 270 * Math.PI / 180 }, 8: { dimensionSwapped: true, scaleX: 1, scaleY: 1, deg: 270, rad: 270 * Math.PI / 180 } });
+ var we = true;
+ var Te = true;
+ if ("object" == typeof navigator) {
+ let e3 = navigator.userAgent;
+ if (e3.includes("iPad") || e3.includes("iPhone")) {
+ let t2 = e3.match(/OS (\d+)_(\d+)/);
+ if (t2) {
+ let [, e4, i3] = t2, n3 = Number(e4) + 0.1 * Number(i3);
+ we = n3 < 13.4, Te = false;
}
- return t2;
- }
- 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);
+ } else if (e3.includes("OS X 10")) {
+ let [, t2] = e3.match(/OS X 10[_.](\d+)/);
+ we = Te = Number(t2) < 15;
}
- 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;
+ if (e3.includes("Chrome/")) {
+ let [, t2] = e3.match(/Chrome\/(\d+)/);
+ we = Te = Number(t2) < 81;
+ } else if (e3.includes("Firefox/")) {
+ let [, t2] = e3.match(/Firefox\/(\d+)/);
+ we = Te = Number(t2) < 77;
}
- 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;
+ }
+ async function Ae(e3) {
+ let t2 = await Pe(e3);
+ return Object.assign({ canvas: we, css: Te }, ke[t2]);
+ }
+ var De = class extends I {
+ constructor(...e3) {
+ super(...e3), c(this, "ranges", new Oe()), 0 !== this.byteLength && this.ranges.add(0, this.byteLength);
}
- function update() {
- var viewer = context.container().select(".photoviewer");
- var selected = viewer.empty() ? void 0 : viewer.datum();
- var z2 = ~~context.map().zoom();
- var showMarkers = z2 >= minMarkerZoom;
- var showViewfields = z2 >= minViewfieldZoom2;
- 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(d2) {
- return d2.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(d2) {
- return d2.key + (d2.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(a2, b2) {
- return a2 === selected ? 1 : b2 === selected ? -1 : b2.loc[1] - a2.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 d2 = this.parentNode.__data__;
- if (d2.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";
+ _tryExtend(e3, t2, i3) {
+ if (0 === e3 && 0 === this.byteLength && i3) {
+ let e4 = new DataView(i3.buffer || i3, i3.byteOffset, i3.byteLength);
+ this._swapDataView(e4);
+ } else {
+ let i4 = e3 + t2;
+ if (i4 > this.byteLength) {
+ let { dataView: e4 } = this._extend(i4);
+ this._swapDataView(e4);
}
}
}
- 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() >= minZoom4) {
- editOn();
- update();
- service.loadBubbles(projection2);
- } else {
- editOff();
- }
- }
+ _extend(e3) {
+ let t2;
+ t2 = a ? s.allocUnsafe(e3) : new Uint8Array(e3);
+ let i3 = new DataView(t2.buffer, t2.byteOffset, t2.byteLength);
+ return t2.set(new Uint8Array(this.buffer, this.byteOffset, this.byteLength), 0), { uintView: t2, dataView: i3 };
}
- drawImages.enabled = function(_2) {
- if (!arguments.length)
- return svgStreetside.enabled;
- svgStreetside.enabled = _2;
- if (svgStreetside.enabled) {
- showLayer();
- context.photos().on("change.streetside", update);
- } else {
- hideLayer();
- context.photos().on("change.streetside", null);
- }
- dispatch14.call("change");
- return this;
- };
- drawImages.supported = function() {
- return !!getService();
- };
- drawImages.rendered = function(zoom) {
- return zoom >= minZoom4;
- };
- init2();
- return drawImages;
+ subarray(e3, t2, i3 = false) {
+ return t2 = t2 || this._lengthToEnd(e3), i3 && this._tryExtend(e3, t2), this.ranges.add(e3, t2), super.subarray(e3, t2);
+ }
+ set(e3, t2, i3 = false) {
+ i3 && this._tryExtend(t2, e3.byteLength, e3);
+ let n3 = super.set(e3, t2);
+ return this.ranges.add(t2, n3.byteLength), n3;
+ }
+ async ensureChunk(e3, t2) {
+ this.chunked && (this.ranges.available(e3, t2) || await this.readChunk(e3, t2));
+ }
+ available(e3, t2) {
+ return this.ranges.available(e3, t2);
+ }
+ };
+ var Oe = class {
+ constructor() {
+ c(this, "list", []);
+ }
+ get length() {
+ return this.list.length;
+ }
+ add(e3, t2, i3 = 0) {
+ let n3 = e3 + t2, s2 = this.list.filter((t3) => xe(e3, t3.offset, n3) || xe(e3, t3.end, n3));
+ if (s2.length > 0) {
+ e3 = Math.min(e3, ...s2.map((e4) => e4.offset)), n3 = Math.max(n3, ...s2.map((e4) => e4.end)), t2 = n3 - e3;
+ let i4 = s2.shift();
+ i4.offset = e3, i4.length = t2, i4.end = n3, this.list = this.list.filter((e4) => !s2.includes(e4));
+ } else
+ this.list.push({ offset: e3, length: t2, end: n3 });
+ }
+ available(e3, t2) {
+ let i3 = e3 + t2;
+ return this.list.some((t3) => t3.offset <= e3 && i3 <= t3.end);
+ }
+ };
+ function xe(e3, t2, i3) {
+ return e3 <= t2 && t2 <= i3;
}
-
- // modules/svg/vegbilder.js
- function svgVegbilder(projection2, context, dispatch14) {
- const throttledRedraw = throttle_default(() => dispatch14.call("change"), 1e3);
- const minZoom4 = 14;
- const minMarkerZoom = 16;
- const minViewfieldZoom2 = 18;
- let layer = select_default2(null);
- let _viewerYaw = 0;
- let _vegbilder;
- function init2() {
- if (svgVegbilder.initialized)
- return;
- svgVegbilder.enabled = false;
- svgVegbilder.initialized = true;
+ var ve = class extends De {
+ constructor(e3, t2) {
+ super(0), c(this, "chunksRead", 0), this.input = e3, this.options = t2;
+ }
+ async readWhole() {
+ this.chunked = false, await this.readChunk(this.nextChunkOffset);
+ }
+ async readChunked() {
+ this.chunked = true, await this.readChunk(0, this.options.firstChunkSize);
}
- function getService() {
- if (services.vegbilder && !_vegbilder) {
- _vegbilder = services.vegbilder;
- _vegbilder.event.on("viewerChanged.svgVegbilder", viewerChanged).on("loadedImages.svgVegbilder", throttledRedraw);
- } else if (!services.vegbilder && _vegbilder) {
- _vegbilder = null;
- }
- return _vegbilder;
+ async readNextChunk(e3 = this.nextChunkOffset) {
+ if (this.fullyRead)
+ return this.chunksRead++, false;
+ let t2 = this.options.chunkSize, i3 = await this.readChunk(e3, t2);
+ return !!i3 && i3.byteLength === t2;
}
- function showLayer() {
- const service = getService();
- if (!service)
- return;
- editOn();
- layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", () => dispatch14.call("change"));
+ async readChunk(e3, t2) {
+ if (this.chunksRead++, 0 !== (t2 = this.safeWrapAddress(e3, t2)))
+ return this._readChunk(e3, t2);
}
- function hideLayer() {
- throttledRedraw.cancel();
- layer.transition().duration(250).style("opacity", 0).on("end", editOff);
+ safeWrapAddress(e3, t2) {
+ return void 0 !== this.size && e3 + t2 > this.size ? Math.max(0, this.size - e3) : t2;
}
- function editOn() {
- layer.style("display", "block");
+ get nextChunkOffset() {
+ if (0 !== this.ranges.list.length)
+ return this.ranges.list[0].length;
}
- function editOff() {
- layer.selectAll(".viewfield-group").remove();
- layer.style("display", "none");
+ get canReadNextChunk() {
+ return this.chunksRead < this.options.chunkLimit;
}
- function click(d3_event, d2) {
- const service = getService();
- if (!service)
- return;
- service.ensureViewerLoaded(context).then(() => {
- service.selectImage(context, d2.key).showViewer(context);
- });
- context.map().centerEase(d2.loc);
+ get fullyRead() {
+ return void 0 !== this.size && this.nextChunkOffset === this.size;
}
- function mouseover(d3_event, d2) {
- const service = getService();
- if (service)
- service.setStyles(context, d2);
+ read() {
+ return this.options.chunked ? this.readChunked() : this.readWhole();
}
- function mouseout() {
- const service = getService();
- if (service)
- service.setStyles(context, null);
+ close() {
}
- function transform2(d2, selected) {
- let t2 = svgPointTransform(projection2)(d2);
- let rot = d2.ca;
- if (d2 === selected) {
- rot += _viewerYaw;
- }
- if (rot) {
- t2 += " rotate(" + Math.floor(rot) + ",0,0)";
+ };
+ A.set("blob", class extends ve {
+ async readWhole() {
+ this.chunked = false;
+ let e3 = await R(this.input);
+ this._swapArrayBuffer(e3);
+ }
+ readChunked() {
+ return this.chunked = true, this.size = this.input.size, super.readChunked();
+ }
+ async _readChunk(e3, t2) {
+ let i3 = t2 ? e3 + t2 : void 0, n3 = this.input.slice(e3, i3), s2 = await R(n3);
+ return this.set(s2, e3, true);
+ }
+ });
+ var Me = Object.freeze({ __proto__: null, default: pe, Exifr: te, fileParsers: w, segmentParsers: T, fileReaders: A, tagKeys: E, tagValues: B, tagRevivers: N, createDictionary: U, extendDictionary: F, fetchUrlAsArrayBuffer: M, readBlobAsArrayBuffer: R, chunkedProps: G, otherSegments: V, segments: z, tiffBlocks: H, segmentsAndBlocks: j, tiffExtractables: W, inheritables: K, allFormatters: X, Options: q, parse: ie, gpsOnlyOptions: me, gps: Se, thumbnailOnlyOptions: Ce, thumbnail: ye, thumbnailUrl: be, orientationOnlyOptions: Ie, orientation: Pe, rotations: ke, get rotateCanvas() {
+ return we;
+ }, get rotateCss() {
+ return Te;
+ }, rotation: Ae });
+ A.set("url", class extends ve {
+ async readWhole() {
+ this.chunked = false;
+ let e3 = await M(this.input);
+ e3 instanceof ArrayBuffer ? this._swapArrayBuffer(e3) : e3 instanceof Uint8Array && this._swapBuffer(e3);
+ }
+ async _readChunk(e3, t2) {
+ let i3 = t2 ? e3 + t2 - 1 : void 0, n3 = this.options.httpHeaders || {};
+ (e3 || i3) && (n3.range = "bytes=".concat([e3, i3].join("-")));
+ let s2 = await h(this.input, { headers: n3 }), r2 = await s2.arrayBuffer(), a2 = r2.byteLength;
+ if (416 !== s2.status)
+ return a2 !== t2 && (this.size = e3 + a2), this.set(r2, e3, true);
+ }
+ });
+ I.prototype.getUint64 = function(e3) {
+ let t2 = this.getUint32(e3), i3 = this.getUint32(e3 + 4);
+ return t2 < 1048575 ? t2 << 32 | i3 : void 0 !== typeof r ? (console.warn("Using BigInt because of type 64uint but JS can only handle 53b numbers."), r(t2) << r(32) | r(i3)) : void g2("Trying to read 64b value but JS can only handle 53b numbers.");
+ };
+ var Re = class extends se {
+ parseBoxes(e3 = 0) {
+ let t2 = [];
+ for (; e3 < this.file.byteLength - 4; ) {
+ let i3 = this.parseBoxHead(e3);
+ if (t2.push(i3), 0 === i3.length)
+ break;
+ e3 += i3.length;
}
return t2;
}
- function viewerChanged() {
- const service = getService();
- if (!service)
+ parseSubBoxes(e3) {
+ e3.boxes = this.parseBoxes(e3.start);
+ }
+ findBox(e3, t2) {
+ return void 0 === e3.boxes && this.parseSubBoxes(e3), e3.boxes.find((e4) => e4.kind === t2);
+ }
+ parseBoxHead(e3) {
+ let t2 = this.file.getUint32(e3), i3 = this.file.getString(e3 + 4, 4), n3 = e3 + 8;
+ return 1 === t2 && (t2 = this.file.getUint64(e3 + 8), n3 += 8), { offset: e3, length: t2, kind: i3, start: n3 };
+ }
+ parseBoxFullHead(e3) {
+ if (void 0 !== e3.version)
return;
- const frame2 = service.photoFrame();
- _viewerYaw = frame2.getYaw();
- if (context.map().isTransformed())
+ let t2 = this.file.getUint32(e3.start);
+ e3.version = t2 >> 24, e3.start += 4;
+ }
+ };
+ var Le = class extends Re {
+ static canHandle(e3, t2) {
+ if (0 !== t2)
+ return false;
+ let i3 = e3.getUint16(2);
+ if (i3 > 50)
+ return false;
+ let n3 = 16, s2 = [];
+ for (; n3 < i3; )
+ s2.push(e3.getString(n3, 4)), n3 += 4;
+ return s2.includes(this.type);
+ }
+ async parse() {
+ let e3 = this.file.getUint32(0), t2 = this.parseBoxHead(e3);
+ for (; "meta" !== t2.kind; )
+ e3 += t2.length, await this.file.ensureChunk(e3, 16), t2 = this.parseBoxHead(e3);
+ await this.file.ensureChunk(t2.offset, t2.length), this.parseBoxFullHead(t2), this.parseSubBoxes(t2), this.options.icc.enabled && await this.findIcc(t2), this.options.tiff.enabled && await this.findExif(t2);
+ }
+ async registerSegment(e3, t2, i3) {
+ await this.file.ensureChunk(t2, i3);
+ let n3 = this.file.subarray(t2, i3);
+ this.createParser(e3, n3);
+ }
+ async findIcc(e3) {
+ let t2 = this.findBox(e3, "iprp");
+ if (void 0 === t2)
return;
- layer.selectAll(".viewfield-group.currentView").attr("transform", (d2) => transform2(d2, d2));
+ let i3 = this.findBox(t2, "ipco");
+ if (void 0 === i3)
+ return;
+ let n3 = this.findBox(i3, "colr");
+ void 0 !== n3 && await this.registerSegment("icc", n3.offset + 12, n3.length);
}
- function filterImages(images) {
- const photoContext = context.photos();
- const fromDateString = photoContext.fromDate();
- const toDateString = photoContext.toDate();
- const showsFlat = photoContext.showsFlat();
- const showsPano = photoContext.showsPanoramic();
- if (fromDateString) {
- const fromDate = new Date(fromDateString);
- images = images.filter((image) => image.captured_at.getTime() >= fromDate.getTime());
- }
- if (toDateString) {
- const toDate = new Date(toDateString);
- images = images.filter((image) => image.captured_at.getTime() <= toDate.getTime());
- }
- if (!showsPano) {
- images = images.filter((image) => !image.is_sphere);
- }
- if (!showsFlat) {
- images = images.filter((image) => image.is_sphere);
- }
- return images;
+ async findExif(e3) {
+ let t2 = this.findBox(e3, "iinf");
+ if (void 0 === t2)
+ return;
+ let i3 = this.findBox(e3, "iloc");
+ if (void 0 === i3)
+ return;
+ let n3 = this.findExifLocIdInIinf(t2), s2 = this.findExtentInIloc(i3, n3);
+ if (void 0 === s2)
+ return;
+ let [r2, a2] = s2;
+ await this.file.ensureChunk(r2, a2);
+ let o2 = 4 + this.file.getUint32(r2);
+ r2 += o2, a2 -= o2, await this.registerSegment("tiff", r2, a2);
}
- function filterSequences(sequences) {
- const photoContext = context.photos();
- const fromDateString = photoContext.fromDate();
- const toDateString = photoContext.toDate();
- const showsFlat = photoContext.showsFlat();
- const showsPano = photoContext.showsPanoramic();
- if (fromDateString) {
- const fromDate = new Date(fromDateString);
- sequences = sequences.filter(({ images }) => images[0].captured_at.getTime() >= fromDate.getTime());
- }
- if (toDateString) {
- const toDate = new Date(toDateString);
- sequences = sequences.filter(({ images }) => images[images.length - 1].captured_at.getTime() <= toDate.getTime());
- }
- if (!showsPano) {
- sequences = sequences.filter(({ images }) => !images[0].is_sphere);
- }
- if (!showsFlat) {
- sequences = sequences.filter(({ images }) => images[0].is_sphere);
+ findExifLocIdInIinf(e3) {
+ this.parseBoxFullHead(e3);
+ let t2, i3, n3, s2, r2 = e3.start, a2 = this.file.getUint16(r2);
+ for (r2 += 2; a2--; ) {
+ if (t2 = this.parseBoxHead(r2), this.parseBoxFullHead(t2), i3 = t2.start, t2.version >= 2 && (n3 = 3 === t2.version ? 4 : 2, s2 = this.file.getString(i3 + n3 + 2, 4), "Exif" === s2))
+ return this.file.getUintBytes(i3, n3);
+ r2 += t2.length;
}
- return sequences;
}
- function update() {
- const viewer = context.container().select(".photoviewer");
- const selected = viewer.empty() ? void 0 : viewer.datum();
- const z2 = ~~context.map().zoom();
- const showMarkers = z2 >= minMarkerZoom;
- const showViewfields = z2 >= minViewfieldZoom2;
- const service = getService();
- let sequences = [];
- let images = [];
- if (service) {
- service.loadImages(context);
- sequences = service.sequences(projection2);
- images = showMarkers ? service.images(projection2) : [];
- images = filterImages(images);
- sequences = filterSequences(sequences);
- }
- let traces = layer.selectAll(".sequences").selectAll(".sequence").data(sequences, (d2) => d2.key);
- traces.exit().remove();
- traces.enter().append("path").attr("class", "sequence").merge(traces).attr("d", svgPath(projection2).geojson);
- const groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(images, (d2) => d2.key);
- 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((a2, b2) => {
- return a2 === selected ? 1 : b2 === selected ? -1 : b2.loc[1] - a2.loc[1];
- }).attr("transform", (d2) => transform2(d2, selected)).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", viewfieldPath);
- function viewfieldPath() {
- const d2 = this.parentNode.__data__;
- if (d2.is_sphere) {
- 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";
- }
- }
+ get8bits(e3) {
+ let t2 = this.file.getUint8(e3);
+ return [t2 >> 4, 15 & t2];
}
- function drawImages(selection2) {
- const enabled = svgVegbilder.enabled;
- const service = getService();
- layer = selection2.selectAll(".layer-vegbilder").data(service ? [0] : []);
- layer.exit().remove();
- const layerEnter = layer.enter().append("g").attr("class", "layer-vegbilder").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() >= minZoom4) {
- editOn();
- update();
- service.loadImages(context);
- } else {
- editOff();
- }
+ findExtentInIloc(e3, t2) {
+ this.parseBoxFullHead(e3);
+ let i3 = e3.start, [n3, s2] = this.get8bits(i3++), [r2, a2] = this.get8bits(i3++), o2 = 2 === e3.version ? 4 : 2, l2 = 1 === e3.version || 2 === e3.version ? 2 : 0, h2 = a2 + n3 + s2, u2 = 2 === e3.version ? 4 : 2, c2 = this.file.getUintBytes(i3, u2);
+ for (i3 += u2; c2--; ) {
+ let e4 = this.file.getUintBytes(i3, o2);
+ i3 += o2 + l2 + 2 + r2;
+ let u3 = this.file.getUint16(i3);
+ if (i3 += 2, e4 === t2)
+ return u3 > 1 && console.warn("ILOC box has more than one extent but we're only processing one\nPlease create an issue at https://github.com/MikeKovarik/exifr with this file"), [this.file.getUintBytes(i3 + a2, n3), this.file.getUintBytes(i3 + a2 + n3, s2)];
+ i3 += u3 * h2;
}
}
- drawImages.enabled = function(_2) {
- if (!arguments.length)
- return svgVegbilder.enabled;
- svgVegbilder.enabled = _2;
- if (svgVegbilder.enabled) {
- showLayer();
- context.photos().on("change.vegbilder", update);
- } else {
- hideLayer();
- context.photos().on("change.vegbilder", null);
- }
- dispatch14.call("change");
- return this;
- };
- drawImages.supported = function() {
- return !!getService();
- };
- drawImages.rendered = function(zoom) {
- return zoom >= minZoom4;
- };
- drawImages.validHere = function(extent, zoom) {
- return zoom >= minZoom4 - 2 && getService().validHere(extent);
- };
- init2();
- return drawImages;
+ };
+ var Ue = class extends Le {
+ };
+ c(Ue, "type", "heic");
+ var Fe = class extends Le {
+ };
+ c(Fe, "type", "avif"), w.set("heic", Ue), w.set("avif", Fe), U(E, ["ifd0", "ifd1"], [[256, "ImageWidth"], [257, "ImageHeight"], [258, "BitsPerSample"], [259, "Compression"], [262, "PhotometricInterpretation"], [270, "ImageDescription"], [271, "Make"], [272, "Model"], [273, "StripOffsets"], [274, "Orientation"], [277, "SamplesPerPixel"], [278, "RowsPerStrip"], [279, "StripByteCounts"], [282, "XResolution"], [283, "YResolution"], [284, "PlanarConfiguration"], [296, "ResolutionUnit"], [301, "TransferFunction"], [305, "Software"], [306, "ModifyDate"], [315, "Artist"], [316, "HostComputer"], [317, "Predictor"], [318, "WhitePoint"], [319, "PrimaryChromaticities"], [513, "ThumbnailOffset"], [514, "ThumbnailLength"], [529, "YCbCrCoefficients"], [530, "YCbCrSubSampling"], [531, "YCbCrPositioning"], [532, "ReferenceBlackWhite"], [700, "ApplicationNotes"], [33432, "Copyright"], [33723, "IPTC"], [34665, "ExifIFD"], [34675, "ICC"], [34853, "GpsIFD"], [330, "SubIFD"], [40965, "InteropIFD"], [40091, "XPTitle"], [40092, "XPComment"], [40093, "XPAuthor"], [40094, "XPKeywords"], [40095, "XPSubject"]]), U(E, "exif", [[33434, "ExposureTime"], [33437, "FNumber"], [34850, "ExposureProgram"], [34852, "SpectralSensitivity"], [34855, "ISO"], [34858, "TimeZoneOffset"], [34859, "SelfTimerMode"], [34864, "SensitivityType"], [34865, "StandardOutputSensitivity"], [34866, "RecommendedExposureIndex"], [34867, "ISOSpeed"], [34868, "ISOSpeedLatitudeyyy"], [34869, "ISOSpeedLatitudezzz"], [36864, "ExifVersion"], [36867, "DateTimeOriginal"], [36868, "CreateDate"], [36873, "GooglePlusUploadCode"], [36880, "OffsetTime"], [36881, "OffsetTimeOriginal"], [36882, "OffsetTimeDigitized"], [37121, "ComponentsConfiguration"], [37122, "CompressedBitsPerPixel"], [37377, "ShutterSpeedValue"], [37378, "ApertureValue"], [37379, "BrightnessValue"], [37380, "ExposureCompensation"], [37381, "MaxApertureValue"], [37382, "SubjectDistance"], [37383, "MeteringMode"], [37384, "LightSource"], [37385, "Flash"], [37386, "FocalLength"], [37393, "ImageNumber"], [37394, "SecurityClassification"], [37395, "ImageHistory"], [37396, "SubjectArea"], [37500, "MakerNote"], [37510, "UserComment"], [37520, "SubSecTime"], [37521, "SubSecTimeOriginal"], [37522, "SubSecTimeDigitized"], [37888, "AmbientTemperature"], [37889, "Humidity"], [37890, "Pressure"], [37891, "WaterDepth"], [37892, "Acceleration"], [37893, "CameraElevationAngle"], [40960, "FlashpixVersion"], [40961, "ColorSpace"], [40962, "ExifImageWidth"], [40963, "ExifImageHeight"], [40964, "RelatedSoundFile"], [41483, "FlashEnergy"], [41486, "FocalPlaneXResolution"], [41487, "FocalPlaneYResolution"], [41488, "FocalPlaneResolutionUnit"], [41492, "SubjectLocation"], [41493, "ExposureIndex"], [41495, "SensingMethod"], [41728, "FileSource"], [41729, "SceneType"], [41730, "CFAPattern"], [41985, "CustomRendered"], [41986, "ExposureMode"], [41987, "WhiteBalance"], [41988, "DigitalZoomRatio"], [41989, "FocalLengthIn35mmFormat"], [41990, "SceneCaptureType"], [41991, "GainControl"], [41992, "Contrast"], [41993, "Saturation"], [41994, "Sharpness"], [41996, "SubjectDistanceRange"], [42016, "ImageUniqueID"], [42032, "OwnerName"], [42033, "SerialNumber"], [42034, "LensInfo"], [42035, "LensMake"], [42036, "LensModel"], [42037, "LensSerialNumber"], [42080, "CompositeImage"], [42081, "CompositeImageCount"], [42082, "CompositeImageExposureTimes"], [42240, "Gamma"], [59932, "Padding"], [59933, "OffsetSchema"], [65e3, "OwnerName"], [65001, "SerialNumber"], [65002, "Lens"], [65100, "RawFile"], [65101, "Converter"], [65102, "WhiteBalance"], [65105, "Exposure"], [65106, "Shadows"], [65107, "Brightness"], [65108, "Contrast"], [65109, "Saturation"], [65110, "Sharpness"], [65111, "Smoothness"], [65112, "MoireFilter"], [40965, "InteropIFD"]]), U(E, "gps", [[0, "GPSVersionID"], [1, "GPSLatitudeRef"], [2, "GPSLatitude"], [3, "GPSLongitudeRef"], [4, "GPSLongitude"], [5, "GPSAltitudeRef"], [6, "GPSAltitude"], [7, "GPSTimeStamp"], [8, "GPSSatellites"], [9, "GPSStatus"], [10, "GPSMeasureMode"], [11, "GPSDOP"], [12, "GPSSpeedRef"], [13, "GPSSpeed"], [14, "GPSTrackRef"], [15, "GPSTrack"], [16, "GPSImgDirectionRef"], [17, "GPSImgDirection"], [18, "GPSMapDatum"], [19, "GPSDestLatitudeRef"], [20, "GPSDestLatitude"], [21, "GPSDestLongitudeRef"], [22, "GPSDestLongitude"], [23, "GPSDestBearingRef"], [24, "GPSDestBearing"], [25, "GPSDestDistanceRef"], [26, "GPSDestDistance"], [27, "GPSProcessingMethod"], [28, "GPSAreaInformation"], [29, "GPSDateStamp"], [30, "GPSDifferential"], [31, "GPSHPositioningError"]]), U(B, ["ifd0", "ifd1"], [[274, { 1: "Horizontal (normal)", 2: "Mirror horizontal", 3: "Rotate 180", 4: "Mirror vertical", 5: "Mirror horizontal and rotate 270 CW", 6: "Rotate 90 CW", 7: "Mirror horizontal and rotate 90 CW", 8: "Rotate 270 CW" }], [296, { 1: "None", 2: "inches", 3: "cm" }]]);
+ var Ee = U(B, "exif", [[34850, { 0: "Not defined", 1: "Manual", 2: "Normal program", 3: "Aperture priority", 4: "Shutter priority", 5: "Creative program", 6: "Action program", 7: "Portrait mode", 8: "Landscape mode" }], [37121, { 0: "-", 1: "Y", 2: "Cb", 3: "Cr", 4: "R", 5: "G", 6: "B" }], [37383, { 0: "Unknown", 1: "Average", 2: "CenterWeightedAverage", 3: "Spot", 4: "MultiSpot", 5: "Pattern", 6: "Partial", 255: "Other" }], [37384, { 0: "Unknown", 1: "Daylight", 2: "Fluorescent", 3: "Tungsten (incandescent light)", 4: "Flash", 9: "Fine weather", 10: "Cloudy weather", 11: "Shade", 12: "Daylight fluorescent (D 5700 - 7100K)", 13: "Day white fluorescent (N 4600 - 5400K)", 14: "Cool white fluorescent (W 3900 - 4500K)", 15: "White fluorescent (WW 3200 - 3700K)", 17: "Standard light A", 18: "Standard light B", 19: "Standard light C", 20: "D55", 21: "D65", 22: "D75", 23: "D50", 24: "ISO studio tungsten", 255: "Other" }], [37385, { 0: "Flash did not fire", 1: "Flash fired", 5: "Strobe return light not detected", 7: "Strobe return light detected", 9: "Flash fired, compulsory flash mode", 13: "Flash fired, compulsory flash mode, return light not detected", 15: "Flash fired, compulsory flash mode, return light detected", 16: "Flash did not fire, compulsory flash mode", 24: "Flash did not fire, auto mode", 25: "Flash fired, auto mode", 29: "Flash fired, auto mode, return light not detected", 31: "Flash fired, auto mode, return light detected", 32: "No flash function", 65: "Flash fired, red-eye reduction mode", 69: "Flash fired, red-eye reduction mode, return light not detected", 71: "Flash fired, red-eye reduction mode, return light detected", 73: "Flash fired, compulsory flash mode, red-eye reduction mode", 77: "Flash fired, compulsory flash mode, red-eye reduction mode, return light not detected", 79: "Flash fired, compulsory flash mode, red-eye reduction mode, return light detected", 89: "Flash fired, auto mode, red-eye reduction mode", 93: "Flash fired, auto mode, return light not detected, red-eye reduction mode", 95: "Flash fired, auto mode, return light detected, red-eye reduction mode" }], [41495, { 1: "Not defined", 2: "One-chip color area sensor", 3: "Two-chip color area sensor", 4: "Three-chip color area sensor", 5: "Color sequential area sensor", 7: "Trilinear sensor", 8: "Color sequential linear sensor" }], [41728, { 1: "Film Scanner", 2: "Reflection Print Scanner", 3: "Digital Camera" }], [41729, { 1: "Directly photographed" }], [41985, { 0: "Normal", 1: "Custom", 2: "HDR (no original saved)", 3: "HDR (original saved)", 4: "Original (for HDR)", 6: "Panorama", 7: "Portrait HDR", 8: "Portrait" }], [41986, { 0: "Auto", 1: "Manual", 2: "Auto bracket" }], [41987, { 0: "Auto", 1: "Manual" }], [41990, { 0: "Standard", 1: "Landscape", 2: "Portrait", 3: "Night", 4: "Other" }], [41991, { 0: "None", 1: "Low gain up", 2: "High gain up", 3: "Low gain down", 4: "High gain down" }], [41996, { 0: "Unknown", 1: "Macro", 2: "Close", 3: "Distant" }], [42080, { 0: "Unknown", 1: "Not a Composite Image", 2: "General Composite Image", 3: "Composite Image Captured While Shooting" }]]);
+ var Be = { 1: "No absolute unit of measurement", 2: "Inch", 3: "Centimeter" };
+ Ee.set(37392, Be), Ee.set(41488, Be);
+ var Ne = { 0: "Normal", 1: "Low", 2: "High" };
+ function Ge(e3) {
+ return "object" == typeof e3 && void 0 !== e3.length ? e3[0] : e3;
}
-
- // modules/svg/mapillary_images.js
- function svgMapillaryImages(projection2, context, dispatch14) {
- const throttledRedraw = throttle_default(function() {
- dispatch14.call("change");
- }, 1e3);
- const minZoom4 = 12;
- const minMarkerZoom = 16;
- const minViewfieldZoom2 = 18;
- let layer = select_default2(null);
- let _mapillary;
- function init2() {
- if (svgMapillaryImages.initialized)
- return;
- svgMapillaryImages.enabled = false;
- svgMapillaryImages.initialized = true;
+ function Ve(e3) {
+ let t2 = Array.from(e3).slice(1);
+ return t2[1] > 15 && (t2 = t2.map((e4) => String.fromCharCode(e4))), "0" !== t2[2] && 0 !== t2[2] || t2.pop(), t2.join(".");
+ }
+ function ze(e3) {
+ if ("string" == typeof e3) {
+ var [t2, i3, n3, s2, r2, a2] = e3.trim().split(/[-: ]/g).map(Number), o2 = new Date(t2, i3 - 1, n3);
+ return Number.isNaN(s2) || Number.isNaN(r2) || Number.isNaN(a2) || (o2.setHours(s2), o2.setMinutes(r2), o2.setSeconds(a2)), Number.isNaN(+o2) ? e3 : o2;
}
- function getService() {
- if (services.mapillary && !_mapillary) {
- _mapillary = services.mapillary;
- _mapillary.event.on("loadedImages", throttledRedraw);
- } else if (!services.mapillary && _mapillary) {
- _mapillary = null;
- }
- return _mapillary;
+ }
+ function He(e3) {
+ if ("string" == typeof e3)
+ return e3;
+ let t2 = [];
+ if (0 === e3[1] && 0 === e3[e3.length - 1])
+ for (let i3 = 0; i3 < e3.length; i3 += 2)
+ t2.push(je(e3[i3 + 1], e3[i3]));
+ else
+ for (let i3 = 0; i3 < e3.length; i3 += 2)
+ t2.push(je(e3[i3], e3[i3 + 1]));
+ return m(String.fromCodePoint(...t2));
+ }
+ function je(e3, t2) {
+ return e3 << 8 | t2;
+ }
+ Ee.set(41992, Ne), Ee.set(41993, Ne), Ee.set(41994, Ne), U(N, ["ifd0", "ifd1"], [[50827, function(e3) {
+ return "string" != typeof e3 ? b(e3) : e3;
+ }], [306, ze], [40091, He], [40092, He], [40093, He], [40094, He], [40095, He]]), U(N, "exif", [[40960, Ve], [36864, Ve], [36867, ze], [36868, ze], [40962, Ge], [40963, Ge]]), U(N, "gps", [[0, (e3) => Array.from(e3).join(".")], [7, (e3) => Array.from(e3).join(":")]]);
+ var We = class extends re2 {
+ static canHandle(e3, t2) {
+ return 225 === e3.getUint8(t2 + 1) && 1752462448 === e3.getUint32(t2 + 4) && "http://ns.adobe.com/" === e3.getString(t2 + 4, "http://ns.adobe.com/".length);
}
- function showLayer() {
- const service = getService();
- if (!service)
- return;
- editOn();
- layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
- dispatch14.call("change");
- });
+ static headerLength(e3, t2) {
+ return "http://ns.adobe.com/xmp/extension/" === e3.getString(t2 + 4, "http://ns.adobe.com/xmp/extension/".length) ? 79 : 4 + "http://ns.adobe.com/xap/1.0/".length + 1;
}
- function hideLayer() {
- throttledRedraw.cancel();
- layer.transition().duration(250).style("opacity", 0).on("end", editOff);
+ static findPosition(e3, t2) {
+ let i3 = super.findPosition(e3, t2);
+ return i3.multiSegment = i3.extended = 79 === i3.headerLength, i3.multiSegment ? (i3.chunkCount = e3.getUint8(t2 + 72), i3.chunkNumber = e3.getUint8(t2 + 76), 0 !== e3.getUint8(t2 + 77) && i3.chunkNumber++) : (i3.chunkCount = 1 / 0, i3.chunkNumber = -1), i3;
}
- function editOn() {
- layer.style("display", "block");
+ static handleMultiSegments(e3) {
+ return e3.map((e4) => e4.chunk.getString()).join("");
+ }
+ normalizeInput(e3) {
+ return "string" == typeof e3 ? e3 : I.from(e3).getString();
+ }
+ parse(e3 = this.chunk) {
+ if (!this.localOptions.parse)
+ return e3;
+ e3 = function(e4) {
+ let t3 = {}, i4 = {};
+ for (let e6 of Ze)
+ t3[e6] = [], i4[e6] = 0;
+ return e4.replace(et, (e6, n4, s2) => {
+ if ("<" === n4) {
+ let n5 = ++i4[s2];
+ return t3[s2].push(n5), "".concat(e6, "#").concat(n5);
+ }
+ return "".concat(e6, "#").concat(t3[s2].pop());
+ });
+ }(e3);
+ let t2 = Xe.findAll(e3, "rdf", "Description");
+ 0 === t2.length && t2.push(new Xe("rdf", "Description", void 0, e3));
+ let i3, n3 = {};
+ for (let e4 of t2)
+ for (let t3 of e4.properties)
+ i3 = Je(t3.ns, n3), _e(t3, i3);
+ return function(e4) {
+ let t3;
+ for (let i4 in e4)
+ t3 = e4[i4] = f(e4[i4]), void 0 === t3 && delete e4[i4];
+ return f(e4);
+ }(n3);
+ }
+ assignToOutput(e3, t2) {
+ if (this.localOptions.parse)
+ for (let [i3, n3] of Object.entries(t2))
+ switch (i3) {
+ case "tiff":
+ this.assignObjectToOutput(e3, "ifd0", n3);
+ break;
+ case "exif":
+ this.assignObjectToOutput(e3, "exif", n3);
+ break;
+ case "xmlns":
+ break;
+ default:
+ this.assignObjectToOutput(e3, i3, n3);
+ }
+ else
+ e3.xmp = t2;
+ }
+ };
+ c(We, "type", "xmp"), c(We, "multiSegment", true), T.set("xmp", We);
+ var Ke = class _Ke {
+ static findAll(e3) {
+ return qe(e3, /([a-zA-Z0-9-]+):([a-zA-Z0-9-]+)=("[^"]*"|'[^']*')/gm).map(_Ke.unpackMatch);
+ }
+ static unpackMatch(e3) {
+ let t2 = e3[1], i3 = e3[2], n3 = e3[3].slice(1, -1);
+ return n3 = Qe(n3), new _Ke(t2, i3, n3);
+ }
+ constructor(e3, t2, i3) {
+ this.ns = e3, this.name = t2, this.value = i3;
+ }
+ serialize() {
+ return this.value;
+ }
+ };
+ var Xe = class _Xe {
+ static findAll(e3, t2, i3) {
+ if (void 0 !== t2 || void 0 !== i3) {
+ t2 = t2 || "[\\w\\d-]+", i3 = i3 || "[\\w\\d-]+";
+ var n3 = new RegExp("<(".concat(t2, "):(").concat(i3, ")(#\\d+)?((\\s+?[\\w\\d-:]+=(\"[^\"]*\"|'[^']*'))*\\s*)(\\/>|>([\\s\\S]*?)<\\/\\1:\\2\\3>)"), "gm");
+ } else
+ n3 = /<([\w\d-]+):([\w\d-]+)(#\d+)?((\s+?[\w\d-:]+=("[^"]*"|'[^']*'))*\s*)(\/>|>([\s\S]*?)<\/\1:\2\3>)/gm;
+ return qe(e3, n3).map(_Xe.unpackMatch);
+ }
+ static unpackMatch(e3) {
+ let t2 = e3[1], i3 = e3[2], n3 = e3[4], s2 = e3[8];
+ return new _Xe(t2, i3, n3, s2);
+ }
+ constructor(e3, t2, i3, n3) {
+ this.ns = e3, this.name = t2, this.attrString = i3, this.innerXml = n3, this.attrs = Ke.findAll(i3), this.children = _Xe.findAll(n3), this.value = 0 === this.children.length ? Qe(n3) : void 0, this.properties = [...this.attrs, ...this.children];
+ }
+ get isPrimitive() {
+ return void 0 !== this.value && 0 === this.attrs.length && 0 === this.children.length;
+ }
+ get isListContainer() {
+ return 1 === this.children.length && this.children[0].isList;
+ }
+ get isList() {
+ let { ns: e3, name: t2 } = this;
+ return "rdf" === e3 && ("Seq" === t2 || "Bag" === t2 || "Alt" === t2);
+ }
+ get isListItem() {
+ return "rdf" === this.ns && "li" === this.name;
+ }
+ serialize() {
+ if (0 === this.properties.length && void 0 === this.value)
+ return;
+ if (this.isPrimitive)
+ return this.value;
+ if (this.isListContainer)
+ return this.children[0].serialize();
+ if (this.isList)
+ return $e(this.children.map(Ye));
+ if (this.isListItem && 1 === this.children.length && 0 === this.attrs.length)
+ return this.children[0].serialize();
+ let e3 = {};
+ for (let t2 of this.properties)
+ _e(t2, e3);
+ return void 0 !== this.value && (e3.value = this.value), f(e3);
}
- function editOff() {
- layer.selectAll(".viewfield-group").remove();
- layer.style("display", "none");
+ };
+ function _e(e3, t2) {
+ let i3 = e3.serialize();
+ void 0 !== i3 && (t2[e3.name] = i3);
+ }
+ var Ye = (e3) => e3.serialize();
+ var $e = (e3) => 1 === e3.length ? e3[0] : e3;
+ var Je = (e3, t2) => t2[e3] ? t2[e3] : t2[e3] = {};
+ function qe(e3, t2) {
+ let i3, n3 = [];
+ if (!e3)
+ return n3;
+ for (; null !== (i3 = t2.exec(e3)); )
+ n3.push(i3);
+ return n3;
+ }
+ function Qe(e3) {
+ if (function(e4) {
+ return null == e4 || "null" === e4 || "undefined" === e4 || "" === e4 || "" === e4.trim();
+ }(e3))
+ return;
+ let t2 = Number(e3);
+ if (!Number.isNaN(t2))
+ return t2;
+ let i3 = e3.toLowerCase();
+ return "true" === i3 || "false" !== i3 && e3.trim();
+ }
+ var Ze = ["rdf:li", "rdf:Seq", "rdf:Bag", "rdf:Alt", "rdf:Description"];
+ var et = new RegExp("(<|\\/)(".concat(Ze.join("|"), ")"), "g");
+ var tt = Object.freeze({ __proto__: null, default: Me, Exifr: te, fileParsers: w, segmentParsers: T, fileReaders: A, tagKeys: E, tagValues: B, tagRevivers: N, createDictionary: U, extendDictionary: F, fetchUrlAsArrayBuffer: M, readBlobAsArrayBuffer: R, chunkedProps: G, otherSegments: V, segments: z, tiffBlocks: H, segmentsAndBlocks: j, tiffExtractables: W, inheritables: K, allFormatters: X, Options: q, parse: ie, gpsOnlyOptions: me, gps: Se, thumbnailOnlyOptions: Ce, thumbnail: ye, thumbnailUrl: be, orientationOnlyOptions: Ie, orientation: Pe, rotations: ke, get rotateCanvas() {
+ return we;
+ }, get rotateCss() {
+ return Te;
+ }, rotation: Ae });
+ var at = l("fs", (e3) => e3.promises);
+ A.set("fs", class extends ve {
+ async readWhole() {
+ this.chunked = false, this.fs = await at;
+ let e3 = await this.fs.readFile(this.input);
+ this._swapBuffer(e3);
}
- 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);
+ async readChunked() {
+ this.chunked = true, this.fs = await at, await this.open(), await this.readChunk(0, this.options.firstChunkSize);
}
- function mouseover(d3_event, image) {
- const service = getService();
- if (service)
- service.setStyles(context, image);
+ async open() {
+ void 0 === this.fh && (this.fh = await this.fs.open(this.input, "r"), this.size = (await this.fh.stat(this.input)).size);
}
- function mouseout() {
- const service = getService();
- if (service)
- service.setStyles(context, null);
+ async _readChunk(e3, t2) {
+ void 0 === this.fh && await this.open(), e3 + t2 > this.size && (t2 = this.size - e3);
+ var i3 = this.subarray(e3, t2, true);
+ return await this.fh.read(i3.dataView, 0, t2, e3), i3;
}
- function transform2(d2) {
- let t2 = svgPointTransform(projection2)(d2);
- if (d2.ca) {
- t2 += " rotate(" + Math.floor(d2.ca) + ",0,0)";
+ async close() {
+ if (this.fh) {
+ let e3 = this.fh;
+ this.fh = void 0, await e3.close();
}
- return t2;
}
- 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;
+ });
+ A.set("base64", class extends ve {
+ constructor(...e3) {
+ super(...e3), this.input = this.input.replace(/^data:([^;]+);base64,/gim, ""), this.size = this.input.length / 4 * 3, this.input.endsWith("==") ? this.size -= 2 : this.input.endsWith("=") && (this.size -= 1);
}
- 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();
- });
+ async _readChunk(e3, t2) {
+ let i3, n3, r2 = this.input;
+ void 0 === e3 ? (e3 = 0, i3 = 0, n3 = 0) : (i3 = 4 * Math.floor(e3 / 3), n3 = e3 - i3 / 4 * 3), void 0 === t2 && (t2 = this.size);
+ let o2 = e3 + t2, l2 = i3 + 4 * Math.ceil(o2 / 3);
+ r2 = r2.slice(i3, l2);
+ let h2 = Math.min(t2, this.size - e3);
+ if (a) {
+ let t3 = s.from(r2, "base64").slice(n3, n3 + h2);
+ return this.set(t3, e3, true);
}
- if (toDate) {
- sequences = sequences.filter(function(sequence) {
- return new Date(sequence.properties.captured_at).getTime() <= new Date(toDate).getTime().toString();
- });
+ {
+ let t3 = this.subarray(e3, h2, true), i4 = atob(r2), s2 = t3.toUint8();
+ for (let e4 = 0; e4 < h2; e4++)
+ s2[e4] = i4.charCodeAt(n3 + e4);
+ return t3;
}
- return sequences;
}
- function update() {
- const z2 = ~~context.map().zoom();
- const showMarkers = z2 >= minMarkerZoom;
- const showViewfields = z2 >= minViewfieldZoom2;
- 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(d2) {
- return d2.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(d2) {
- return d2.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(a2, b2) {
- return b2.loc[1] - a2.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";
- }
- }
+ });
+ var ot = class extends se {
+ static canHandle(e3, t2) {
+ return 18761 === t2 || 19789 === t2;
}
- 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() >= minZoom4) {
- editOn();
- update();
- service.loadImages(projection2);
- } else {
- editOff();
- }
- }
+ extendOptions(e3) {
+ let { ifd0: t2, xmp: i3, iptc: n3, icc: s2 } = e3;
+ i3.enabled && t2.deps.add(700), n3.enabled && t2.deps.add(33723), s2.enabled && t2.deps.add(34675), t2.finalizeFilters();
}
- drawImages.enabled = function(_2) {
- if (!arguments.length)
- return svgMapillaryImages.enabled;
- svgMapillaryImages.enabled = _2;
- if (svgMapillaryImages.enabled) {
- showLayer();
- context.photos().on("change.mapillary_images", update);
- } else {
- hideLayer();
- context.photos().on("change.mapillary_images", null);
+ async parse() {
+ let { tiff: e3, xmp: t2, iptc: i3, icc: n3 } = this.options;
+ if (e3.enabled || t2.enabled || i3.enabled || n3.enabled) {
+ let e4 = Math.max(S(this.options), this.options.chunkSize);
+ await this.file.ensureChunk(0, e4), this.createParser("tiff", this.file), this.parsers.tiff.parseHeader(), await this.parsers.tiff.parseIfd0Block(), this.adaptTiffPropAsSegment("xmp"), this.adaptTiffPropAsSegment("iptc"), this.adaptTiffPropAsSegment("icc");
}
- dispatch14.call("change");
- return this;
- };
- drawImages.supported = function() {
- return !!getService();
- };
- drawImages.rendered = function(zoom) {
- return zoom >= minZoom4;
- };
- init2();
- return drawImages;
- }
-
- // modules/svg/mapillary_position.js
- function svgMapillaryPosition(projection2, context) {
- const throttledRedraw = throttle_default(function() {
- update();
- }, 1e3);
- const minZoom4 = 12;
- const minViewfieldZoom2 = 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(e3) {
- viewerCompassAngle = e3.bearing;
- if (context.map().isTransformed())
- return;
- layer.selectAll(".viewfield-group.currentView").filter(function(d2) {
- return d2.is_pano;
- }).attr("transform", transform2);
- });
- } else if (!services.mapillary && _mapillary) {
- _mapillary = null;
+ adaptTiffPropAsSegment(e3) {
+ if (this.parsers.tiff[e3]) {
+ let t2 = this.parsers.tiff[e3];
+ this.injectSegment(e3, t2);
}
- return _mapillary;
- }
- function editOn() {
- layer.style("display", "block");
}
- function editOff() {
- layer.selectAll(".viewfield-group").remove();
- layer.style("display", "none");
+ };
+ c(ot, "type", "tiff"), w.set("tiff", ot);
+ var lt = l("zlib");
+ var ht = ["ihdr", "iccp", "text", "itxt", "exif"];
+ var ut = class extends se {
+ constructor(...e3) {
+ super(...e3), c(this, "catchError", (e4) => this.errors.push(e4)), c(this, "metaChunks", []), c(this, "unknownChunks", []);
}
- function transform2(d2) {
- let t2 = svgPointTransform(projection2)(d2);
- if (d2.is_pano && viewerCompassAngle !== null && isFinite(viewerCompassAngle)) {
- t2 += " rotate(" + Math.floor(viewerCompassAngle) + ",0,0)";
- } else if (d2.ca) {
- t2 += " rotate(" + Math.floor(d2.ca) + ",0,0)";
- }
- return t2;
+ static canHandle(e3, t2) {
+ return 35152 === t2 && 2303741511 === e3.getUint32(0) && 218765834 === e3.getUint32(4);
}
- function update() {
- const z2 = ~~context.map().zoom();
- const showViewfields = z2 >= minViewfieldZoom2;
- const service = getService();
- const image = service && service.getActiveImage();
- const groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(image ? [image] : [], function(d2) {
- return d2.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");
+ async parse() {
+ let { file: e3 } = this;
+ await this.findPngChunksInRange("\x89PNG\r\n\n".length, e3.byteLength), await this.readSegments(this.metaChunks), this.findIhdr(), this.parseTextChunks(), await this.findExif().catch(this.catchError), await this.findXmp().catch(this.catchError), await this.findIcc().catch(this.catchError);
}
- 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() >= minZoom4) {
- editOn();
- update();
- } else {
- editOff();
+ async findPngChunksInRange(e3, t2) {
+ let { file: i3 } = this;
+ for (; e3 < t2; ) {
+ let t3 = i3.getUint32(e3), n3 = i3.getUint32(e3 + 4), s2 = i3.getString(e3 + 4, 4).toLowerCase(), r2 = t3 + 4 + 4 + 4, a2 = { type: s2, offset: e3, length: r2, start: e3 + 4 + 4, size: t3, marker: n3 };
+ ht.includes(s2) ? this.metaChunks.push(a2) : this.unknownChunks.push(a2), e3 += r2;
}
}
- drawImages.enabled = function() {
- update();
- return this;
- };
- drawImages.supported = function() {
- return !!getService();
- };
- drawImages.rendered = function(zoom) {
- return zoom >= minZoom4;
- };
- init2();
- return drawImages;
- }
-
- // modules/svg/mapillary_signs.js
- function svgMapillarySigns(projection2, context, dispatch14) {
- const throttledRedraw = throttle_default(function() {
- dispatch14.call("change");
- }, 1e3);
- const minZoom4 = 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;
+ parseTextChunks() {
+ let e3 = this.metaChunks.filter((e4) => "text" === e4.type);
+ for (let t2 of e3) {
+ let [e4, i3] = this.file.getString(t2.start, t2.size).split("\0");
+ this.injectKeyValToIhdr(e4, i3);
}
- 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");
+ injectKeyValToIhdr(e3, t2) {
+ let i3 = this.parsers.ihdr;
+ i3 && i3.raw.set(e3, t2);
}
- function editOff() {
- layer.selectAll(".icon-sign").remove();
- layer.style("display", "none");
+ findIhdr() {
+ let e3 = this.metaChunks.find((e4) => "ihdr" === e4.type);
+ e3 && false !== this.options.ihdr.enabled && this.createParser("ihdr", e3.chunk);
}
- function click(d3_event, d2) {
- const service = getService();
- if (!service)
- return;
- context.map().centerEase(d2.loc);
- const selectedImageId = service.getActiveImage() && service.getActiveImage().id;
- service.getDetections(d2.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);
- });
- }
- }
- });
+ async findExif() {
+ let e3 = this.metaChunks.find((e4) => "exif" === e4.type);
+ e3 && this.injectSegment("tiff", e3.chunk);
}
- 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;
- });
+ async findXmp() {
+ let e3 = this.metaChunks.filter((e4) => "itxt" === e4.type);
+ for (let t2 of e3) {
+ "XML:com.adobe.xmp" === t2.chunk.getString(0, "XML:com.adobe.xmp".length) && this.injectSegment("xmp", t2.chunk);
}
- 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(d2) {
- return d2.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(d2) {
- return "#" + d2.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() >= minZoom4) {
- editOn();
- update();
- service.loadSigns(projection2);
- service.showSignDetections(true);
- } else {
- editOff();
- }
- } else if (service) {
- service.showSignDetections(false);
+ async findIcc() {
+ let e3 = this.metaChunks.find((e4) => "iccp" === e4.type);
+ if (!e3)
+ return;
+ let { chunk: t2 } = e3, i3 = t2.getUint8Array(0, 81), s2 = 0;
+ for (; s2 < 80 && 0 !== i3[s2]; )
+ s2++;
+ let r2 = s2 + 2, a2 = t2.getString(0, s2);
+ if (this.injectKeyValToIhdr("ProfileName", a2), n2) {
+ let e4 = await lt, i4 = t2.getUint8Array(r2);
+ i4 = e4.inflateSync(i4), this.injectSegment("icc", i4);
}
}
- drawSigns.enabled = function(_2) {
- if (!arguments.length)
- return svgMapillarySigns.enabled;
- svgMapillarySigns.enabled = _2;
- if (svgMapillarySigns.enabled) {
- showLayer();
- context.photos().on("change.mapillary_signs", update);
- } else {
- hideLayer();
- context.photos().on("change.mapillary_signs", null);
- }
- dispatch14.call("change");
- return this;
- };
- drawSigns.supported = function() {
- return !!getService();
- };
- drawSigns.rendered = function(zoom) {
- return zoom >= minZoom4;
- };
- init2();
- return drawSigns;
- }
-
- // modules/svg/mapillary_map_features.js
- function svgMapillaryMapFeatures(projection2, context, dispatch14) {
- const throttledRedraw = throttle_default(function() {
- dispatch14.call("change");
- }, 1e3);
- const minZoom4 = 12;
- let layer = select_default2(null);
- let _mapillary;
- function init2() {
- if (svgMapillaryMapFeatures.initialized)
- return;
- svgMapillaryMapFeatures.enabled = false;
- svgMapillaryMapFeatures.initialized = true;
+ };
+ c(ut, "type", "png"), w.set("png", ut), U(E, "interop", [[1, "InteropIndex"], [2, "InteropVersion"], [4096, "RelatedImageFileFormat"], [4097, "RelatedImageWidth"], [4098, "RelatedImageHeight"]]), F(E, "ifd0", [[11, "ProcessingSoftware"], [254, "SubfileType"], [255, "OldSubfileType"], [263, "Thresholding"], [264, "CellWidth"], [265, "CellLength"], [266, "FillOrder"], [269, "DocumentName"], [280, "MinSampleValue"], [281, "MaxSampleValue"], [285, "PageName"], [286, "XPosition"], [287, "YPosition"], [290, "GrayResponseUnit"], [297, "PageNumber"], [321, "HalftoneHints"], [322, "TileWidth"], [323, "TileLength"], [332, "InkSet"], [337, "TargetPrinter"], [18246, "Rating"], [18249, "RatingPercent"], [33550, "PixelScale"], [34264, "ModelTransform"], [34377, "PhotoshopSettings"], [50706, "DNGVersion"], [50707, "DNGBackwardVersion"], [50708, "UniqueCameraModel"], [50709, "LocalizedCameraModel"], [50736, "DNGLensInfo"], [50739, "ShadowScale"], [50740, "DNGPrivateData"], [33920, "IntergraphMatrix"], [33922, "ModelTiePoint"], [34118, "SEMInfo"], [34735, "GeoTiffDirectory"], [34736, "GeoTiffDoubleParams"], [34737, "GeoTiffAsciiParams"], [50341, "PrintIM"], [50721, "ColorMatrix1"], [50722, "ColorMatrix2"], [50723, "CameraCalibration1"], [50724, "CameraCalibration2"], [50725, "ReductionMatrix1"], [50726, "ReductionMatrix2"], [50727, "AnalogBalance"], [50728, "AsShotNeutral"], [50729, "AsShotWhiteXY"], [50730, "BaselineExposure"], [50731, "BaselineNoise"], [50732, "BaselineSharpness"], [50734, "LinearResponseLimit"], [50735, "CameraSerialNumber"], [50741, "MakerNoteSafety"], [50778, "CalibrationIlluminant1"], [50779, "CalibrationIlluminant2"], [50781, "RawDataUniqueID"], [50827, "OriginalRawFileName"], [50828, "OriginalRawFileData"], [50831, "AsShotICCProfile"], [50832, "AsShotPreProfileMatrix"], [50833, "CurrentICCProfile"], [50834, "CurrentPreProfileMatrix"], [50879, "ColorimetricReference"], [50885, "SRawType"], [50898, "PanasonicTitle"], [50899, "PanasonicTitle2"], [50931, "CameraCalibrationSig"], [50932, "ProfileCalibrationSig"], [50933, "ProfileIFD"], [50934, "AsShotProfileName"], [50936, "ProfileName"], [50937, "ProfileHueSatMapDims"], [50938, "ProfileHueSatMapData1"], [50939, "ProfileHueSatMapData2"], [50940, "ProfileToneCurve"], [50941, "ProfileEmbedPolicy"], [50942, "ProfileCopyright"], [50964, "ForwardMatrix1"], [50965, "ForwardMatrix2"], [50966, "PreviewApplicationName"], [50967, "PreviewApplicationVersion"], [50968, "PreviewSettingsName"], [50969, "PreviewSettingsDigest"], [50970, "PreviewColorSpace"], [50971, "PreviewDateTime"], [50972, "RawImageDigest"], [50973, "OriginalRawFileDigest"], [50981, "ProfileLookTableDims"], [50982, "ProfileLookTableData"], [51043, "TimeCodes"], [51044, "FrameRate"], [51058, "TStop"], [51081, "ReelName"], [51089, "OriginalDefaultFinalSize"], [51090, "OriginalBestQualitySize"], [51091, "OriginalDefaultCropSize"], [51105, "CameraLabel"], [51107, "ProfileHueSatMapEncoding"], [51108, "ProfileLookTableEncoding"], [51109, "BaselineExposureOffset"], [51110, "DefaultBlackRender"], [51111, "NewRawImageDigest"], [51112, "RawToPreviewGain"]]);
+ var ct = [[273, "StripOffsets"], [279, "StripByteCounts"], [288, "FreeOffsets"], [289, "FreeByteCounts"], [291, "GrayResponseCurve"], [292, "T4Options"], [293, "T6Options"], [300, "ColorResponseUnit"], [320, "ColorMap"], [324, "TileOffsets"], [325, "TileByteCounts"], [326, "BadFaxLines"], [327, "CleanFaxData"], [328, "ConsecutiveBadFaxLines"], [330, "SubIFD"], [333, "InkNames"], [334, "NumberofInks"], [336, "DotRange"], [338, "ExtraSamples"], [339, "SampleFormat"], [340, "SMinSampleValue"], [341, "SMaxSampleValue"], [342, "TransferRange"], [343, "ClipPath"], [344, "XClipPathUnits"], [345, "YClipPathUnits"], [346, "Indexed"], [347, "JPEGTables"], [351, "OPIProxy"], [400, "GlobalParametersIFD"], [401, "ProfileType"], [402, "FaxProfile"], [403, "CodingMethods"], [404, "VersionYear"], [405, "ModeNumber"], [433, "Decode"], [434, "DefaultImageColor"], [435, "T82Options"], [437, "JPEGTables"], [512, "JPEGProc"], [515, "JPEGRestartInterval"], [517, "JPEGLosslessPredictors"], [518, "JPEGPointTransforms"], [519, "JPEGQTables"], [520, "JPEGDCTables"], [521, "JPEGACTables"], [559, "StripRowCounts"], [999, "USPTOMiscellaneous"], [18247, "XP_DIP_XML"], [18248, "StitchInfo"], [28672, "SonyRawFileType"], [28688, "SonyToneCurve"], [28721, "VignettingCorrection"], [28722, "VignettingCorrParams"], [28724, "ChromaticAberrationCorrection"], [28725, "ChromaticAberrationCorrParams"], [28726, "DistortionCorrection"], [28727, "DistortionCorrParams"], [29895, "SonyCropTopLeft"], [29896, "SonyCropSize"], [32781, "ImageID"], [32931, "WangTag1"], [32932, "WangAnnotation"], [32933, "WangTag3"], [32934, "WangTag4"], [32953, "ImageReferencePoints"], [32954, "RegionXformTackPoint"], [32955, "WarpQuadrilateral"], [32956, "AffineTransformMat"], [32995, "Matteing"], [32996, "DataType"], [32997, "ImageDepth"], [32998, "TileDepth"], [33300, "ImageFullWidth"], [33301, "ImageFullHeight"], [33302, "TextureFormat"], [33303, "WrapModes"], [33304, "FovCot"], [33305, "MatrixWorldToScreen"], [33306, "MatrixWorldToCamera"], [33405, "Model2"], [33421, "CFARepeatPatternDim"], [33422, "CFAPattern2"], [33423, "BatteryLevel"], [33424, "KodakIFD"], [33445, "MDFileTag"], [33446, "MDScalePixel"], [33447, "MDColorTable"], [33448, "MDLabName"], [33449, "MDSampleInfo"], [33450, "MDPrepDate"], [33451, "MDPrepTime"], [33452, "MDFileUnits"], [33589, "AdventScale"], [33590, "AdventRevision"], [33628, "UIC1Tag"], [33629, "UIC2Tag"], [33630, "UIC3Tag"], [33631, "UIC4Tag"], [33918, "IntergraphPacketData"], [33919, "IntergraphFlagRegisters"], [33921, "INGRReserved"], [34016, "Site"], [34017, "ColorSequence"], [34018, "IT8Header"], [34019, "RasterPadding"], [34020, "BitsPerRunLength"], [34021, "BitsPerExtendedRunLength"], [34022, "ColorTable"], [34023, "ImageColorIndicator"], [34024, "BackgroundColorIndicator"], [34025, "ImageColorValue"], [34026, "BackgroundColorValue"], [34027, "PixelIntensityRange"], [34028, "TransparencyIndicator"], [34029, "ColorCharacterization"], [34030, "HCUsage"], [34031, "TrapIndicator"], [34032, "CMYKEquivalent"], [34152, "AFCP_IPTC"], [34232, "PixelMagicJBIGOptions"], [34263, "JPLCartoIFD"], [34306, "WB_GRGBLevels"], [34310, "LeafData"], [34687, "TIFF_FXExtensions"], [34688, "MultiProfiles"], [34689, "SharedData"], [34690, "T88Options"], [34732, "ImageLayer"], [34750, "JBIGOptions"], [34856, "Opto-ElectricConvFactor"], [34857, "Interlace"], [34908, "FaxRecvParams"], [34909, "FaxSubAddress"], [34910, "FaxRecvTime"], [34929, "FedexEDR"], [34954, "LeafSubIFD"], [37387, "FlashEnergy"], [37388, "SpatialFrequencyResponse"], [37389, "Noise"], [37390, "FocalPlaneXResolution"], [37391, "FocalPlaneYResolution"], [37392, "FocalPlaneResolutionUnit"], [37397, "ExposureIndex"], [37398, "TIFF-EPStandardID"], [37399, "SensingMethod"], [37434, "CIP3DataFile"], [37435, "CIP3Sheet"], [37436, "CIP3Side"], [37439, "StoNits"], [37679, "MSDocumentText"], [37680, "MSPropertySetStorage"], [37681, "MSDocumentTextPosition"], [37724, "ImageSourceData"], [40965, "InteropIFD"], [40976, "SamsungRawPointersOffset"], [40977, "SamsungRawPointersLength"], [41217, "SamsungRawByteOrder"], [41218, "SamsungRawUnknown"], [41484, "SpatialFrequencyResponse"], [41485, "Noise"], [41489, "ImageNumber"], [41490, "SecurityClassification"], [41491, "ImageHistory"], [41494, "TIFF-EPStandardID"], [41995, "DeviceSettingDescription"], [42112, "GDALMetadata"], [42113, "GDALNoData"], [44992, "ExpandSoftware"], [44993, "ExpandLens"], [44994, "ExpandFilm"], [44995, "ExpandFilterLens"], [44996, "ExpandScanner"], [44997, "ExpandFlashLamp"], [46275, "HasselbladRawImage"], [48129, "PixelFormat"], [48130, "Transformation"], [48131, "Uncompressed"], [48132, "ImageType"], [48256, "ImageWidth"], [48257, "ImageHeight"], [48258, "WidthResolution"], [48259, "HeightResolution"], [48320, "ImageOffset"], [48321, "ImageByteCount"], [48322, "AlphaOffset"], [48323, "AlphaByteCount"], [48324, "ImageDataDiscard"], [48325, "AlphaDataDiscard"], [50215, "OceScanjobDesc"], [50216, "OceApplicationSelector"], [50217, "OceIDNumber"], [50218, "OceImageLogic"], [50255, "Annotations"], [50459, "HasselbladExif"], [50547, "OriginalFileName"], [50560, "USPTOOriginalContentType"], [50656, "CR2CFAPattern"], [50710, "CFAPlaneColor"], [50711, "CFALayout"], [50712, "LinearizationTable"], [50713, "BlackLevelRepeatDim"], [50714, "BlackLevel"], [50715, "BlackLevelDeltaH"], [50716, "BlackLevelDeltaV"], [50717, "WhiteLevel"], [50718, "DefaultScale"], [50719, "DefaultCropOrigin"], [50720, "DefaultCropSize"], [50733, "BayerGreenSplit"], [50737, "ChromaBlurRadius"], [50738, "AntiAliasStrength"], [50752, "RawImageSegmentation"], [50780, "BestQualityScale"], [50784, "AliasLayerMetadata"], [50829, "ActiveArea"], [50830, "MaskedAreas"], [50935, "NoiseReductionApplied"], [50974, "SubTileBlockSize"], [50975, "RowInterleaveFactor"], [51008, "OpcodeList1"], [51009, "OpcodeList2"], [51022, "OpcodeList3"], [51041, "NoiseProfile"], [51114, "CacheVersion"], [51125, "DefaultUserCrop"], [51157, "NikonNEFInfo"], [65024, "KdcIFD"]];
+ F(E, "ifd0", ct), F(E, "exif", ct), U(B, "gps", [[23, { M: "Magnetic North", T: "True North" }], [25, { K: "Kilometers", M: "Miles", N: "Nautical Miles" }]]);
+ var ft = class extends re2 {
+ static canHandle(e3, t2) {
+ return 224 === e3.getUint8(t2 + 1) && 1246120262 === e3.getUint32(t2 + 4) && 0 === e3.getUint8(t2 + 8);
}
- function getService() {
- if (services.mapillary && !_mapillary) {
- _mapillary = services.mapillary;
- _mapillary.event.on("loadedMapFeatures", throttledRedraw);
- } else if (!services.mapillary && _mapillary) {
- _mapillary = null;
- }
- return _mapillary;
+ parse() {
+ return this.parseTags(), this.translate(), this.output;
}
- function showLayer() {
- const service = getService();
- if (!service)
- return;
- service.loadObjectResources(context);
- editOn();
+ parseTags() {
+ this.raw = /* @__PURE__ */ new Map([[0, this.chunk.getUint16(0)], [2, this.chunk.getUint8(2)], [3, this.chunk.getUint16(3)], [5, this.chunk.getUint16(5)], [7, this.chunk.getUint8(7)], [8, this.chunk.getUint8(8)]]);
}
- function hideLayer() {
- throttledRedraw.cancel();
- editOff();
+ };
+ c(ft, "type", "jfif"), c(ft, "headerLength", 9), T.set("jfif", ft), U(E, "jfif", [[0, "JFIFVersion"], [2, "ResolutionUnit"], [3, "XResolution"], [5, "YResolution"], [7, "ThumbnailWidth"], [8, "ThumbnailHeight"]]);
+ var dt = class extends re2 {
+ parse() {
+ return this.parseTags(), this.translate(), this.output;
}
- function editOn() {
- layer.style("display", "block");
+ parseTags() {
+ this.raw = new Map([[0, this.chunk.getUint32(0)], [4, this.chunk.getUint32(4)], [8, this.chunk.getUint8(8)], [9, this.chunk.getUint8(9)], [10, this.chunk.getUint8(10)], [11, this.chunk.getUint8(11)], [12, this.chunk.getUint8(12)], ...Array.from(this.raw)]);
}
- function editOff() {
- layer.selectAll(".icon-map-feature").remove();
- layer.style("display", "none");
+ };
+ c(dt, "type", "ihdr"), T.set("ihdr", dt), U(E, "ihdr", [[0, "ImageWidth"], [4, "ImageHeight"], [8, "BitDepth"], [9, "ColorType"], [10, "Compression"], [11, "Filter"], [12, "Interlace"]]), U(B, "ihdr", [[9, { 0: "Grayscale", 2: "RGB", 3: "Palette", 4: "Grayscale with Alpha", 6: "RGB with Alpha", DEFAULT: "Unknown" }], [10, { 0: "Deflate/Inflate", DEFAULT: "Unknown" }], [11, { 0: "Adaptive", DEFAULT: "Unknown" }], [12, { 0: "Noninterlaced", 1: "Adam7 Interlace", DEFAULT: "Unknown" }]]);
+ var pt = class extends re2 {
+ static canHandle(e3, t2) {
+ return 226 === e3.getUint8(t2 + 1) && 1229144927 === e3.getUint32(t2 + 4);
}
- function click(d3_event, d2) {
- const service = getService();
- if (!service)
- return;
- context.map().centerEase(d2.loc);
- const selectedImageId = service.getActiveImage() && service.getActiveImage().id;
- service.getDetections(d2.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);
- });
- }
- }
- });
+ static findPosition(e3, t2) {
+ let i3 = super.findPosition(e3, t2);
+ return i3.chunkNumber = e3.getUint8(t2 + 16), i3.chunkCount = e3.getUint8(t2 + 17), i3.multiSegment = i3.chunkCount > 1, i3;
}
- 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;
+ static handleMultiSegments(e3) {
+ return function(e4) {
+ let t2 = function(e6) {
+ let t3 = e6[0].constructor, i3 = 0;
+ for (let t4 of e6)
+ i3 += t4.length;
+ let n3 = new t3(i3), s2 = 0;
+ for (let t4 of e6)
+ n3.set(t4, s2), s2 += t4.length;
+ return n3;
+ }(e4.map((e6) => e6.chunk.toUint8()));
+ return new I(t2);
+ }(e3);
}
- 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(d2) {
- return d2.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(d2) {
- var id2 = d2.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(d2) {
- if (d2.value === "object--billboard") {
- return "#object--sign--advertisement";
- }
- return "#" + d2.value;
- });
- enter.append("rect").attr("width", "24px").attr("height", "24px").attr("x", "-12px").attr("y", "-12px");
- mapFeatures.merge(enter).attr("transform", transform2);
+ parse() {
+ return this.raw = /* @__PURE__ */ new Map(), this.parseHeader(), this.parseTags(), this.translate(), this.output;
}
- 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() >= minZoom4) {
- editOn();
- update();
- service.loadMapFeatures(projection2);
- service.showFeatureDetections(true);
- } else {
- editOff();
- }
- } else if (service) {
- service.showFeatureDetections(false);
+ parseHeader() {
+ let { raw: e3 } = this;
+ this.chunk.byteLength < 84 && g2("ICC header is too short");
+ for (let [t2, i3] of Object.entries(gt)) {
+ t2 = parseInt(t2, 10);
+ let n3 = i3(this.chunk, t2);
+ "\0\0\0\0" !== n3 && e3.set(t2, n3);
}
}
- drawMapFeatures.enabled = function(_2) {
- if (!arguments.length)
- return svgMapillaryMapFeatures.enabled;
- svgMapillaryMapFeatures.enabled = _2;
- if (svgMapillaryMapFeatures.enabled) {
- showLayer();
- context.photos().on("change.mapillary_map_features", update);
- } else {
- hideLayer();
- context.photos().on("change.mapillary_map_features", null);
+ parseTags() {
+ let e3, t2, i3, n3, s2, { raw: r2 } = this, a2 = this.chunk.getUint32(128), o2 = 132, l2 = this.chunk.byteLength;
+ for (; a2--; ) {
+ if (e3 = this.chunk.getString(o2, 4), t2 = this.chunk.getUint32(o2 + 4), i3 = this.chunk.getUint32(o2 + 8), n3 = this.chunk.getString(t2, 4), t2 + i3 > l2)
+ return void console.warn("reached the end of the first ICC chunk. Enable options.tiff.multiSegment to read all ICC segments.");
+ s2 = this.parseTag(n3, t2, i3), void 0 !== s2 && "\0\0\0\0" !== s2 && r2.set(e3, s2), o2 += 12;
}
- dispatch14.call("change");
- return this;
- };
- drawMapFeatures.supported = function() {
- return !!getService();
- };
- drawMapFeatures.rendered = function(zoom) {
- return zoom >= minZoom4;
- };
- init2();
- return drawMapFeatures;
- }
-
- // modules/svg/kartaview_images.js
- function svgKartaviewImages(projection2, context, dispatch14) {
- var throttledRedraw = throttle_default(function() {
- dispatch14.call("change");
- }, 1e3);
- var minZoom4 = 12;
- var minMarkerZoom = 16;
- var minViewfieldZoom2 = 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;
+ parseTag(e3, t2, i3) {
+ switch (e3) {
+ case "desc":
+ return this.parseDesc(t2);
+ case "mluc":
+ return this.parseMluc(t2);
+ case "text":
+ return this.parseText(t2, i3);
+ case "sig ":
+ return this.parseSig(t2);
}
- return _kartaview;
+ if (!(t2 + i3 > this.chunk.byteLength))
+ return this.chunk.getUint8Array(t2, i3);
}
- function showLayer() {
- var service = getService();
- if (!service)
- return;
- editOn();
- layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
- dispatch14.call("change");
- });
+ parseDesc(e3) {
+ let t2 = this.chunk.getUint32(e3 + 8) - 1;
+ return m(this.chunk.getString(e3 + 12, t2));
}
- function hideLayer() {
- throttledRedraw.cancel();
- layer.transition().duration(250).style("opacity", 0).on("end", editOff);
+ parseText(e3, t2) {
+ return m(this.chunk.getString(e3 + 8, t2 - 8));
}
- function editOn() {
- layer.style("display", "block");
+ parseSig(e3) {
+ return m(this.chunk.getString(e3 + 8, 4));
}
- function editOff() {
- layer.selectAll(".viewfield-group").remove();
- layer.style("display", "none");
+ parseMluc(e3) {
+ let { chunk: t2 } = this, i3 = t2.getUint32(e3 + 8), n3 = t2.getUint32(e3 + 12), s2 = e3 + 16, r2 = [];
+ for (let a2 = 0; a2 < i3; a2++) {
+ let i4 = t2.getString(s2 + 0, 2), a3 = t2.getString(s2 + 2, 2), o2 = t2.getUint32(s2 + 4), l2 = t2.getUint32(s2 + 8) + e3, h2 = m(t2.getUnicodeString(l2, o2));
+ r2.push({ lang: i4, country: a3, text: h2 }), s2 += n3;
+ }
+ return 1 === i3 ? r2[0].text : r2;
}
- function click(d3_event, d2) {
- var service = getService();
- if (!service)
- return;
- service.ensureViewerLoaded(context).then(function() {
- service.selectImage(context, d2.key).showViewer(context);
- });
- context.map().centerEase(d2.loc);
+ translateValue(e3, t2) {
+ return "string" == typeof e3 ? t2[e3] || t2[e3.toLowerCase()] || e3 : t2[e3] || e3;
}
- function mouseover(d3_event, d2) {
- var service = getService();
- if (service)
- service.setStyles(context, d2);
+ };
+ c(pt, "type", "icc"), c(pt, "multiSegment", true), c(pt, "headerLength", 18);
+ var gt = { 4: mt, 8: function(e3, t2) {
+ return [e3.getUint8(t2), e3.getUint8(t2 + 1) >> 4, e3.getUint8(t2 + 1) % 16].map((e4) => e4.toString(10)).join(".");
+ }, 12: mt, 16: mt, 20: mt, 24: function(e3, t2) {
+ const i3 = e3.getUint16(t2), n3 = e3.getUint16(t2 + 2) - 1, s2 = e3.getUint16(t2 + 4), r2 = e3.getUint16(t2 + 6), a2 = e3.getUint16(t2 + 8), o2 = e3.getUint16(t2 + 10);
+ return new Date(Date.UTC(i3, n3, s2, r2, a2, o2));
+ }, 36: mt, 40: mt, 48: mt, 52: mt, 64: (e3, t2) => e3.getUint32(t2), 80: mt };
+ function mt(e3, t2) {
+ return m(e3.getString(t2, 4));
+ }
+ T.set("icc", pt), U(E, "icc", [[4, "ProfileCMMType"], [8, "ProfileVersion"], [12, "ProfileClass"], [16, "ColorSpaceData"], [20, "ProfileConnectionSpace"], [24, "ProfileDateTime"], [36, "ProfileFileSignature"], [40, "PrimaryPlatform"], [44, "CMMFlags"], [48, "DeviceManufacturer"], [52, "DeviceModel"], [56, "DeviceAttributes"], [64, "RenderingIntent"], [68, "ConnectionSpaceIlluminant"], [80, "ProfileCreator"], [84, "ProfileID"], ["Header", "ProfileHeader"], ["MS00", "WCSProfiles"], ["bTRC", "BlueTRC"], ["bXYZ", "BlueMatrixColumn"], ["bfd", "UCRBG"], ["bkpt", "MediaBlackPoint"], ["calt", "CalibrationDateTime"], ["chad", "ChromaticAdaptation"], ["chrm", "Chromaticity"], ["ciis", "ColorimetricIntentImageState"], ["clot", "ColorantTableOut"], ["clro", "ColorantOrder"], ["clrt", "ColorantTable"], ["cprt", "ProfileCopyright"], ["crdi", "CRDInfo"], ["desc", "ProfileDescription"], ["devs", "DeviceSettings"], ["dmdd", "DeviceModelDesc"], ["dmnd", "DeviceMfgDesc"], ["dscm", "ProfileDescriptionML"], ["fpce", "FocalPlaneColorimetryEstimates"], ["gTRC", "GreenTRC"], ["gXYZ", "GreenMatrixColumn"], ["gamt", "Gamut"], ["kTRC", "GrayTRC"], ["lumi", "Luminance"], ["meas", "Measurement"], ["meta", "Metadata"], ["mmod", "MakeAndModel"], ["ncl2", "NamedColor2"], ["ncol", "NamedColor"], ["ndin", "NativeDisplayInfo"], ["pre0", "Preview0"], ["pre1", "Preview1"], ["pre2", "Preview2"], ["ps2i", "PS2RenderingIntent"], ["ps2s", "PostScript2CSA"], ["psd0", "PostScript2CRD0"], ["psd1", "PostScript2CRD1"], ["psd2", "PostScript2CRD2"], ["psd3", "PostScript2CRD3"], ["pseq", "ProfileSequenceDesc"], ["psid", "ProfileSequenceIdentifier"], ["psvm", "PS2CRDVMSize"], ["rTRC", "RedTRC"], ["rXYZ", "RedMatrixColumn"], ["resp", "OutputResponse"], ["rhoc", "ReflectionHardcopyOrigColorimetry"], ["rig0", "PerceptualRenderingIntentGamut"], ["rig2", "SaturationRenderingIntentGamut"], ["rpoc", "ReflectionPrintOutputColorimetry"], ["sape", "SceneAppearanceEstimates"], ["scoe", "SceneColorimetryEstimates"], ["scrd", "ScreeningDesc"], ["scrn", "Screening"], ["targ", "CharTarget"], ["tech", "Technology"], ["vcgt", "VideoCardGamma"], ["view", "ViewingConditions"], ["vued", "ViewingCondDesc"], ["wtpt", "MediaWhitePoint"]]);
+ var St = { "4d2p": "Erdt Systems", AAMA: "Aamazing Technologies", ACER: "Acer", ACLT: "Acolyte Color Research", ACTI: "Actix Sytems", ADAR: "Adara Technology", ADBE: "Adobe", ADI: "ADI Systems", AGFA: "Agfa Graphics", ALMD: "Alps Electric", ALPS: "Alps Electric", ALWN: "Alwan Color Expertise", AMTI: "Amiable Technologies", AOC: "AOC International", APAG: "Apago", APPL: "Apple Computer", AST: "AST", "AT&T": "AT&T", BAEL: "BARBIERI electronic", BRCO: "Barco NV", BRKP: "Breakpoint", BROT: "Brother", BULL: "Bull", BUS: "Bus Computer Systems", "C-IT": "C-Itoh", CAMR: "Intel", CANO: "Canon", CARR: "Carroll Touch", CASI: "Casio", CBUS: "Colorbus PL", CEL: "Crossfield", CELx: "Crossfield", CGS: "CGS Publishing Technologies International", CHM: "Rochester Robotics", CIGL: "Colour Imaging Group, London", CITI: "Citizen", CL00: "Candela", CLIQ: "Color IQ", CMCO: "Chromaco", CMiX: "CHROMiX", COLO: "Colorgraphic Communications", COMP: "Compaq", COMp: "Compeq/Focus Technology", CONR: "Conrac Display Products", CORD: "Cordata Technologies", CPQ: "Compaq", CPRO: "ColorPro", CRN: "Cornerstone", CTX: "CTX International", CVIS: "ColorVision", CWC: "Fujitsu Laboratories", DARI: "Darius Technology", DATA: "Dataproducts", DCP: "Dry Creek Photo", DCRC: "Digital Contents Resource Center, Chung-Ang University", DELL: "Dell Computer", DIC: "Dainippon Ink and Chemicals", DICO: "Diconix", DIGI: "Digital", "DL&C": "Digital Light & Color", DPLG: "Doppelganger", DS: "Dainippon Screen", DSOL: "DOOSOL", DUPN: "DuPont", EPSO: "Epson", ESKO: "Esko-Graphics", ETRI: "Electronics and Telecommunications Research Institute", EVER: "Everex Systems", EXAC: "ExactCODE", Eizo: "Eizo", FALC: "Falco Data Products", FF: "Fuji Photo Film", FFEI: "FujiFilm Electronic Imaging", FNRD: "Fnord Software", FORA: "Fora", FORE: "Forefront Technology", FP: "Fujitsu", FPA: "WayTech Development", FUJI: "Fujitsu", FX: "Fuji Xerox", GCC: "GCC Technologies", GGSL: "Global Graphics Software", GMB: "Gretagmacbeth", GMG: "GMG", GOLD: "GoldStar Technology", GOOG: "Google", GPRT: "Giantprint", GTMB: "Gretagmacbeth", GVC: "WayTech Development", GW2K: "Sony", HCI: "HCI", HDM: "Heidelberger Druckmaschinen", HERM: "Hermes", HITA: "Hitachi America", HP: "Hewlett-Packard", HTC: "Hitachi", HiTi: "HiTi Digital", IBM: "IBM", IDNT: "Scitex", IEC: "Hewlett-Packard", IIYA: "Iiyama North America", IKEG: "Ikegami Electronics", IMAG: "Image Systems", IMI: "Ingram Micro", INTC: "Intel", INTL: "N/A (INTL)", INTR: "Intra Electronics", IOCO: "Iocomm International Technology", IPS: "InfoPrint Solutions Company", IRIS: "Scitex", ISL: "Ichikawa Soft Laboratory", ITNL: "N/A (ITNL)", IVM: "IVM", IWAT: "Iwatsu Electric", Idnt: "Scitex", Inca: "Inca Digital Printers", Iris: "Scitex", JPEG: "Joint Photographic Experts Group", JSFT: "Jetsoft Development", JVC: "JVC Information Products", KART: "Scitex", KFC: "KFC Computek Components", KLH: "KLH Computers", KMHD: "Konica Minolta", KNCA: "Konica", KODA: "Kodak", KYOC: "Kyocera", Kart: "Scitex", LCAG: "Leica", LCCD: "Leeds Colour", LDAK: "Left Dakota", LEAD: "Leading Technology", LEXM: "Lexmark International", LINK: "Link Computer", LINO: "Linotronic", LITE: "Lite-On", Leaf: "Leaf", Lino: "Linotronic", MAGC: "Mag Computronic", MAGI: "MAG Innovision", MANN: "Mannesmann", MICN: "Micron Technology", MICR: "Microtek", MICV: "Microvitec", MINO: "Minolta", MITS: "Mitsubishi Electronics America", MITs: "Mitsuba", MNLT: "Minolta", MODG: "Modgraph", MONI: "Monitronix", MONS: "Monaco Systems", MORS: "Morse Technology", MOTI: "Motive Systems", MSFT: "Microsoft", MUTO: "MUTOH INDUSTRIES", Mits: "Mitsubishi Electric", NANA: "NANAO", NEC: "NEC", NEXP: "NexPress Solutions", NISS: "Nissei Sangyo America", NKON: "Nikon", NONE: "none", OCE: "Oce Technologies", OCEC: "OceColor", OKI: "Oki", OKID: "Okidata", OKIP: "Okidata", OLIV: "Olivetti", OLYM: "Olympus", ONYX: "Onyx Graphics", OPTI: "Optiquest", PACK: "Packard Bell", PANA: "Matsushita Electric Industrial", PANT: "Pantone", PBN: "Packard Bell", PFU: "PFU", PHIL: "Philips Consumer Electronics", PNTX: "HOYA", POne: "Phase One A/S", PREM: "Premier Computer Innovations", PRIN: "Princeton Graphic Systems", PRIP: "Princeton Publishing Labs", QLUX: "Hong Kong", QMS: "QMS", QPCD: "QPcard AB", QUAD: "QuadLaser", QUME: "Qume", RADI: "Radius", RDDx: "Integrated Color Solutions", RDG: "Roland DG", REDM: "REDMS Group", RELI: "Relisys", RGMS: "Rolf Gierling Multitools", RICO: "Ricoh", RNLD: "Edmund Ronald", ROYA: "Royal", RPC: "Ricoh Printing Systems", RTL: "Royal Information Electronics", SAMP: "Sampo", SAMS: "Samsung", SANT: "Jaime Santana Pomares", SCIT: "Scitex", SCRN: "Dainippon Screen", SDP: "Scitex", SEC: "Samsung", SEIK: "Seiko Instruments", SEIk: "Seikosha", SGUY: "ScanGuy.com", SHAR: "Sharp Laboratories", SICC: "International Color Consortium", SONY: "Sony", SPCL: "SpectraCal", STAR: "Star", STC: "Sampo Technology", Scit: "Scitex", Sdp: "Scitex", Sony: "Sony", TALO: "Talon Technology", TAND: "Tandy", TATU: "Tatung", TAXA: "TAXAN America", TDS: "Tokyo Denshi Sekei", TECO: "TECO Information Systems", TEGR: "Tegra", TEKT: "Tektronix", TI: "Texas Instruments", TMKR: "TypeMaker", TOSB: "Toshiba", TOSH: "Toshiba", TOTK: "TOTOKU ELECTRIC", TRIU: "Triumph", TSBT: "Toshiba", TTX: "TTX Computer Products", TVM: "TVM Professional Monitor", TW: "TW Casper", ULSX: "Ulead Systems", UNIS: "Unisys", UTZF: "Utz Fehlau & Sohn", VARI: "Varityper", VIEW: "Viewsonic", VISL: "Visual communication", VIVO: "Vivo Mobile Communication", WANG: "Wang", WLBR: "Wilbur Imaging", WTG2: "Ware To Go", WYSE: "WYSE Technology", XERX: "Xerox", XRIT: "X-Rite", ZRAN: "Zoran", Zebr: "Zebra Technologies", appl: "Apple Computer", bICC: "basICColor", berg: "bergdesign", ceyd: "Integrated Color Solutions", clsp: "MacDermid ColorSpan", ds: "Dainippon Screen", dupn: "DuPont", ffei: "FujiFilm Electronic Imaging", flux: "FluxData", iris: "Scitex", kart: "Scitex", lcms: "Little CMS", lino: "Linotronic", none: "none", ob4d: "Erdt Systems", obic: "Medigraph", quby: "Qubyx Sarl", scit: "Scitex", scrn: "Dainippon Screen", sdp: "Scitex", siwi: "SIWI GRAFIKA", yxym: "YxyMaster" };
+ var Ct = { scnr: "Scanner", mntr: "Monitor", prtr: "Printer", link: "Device Link", abst: "Abstract", spac: "Color Space Conversion Profile", nmcl: "Named Color", cenc: "ColorEncodingSpace profile", mid: "MultiplexIdentification profile", mlnk: "MultiplexLink profile", mvis: "MultiplexVisualization profile", nkpf: "Nikon Input Device Profile (NON-STANDARD!)" };
+ U(B, "icc", [[4, St], [12, Ct], [40, Object.assign({}, St, Ct)], [48, St], [80, St], [64, { 0: "Perceptual", 1: "Relative Colorimetric", 2: "Saturation", 3: "Absolute Colorimetric" }], ["tech", { amd: "Active Matrix Display", crt: "Cathode Ray Tube Display", kpcd: "Photo CD", pmd: "Passive Matrix Display", dcam: "Digital Camera", dcpj: "Digital Cinema Projector", dmpc: "Digital Motion Picture Camera", dsub: "Dye Sublimation Printer", epho: "Electrophotographic Printer", esta: "Electrostatic Printer", flex: "Flexography", fprn: "Film Writer", fscn: "Film Scanner", grav: "Gravure", ijet: "Ink Jet Printer", imgs: "Photo Image Setter", mpfr: "Motion Picture Film Recorder", mpfs: "Motion Picture Film Scanner", offs: "Offset Lithography", pjtv: "Projection Television", rpho: "Photographic Paper Printer", rscn: "Reflective Scanner", silk: "Silkscreen", twax: "Thermal Wax Printer", vidc: "Video Camera", vidm: "Video Monitor" }]]);
+ var yt = class extends re2 {
+ static canHandle(e3, t2, i3) {
+ return 237 === e3.getUint8(t2 + 1) && "Photoshop" === e3.getString(t2 + 4, 9) && void 0 !== this.containsIptc8bim(e3, t2, i3);
}
- function mouseout() {
- var service = getService();
- if (service)
- service.setStyles(context, null);
+ static headerLength(e3, t2, i3) {
+ let n3, s2 = this.containsIptc8bim(e3, t2, i3);
+ if (void 0 !== s2)
+ return n3 = e3.getUint8(t2 + s2 + 7), n3 % 2 != 0 && (n3 += 1), 0 === n3 && (n3 = 4), s2 + 8 + n3;
}
- function transform2(d2) {
- var t2 = svgPointTransform(projection2)(d2);
- if (d2.ca) {
- t2 += " rotate(" + Math.floor(d2.ca) + ",0,0)";
- }
- return t2;
+ static containsIptc8bim(e3, t2, i3) {
+ for (let n3 = 0; n3 < i3; n3++)
+ if (this.isIptcSegmentHead(e3, t2 + n3))
+ return n3;
}
- 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;
+ static isIptcSegmentHead(e3, t2) {
+ return 56 === e3.getUint8(t2) && 943868237 === e3.getUint32(t2) && 1028 === e3.getUint16(t2 + 4);
}
- 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;
+ parse() {
+ let { raw: e3 } = this, t2 = this.chunk.byteLength - 1, i3 = false;
+ for (let n3 = 0; n3 < t2; n3++)
+ if (28 === this.chunk.getUint8(n3) && 2 === this.chunk.getUint8(n3 + 1)) {
+ i3 = true;
+ let t3 = this.chunk.getUint16(n3 + 3), s2 = this.chunk.getUint8(n3 + 2), r2 = this.chunk.getLatin1String(n3 + 5, t3);
+ e3.set(s2, this.pluralizeValue(e3.get(s2), r2)), n3 += 4 + t3;
+ } else if (i3)
+ break;
+ return this.translate(), this.output;
}
- function update() {
- var viewer = context.container().select(".photoviewer");
- var selected = viewer.empty() ? void 0 : viewer.datum();
- var z2 = ~~context.map().zoom();
- var showMarkers = z2 >= minMarkerZoom;
- var showViewfields = z2 >= minViewfieldZoom2;
- 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(d2) {
- return d2.properties.key;
+ pluralizeValue(e3, t2) {
+ return void 0 !== e3 ? e3 instanceof Array ? (e3.push(t2), e3) : [e3, t2] : t2;
+ }
+ };
+ c(yt, "type", "iptc"), c(yt, "translateValues", false), c(yt, "reviveValues", false), T.set("iptc", yt), U(E, "iptc", [[0, "ApplicationRecordVersion"], [3, "ObjectTypeReference"], [4, "ObjectAttributeReference"], [5, "ObjectName"], [7, "EditStatus"], [8, "EditorialUpdate"], [10, "Urgency"], [12, "SubjectReference"], [15, "Category"], [20, "SupplementalCategories"], [22, "FixtureIdentifier"], [25, "Keywords"], [26, "ContentLocationCode"], [27, "ContentLocationName"], [30, "ReleaseDate"], [35, "ReleaseTime"], [37, "ExpirationDate"], [38, "ExpirationTime"], [40, "SpecialInstructions"], [42, "ActionAdvised"], [45, "ReferenceService"], [47, "ReferenceDate"], [50, "ReferenceNumber"], [55, "DateCreated"], [60, "TimeCreated"], [62, "DigitalCreationDate"], [63, "DigitalCreationTime"], [65, "OriginatingProgram"], [70, "ProgramVersion"], [75, "ObjectCycle"], [80, "Byline"], [85, "BylineTitle"], [90, "City"], [92, "Sublocation"], [95, "State"], [100, "CountryCode"], [101, "Country"], [103, "OriginalTransmissionReference"], [105, "Headline"], [110, "Credit"], [115, "Source"], [116, "CopyrightNotice"], [118, "Contact"], [120, "Caption"], [121, "LocalCaption"], [122, "Writer"], [125, "RasterizedCaption"], [130, "ImageType"], [131, "ImageOrientation"], [135, "LanguageIdentifier"], [150, "AudioType"], [151, "AudioSamplingRate"], [152, "AudioSamplingResolution"], [153, "AudioDuration"], [154, "AudioOutcue"], [184, "JobID"], [185, "MasterDocumentID"], [186, "ShortDocumentID"], [187, "UniqueDocumentID"], [188, "OwnerID"], [200, "ObjectPreviewFileFormat"], [201, "ObjectPreviewFileVersion"], [202, "ObjectPreviewData"], [221, "Prefs"], [225, "ClassifyState"], [228, "SimilarityIndex"], [230, "DocumentNotes"], [231, "DocumentHistory"], [232, "ExifCameraInfo"], [255, "CatalogSets"]]), U(B, "iptc", [[10, { 0: "0 (reserved)", 1: "1 (most urgent)", 2: "2", 3: "3", 4: "4", 5: "5 (normal urgency)", 6: "6", 7: "7", 8: "8 (least urgent)", 9: "9 (user-defined priority)" }], [75, { a: "Morning", b: "Both Morning and Evening", p: "Evening" }], [131, { L: "Landscape", P: "Portrait", S: "Square" }]]);
+ var full_esm_default = tt;
+
+ // modules/services/plane_photo.js
+ var dispatch6 = dispatch_default("viewerChanged");
+ var _photo;
+ var _wrapper;
+ var imgZoom;
+ var _widthOverflow;
+ function zoomPan(d3_event) {
+ let t2 = d3_event.transform;
+ _photo.call(utilSetTransform, t2.x, t2.y, t2.k);
+ }
+ function zoomBeahvior() {
+ const { width: wrapperWidth, height: wrapperHeight } = _wrapper.node().getBoundingClientRect();
+ const { naturalHeight, naturalWidth } = _photo.node();
+ const intrinsicRatio = naturalWidth / naturalHeight;
+ _widthOverflow = wrapperHeight * intrinsicRatio - wrapperWidth;
+ return zoom_default2().extent([[0, 0], [wrapperWidth, wrapperHeight]]).translateExtent([[0, 0], [wrapperWidth + _widthOverflow, wrapperHeight]]).scaleExtent([1, 15]).on("zoom", zoomPan);
+ }
+ function loadImage(selection2, path) {
+ return new Promise((resolve) => {
+ selection2.attr("src", path);
+ selection2.on("load", () => {
+ resolve(selection2);
});
- 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(d2) {
- return d2.key;
+ });
+ }
+ var plane_photo_default = {
+ init: async function(context, selection2) {
+ this.event = utilRebind(this, dispatch6, "on");
+ _wrapper = selection2.append("div").attr("class", "photo-frame plane-frame").classed("hide", true);
+ _photo = _wrapper.append("img").attr("class", "plane-photo");
+ context.ui().photoviewer.on("resize.plane", () => {
+ imgZoom = zoomBeahvior();
+ _wrapper.call(imgZoom);
});
- 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(a2, b2) {
- return a2 === selected ? 1 : b2 === selected ? -1 : b2.loc[1] - a2.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() >= minZoom4) {
- editOn();
- update();
- service.loadImages(projection2);
- } else {
- editOff();
- }
- }
- }
- drawImages.enabled = function(_2) {
- if (!arguments.length)
- return svgKartaviewImages.enabled;
- svgKartaviewImages.enabled = _2;
- if (svgKartaviewImages.enabled) {
- showLayer();
- context.photos().on("change.kartaview_images", update);
- } else {
- hideLayer();
- context.photos().on("change.kartaview_images", null);
+ await Promise.resolve();
+ return this;
+ },
+ showPhotoFrame: function(context) {
+ const isHidden = context.selectAll(".photo-frame.plane-frame.hide").size();
+ if (isHidden) {
+ context.selectAll(".photo-frame:not(.plane-frame)").classed("hide", true);
+ context.selectAll(".photo-frame.plane-frame").classed("hide", false);
}
- dispatch14.call("change");
return this;
- };
- drawImages.supported = function() {
- return !!getService();
- };
- drawImages.rendered = function(zoom) {
- return zoom >= minZoom4;
- };
- init2();
- return drawImages;
- }
+ },
+ hidePhotoFrame: function(context) {
+ context.select("photo-frame.plane-frame").classed("hide", false);
+ return this;
+ },
+ selectPhoto: function(data, keepOrientation) {
+ dispatch6.call("viewerChanged");
+ loadImage(_photo, "");
+ loadImage(_photo, data.image_path).then(() => {
+ if (!keepOrientation) {
+ imgZoom = zoomBeahvior();
+ _wrapper.call(imgZoom);
+ _wrapper.call(imgZoom.transform, identity2.translate(-_widthOverflow / 2, 0));
+ }
+ });
+ return this;
+ },
+ getYaw: function() {
+ return 0;
+ }
+ };
- // modules/svg/mapilio_images.js
- function svgMapilioImages(projection2, context, dispatch14) {
- const throttledRedraw = throttle_default(function() {
- dispatch14.call("change");
- }, 1e3);
- const minZoom4 = 12;
+ // modules/svg/local_photos.js
+ var _initialized2 = false;
+ var _enabled2 = false;
+ var minViewfieldZoom = 16;
+ function svgLocalPhotos(projection2, context, dispatch14) {
+ const detected = utilDetect();
let layer = select_default2(null);
- let _mapilio;
- const viewFieldZoomLevel = 18;
+ let _fileList;
+ let _photos = [];
+ let _idAutoinc = 0;
+ let _photoFrame;
function init2() {
- if (svgMapilioImages.initialized)
+ if (_initialized2)
return;
- svgMapilioImages.enabled = false;
- svgMapilioImages.initialized = true;
+ _enabled2 = true;
+ function over(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ d3_event.dataTransfer.dropEffect = "copy";
+ }
+ context.container().attr("dropzone", "copy").on("drop.svgLocalPhotos", function(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ if (!detected.filedrop)
+ return;
+ drawPhotos.fileList(d3_event.dataTransfer.files, (loaded) => {
+ if (loaded.length > 0) {
+ drawPhotos.fitZoom(false);
+ }
+ });
+ }).on("dragenter.svgLocalPhotos", over).on("dragexit.svgLocalPhotos", over).on("dragover.svgLocalPhotos", over);
+ _initialized2 = true;
}
- function getService() {
- if (services.mapilio && !_mapilio) {
- _mapilio = services.mapilio;
- _mapilio.event.on("loadedImages", throttledRedraw);
- } else if (!services.mapilio && _mapilio) {
- _mapilio = null;
+ function ensureViewerLoaded(context2) {
+ if (_photoFrame) {
+ return Promise.resolve(_photoFrame);
}
- return _mapilio;
- }
- function showLayer() {
- const service = getService();
- if (!service)
- return;
- editOn();
- layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
- dispatch14.call("change");
+ const viewer = context2.container().select(".photoviewer").selectAll(".local-photos-wrapper").data([0]);
+ const viewerEnter = viewer.enter().append("div").attr("class", "photo-wrapper local-photos-wrapper").classed("hide", true);
+ viewerEnter.append("div").attr("class", "photo-attribution photo-attribution-dual fillD");
+ return plane_photo_default.init(context2, viewerEnter).then((planePhotoFrame) => {
+ _photoFrame = planePhotoFrame;
});
}
- function hideLayer() {
- throttledRedraw.cancel();
- layer.transition().duration(250).style("opacity", 0).on("end", editOff);
- }
- function transform2(d2) {
- let t2 = svgPointTransform(projection2)(d2);
- if (d2.heading) {
- t2 += " rotate(" + Math.floor(d2.heading) + ",0,0)";
- }
- return t2;
- }
- 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, image.id).then(function() {
- service.selectImage(context, image.id).showViewer(context);
+ function click(d3_event, image, zoomTo) {
+ ensureViewerLoaded(context).then(() => {
+ const viewer = context.container().select(".photoviewer").datum(image).classed("hide", false);
+ const viewerWrap = viewer.select(".local-photos-wrapper").classed("hide", false);
+ const attribution = viewerWrap.selectAll(".photo-attribution").text("");
+ if (image.date) {
+ attribution.append("span").text(image.date.toLocaleString());
+ }
+ if (image.name) {
+ attribution.append("span").classed("filename", true).text(image.name);
+ }
+ _photoFrame.selectPhoto({ image_path: "" });
+ image.getSrc().then((src) => {
+ _photoFrame.selectPhoto({ image_path: src }).showPhotoFrame(viewerWrap);
+ setStyles();
+ });
});
- context.map().centerEase(image.loc);
+ if (zoomTo) {
+ context.map().centerEase(image.loc);
+ }
}
- function mouseover(d3_event, image) {
- const service = getService();
- if (service)
- service.setStyles(context, image);
+ function transform2(d2) {
+ var svgpoint = projection2(d2.loc);
+ return "translate(" + svgpoint[0] + "," + svgpoint[1] + ")";
}
- function mouseout() {
- const service = getService();
- if (service)
- service.setStyles(context, null);
+ function setStyles(hovered) {
+ const viewer = context.container().select(".photoviewer");
+ const selected = viewer.empty() ? void 0 : viewer.datum();
+ context.container().selectAll(".layer-local-photos .viewfield-group").classed("hovered", (d2) => d2.id === (hovered == null ? void 0 : hovered.id)).classed("highlighted", (d2) => d2.id === (hovered == null ? void 0 : hovered.id) || d2.id === (selected == null ? void 0 : selected.id)).classed("currentView", (d2) => d2.id === (selected == null ? void 0 : selected.id));
}
- function update() {
- const z2 = ~~context.map().zoom();
- const showViewfields = z2 >= viewFieldZoomLevel;
- const service = getService();
- let sequences = service ? service.sequences(projection2) : [];
- let images = service ? service.images(projection2) : [];
- let traces = layer.selectAll(".sequences").selectAll(".sequence").data(sequences, function(d2) {
- return d2.properties.id;
- });
- traces.exit().remove();
- 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(d2) {
+ function display_markers(imageList) {
+ imageList = imageList.filter((image) => isArray_default(image.loc) && isNumber_default(image.loc[0]) && isNumber_default(image.loc[1]));
+ const groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(imageList, function(d2) {
return d2.id;
});
groups.exit().remove();
- const groupsEnter = groups.enter().append("g").attr("class", "viewfield-group").on("mouseenter", mouseover).on("mouseleave", mouseout).on("click", click);
+ const groupsEnter = groups.enter().append("g").attr("class", "viewfield-group").on("mouseenter", (d3_event, d2) => setStyles(d2)).on("mouseleave", () => setStyles(null)).on("click", click);
groupsEnter.append("g").attr("class", "viewfield-scale");
- const markers = groups.merge(groupsEnter).sort(function(a2, b2) {
- return b2.loc[1] - a2.loc[1];
- }).attr("transform", transform2).select(".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 showViewfields = context.map().zoom() >= minViewfieldZoom;
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", viewfieldPath);
- function viewfieldPath() {
- if (this.parentNode.__data__.isPano) {
- 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";
- }
- }
+ viewfields.enter().insert("path", "circle").attr("class", "viewfield").attr("transform", function() {
+ var _a2;
+ const d2 = this.parentNode.__data__;
+ return "rotate(".concat(Math.round((_a2 = d2.direction) != null ? _a2 : 0), ",0,0),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").style("visibility", function() {
+ const d2 = this.parentNode.__data__;
+ return isNumber_default(d2.direction) ? "visible" : "hidden";
+ });
}
- function drawImages(selection2) {
- const enabled = svgMapilioImages.enabled;
- const service = getService();
- layer = selection2.selectAll(".layer-mapilio").data(service ? [0] : []);
+ function drawPhotos(selection2) {
+ layer = selection2.selectAll(".layer-local-photos").data(_photos ? [0] : []);
layer.exit().remove();
- const layerEnter = layer.enter().append("g").attr("class", "layer-mapilio").style("display", enabled ? "block" : "none");
- layerEnter.append("g").attr("class", "sequences");
+ const layerEnter = layer.enter().append("g").attr("class", "layer-local-photos");
layerEnter.append("g").attr("class", "markers");
layer = layerEnter.merge(layer);
- if (enabled) {
- if (service && ~~context.map().zoom() >= minZoom4) {
- editOn();
- update();
- service.loadImages(projection2);
- service.loadLines(projection2);
- } else {
- editOff();
- }
+ if (_photos) {
+ display_markers(_photos);
}
}
- drawImages.enabled = function(_2) {
- if (!arguments.length)
- return svgMapilioImages.enabled;
- svgMapilioImages.enabled = _2;
- if (svgMapilioImages.enabled) {
- showLayer();
- context.photos().on("change.mapilio_images", null);
- } else {
- hideLayer();
- context.photos().on("change.mapilio_images", null);
+ function readFileAsDataURL(file) {
+ return new Promise((resolve, reject) => {
+ const reader = new FileReader();
+ reader.onload = () => resolve(reader.result);
+ reader.onerror = (error) => reject(error);
+ reader.readAsDataURL(file);
+ });
+ }
+ async function readmultifiles(files, callback) {
+ const loaded = [];
+ for (const file of files) {
+ try {
+ const exifData = await full_esm_default.parse(file);
+ const photo = {
+ id: _idAutoinc++,
+ name: file.name,
+ getSrc: () => readFileAsDataURL(file),
+ file,
+ loc: [exifData.longitude, exifData.latitude],
+ direction: exifData.GPSImgDirection,
+ date: exifData.CreateDate || exifData.DateTimeOriginal || exifData.ModifyDate
+ };
+ loaded.push(photo);
+ const sameName = _photos.filter((i3) => i3.name === photo.name);
+ if (sameName.length === 0) {
+ _photos.push(photo);
+ } else {
+ const thisContent = await photo.getSrc();
+ const sameNameContent = await Promise.allSettled(sameName.map((i3) => i3.getSrc()));
+ if (!sameNameContent.some((i3) => i3.value === thisContent)) {
+ _photos.push(photo);
+ }
+ }
+ } catch (err) {
+ }
}
+ if (typeof callback === "function")
+ callback(loaded);
dispatch14.call("change");
+ }
+ drawPhotos.setFiles = function(fileList, callback) {
+ readmultifiles(Array.from(fileList), callback);
return this;
};
- drawImages.supported = function() {
- return !!getService();
+ drawPhotos.fileList = function(fileList, callback) {
+ if (!arguments.length)
+ return _fileList;
+ _fileList = fileList;
+ if (!fileList || !fileList.length)
+ return this;
+ drawPhotos.setFiles(_fileList, callback);
+ return this;
};
- drawImages.rendered = function(zoom) {
- return zoom >= minZoom4;
+ drawPhotos.getPhotos = function() {
+ return _photos;
+ };
+ drawPhotos.removePhoto = function(id2) {
+ _photos = _photos.filter((i3) => i3.id !== id2);
+ dispatch14.call("change");
+ return _photos;
+ };
+ drawPhotos.openPhoto = click;
+ drawPhotos.fitZoom = function(force) {
+ const coords = _photos.map((image) => image.loc).filter((l2) => isArray_default(l2) && isNumber_default(l2[0]) && isNumber_default(l2[1]));
+ if (coords.length === 0)
+ return;
+ const extent = coords.map((l2) => geoExtent(l2, l2)).reduce((a2, b2) => a2.extend(b2));
+ const map2 = context.map();
+ var viewport = map2.trimmedExtent().polygon();
+ if (force !== false || !geoPolygonIntersectsPolygon(viewport, coords, true)) {
+ map2.centerZoom(extent.center(), Math.min(18, map2.trimmedExtentZoom(extent)));
+ }
};
- init2();
- return drawImages;
- }
-
- // modules/svg/osm.js
- function svgOsm(projection2, context, dispatch14) {
- var enabled = true;
- function drawOsm(selection2) {
- selection2.selectAll(".layer-osm").data(["covered", "areas", "lines", "points", "labels"]).enter().append("g").attr("class", function(d2) {
- return "layer-osm " + d2;
- });
- selection2.selectAll(".layer-osm.points").selectAll(".points-group").data(["points", "midpoints", "vertices", "turns"]).enter().append("g").attr("class", function(d2) {
- return "points-group " + d2;
- });
- }
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() {
+ layer.style("display", "block");
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
dispatch14.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);
- dispatch14.call("change");
+ layer.transition().duration(250).style("opacity", 0).on("end", () => {
+ layer.selectAll(".viewfield-group").remove();
+ layer.style("display", "none");
});
}
- drawOsm.enabled = function(val) {
+ drawPhotos.enabled = function(val) {
if (!arguments.length)
- return enabled;
- enabled = val;
- if (enabled) {
+ return _enabled2;
+ _enabled2 = val;
+ if (_enabled2) {
showLayer();
} else {
hideLayer();
@@ -53129,5674 +54190,5881 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
dispatch14.call("change");
return this;
};
- return drawOsm;
+ drawPhotos.hasData = function() {
+ return isArray_default(_photos) && _photos.length > 0;
+ };
+ init2();
+ return drawPhotos;
}
- // modules/svg/notes.js
- var _notesEnabled = false;
- var _osmService;
- function svgNotes(projection2, context, dispatch14) {
- if (!dispatch14) {
- dispatch14 = dispatch_default("change");
- }
- var throttledRedraw = throttle_default(function() {
- dispatch14.call("change");
- }, 1e3);
- var minZoom4 = 12;
- var touchLayer = select_default2(null);
- var drawLayer = select_default2(null);
- var _notesVisible = false;
+ // modules/svg/improveOSM.js
+ var _layerEnabled2 = false;
+ var _qaService2;
+ function svgImproveOSM(projection2, context, dispatch14) {
+ const throttledRedraw = throttle_default(() => dispatch14.call("change"), 1e3);
+ const minZoom4 = 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(-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");
+ 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.osm && !_osmService) {
- _osmService = services.osm;
- _osmService.on("loadedNotes", throttledRedraw);
- } else if (!services.osm && _osmService) {
- _osmService = null;
+ if (services.improveOSM && !_qaService2) {
+ _qaService2 = services.improveOSM;
+ _qaService2.on("loaded", throttledRedraw);
+ } else if (!services.improveOSM && _qaService2) {
+ _qaService2 = null;
}
- return _osmService;
+ return _qaService2;
}
function editOn() {
- if (!_notesVisible) {
- _notesVisible = true;
+ if (!layerVisible) {
+ layerVisible = true;
drawLayer.style("display", "block");
}
}
function editOff() {
- if (_notesVisible) {
- _notesVisible = false;
+ if (layerVisible) {
+ layerVisible = false;
drawLayer.style("display", "none");
- drawLayer.selectAll(".note").remove();
- touchLayer.selectAll(".note").remove();
+ 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", function() {
- dispatch14.call("change");
- });
+ drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch14.call("change"));
}
function layerOff() {
throttledRedraw.cancel();
drawLayer.interrupt();
- touchLayer.selectAll(".note").remove();
- drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", function() {
+ touchLayer.selectAll(".qaItem.improveOSM").remove();
+ drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
editOff();
dispatch14.call("change");
});
}
function updateMarkers() {
- if (!_notesVisible || !_notesEnabled)
+ if (!layerVisible || !_layerEnabled2)
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(d2) {
- return d2.status + d2.id;
- });
- notes.exit().remove();
- var notesEnter = notes.enter().append("g").attr("class", function(d2) {
- return "note note-" + d2.id + " " + d2.status;
- }).classed("new", function(d2) {
- return d2.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(d2) {
- return [d2];
- }).enter().append("use").attr("class", "icon-annotation").attr("width", "10px").attr("height", "10px").attr("x", "-3px").attr("y", "-19px").attr("xlink:href", function(d2) {
- if (d2.id < 0)
- return "#iD-icon-plus";
- if (d2.status === "open")
- return "#iD-icon-close";
- return "#iD-icon-apply";
- });
- notes.merge(notesEnter).sort(sortY).classed("selected", function(d2) {
- var mode = context.mode();
- var isMoving = mode && mode.id === "drag-note";
- return !isMoving && d2.id === selectedID;
- }).attr("transform", getTransform);
+ 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, (d2) => d2.id);
+ markers.exit().remove();
+ const markersEnter = markers.enter().append("g").attr("class", (d2) => "qaItem ".concat(d2.service, " itemId-").concat(d2.id, " itemType-").concat(d2.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", (d2) => d2.icon ? "#" + d2.icon : "");
+ markers.merge(markersEnter).sort(sortY).classed("selected", (d2) => d2.id === selectedID).attr("transform", getTransform);
if (touchLayer.empty())
return;
- var fillClass = context.getDebug("target") ? "pink " : "nocolor ";
- var targets = touchLayer.selectAll(".note").data(data, function(d2) {
- return d2.id;
- });
+ const fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ const targets = touchLayer.selectAll(".qaItem.improveOSM").data(data, (d2) => d2.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(d2) {
- var newClass = d2.id < 0 ? "new" : "";
- return "note target note-" + d2.id + " " + fillClass + newClass;
- }).attr("transform", getTransform);
+ targets.enter().append("rect").attr("width", "20px").attr("height", "30px").attr("x", "-10px").attr("y", "-28px").merge(targets).sort(sortY).attr("class", (d2) => "qaItem ".concat(d2.service, " target ").concat(fillClass, " itemId-").concat(d2.id)).attr("transform", getTransform);
function sortY(a2, b2) {
- if (a2.id === selectedID)
- return 1;
- if (b2.id === selectedID)
- return -1;
- return b2.loc[1] - a2.loc[1];
+ return a2.id === selectedID ? 1 : b2.id === selectedID ? -1 : b2.loc[1] - a2.loc[1];
}
}
- function drawNotes(selection2) {
- var service = getService();
- var surface = context.surface();
+ 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-notes").data(service ? [0] : []);
+ drawLayer = selection2.selectAll(".layer-improveOSM").data(service ? [0] : []);
drawLayer.exit().remove();
- drawLayer = drawLayer.enter().append("g").attr("class", "layer-notes").style("display", _notesEnabled ? "block" : "none").merge(drawLayer);
- if (_notesEnabled) {
+ drawLayer = drawLayer.enter().append("g").attr("class", "layer-improveOSM").style("display", _layerEnabled2 ? "block" : "none").merge(drawLayer);
+ if (_layerEnabled2) {
if (service && ~~context.map().zoom() >= minZoom4) {
editOn();
- service.loadNotes(projection2);
+ service.loadIssues(projection2);
updateMarkers();
} else {
editOff();
}
}
}
- drawNotes.enabled = function(val) {
+ drawImproveOSM.enabled = function(val) {
if (!arguments.length)
- return _notesEnabled;
- _notesEnabled = val;
- if (_notesEnabled) {
+ return _layerEnabled2;
+ _layerEnabled2 = val;
+ if (_layerEnabled2) {
layerOn();
} else {
layerOff();
- if (context.selectedNoteID()) {
+ if (context.selectedErrorID()) {
context.enter(modeBrowse(context));
}
}
dispatch14.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(d2) {
- return "layer-touch " + d2;
- });
- }
- 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 dispatch14 = dispatch_default("change");
- var svg2 = select_default2(null);
- var _layers = [
- { id: "osm", layer: svgOsm(projection2, context, dispatch14) },
- { id: "notes", layer: svgNotes(projection2, context, dispatch14) },
- { id: "data", layer: svgData(projection2, context, dispatch14) },
- { id: "keepRight", layer: svgKeepRight(projection2, context, dispatch14) },
- { id: "improveOSM", layer: svgImproveOSM(projection2, context, dispatch14) },
- { id: "osmose", layer: svgOsmose(projection2, context, dispatch14) },
- { id: "streetside", layer: svgStreetside(projection2, context, dispatch14) },
- { id: "mapillary", layer: svgMapillaryImages(projection2, context, dispatch14) },
- { id: "mapillary-position", layer: svgMapillaryPosition(projection2, context, dispatch14) },
- { id: "mapillary-map-features", layer: svgMapillaryMapFeatures(projection2, context, dispatch14) },
- { id: "mapillary-signs", layer: svgMapillarySigns(projection2, context, dispatch14) },
- { id: "kartaview", layer: svgKartaviewImages(projection2, context, dispatch14) },
- { id: "mapilio", layer: svgMapilioImages(projection2, context, dispatch14) },
- { id: "vegbilder", layer: svgVegbilder(projection2, context, dispatch14) },
- { id: "local-photos", layer: svgLocalPhotos(projection2, context, dispatch14) },
- { id: "debug", layer: svgDebug(projection2, context, dispatch14) },
- { id: "geolocate", layer: svgGeolocate(projection2, context, dispatch14) },
- { id: "touch", layer: svgTouch(projection2, context, dispatch14) }
- ];
- 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(d2) {
- return "data-layer " + d2.id;
- }).merge(groups).each(function(d2) {
- select_default2(this).call(d2.layer);
- });
- }
- drawLayers.all = function() {
- return _layers;
- };
- drawLayers.layer = function(id2) {
- var obj = _layers.find(function(o2) {
- return o2.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(o2) {
- return o2.id !== id2;
- });
- });
- dispatch14.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);
- }
- });
- dispatch14.call("change");
- return this;
- };
- drawLayers.dimensions = function(val) {
- if (!arguments.length)
- return utilGetDimensions(svg2);
- utilSetDimensions(svg2, val);
- return this;
- };
- return utilRebind(drawLayers, dispatch14, "on");
+ drawImproveOSM.supported = () => !!getService();
+ return drawImproveOSM;
}
- // 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(d2) {
- return filter2(d2.properties.entity);
- }).data(targetData, function key(d2) {
- return d2.id;
- });
- targets.exit().remove();
- var segmentWasEdited = function(d2) {
- var wayID = d2.properties.entity.id;
- if (!base.entities[wayID] || !(0, import_fast_deep_equal6.default)(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
- return false;
- }
- return d2.properties.nodes.some(function(n3) {
- return !base.entities[n3.id] || !(0, import_fast_deep_equal6.default)(graph.entities[n3.id].loc, base.entities[n3.id].loc);
- });
- };
- targets.enter().append("path").merge(targets).attr("d", getPath).attr("class", function(d2) {
- return "way line target target-allowed " + targetClass + d2.id;
- }).classed("segment-edited", segmentWasEdited);
- var nopeData = data.nopes.filter(getPath);
- var nopes = selection2.selectAll(".line.target-nope").filter(function(d2) {
- return filter2(d2.properties.entity);
- }).data(nopeData, function key(d2) {
- return d2.id;
- });
- nopes.exit().remove();
- nopes.enter().append("path").merge(nopes).attr("d", getPath).attr("class", function(d2) {
- return "way line target target-nope " + nopeClass + d2.id;
- }).classed("segment-edited", segmentWasEdited);
+ // modules/svg/osmose.js
+ var _layerEnabled3 = false;
+ var _qaService3;
+ function svgOsmose(projection2, context, dispatch14) {
+ const throttledRedraw = throttle_default(() => dispatch14.call("change"), 1e3);
+ const minZoom4 = 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 drawLines(selection2, graph, entities, filter2) {
- var base = context.history().base();
- function waystack(a2, b2) {
- var selected = context.selectedIDs();
- var scoreA = selected.indexOf(a2.id) !== -1 ? 20 : 0;
- var scoreB = selected.indexOf(b2.id) !== -1 ? 20 : 0;
- if (a2.tags.highway) {
- scoreA -= highway_stack[a2.tags.highway];
- }
- if (b2.tags.highway) {
- scoreB -= highway_stack[b2.tags.highway];
- }
- return scoreA - scoreB;
+ function getService() {
+ if (services.osmose && !_qaService3) {
+ _qaService3 = services.osmose;
+ _qaService3.on("loaded", throttledRedraw);
+ } else if (!services.osmose && _qaService3) {
+ _qaService3 = null;
}
- 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(d2) {
- var prefix = "way line";
- if (!d2.hasInterestingTags()) {
- var parentRelations = graph.parentRelations(d2);
- 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[d2.id] ? "old-multipolygon " : "";
- return prefix + " " + klass + " " + selectedClass + oldMPClass + d2.id;
- }).classed("added", function(d2) {
- return !base.entities[d2.id];
- }).classed("geometry-edited", function(d2) {
- return graph.entities[d2.id] && base.entities[d2.id] && !(0, import_fast_deep_equal6.default)(graph.entities[d2.id].nodes, base.entities[d2.id].nodes);
- }).classed("retagged", function(d2) {
- return graph.entities[d2.id] && base.entities[d2.id] && !(0, import_fast_deep_equal6.default)(graph.entities[d2.id].tags, base.entities[d2.id].tags);
- }).call(svgTagClasses()).merge(lines).sort(waystack).attr("d", getPath).call(svgTagClasses().tags(svgRelationMemberTags(graph)));
- return selection3;
+ return _qaService3;
+ }
+ function editOn() {
+ if (!layerVisible) {
+ layerVisible = true;
+ drawLayer.style("display", "block");
}
- function getPathData(isSelected) {
- return function() {
- var layer = this.parentNode.__data__;
- var data = pathdata[layer] || [];
- return data.filter(function(d2) {
- if (isSelected) {
- return context.selectedIDs().indexOf(d2.id) !== -1;
- } else {
- return context.selectedIDs().indexOf(d2.id) === -1;
- }
- });
- };
+ }
+ function editOff() {
+ if (layerVisible) {
+ layerVisible = false;
+ drawLayer.style("display", "none");
+ drawLayer.selectAll(".qaItem.osmose").remove();
+ touchLayer.selectAll(".qaItem.osmose").remove();
}
- 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(d2) {
- return [d2.id, d2.index];
- }
- );
- markers.exit().remove();
- markers = markers.enter().append("path").attr("class", pathclass).merge(markers).attr("marker-mid", marker).attr("d", function(d2) {
- return d2.d;
- });
- if (detected.ie) {
- markers.each(function() {
- this.parentNode.insertBefore(this, this);
- });
- }
+ }
+ function layerOn() {
+ editOn();
+ drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", () => dispatch14.call("change"));
+ }
+ function layerOff() {
+ throttledRedraw.cancel();
+ drawLayer.interrupt();
+ touchLayer.selectAll(".qaItem.osmose").remove();
+ drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", () => {
+ editOff();
+ dispatch14.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, (d2) => d2.id);
+ markers.exit().remove();
+ const markersEnter = markers.enter().append("g").attr("class", (d2) => "qaItem ".concat(d2.service, " itemId-").concat(d2.id, " itemType-").concat(d2.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", (d2) => service.getColor(d2.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", (d2) => d2.icon ? "#" + d2.icon : "");
+ markers.merge(markersEnter).sort(sortY).classed("selected", (d2) => d2.id === selectedID).attr("transform", getTransform);
+ if (touchLayer.empty())
+ return;
+ const fillClass = context.getDebug("target") ? "pink" : "nocolor";
+ const targets = touchLayer.selectAll(".qaItem.osmose").data(data, (d2) => d2.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", (d2) => "qaItem ".concat(d2.service, " target ").concat(fillClass, " itemId-").concat(d2.id)).attr("transform", getTransform);
+ function sortY(a2, b2) {
+ return a2.id === selectedID ? 1 : b2.id === selectedID ? -1 : b2.loc[1] - a2.loc[1];
}
- var getPath = svgPath(projection2, graph);
- var ways = [];
- var onewaydata = {};
- var sideddata = {};
- var oldMultiPolygonOuters = {};
- for (var i3 = 0; i3 < entities.length; i3++) {
- var entity = entities[i3];
- 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);
+ }
+ 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() >= minZoom4) {
+ editOn();
+ service.loadIssues(projection2);
+ updateMarkers();
+ } else {
+ editOff();
}
}
- ways = ways.filter(getPath);
- var pathdata = utilArrayGroupBy(ways, function(way) {
- return way.layer();
- });
- Object.keys(pathdata).forEach(function(k2) {
- var v2 = pathdata[k2];
- var onewayArr = v2.filter(function(d2) {
- return d2.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[k2] = utilArrayFlatten(onewayArr.map(onewaySegments));
- var sidedArr = v2.filter(function(d2) {
- return d2.isSided();
- });
- var sidedSegments = svgMarkerSegments(
- projection2,
- graph,
- 30,
- function shouldReverse() {
- return false;
- },
- function bothDirections() {
- return false;
- }
- );
- sideddata[k2] = 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(d2) {
- return "layergroup layer" + String(d2);
- }).merge(layergroup);
- layergroup.selectAll("g.linegroup").data(["shadow", "casing", "stroke", "shadow-highlighted", "casing-highlighted", "stroke-highlighted"]).enter().append("g").attr("class", function(d2) {
- return "linegroup line-" + d2;
- });
- 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(d2) {
- var category = graph.entity(d2.id).sidednessIdentifier();
- return "url(#ideditor-sided-marker-" + category + ")";
- }
- );
- });
- touchLayer.call(drawTargets, graph, ways, filter2);
}
- return drawLines;
+ 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));
+ }
+ }
+ dispatch14.call("change");
+ return this;
+ };
+ drawOsmose.supported = () => !!getService();
+ return drawOsmose;
}
- // 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
- }
- };
+ // modules/svg/streetside.js
+ function svgStreetside(projection2, context, dispatch14) {
+ var throttledRedraw = throttle_default(function() {
+ dispatch14.call("change");
+ }, 1e3);
+ var minZoom4 = 14;
+ var minMarkerZoom = 16;
+ var minViewfieldZoom2 = 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() {
+ dispatch14.call("change");
});
- var targets = selection2.selectAll(".midpoint.target").filter(function(d2) {
- return filter2(d2.properties.entity);
- }).data(data, function key(d2) {
- return d2.id;
+ }
+ 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, d2) {
+ var service = getService();
+ if (!service)
+ return;
+ if (d2.sequenceKey !== _selectedSequence) {
+ _viewerYaw = 0;
+ }
+ _selectedSequence = d2.sequenceKey;
+ service.ensureViewerLoaded(context).then(function() {
+ service.selectImage(context, d2.key).yaw(_viewerYaw).showViewer(context);
});
- targets.exit().remove();
- targets.enter().append("circle").attr("r", targetRadius).merge(targets).attr("class", function(d2) {
- return "node midpoint target " + fillClass + d2.id;
- }).attr("transform", getTransform);
+ context.map().centerEase(d2.loc);
}
- 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();
+ function mouseover(d3_event, d2) {
+ var service = getService();
+ if (service)
+ service.setStyles(context, d2);
+ }
+ function mouseout() {
+ var service = getService();
+ if (service)
+ service.setStyles(context, null);
+ }
+ function transform2(d2) {
+ var t2 = svgPointTransform(projection2)(d2);
+ var rot = d2.ca + _viewerYaw;
+ if (rot) {
+ t2 += " rotate(" + Math.floor(rot) + ",0,0)";
+ }
+ return t2;
+ }
+ 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;
+ });
}
- var poly = extent.polygon();
- var midpoints = {};
- for (var i3 = 0; i3 < entities.length; i3++) {
- var entity = entities[i3];
- if (entity.type !== "way")
- continue;
- if (!filter2(entity))
- continue;
- if (context.selectedIDs().indexOf(entity.id) < 0)
- continue;
- var nodes = graph.childNodes(entity);
- for (var j3 = 0; j3 < nodes.length - 1; j3++) {
- var a2 = nodes[j3];
- var b2 = nodes[j3 + 1];
- var id2 = [a2.id, b2.id].sort().join("-");
- if (midpoints[id2]) {
- midpoints[id2].parents.push(entity);
- } else if (geoVecLength(projection2(a2.loc), projection2(b2.loc)) > 40) {
- var point2 = geoVecInterp(a2.loc, b2.loc, 0.5);
- var loc = null;
- if (extent.intersects(point2)) {
- loc = point2;
- } else {
- for (var k2 = 0; k2 < 4; k2++) {
- point2 = geoLineIntersection([a2.loc, b2.loc], [poly[k2], poly[k2 + 1]]);
- if (point2 && geoVecLength(projection2(a2.loc), projection2(point2)) > 20 && geoVecLength(projection2(b2.loc), projection2(point2)) > 20) {
- loc = point2;
- break;
- }
- }
- }
- if (loc) {
- midpoints[id2] = {
- type: "midpoint",
- id: id2,
- loc,
- edge: [a2.id, b2.id],
- parents: [entity]
- };
- }
- }
- }
+ 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;
+ });
}
- function midpointFilter(d2) {
- if (midpoints[d2.id])
- return true;
- for (var i4 = 0; i4 < d2.parents.length; i4++) {
- if (filter2(d2.parents[i4])) {
- return true;
- }
- }
- return false;
+ if (usernames) {
+ sequences = sequences.filter(function(sequences2) {
+ return usernames.indexOf(sequences2.properties.captured_by) !== -1;
+ });
}
- var groups = drawLayer.selectAll(".midpoint").filter(midpointFilter).data(Object.values(midpoints), function(d2) {
- return d2.id;
+ return sequences;
+ }
+ function update() {
+ var viewer = context.container().select(".photoviewer");
+ var selected = viewer.empty() ? void 0 : viewer.datum();
+ var z2 = ~~context.map().zoom();
+ var showMarkers = z2 >= minMarkerZoom;
+ var showViewfields = z2 >= minViewfieldZoom2;
+ 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(d2) {
+ return d2.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(d2) {
+ return d2.key + (d2.sequenceKey ? "v1" : "v0");
});
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(d2) {
- var translate = svgPointTransform(projection2);
- var a3 = graph.entity(d2.edge[0]);
- var b3 = graph.entity(d2.edge[1]);
- var angle2 = geoAngle(a3, b3, projection2) * (180 / Math.PI);
- return translate(d2) + " rotate(" + angle2 + ")";
- }).call(svgTagClasses().tags(
- function(d2) {
- return d2.parents[0].tags;
+ 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(a2, b2) {
+ return a2 === selected ? 1 : b2 === selected ? -1 : b2.loc[1] - a2.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 d2 = this.parentNode.__data__;
+ if (d2.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";
}
- ));
- groups.select("polygon.shadow");
- groups.select("polygon.fill");
- touchLayer.call(drawTargets, graph, Object.values(midpoints), midpointFilter);
+ }
}
- return drawMidpoints;
+ 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() >= minZoom4) {
+ editOn();
+ update();
+ service.loadBubbles(projection2);
+ } else {
+ editOff();
+ }
+ }
+ }
+ drawImages.enabled = function(_2) {
+ if (!arguments.length)
+ return svgStreetside.enabled;
+ svgStreetside.enabled = _2;
+ if (svgStreetside.enabled) {
+ showLayer();
+ context.photos().on("change.streetside", update);
+ } else {
+ hideLayer();
+ context.photos().on("change.streetside", null);
+ }
+ dispatch14.call("change");
+ return this;
+ };
+ drawImages.supported = function() {
+ return !!getService();
+ };
+ drawImages.rendered = function(zoom) {
+ return zoom >= minZoom4;
+ };
+ init2();
+ return drawImages;
}
- // 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");
+ // modules/svg/vegbilder.js
+ function svgVegbilder(projection2, context, dispatch14) {
+ const throttledRedraw = throttle_default(() => dispatch14.call("change"), 1e3);
+ const minZoom4 = 14;
+ const minMarkerZoom = 16;
+ const minViewfieldZoom2 = 18;
+ let layer = select_default2(null);
+ let _viewerYaw = 0;
+ let _vegbilder;
+ function init2() {
+ if (svgVegbilder.initialized)
+ return;
+ svgVegbilder.enabled = false;
+ svgVegbilder.initialized = true;
}
- function sortY(a2, b2) {
- return b2.loc[1] - a2.loc[1];
+ function getService() {
+ if (services.vegbilder && !_vegbilder) {
+ _vegbilder = services.vegbilder;
+ _vegbilder.event.on("viewerChanged.svgVegbilder", viewerChanged).on("loadedImages.svgVegbilder", throttledRedraw);
+ } else if (!services.vegbilder && _vegbilder) {
+ _vegbilder = null;
+ }
+ return _vegbilder;
}
- function fastEntityKey(d2) {
- var mode = context.mode();
- var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
- return isMoving ? d2.id : osmEntity.key(d2);
+ function showLayer() {
+ const service = getService();
+ if (!service)
+ return;
+ editOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", () => dispatch14.call("change"));
}
- 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(d2) {
- return filter2(d2.properties.entity);
- }).data(data, function key(d2) {
- return d2.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(d2) {
- return "node point target " + fillClass + d2.id;
- }).attr("transform", getTransform);
+ function hideLayer() {
+ throttledRedraw.cancel();
+ layer.transition().duration(250).style("opacity", 0).on("end", editOff);
}
- 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(d2) {
- return "node point " + d2.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(d2) {
- return !base.entities[d2.id];
- }).classed("moved", function(d2) {
- return base.entities[d2.id] && !(0, import_fast_deep_equal7.default)(graph.entities[d2.id].loc, base.entities[d2.id].loc);
- }).classed("retagged", function(d2) {
- return base.entities[d2.id] && !(0, import_fast_deep_equal7.default)(graph.entities[d2.id].tags, base.entities[d2.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);
+ function editOn() {
+ layer.style("display", "block");
}
- return drawPoints;
- }
-
- // modules/svg/turns.js
- function svgTurns(projection2, context) {
- function icon2(turn) {
- var u2 = turn.u ? "-u" : "";
- if (turn.no)
- return "#iD-turn-no" + u2;
- if (turn.only)
- return "#iD-turn-only" + u2;
- return "#iD-turn-yes" + u2;
+ function editOff() {
+ layer.selectAll(".viewfield-group").remove();
+ layer.style("display", "none");
}
- function drawTurns(selection2, graph, turns) {
- function turnTransform(d2) {
- var pxRadius = 50;
- var toWay = graph.entity(d2.to.way);
- var toPoints = graph.childNodes(toWay).map(function(n3) {
- return n3.loc;
- }).map(projection2);
- var toLength = geoPathLength(toPoints);
- var mid = toLength / 2;
- var toNode = graph.entity(d2.to.node);
- var toVertex = graph.entity(d2.to.vertex);
- var a2 = geoAngle(toVertex, toNode, projection2);
- var o2 = projection2(toVertex.loc);
- var r2 = d2.u ? 0 : !toWay.__via ? pxRadius : Math.min(mid, pxRadius);
- return "translate(" + (r2 * Math.cos(a2) + o2[0]) + "," + (r2 * Math.sin(a2) + o2[1]) + ") rotate(" + a2 * 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(d2) {
- return d2.key;
- });
- groups.exit().remove();
- var groupsEnter = groups.enter().append("g").attr("class", function(d2) {
- return "turn " + d2.key;
- });
- var turnsEnter = groupsEnter.filter(function(d2) {
- return !d2.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(d2) {
- return d2.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(d2) {
- return d2.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(d2) {
- return d2.key;
- });
- groups.exit().remove();
- groupsEnter = groups.enter().append("g").attr("class", function(d2) {
- return "turn " + d2.key;
- });
- turnsEnter = groupsEnter.filter(function(d2) {
- return !d2.u;
- });
- turnsEnter.append("rect").attr("class", "target " + fillClass).attr("transform", "translate(-22, -12)").attr("width", "44").attr("height", "24");
- uEnter = groupsEnter.filter(function(d2) {
- return d2.u;
+ function click(d3_event, d2) {
+ const service = getService();
+ if (!service)
+ return;
+ service.ensureViewerLoaded(context).then(() => {
+ service.selectImage(context, d2.key).showViewer(context);
});
- 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;
+ context.map().centerEase(d2.loc);
}
- 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(a2, b2) {
- return b2.loc[1] - a2.loc[1];
+ function mouseover(d3_event, d2) {
+ const service = getService();
+ if (service)
+ service.setStyles(context, d2);
}
- function fastEntityKey(d2) {
- var mode = context.mode();
- var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
- return isMoving ? d2.id : osmEntity.key(d2);
+ function mouseout() {
+ const service = getService();
+ if (service)
+ service.setStyles(context, null);
}
- 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 z2 = zoom < 17 ? 0 : zoom < 18 ? 1 : 2;
- var activeID = context.activeID();
- var base = context.history().base();
- function getIcon(d2) {
- var entity = graph.entity(d2.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 transform2(d2, selected) {
+ let t2 = svgPointTransform(projection2)(d2);
+ let rot = d2.ca;
+ if (d2 === selected) {
+ rot += _viewerYaw;
}
- function updateAttributes(selection3) {
- ["shadow", "stroke", "fill"].forEach(function(klass) {
- var rads = radiuses[klass];
- selection3.selectAll("." + klass).each(function(entity) {
- var i3 = z2 && getIcon(entity);
- var r2 = rads[i3 ? 3 : z2];
- if (entity.id !== activeID && entity.isEndpoint(graph) && !entity.isConnected(graph)) {
- r2 += 1.5;
- }
- if (klass === "shadow") {
- _radii[entity.id] = r2;
- }
- select_default2(this).attr("r", r2).attr("visibility", i3 && klass === "fill" ? "hidden" : null);
- });
- });
+ if (rot) {
+ t2 += " rotate(" + Math.floor(rot) + ",0,0)";
}
- 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(d2) {
- return "node vertex " + d2.id;
- }).order();
- enter.append("circle").attr("class", "shadow");
- enter.append("circle").attr("class", "stroke");
- enter.filter(function(d2) {
- return d2.hasInterestingTags();
- }).append("circle").attr("class", "fill");
- groups = groups.merge(enter).attr("transform", svgPointTransform(projection2)).classed("sibling", function(d2) {
- return d2.id in sets2.selected;
- }).classed("shared", function(d2) {
- return graph.isShared(d2);
- }).classed("endpoint", function(d2) {
- return d2.isEndpoint(graph);
- }).classed("added", function(d2) {
- return !base.entities[d2.id];
- }).classed("moved", function(d2) {
- return base.entities[d2.id] && !(0, import_fast_deep_equal8.default)(graph.entities[d2.id].loc, base.entities[d2.id].loc);
- }).classed("retagged", function(d2) {
- return base.entities[d2.id] && !(0, import_fast_deep_equal8.default)(graph.entities[d2.id].tags, base.entities[d2.id].tags);
- }).call(updateAttributes);
- var iconUse = groups.selectAll(".icon").data(function data(d2) {
- return zoom >= 17 && getIcon(d2) ? [d2] : [];
- }, 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(d2) {
- var picon = getIcon(d2);
- return picon ? "#" + picon : "";
- });
- var dgroups = groups.selectAll(".viewfieldgroup").data(function data(d2) {
- return zoom >= 18 && getDirections(d2) ? [d2] : [];
- }, fastEntityKey);
- dgroups.exit().remove();
- dgroups = dgroups.enter().insert("g", ".shadow").attr("class", "viewfieldgroup").merge(dgroups);
- var viewfields = dgroups.selectAll(".viewfield").data(getDirections, function key(d2) {
- return osmEntity.key(d2);
- });
- 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(d2) {
- return "rotate(" + d2 + ")";
- });
+ return t2;
}
- 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(d2) {
- return filter2(d2.properties.entity);
- }).data(data.targets, function key(d2) {
- return d2.id;
- });
- targets.exit().remove();
- targets.enter().append("circle").attr("r", function(d2) {
- return _radii[d2.id] || radiuses.shadow[3];
- }).merge(targets).attr("class", function(d2) {
- return "node vertex target target-allowed " + targetClass + d2.id;
- }).attr("transform", getTransform);
- var nopes = selection2.selectAll(".vertex.target-nope").filter(function(d2) {
- return filter2(d2.properties.entity);
- }).data(data.nopes, function key(d2) {
- return d2.id;
- });
- nopes.exit().remove();
- nopes.enter().append("circle").attr("r", function(d2) {
- return _radii[d2.properties.entity.id] || radiuses.shadow[3];
- }).merge(nopes).attr("class", function(d2) {
- return "node vertex target target-nope " + nopeClass + d2.id;
- }).attr("transform", getTransform);
+ function viewerChanged() {
+ const service = getService();
+ if (!service)
+ return;
+ const frame2 = service.photoFrame();
+ _viewerYaw = frame2.getYaw();
+ if (context.map().isTransformed())
+ return;
+ layer.selectAll(".viewfield-group.currentView").attr("transform", (d2) => transform2(d2, d2));
}
- 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 filterImages(images) {
+ const photoContext = context.photos();
+ const fromDateString = photoContext.fromDate();
+ const toDateString = photoContext.toDate();
+ const showsFlat = photoContext.showsFlat();
+ const showsPano = photoContext.showsPanoramic();
+ if (fromDateString) {
+ const fromDate = new Date(fromDateString);
+ images = images.filter((image) => image.captured_at.getTime() >= fromDate.getTime());
+ }
+ if (toDateString) {
+ const toDate = new Date(toDateString);
+ images = images.filter((image) => image.captured_at.getTime() <= toDate.getTime());
+ }
+ if (!showsPano) {
+ images = images.filter((image) => !image.is_sphere);
+ }
+ if (!showsFlat) {
+ images = images.filter((image) => image.is_sphere);
+ }
+ return images;
}
- 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 filterSequences(sequences) {
+ const photoContext = context.photos();
+ const fromDateString = photoContext.fromDate();
+ const toDateString = photoContext.toDate();
+ const showsFlat = photoContext.showsFlat();
+ const showsPano = photoContext.showsPanoramic();
+ if (fromDateString) {
+ const fromDate = new Date(fromDateString);
+ sequences = sequences.filter(({ images }) => images[0].captured_at.getTime() >= fromDate.getTime());
+ }
+ if (toDateString) {
+ const toDate = new Date(toDateString);
+ sequences = sequences.filter(({ images }) => images[images.length - 1].captured_at.getTime() <= toDate.getTime());
+ }
+ if (!showsPano) {
+ sequences = sequences.filter(({ images }) => !images[0].is_sphere);
+ }
+ if (!showsFlat) {
+ sequences = sequences.filter(({ images }) => images[0].is_sphere);
+ }
+ return sequences;
}
- 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 i3;
- if (entity.type === "way") {
- for (i3 = 0; i3 < entity.nodes.length; i3++) {
- var child = graph.hasEntity(entity.nodes[i3]);
- if (child) {
- addChildVertices(child);
- }
- }
- } else if (entity.type === "relation") {
- for (i3 = 0; i3 < entity.members.length; i3++) {
- var member = graph.hasEntity(entity.members[i3].id);
- if (member) {
- addChildVertices(member);
- }
- }
- } else if (renderAsVertex(entity, graph, wireframe, zoom)) {
- results[entity.id] = entity;
- }
- }
+ function update() {
+ const viewer = context.container().select(".photoviewer");
+ const selected = viewer.empty() ? void 0 : viewer.datum();
+ const z2 = ~~context.map().zoom();
+ const showMarkers = z2 >= minMarkerZoom;
+ const showViewfields = z2 >= minViewfieldZoom2;
+ const service = getService();
+ let sequences = [];
+ let images = [];
+ if (service) {
+ service.loadImages(context);
+ sequences = service.sequences(projection2);
+ images = showMarkers ? service.images(projection2) : [];
+ images = filterImages(images);
+ sequences = filterSequences(sequences);
}
- 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);
- });
- }
+ let traces = layer.selectAll(".sequences").selectAll(".sequence").data(sequences, (d2) => d2.key);
+ traces.exit().remove();
+ traces.enter().append("path").attr("class", "sequence").merge(traces).attr("d", svgPath(projection2).geojson);
+ const groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(images, (d2) => d2.key);
+ 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((a2, b2) => {
+ return a2 === selected ? 1 : b2 === selected ? -1 : b2.loc[1] - a2.loc[1];
+ }).attr("transform", (d2) => transform2(d2, selected)).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", viewfieldPath);
+ function viewfieldPath() {
+ const d2 = this.parentNode.__data__;
+ if (d2.is_sphere) {
+ return "M 8,13 m -10,0 a 10,10 0 1,0 20,0 a 10,10 0 1,0 -20,0";
} else {
- addChildVertices(entity);
+ 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 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 i3 = 0; i3 < entities.length; i3++) {
- var entity = entities[i3];
- 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];
+ }
+ function drawImages(selection2) {
+ const enabled = svgVegbilder.enabled;
+ const service = getService();
+ layer = selection2.selectAll(".layer-vegbilder").data(service ? [0] : []);
+ layer.exit().remove();
+ const layerEnter = layer.enter().append("g").attr("class", "layer-vegbilder").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() >= minZoom4) {
+ editOn();
+ update();
+ service.loadImages(context);
+ } else {
+ editOff();
}
}
- 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(d2) {
- return d2.id in _currPersistent || d2.id in _currSelected || d2.id in _currHover || filter2(d2);
- };
- drawLayer.call(draw, graph, currentVisible(all), sets2, filterRendered);
- var filterTouch = function(d2) {
- return isMoving ? true : filterRendered(d2);
- };
- 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;
- }
- }
- });
+ drawImages.enabled = function(_2) {
+ if (!arguments.length)
+ return svgVegbilder.enabled;
+ svgVegbilder.enabled = _2;
+ if (svgVegbilder.enabled) {
+ showLayer();
+ context.photos().on("change.vegbilder", update);
} else {
- _currSelected = getSiblingAndChildVertices(context.selectedIDs(), graph, wireframe, zoom);
+ hideLayer();
+ context.photos().on("change.vegbilder", null);
}
- var filter2 = function(d2) {
- return d2.id in _prevSelected;
- };
- drawVertices(selection2, graph, Object.values(_prevSelected), filter2, extent, false);
+ dispatch14.call("change");
+ return this;
};
- 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(d2) {
- return d2.id in _prevHover;
- };
- drawVertices(selection2, graph, Object.values(_prevHover), filter2, extent, false);
+ drawImages.supported = function() {
+ return !!getService();
};
- return drawVertices;
+ drawImages.rendered = function(zoom) {
+ return zoom >= minZoom4;
+ };
+ drawImages.validHere = function(extent, zoom) {
+ return zoom >= minZoom4 - 2 && getService().validHere(extent);
+ };
+ init2();
+ return drawImages;
}
- // 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);
+ // modules/svg/mapillary_images.js
+ function svgMapillaryImages(projection2, context, dispatch14) {
+ const throttledRedraw = throttle_default(function() {
+ dispatch14.call("change");
+ }, 1e3);
+ const minZoom4 = 12;
+ const minMarkerZoom = 16;
+ const minViewfieldZoom2 = 18;
+ let layer = select_default2(null);
+ let _mapillary;
+ function init2() {
+ if (svgMapillaryImages.initialized)
+ return;
+ svgMapillaryImages.enabled = false;
+ svgMapillaryImages.initialized = true;
}
- 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 e3 = this;
- if (e3 instanceof SVGElement) {
- e3 = e3.ownerSVGElement || e3;
- if (e3.hasAttribute("viewBox")) {
- e3 = e3.viewBox.baseVal;
- return [[e3.x, e3.y], [e3.x + e3.width, e3.y + e3.height]];
+ function getService() {
+ if (services.mapillary && !_mapillary) {
+ _mapillary = services.mapillary;
+ _mapillary.event.on("loadedImages", throttledRedraw);
+ } else if (!services.mapillary && _mapillary) {
+ _mapillary = null;
}
- return [[0, 0], [e3.width.baseVal.value, e3.height.baseVal.value]];
- }
- return [[0, 0], [e3.clientWidth, e3.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, dispatch14 = 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);
+ return _mapillary;
}
- 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, k2, p2) {
- zoom.scaleTo(selection2, function() {
- var k0 = _transform.k, k1 = typeof k2 === "function" ? k2.apply(this, arguments) : k2;
- return k0 * k1;
- }, p2);
- };
- zoom.scaleTo = function(selection2, k2, p2) {
- zoom.transform(selection2, function() {
- var e3 = extent.apply(this, arguments), t0 = _transform, p02 = !p2 ? centroid(e3) : typeof p2 === "function" ? p2.apply(this, arguments) : p2, p1 = t0.invert(p02), k1 = typeof k2 === "function" ? k2.apply(this, arguments) : k2;
- return constrain(translate(scale(t0, k1), p02, p1), e3, translateExtent);
- }, p2);
- };
- zoom.translateBy = function(selection2, x2, y2) {
- zoom.transform(selection2, function() {
- return constrain(_transform.translate(
- typeof x2 === "function" ? x2.apply(this, arguments) : x2,
- typeof y2 === "function" ? y2.apply(this, arguments) : y2
- ), extent.apply(this, arguments), translateExtent);
+ function showLayer() {
+ const service = getService();
+ if (!service)
+ return;
+ editOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
+ dispatch14.call("change");
});
- };
- zoom.translateTo = function(selection2, x2, y2, p2) {
- zoom.transform(selection2, function() {
- var e3 = extent.apply(this, arguments), t2 = _transform, p02 = !p2 ? centroid(e3) : typeof p2 === "function" ? p2.apply(this, arguments) : p2;
- return constrain(identity2.translate(p02[0], p02[1]).scale(t2.k).translate(
- typeof x2 === "function" ? -x2.apply(this, arguments) : -x2,
- typeof y2 === "function" ? -y2.apply(this, arguments) : -y2
- ), e3, translateExtent);
- }, p2);
- };
- function scale(transform2, k2) {
- k2 = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k2));
- return k2 === transform2.k ? transform2 : new Transform(k2, transform2.x, transform2.y);
}
- function translate(transform2, p02, p1) {
- var x2 = p02[0] - p1[0] * transform2.k, y2 = p02[1] - p1[1] * transform2.k;
- return x2 === transform2.x && y2 === transform2.y ? transform2 : new Transform(transform2.k, x2, y2);
+ function hideLayer() {
+ throttledRedraw.cancel();
+ layer.transition().duration(250).style("opacity", 0).on("end", editOff);
}
- function centroid(extent2) {
- return [(+extent2[0][0] + +extent2[1][0]) / 2, (+extent2[0][1] + +extent2[1][1]) / 2];
+ function editOn() {
+ layer.style("display", "block");
}
- 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, g3 = gesture(that, args), e3 = extent.apply(that, args), p2 = !point2 ? centroid(e3) : typeof point2 === "function" ? point2.apply(that, args) : point2, w2 = Math.max(e3[1][0] - e3[0][0], e3[1][1] - e3[0][1]), a2 = _transform, b2 = typeof transform2 === "function" ? transform2.apply(that, args) : transform2, i3 = interpolate(a2.invert(p2).concat(w2 / a2.k), b2.invert(p2).concat(w2 / b2.k));
- return function(t2) {
- if (t2 === 1) {
- t2 = b2;
- } else {
- var l2 = i3(t2);
- var k2 = w2 / l2[2];
- t2 = new Transform(k2, p2[0] - l2[0] * k2, p2[1] - l2[1] * k2);
- }
- g3.zoom(null, null, t2);
- };
+ 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 gesture(that, args, clean2) {
- return !clean2 && _activeGesture || new Gesture(that, args);
+ function mouseover(d3_event, image) {
+ const service = getService();
+ if (service)
+ service.setStyles(context, image);
}
- function Gesture(that, args) {
- this.that = that;
- this.args = args;
- this.active = 0;
- this.extent = extent.apply(that, args);
+ function mouseout() {
+ const service = getService();
+ if (service)
+ service.setStyles(context, null);
}
- Gesture.prototype = {
- start: function(d3_event) {
- if (++this.active === 1) {
- _activeGesture = this;
- dispatch14.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;
- dispatch14.call("zoom", this, d3_event, key, transform2);
- return this;
- },
- end: function(d3_event) {
- if (--this.active === 0) {
- _activeGesture = null;
- dispatch14.call("end", this, d3_event);
- }
- return this;
- }
- };
- function wheeled(d3_event) {
- if (!filter2.apply(this, arguments))
- return;
- var g3 = gesture(this, arguments), t2 = _transform, k2 = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t2.k * Math.pow(2, wheelDelta.apply(this, arguments)))), p2 = utilFastMouse(this)(d3_event);
- if (g3.wheel) {
- if (g3.mouse[0][0] !== p2[0] || g3.mouse[0][1] !== p2[1]) {
- g3.mouse[1] = t2.invert(g3.mouse[0] = p2);
- }
- clearTimeout(g3.wheel);
- } else {
- g3.mouse = [p2, t2.invert(p2)];
- interrupt_default(this);
- g3.start(d3_event);
- }
- d3_event.preventDefault();
- d3_event.stopImmediatePropagation();
- g3.wheel = setTimeout(wheelidled, _wheelDelay);
- g3.zoom(d3_event, "mouse", constrain(translate(scale(t2, k2), g3.mouse[0], g3.mouse[1]), g3.extent, translateExtent));
- function wheelidled() {
- g3.wheel = null;
- g3.end(d3_event);
+ function transform2(d2) {
+ let t2 = svgPointTransform(projection2)(d2);
+ if (d2.ca) {
+ t2 += " rotate(" + Math.floor(d2.ca) + ",0,0)";
}
+ return t2;
}
- var _downPointerIDs = /* @__PURE__ */ new Set();
- var _pointerLocGetter;
- function pointerdown(d3_event) {
- _downPointerIDs.add(d3_event.pointerId);
- if (!filter2.apply(this, arguments))
- return;
- var g3 = gesture(this, arguments, _downPointerIDs.size === 1);
- var started;
- d3_event.stopImmediatePropagation();
- _pointerLocGetter = utilFastMouse(this);
- var loc = _pointerLocGetter(d3_event);
- var p2 = [loc, _transform.invert(loc), d3_event.pointerId];
- if (!g3.pointer0) {
- g3.pointer0 = p2;
- started = true;
- } else if (!g3.pointer1 && g3.pointer0[2] !== p2[2]) {
- g3.pointer1 = p2;
+ 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 (started) {
- interrupt_default(this);
- g3.start(d3_event);
+ 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 pointermove(d3_event) {
- if (!_downPointerIDs.has(d3_event.pointerId))
- return;
- if (!_activeGesture || !_pointerLocGetter)
- return;
- var g3 = gesture(this, arguments);
- var isPointer0 = g3.pointer0 && g3.pointer0[2] === d3_event.pointerId;
- var isPointer1 = !isPointer0 && g3.pointer1 && g3.pointer1[2] === d3_event.pointerId;
- if ((isPointer0 || isPointer1) && "buttons" in d3_event && !d3_event.buttons) {
- if (g3.pointer0)
- _downPointerIDs.delete(g3.pointer0[2]);
- if (g3.pointer1)
- _downPointerIDs.delete(g3.pointer1[2]);
- g3.end(d3_event);
- return;
+ 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;
+ });
}
- d3_event.preventDefault();
- d3_event.stopImmediatePropagation();
- var loc = _pointerLocGetter(d3_event);
- var t2, p2, l2;
- if (isPointer0)
- g3.pointer0[0] = loc;
- else if (isPointer1)
- g3.pointer1[0] = loc;
- t2 = _transform;
- if (g3.pointer1) {
- var p02 = g3.pointer0[0], l0 = g3.pointer0[1], p1 = g3.pointer1[0], l1 = g3.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;
- t2 = scale(t2, Math.sqrt(dp / dl));
- p2 = [(p02[0] + p1[0]) / 2, (p02[1] + p1[1]) / 2];
- l2 = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];
- } else if (g3.pointer0) {
- p2 = g3.pointer0[0];
- l2 = g3.pointer0[1];
- } else {
- return;
+ if (fromDate) {
+ sequences = sequences.filter(function(sequence) {
+ return new Date(sequence.properties.captured_at).getTime() >= new Date(fromDate).getTime().toString();
+ });
}
- g3.zoom(d3_event, "touch", constrain(translate(t2, p2, l2), g3.extent, translateExtent));
+ if (toDate) {
+ sequences = sequences.filter(function(sequence) {
+ return new Date(sequence.properties.captured_at).getTime() <= new Date(toDate).getTime().toString();
+ });
+ }
+ return sequences;
}
- function pointerup(d3_event) {
- if (!_downPointerIDs.has(d3_event.pointerId))
- return;
- _downPointerIDs.delete(d3_event.pointerId);
- if (!_activeGesture)
- return;
- var g3 = gesture(this, arguments);
- d3_event.stopImmediatePropagation();
- if (g3.pointer0 && g3.pointer0[2] === d3_event.pointerId)
- delete g3.pointer0;
- else if (g3.pointer1 && g3.pointer1[2] === d3_event.pointerId)
- delete g3.pointer1;
- if (g3.pointer1 && !g3.pointer0) {
- g3.pointer0 = g3.pointer1;
- delete g3.pointer1;
+ function update() {
+ const z2 = ~~context.map().zoom();
+ const showMarkers = z2 >= minMarkerZoom;
+ const showViewfields = z2 >= minViewfieldZoom2;
+ 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(d2) {
+ return d2.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(d2) {
+ return d2.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(a2, b2) {
+ return b2.loc[1] - a2.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";
+ }
}
- if (g3.pointer0) {
- g3.pointer0[1] = _transform.invert(g3.pointer0[0]);
- } else {
- g3.end(d3_event);
+ }
+ 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() >= minZoom4) {
+ editOn();
+ update();
+ service.loadImages(projection2);
+ } else {
+ editOff();
+ }
}
}
- zoom.wheelDelta = function(_2) {
- return arguments.length ? (wheelDelta = utilFunctor(+_2), zoom) : wheelDelta;
- };
- zoom.filter = function(_2) {
- return arguments.length ? (filter2 = utilFunctor(!!_2), zoom) : filter2;
- };
- zoom.extent = function(_2) {
- return arguments.length ? (extent = utilFunctor([[+_2[0][0], +_2[0][1]], [+_2[1][0], +_2[1][1]]]), zoom) : extent;
- };
- zoom.scaleExtent = function(_2) {
- return arguments.length ? (scaleExtent[0] = +_2[0], scaleExtent[1] = +_2[1], zoom) : [scaleExtent[0], scaleExtent[1]];
- };
- zoom.translateExtent = function(_2) {
- return arguments.length ? (translateExtent[0][0] = +_2[0][0], translateExtent[1][0] = +_2[1][0], translateExtent[0][1] = +_2[0][1], translateExtent[1][1] = +_2[1][1], zoom) : [[translateExtent[0][0], translateExtent[0][1]], [translateExtent[1][0], translateExtent[1][1]]];
- };
- zoom.constrain = function(_2) {
- return arguments.length ? (constrain = _2, zoom) : constrain;
+ drawImages.enabled = function(_2) {
+ if (!arguments.length)
+ return svgMapillaryImages.enabled;
+ svgMapillaryImages.enabled = _2;
+ if (svgMapillaryImages.enabled) {
+ showLayer();
+ context.photos().on("change.mapillary_images", update);
+ } else {
+ hideLayer();
+ context.photos().on("change.mapillary_images", null);
+ }
+ dispatch14.call("change");
+ return this;
};
- zoom.interpolate = function(_2) {
- return arguments.length ? (interpolate = _2, zoom) : interpolate;
+ drawImages.supported = function() {
+ return !!getService();
};
- zoom._transform = function(_2) {
- return arguments.length ? (_transform = _2, zoom) : _transform;
+ drawImages.rendered = function(zoom) {
+ return zoom >= minZoom4;
};
- return utilRebind(zoom, dispatch14, "on");
+ init2();
+ return drawImages;
}
- // modules/util/double_up.js
- function utilDoubleUp() {
- var dispatch14 = dispatch_default("doubleUp");
- var _maxTimespan = 500;
- var _maxDistance = 20;
- var _pointer;
- function pointerIsValidFor(loc) {
- return (/* @__PURE__ */ 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)
+ // modules/svg/mapillary_position.js
+ function svgMapillaryPosition(projection2, context) {
+ const throttledRedraw = throttle_default(function() {
+ update();
+ }, 1e3);
+ const minZoom4 = 12;
+ const minViewfieldZoom2 = 18;
+ let layer = select_default2(null);
+ let _mapillary;
+ let viewerCompassAngle;
+ function init2() {
+ if (svgMapillaryPosition.initialized)
return;
- var loc = [d3_event.clientX, d3_event.clientY];
- if (_pointer && !pointerIsValidFor(loc)) {
- _pointer = void 0;
- }
- if (!_pointer) {
- _pointer = {
- startLoc: loc,
- startTime: (/* @__PURE__ */ new Date()).getTime(),
- upCount: 0,
- pointerId: d3_event.pointerId
- };
- } else {
- _pointer.pointerId = d3_event.pointerId;
+ svgMapillaryPosition.initialized = true;
+ }
+ function getService() {
+ if (services.mapillary && !_mapillary) {
+ _mapillary = services.mapillary;
+ _mapillary.event.on("imageChanged", throttledRedraw);
+ _mapillary.event.on("bearingChanged", function(e3) {
+ viewerCompassAngle = e3.bearing;
+ if (context.map().isTransformed())
+ return;
+ layer.selectAll(".viewfield-group.currentView").filter(function(d2) {
+ return d2.is_pano;
+ }).attr("transform", transform2);
+ });
+ } else if (!services.mapillary && _mapillary) {
+ _mapillary = null;
}
+ return _mapillary;
}
- 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);
- dispatch14.call("doubleUp", this, d3_event, locInThis);
- }
- _pointer = void 0;
+ function editOn() {
+ layer.style("display", "block");
+ }
+ function editOff() {
+ layer.selectAll(".viewfield-group").remove();
+ layer.style("display", "none");
+ }
+ function transform2(d2) {
+ let t2 = svgPointTransform(projection2)(d2);
+ if (d2.is_pano && viewerCompassAngle !== null && isFinite(viewerCompassAngle)) {
+ t2 += " rotate(" + Math.floor(viewerCompassAngle) + ",0,0)";
+ } else if (d2.ca) {
+ t2 += " rotate(" + Math.floor(d2.ca) + ",0,0)";
}
+ return t2;
}
- function doubleUp(selection2) {
- if ("PointerEvent" in window) {
- selection2.on("pointerdown.doubleUp", pointerdown).on("pointerup.doubleUp", pointerup);
+ function update() {
+ const z2 = ~~context.map().zoom();
+ const showViewfields = z2 >= minViewfieldZoom2;
+ const service = getService();
+ const image = service && service.getActiveImage();
+ const groups = layer.selectAll(".markers").selectAll(".viewfield-group").data(image ? [image] : [], function(d2) {
+ return d2.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() >= minZoom4) {
+ editOn();
+ update();
} else {
- selection2.on("dblclick.doubleUp", function(d3_event) {
- dispatch14.call("doubleUp", this, d3_event, utilFastMouse(this)(d3_event));
- });
+ editOff();
}
}
- doubleUp.off = function(selection2) {
- selection2.on("pointerdown.doubleUp", null).on("pointerup.doubleUp", null).on("dblclick.doubleUp", null);
+ drawImages.enabled = function() {
+ update();
+ return this;
};
- return utilRebind(doubleUp, dispatch14, "on");
+ drawImages.supported = function() {
+ return !!getService();
+ };
+ drawImages.rendered = function(zoom) {
+ return zoom >= minZoom4;
+ };
+ init2();
+ return drawImages;
}
- // 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 dispatch14 = 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", zoomPan2).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();
+ // modules/svg/mapillary_signs.js
+ function svgMapillarySigns(projection2, context, dispatch14) {
+ const throttledRedraw = throttle_default(function() {
+ dispatch14.call("change");
+ }, 1e3);
+ const minZoom4 = 12;
+ let layer = select_default2(null);
+ let _mapillary;
+ function init2() {
+ if (svgMapillarySigns.initialized)
+ return;
+ svgMapillarySigns.enabled = false;
+ svgMapillarySigns.initialized = true;
}
- 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);
- }
+ function getService() {
+ if (services.mapillary && !_mapillary) {
+ _mapillary = services.mapillary;
+ _mapillary.event.on("loadedSigns", throttledRedraw);
+ } else if (!services.mapillary && _mapillary) {
+ _mapillary = null;
}
- 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());
- dispatch14.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());
- dispatch14.call("drawn", this, { full: false });
+ 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, d2) {
+ const service = getService();
+ if (!service)
+ return;
+ context.map().centerEase(d2.loc);
+ const selectedImageId = service.getActiveImage() && service.getActiveImage().id;
+ service.getDetections(d2.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);
+ });
+ }
}
});
- 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);
+ }
+ 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;
+ });
}
- 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 t2 = projection2.transform();
- var p1 = t2.invert(p02);
- t2 = t2.scale(zoomOut2 ? 0.5 : 2);
- t2.x = p02[0] - p1[0] * t2.k;
- t2.y = p02[1] - p1[1] * t2.k;
- map2.transformEase(t2);
- });
- 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;
- });
- }
- }
+ if (toDate) {
+ var toTimestamp = new Date(toDate).getTime();
+ detectedFeatures = detectedFeatures.filter(function(feature3) {
+ return new Date(feature3.first_seen_at).getTime() <= toTimestamp;
});
- var data = Object.values(selectedAndParents);
- var filter2 = function(d2) {
- return d2.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());
- dispatch14.call("drawn", this, { full: false });
- scheduleRedraw();
+ }
+ 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(d2) {
+ return d2.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(d2) {
+ return "#" + d2.value;
});
- map2.dimensions(utilGetDimensions(selection2));
+ enter.append("rect").attr("width", "24px").attr("height", "24px").attr("x", "-12px").attr("y", "-12px");
+ signs.merge(enter).attr("transform", transform2);
}
- function zoomEventFilter(d3_event) {
- if (d3_event.type === "mousedown") {
- var hasOrphan = false;
- var listeners = window.__on;
- for (var i3 = 0; i3 < listeners.length; i3++) {
- var listener = listeners[i3];
- if (listener.name === "zoom" && listener.type === "mouseup") {
- hasOrphan = true;
- break;
- }
+ 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() >= minZoom4) {
+ editOn();
+ update();
+ service.loadSigns(projection2);
+ service.showSignDetections(true);
+ } else {
+ editOff();
}
- if (hasOrphan) {
- var event = window.CustomEvent;
- if (event) {
- event = new event("mouseup");
+ } else if (service) {
+ service.showSignDetections(false);
+ }
+ }
+ drawSigns.enabled = function(_2) {
+ if (!arguments.length)
+ return svgMapillarySigns.enabled;
+ svgMapillarySigns.enabled = _2;
+ if (svgMapillarySigns.enabled) {
+ showLayer();
+ context.photos().on("change.mapillary_signs", update);
+ } else {
+ hideLayer();
+ context.photos().on("change.mapillary_signs", null);
+ }
+ dispatch14.call("change");
+ return this;
+ };
+ drawSigns.supported = function() {
+ return !!getService();
+ };
+ drawSigns.rendered = function(zoom) {
+ return zoom >= minZoom4;
+ };
+ init2();
+ return drawSigns;
+ }
+
+ // modules/svg/mapillary_map_features.js
+ function svgMapillaryMapFeatures(projection2, context, dispatch14) {
+ const throttledRedraw = throttle_default(function() {
+ dispatch14.call("change");
+ }, 1e3);
+ const minZoom4 = 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, d2) {
+ const service = getService();
+ if (!service)
+ return;
+ context.map().centerEase(d2.loc);
+ const selectedImageId = service.getActiveImage() && service.getActiveImage().id;
+ service.getDetections(d2.id).then((detections) => {
+ if (detections.length) {
+ const imageId = detections[0].image.id;
+ if (imageId === selectedImageId) {
+ service.highlightDetection(detections[0]).selectImage(context, imageId);
} else {
- event = window.document.createEvent("Event");
- event.initEvent("mouseup", false, false);
+ service.ensureViewerLoaded(context).then(function() {
+ service.highlightDetection(detections[0]).selectImage(context, imageId).showViewer(context);
+ });
}
- 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);
+ 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();
});
- 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(d2) {
- return set3.has(d2.id);
- };
- features.clear(data);
- } else {
- if (features.gatherStats(all, graph, _dimensions)) {
- extent = void 0;
+ }
+ 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(d2) {
+ return d2.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(d2) {
+ var id2 = d2.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(d2) {
+ if (d2.value === "object--billboard") {
+ return "#object--sign--advertisement";
}
- if (extent) {
- data = context.history().intersects(map2.extent().intersection(extent));
- set3 = new Set(data.map(function(entity) {
- return entity.id;
- }));
- filter2 = function(d2) {
- return set3.has(d2.id);
- };
+ return "#" + d2.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() >= minZoom4) {
+ editOn();
+ update();
+ service.loadMapFeatures(projection2);
+ service.showFeatureDetections(true);
} else {
- data = all;
- fullRedraw = true;
- filter2 = utilFunctor(true);
+ editOff();
}
+ } else if (service) {
+ service.showFeatureDetections(false);
}
- if (applyFeatureLayerFilters) {
- data = features.filter(data, graph);
+ }
+ drawMapFeatures.enabled = function(_2) {
+ if (!arguments.length)
+ return svgMapillaryMapFeatures.enabled;
+ svgMapillaryMapFeatures.enabled = _2;
+ if (svgMapillaryMapFeatures.enabled) {
+ showLayer();
+ context.photos().on("change.mapillary_map_features", update);
} else {
- context.features().resetStats();
+ hideLayer();
+ context.photos().on("change.mapillary_map_features", null);
}
- if (mode && mode.id === "select") {
- surface.call(drawVertices.drawSelected, graph, map2.extent());
+ dispatch14.call("change");
+ return this;
+ };
+ drawMapFeatures.supported = function() {
+ return !!getService();
+ };
+ drawMapFeatures.rendered = function(zoom) {
+ return zoom >= minZoom4;
+ };
+ init2();
+ return drawMapFeatures;
+ }
+
+ // modules/svg/kartaview_images.js
+ function svgKartaviewImages(projection2, context, dispatch14) {
+ var throttledRedraw = throttle_default(function() {
+ dispatch14.call("change");
+ }, 1e3);
+ var minZoom4 = 12;
+ var minMarkerZoom = 16;
+ var minViewfieldZoom2 = 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;
}
- 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);
- dispatch14.call("drawn", this, { full: true });
+ return _kartaview;
+ }
+ function showLayer() {
+ var service = getService();
+ if (!service)
+ return;
+ editOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
+ dispatch14.call("change");
+ });
+ }
+ function hideLayer() {
+ throttledRedraw.cancel();
+ layer.transition().duration(250).style("opacity", 0).on("end", editOff);
+ }
+ function editOn() {
+ layer.style("display", "block");
}
- 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));
+ layer.selectAll(".viewfield-group").remove();
+ layer.style("display", "none");
+ }
+ function click(d3_event, d2) {
+ var service = getService();
+ if (!service)
+ return;
+ service.ensureViewerLoaded(context).then(function() {
+ service.selectImage(context, d2.key).showViewer(context);
+ });
+ context.map().centerEase(d2.loc);
+ }
+ function mouseover(d3_event, d2) {
+ var service = getService();
+ if (service)
+ service.setStyles(context, d2);
+ }
+ function mouseout() {
+ var service = getService();
+ if (service)
+ service.setStyles(context, null);
+ }
+ function transform2(d2) {
+ var t2 = svgPointTransform(projection2)(d2);
+ if (d2.ca) {
+ t2 += " rotate(" + Math.floor(d2.ca) + ",0,0)";
}
- dispatch14.call("drawn", this, { full: true });
+ return t2;
}
- function gestureChange(d3_event) {
- var e3 = d3_event;
- e3.preventDefault();
- var props = {
- deltaMode: 0,
- // dummy values to ignore in zoomPan
- deltaY: 1,
- // dummy values to ignore in zoomPan
- clientX: e3.clientX,
- clientY: e3.clientY,
- screenX: e3.screenX,
- screenY: e3.screenY,
- x: e3.x,
- y: e3.y
- };
- var e22 = new WheelEvent("wheel", props);
- e22._scale = e3.scale;
- e22._rotation = e3.rotation;
- _selection.node().dispatchEvent(e22);
+ 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 zoomPan2(event, key, transform2) {
- var source = event && event.sourceEvent || event;
- var eventTransform = transform2 || event && event.transform;
- var x2 = eventTransform.x;
- var y2 = eventTransform.y;
- var k2 = eventTransform.k;
- if (source && source.type === "wheel") {
- if (_pointerDown)
- return;
- var detected = utilDetect();
- var dX = source.deltaX;
- var dY = source.deltaY;
- var x22 = x2;
- var y22 = y2;
- var k22 = k2;
- 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);
- k22 = t0.k * Math.pow(2, -dY / 500);
- k22 = clamp(k22, kMin, kMax);
- x22 = p02[0] - p1[0] * k22;
- y22 = p02[1] - p1[1] * k22;
- } else if (source._scale) {
- t0 = _gestureTransformStart;
- p02 = _getMouseCoords(source);
- p1 = t0.invert(p02);
- k22 = t0.k * source._scale;
- k22 = clamp(k22, kMin, kMax);
- x22 = p02[0] - p1[0] * k22;
- y22 = p02[1] - p1[1] * k22;
- } else if (source.ctrlKey && !isInteger(dY)) {
- dY *= 6;
- t0 = _isTransformed ? _transformLast : _transformStart;
- p02 = _getMouseCoords(source);
- p1 = t0.invert(p02);
- k22 = t0.k * Math.pow(2, -dY / 500);
- k22 = clamp(k22, kMin, kMax);
- x22 = p02[0] - p1[0] * k22;
- y22 = p02[1] - p1[1] * k22;
- } else if ((source.altKey || source.shiftKey) && isInteger(dY)) {
- t0 = _isTransformed ? _transformLast : _transformStart;
- p02 = _getMouseCoords(source);
- p1 = t0.invert(p02);
- k22 = t0.k * Math.pow(2, -dY / 500);
- k22 = clamp(k22, kMin, kMax);
- x22 = p02[0] - p1[0] * k22;
- y22 = p02[1] - p1[1] * k22;
- } else if (detected.os === "mac" && detected.browser !== "Firefox" && !source.ctrlKey && isInteger(dX) && isInteger(dY)) {
- p1 = projection2.translate();
- x22 = p1[0] - dX;
- y22 = p1[1] - dY;
- k22 = projection2.scale();
- k22 = clamp(k22, kMin, kMax);
- }
- if (x22 !== x2 || y22 !== y2 || k22 !== k2) {
- x2 = x22;
- y2 = y22;
- k2 = k22;
- eventTransform = identity2.translate(x22, y22).scale(k22);
- if (_zoomerPanner._transform) {
- _zoomerPanner._transform(eventTransform);
- } else {
- _selection.node().__zoom = eventTransform;
- }
+ 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 z2 = ~~context.map().zoom();
+ var showMarkers = z2 >= minMarkerZoom;
+ var showViewfields = z2 >= minViewfieldZoom2;
+ 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(d2) {
+ return d2.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(d2) {
+ return d2.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(a2, b2) {
+ return a2 === selected ? 1 : b2 === selected ? -1 : b2.loc[1] - a2.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() >= minZoom4) {
+ editOn();
+ update();
+ service.loadImages(projection2);
+ } else {
+ editOff();
}
}
- if (_transformStart.x === x2 && _transformStart.y === y2 && _transformStart.k === k2) {
+ }
+ drawImages.enabled = function(_2) {
+ if (!arguments.length)
+ return svgKartaviewImages.enabled;
+ svgKartaviewImages.enabled = _2;
+ if (svgKartaviewImages.enabled) {
+ showLayer();
+ context.photos().on("change.kartaview_images", update);
+ } else {
+ hideLayer();
+ context.photos().on("change.kartaview_images", null);
+ }
+ dispatch14.call("change");
+ return this;
+ };
+ drawImages.supported = function() {
+ return !!getService();
+ };
+ drawImages.rendered = function(zoom) {
+ return zoom >= minZoom4;
+ };
+ init2();
+ return drawImages;
+ }
+
+ // modules/svg/mapilio_images.js
+ function svgMapilioImages(projection2, context, dispatch14) {
+ const throttledRedraw = throttle_default(function() {
+ dispatch14.call("change");
+ }, 1e3);
+ const minZoom4 = 12;
+ let layer = select_default2(null);
+ let _mapilio;
+ const viewFieldZoomLevel = 18;
+ function init2() {
+ if (svgMapilioImages.initialized)
return;
+ svgMapilioImages.enabled = false;
+ svgMapilioImages.initialized = true;
+ }
+ function getService() {
+ if (services.mapilio && !_mapilio) {
+ _mapilio = services.mapilio;
+ _mapilio.event.on("loadedImages", throttledRedraw);
+ } else if (!services.mapilio && _mapilio) {
+ _mapilio = null;
}
- if (geoScaleToZoom(k2, TILESIZE) < _minzoom) {
- surface.interrupt();
- dispatch14.call("hitMinZoom", this, map2);
- setCenterZoom(map2.center(), context.minEditableZoom(), 0, true);
- scheduleRedraw();
- dispatch14.call("move", this, map2);
+ return _mapilio;
+ }
+ function showLayer() {
+ const service = getService();
+ if (!service)
return;
+ editOn();
+ layer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end", function() {
+ dispatch14.call("change");
+ });
+ }
+ function hideLayer() {
+ throttledRedraw.cancel();
+ layer.transition().duration(250).style("opacity", 0).on("end", editOff);
+ }
+ function transform2(d2) {
+ let t2 = svgPointTransform(projection2)(d2);
+ if (d2.heading) {
+ t2 += " rotate(" + Math.floor(d2.heading) + ",0,0)";
}
- projection2.transform(eventTransform);
- var withinEditableZoom = map2.withinEditableZoom();
- if (_lastWithinEditableZoom !== withinEditableZoom) {
- if (_lastWithinEditableZoom !== void 0) {
- dispatch14.call("crossEditableZoom", this, withinEditableZoom);
+ return t2;
+ }
+ 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, image.id).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 update() {
+ const z2 = ~~context.map().zoom();
+ const showViewfields = z2 >= viewFieldZoomLevel;
+ const service = getService();
+ let sequences = service ? service.sequences(projection2) : [];
+ let images = service ? service.images(projection2) : [];
+ let traces = layer.selectAll(".sequences").selectAll(".sequence").data(sequences, function(d2) {
+ return d2.properties.id;
+ });
+ traces.exit().remove();
+ 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(d2) {
+ return d2.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(a2, b2) {
+ return b2.loc[1] - a2.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").attr("transform", "scale(1.5,1.5),translate(-8, -13)").attr("d", viewfieldPath);
+ function viewfieldPath() {
+ if (this.parentNode.__data__.isPano) {
+ 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";
}
- _lastWithinEditableZoom = withinEditableZoom;
- }
- var scale = k2 / _transformStart.k;
- var tX = (x2 / scale - _transformStart.x) * scale;
- var tY = (y2 / scale - _transformStart.y) * scale;
- if (context.inIntro()) {
- curtainProjection.transform({
- x: x2 - tX,
- y: y2 - tY,
- k: k2
- });
}
- if (source) {
- _lastPointerEvent = event;
+ }
+ function drawImages(selection2) {
+ const enabled = svgMapilioImages.enabled;
+ const service = getService();
+ layer = selection2.selectAll(".layer-mapilio").data(service ? [0] : []);
+ layer.exit().remove();
+ const layerEnter = layer.enter().append("g").attr("class", "layer-mapilio").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() >= minZoom4) {
+ editOn();
+ update();
+ service.loadImages(projection2);
+ service.loadLines(projection2);
+ } else {
+ editOff();
+ }
}
- _isTransformed = true;
- _transformLast = eventTransform;
- utilSetTransform(supersurface, tX, tY, scale);
- scheduleRedraw();
- dispatch14.call("move", this, map2);
- function isInteger(val) {
- return typeof val === "number" && isFinite(val) && Math.floor(val) === val;
+ }
+ drawImages.enabled = function(_2) {
+ if (!arguments.length)
+ return svgMapilioImages.enabled;
+ svgMapilioImages.enabled = _2;
+ if (svgMapilioImages.enabled) {
+ showLayer();
+ context.photos().on("change.mapilio_images", null);
+ } else {
+ hideLayer();
+ context.photos().on("change.mapilio_images", null);
}
+ dispatch14.call("change");
+ return this;
+ };
+ drawImages.supported = function() {
+ return !!getService();
+ };
+ drawImages.rendered = function(zoom) {
+ return zoom >= minZoom4;
+ };
+ init2();
+ return drawImages;
+ }
+
+ // modules/svg/osm.js
+ function svgOsm(projection2, context, dispatch14) {
+ var enabled = true;
+ function drawOsm(selection2) {
+ selection2.selectAll(".layer-osm").data(["covered", "areas", "lines", "points", "labels"]).enter().append("g").attr("class", function(d2) {
+ return "layer-osm " + d2;
+ });
+ selection2.selectAll(".layer-osm.points").selectAll(".points-group").data(["points", "midpoints", "vertices", "turns"]).enter().append("g").attr("class", function(d2) {
+ return "points-group " + d2;
+ });
}
- function resetTransform() {
- if (!_isTransformed)
- return false;
- utilSetTransform(supersurface, 0, 0);
- _isTransformed = false;
- if (context.inIntro()) {
- curtainProjection.transform(projection2.transform());
+ 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() {
+ dispatch14.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);
+ dispatch14.call("change");
+ });
+ }
+ drawOsm.enabled = function(val) {
+ if (!arguments.length)
+ return enabled;
+ enabled = val;
+ if (enabled) {
+ showLayer();
+ } else {
+ hideLayer();
}
- return true;
+ dispatch14.call("change");
+ return this;
+ };
+ return drawOsm;
+ }
+
+ // modules/svg/notes.js
+ var _notesEnabled = false;
+ var _osmService;
+ function svgNotes(projection2, context, dispatch14) {
+ if (!dispatch14) {
+ dispatch14 = dispatch_default("change");
}
- function redraw(difference, extent) {
- if (surface.empty() || !_redrawEnabled)
- return;
- if (resetTransform()) {
- difference = extent = void 0;
+ var throttledRedraw = throttle_default(function() {
+ dispatch14.call("change");
+ }, 1e3);
+ var minZoom4 = 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;
}
- var zoom = map2.zoom();
- var z2 = String(~~zoom);
- if (surface.attr("data-zoom") !== z2) {
- surface.attr("data-zoom", z2);
+ return _osmService;
+ }
+ function editOn() {
+ if (!_notesVisible) {
+ _notesVisible = true;
+ drawLayer.style("display", "block");
}
- 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);
+ }
+ function editOff() {
+ if (_notesVisible) {
+ _notesVisible = false;
+ drawLayer.style("display", "none");
+ drawLayer.selectAll(".note").remove();
+ touchLayer.selectAll(".note").remove();
}
- if (map2.editableDataEnabled() || map2.isInWideSelection()) {
- context.loadTiles(projection2);
- drawEditable(difference, extent);
- } else {
+ }
+ function layerOn() {
+ editOn();
+ drawLayer.style("opacity", 0).transition().duration(250).style("opacity", 1).on("end interrupt", function() {
+ dispatch14.call("change");
+ });
+ }
+ function layerOff() {
+ throttledRedraw.cancel();
+ drawLayer.interrupt();
+ touchLayer.selectAll(".note").remove();
+ drawLayer.transition().duration(250).style("opacity", 0).on("end interrupt", function() {
editOff();
+ dispatch14.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(d2) {
+ return d2.status + d2.id;
+ });
+ notes.exit().remove();
+ var notesEnter = notes.enter().append("g").attr("class", function(d2) {
+ return "note note-" + d2.id + " " + d2.status;
+ }).classed("new", function(d2) {
+ return d2.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(d2) {
+ return [d2];
+ }).enter().append("use").attr("class", "icon-annotation").attr("width", "10px").attr("height", "10px").attr("x", "-3px").attr("y", "-19px").attr("xlink:href", function(d2) {
+ if (d2.id < 0)
+ return "#iD-icon-plus";
+ if (d2.status === "open")
+ return "#iD-icon-close";
+ return "#iD-icon-apply";
+ });
+ notes.merge(notesEnter).sort(sortY).classed("selected", function(d2) {
+ var mode = context.mode();
+ var isMoving = mode && mode.id === "drag-note";
+ return !isMoving && d2.id === selectedID;
+ }).attr("transform", getTransform);
+ if (touchLayer.empty())
+ return;
+ var fillClass = context.getDebug("target") ? "pink " : "nocolor ";
+ var targets = touchLayer.selectAll(".note").data(data, function(d2) {
+ return d2.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(d2) {
+ var newClass = d2.id < 0 ? "new" : "";
+ return "note target note-" + d2.id + " " + fillClass + newClass;
+ }).attr("transform", getTransform);
+ function sortY(a2, b2) {
+ if (a2.id === selectedID)
+ return 1;
+ if (b2.id === selectedID)
+ return -1;
+ return b2.loc[1] - a2.loc[1];
}
- _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 s2;
- while (s2 = event.sourceEvent) {
- event = s2;
+ 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() >= minZoom4) {
+ editOn();
+ service.loadNotes(projection2);
+ updateMarkers();
+ } else {
+ editOff();
}
- 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) {
+ }
+ drawNotes.enabled = function(val) {
if (!arguments.length)
- return _redrawEnabled;
- _redrawEnabled = val;
- return map2;
- };
- map2.isTransformed = function() {
- return _isTransformed;
- };
- function setTransform(t2, duration, force) {
- var t3 = projection2.transform();
- if (!force && t2.k === t3.k && t2.x === t3.x && t2.y === t3.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));
+ return _notesEnabled;
+ _notesEnabled = val;
+ if (_notesEnabled) {
+ layerOn();
} else {
- projection2.transform(t2);
- _transformStart = t2;
- _selection.call(_zoomerPanner.transform, _transformStart);
+ layerOff();
+ if (context.selectedNoteID()) {
+ context.enter(modeBrowse(context));
+ }
}
- return true;
+ dispatch14.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(d2) {
+ return "layer-touch " + d2;
+ });
}
- function setCenterZoom(loc2, z2, duration, force) {
- var c2 = map2.center();
- var z3 = map2.zoom();
- if (loc2[0] === c2[0] && loc2[1] === c2[1] && z2 === z3 && !force)
- return false;
- var proj = geoRawMercator().transform(projection2.transform());
- var k2 = clamp(geoZoomToScale(z2, TILESIZE), kMin, kMax);
- proj.scale(k2);
- var t2 = proj.translate();
- var point2 = proj(loc2);
- var center = pxCenter();
- t2[0] += center[0] - point2[0];
- t2[1] += center[1] - point2[1];
- return setTransform(identity2.translate(t2[0], t2[1]).scale(k2), duration, force);
+ 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];
}
- map2.pan = function(delta, duration) {
- var t2 = projection2.translate();
- var k2 = projection2.scale();
- t2[0] += delta[0];
- t2[1] += delta[1];
- if (duration) {
- _selection.transition().duration(duration).on("start", function() {
- map2.startEase();
- }).call(_zoomerPanner.transform, identity2.translate(t2[0], t2[1]).scale(k2));
- } else {
- projection2.translate(t2);
- _transformStart = projection2.transform();
- _selection.call(_zoomerPanner.transform, _transformStart);
- dispatch14.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);
+ 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;
}
- function zoomOut(delta) {
- setCenterZoom(map2.center(), ~~map2.zoom() - delta, 250, true);
+ 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 dispatch14 = dispatch_default("change");
+ var svg2 = select_default2(null);
+ var _layers = [
+ { id: "osm", layer: svgOsm(projection2, context, dispatch14) },
+ { id: "notes", layer: svgNotes(projection2, context, dispatch14) },
+ { id: "data", layer: svgData(projection2, context, dispatch14) },
+ { id: "keepRight", layer: svgKeepRight(projection2, context, dispatch14) },
+ { id: "improveOSM", layer: svgImproveOSM(projection2, context, dispatch14) },
+ { id: "osmose", layer: svgOsmose(projection2, context, dispatch14) },
+ { id: "streetside", layer: svgStreetside(projection2, context, dispatch14) },
+ { id: "mapillary", layer: svgMapillaryImages(projection2, context, dispatch14) },
+ { id: "mapillary-position", layer: svgMapillaryPosition(projection2, context, dispatch14) },
+ { id: "mapillary-map-features", layer: svgMapillaryMapFeatures(projection2, context, dispatch14) },
+ { id: "mapillary-signs", layer: svgMapillarySigns(projection2, context, dispatch14) },
+ { id: "kartaview", layer: svgKartaviewImages(projection2, context, dispatch14) },
+ { id: "mapilio", layer: svgMapilioImages(projection2, context, dispatch14) },
+ { id: "vegbilder", layer: svgVegbilder(projection2, context, dispatch14) },
+ { id: "local-photos", layer: svgLocalPhotos(projection2, context, dispatch14) },
+ { id: "debug", layer: svgDebug(projection2, context, dispatch14) },
+ { id: "geolocate", layer: svgGeolocate(projection2, context, dispatch14) },
+ { id: "touch", layer: svgTouch(projection2, context, dispatch14) }
+ ];
+ 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(d2) {
+ return "data-layer " + d2.id;
+ }).merge(groups).each(function(d2) {
+ select_default2(this).call(d2.layer);
+ });
}
- map2.zoomIn = function() {
- zoomIn(1);
- };
- map2.zoomInFurther = function() {
- zoomIn(4);
- };
- map2.canZoomIn = function() {
- return map2.zoom() < maxZoom;
+ drawLayers.all = function() {
+ return _layers;
};
- map2.zoomOut = function() {
- zoomOut(1);
+ drawLayers.layer = function(id2) {
+ var obj = _layers.find(function(o2) {
+ return o2.id === id2;
+ });
+ return obj && obj.layer;
};
- map2.zoomOutFurther = function() {
- zoomOut(4);
+ drawLayers.only = function(what) {
+ var arr = [].concat(what);
+ var all = _layers.map(function(layer) {
+ return layer.id;
+ });
+ return drawLayers.remove(utilArrayDifference(all, arr));
};
- map2.canZoomOut = function() {
- return map2.zoom() > minZoom2;
+ drawLayers.remove = function(what) {
+ var arr = [].concat(what);
+ arr.forEach(function(id2) {
+ _layers = _layers.filter(function(o2) {
+ return o2.id !== id2;
+ });
+ });
+ dispatch14.call("change");
+ return this;
};
- map2.center = function(loc2) {
- if (!arguments.length) {
- return projection2.invert(pxCenter());
- }
- if (setCenterZoom(loc2, map2.zoom())) {
- dispatch14.call("move", this, map2);
- }
- scheduleRedraw();
- return map2;
+ drawLayers.add = function(what) {
+ var arr = [].concat(what);
+ arr.forEach(function(obj) {
+ if ("id" in obj && "layer" in obj) {
+ _layers.push(obj);
+ }
+ });
+ dispatch14.call("change");
+ return this;
};
- 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);
+ drawLayers.dimensions = function(val) {
+ if (!arguments.length)
+ return utilGetDimensions(svg2);
+ utilSetDimensions(svg2, val);
+ return this;
};
- 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];
+ return utilRebind(drawLayers, dispatch14, "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,
+ busway: 11,
+ footway: 12
};
- map2.zoom = function(z2) {
- if (!arguments.length) {
- return Math.max(geoScaleToZoom(projection2.scale(), TILESIZE), 0);
- }
- if (z2 < _minzoom) {
- surface.interrupt();
- dispatch14.call("hitMinZoom", this, map2);
- z2 = context.minEditableZoom();
+ 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(d2) {
+ return filter2(d2.properties.entity);
+ }).data(targetData, function key(d2) {
+ return d2.id;
+ });
+ targets.exit().remove();
+ var segmentWasEdited = function(d2) {
+ var wayID = d2.properties.entity.id;
+ if (!base.entities[wayID] || !(0, import_fast_deep_equal6.default)(graph.entities[wayID].nodes, base.entities[wayID].nodes)) {
+ return false;
+ }
+ return d2.properties.nodes.some(function(n3) {
+ return !base.entities[n3.id] || !(0, import_fast_deep_equal6.default)(graph.entities[n3.id].loc, base.entities[n3.id].loc);
+ });
+ };
+ targets.enter().append("path").merge(targets).attr("d", getPath).attr("class", function(d2) {
+ return "way line target target-allowed " + targetClass + d2.id;
+ }).classed("segment-edited", segmentWasEdited);
+ var nopeData = data.nopes.filter(getPath);
+ var nopes = selection2.selectAll(".line.target-nope").filter(function(d2) {
+ return filter2(d2.properties.entity);
+ }).data(nopeData, function key(d2) {
+ return d2.id;
+ });
+ nopes.exit().remove();
+ nopes.enter().append("path").merge(nopes).attr("d", getPath).attr("class", function(d2) {
+ return "way line target target-nope " + nopeClass + d2.id;
+ }).classed("segment-edited", segmentWasEdited);
+ }
+ function drawLines(selection2, graph, entities, filter2) {
+ var base = context.history().base();
+ function waystack(a2, b2) {
+ var selected = context.selectedIDs();
+ var scoreA = selected.indexOf(a2.id) !== -1 ? 20 : 0;
+ var scoreB = selected.indexOf(b2.id) !== -1 ? 20 : 0;
+ if (a2.tags.highway) {
+ scoreA -= highway_stack[a2.tags.highway];
+ }
+ if (b2.tags.highway) {
+ scoreB -= highway_stack[b2.tags.highway];
+ }
+ return scoreA - scoreB;
}
- if (setCenterZoom(map2.center(), z2)) {
- dispatch14.call("move", this, map2);
+ 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(d2) {
+ var prefix = "way line";
+ if (!d2.hasInterestingTags()) {
+ var parentRelations = graph.parentRelations(d2);
+ 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[d2.id] ? "old-multipolygon " : "";
+ return prefix + " " + klass + " " + selectedClass + oldMPClass + d2.id;
+ }).classed("added", function(d2) {
+ return !base.entities[d2.id];
+ }).classed("geometry-edited", function(d2) {
+ return graph.entities[d2.id] && base.entities[d2.id] && !(0, import_fast_deep_equal6.default)(graph.entities[d2.id].nodes, base.entities[d2.id].nodes);
+ }).classed("retagged", function(d2) {
+ return graph.entities[d2.id] && base.entities[d2.id] && !(0, import_fast_deep_equal6.default)(graph.entities[d2.id].tags, base.entities[d2.id].tags);
+ }).call(svgTagClasses()).merge(lines).sort(waystack).attr("d", getPath).call(svgTagClasses().tags(svgRelationMemberTags(graph)));
+ return selection3;
}
- scheduleRedraw();
- return map2;
- };
- map2.centerZoom = function(loc2, z2) {
- if (setCenterZoom(loc2, z2)) {
- dispatch14.call("move", this, map2);
+ function getPathData(isSelected) {
+ return function() {
+ var layer = this.parentNode.__data__;
+ var data = pathdata[layer] || [];
+ return data.filter(function(d2) {
+ if (isSelected) {
+ return context.selectedIDs().indexOf(d2.id) !== -1;
+ } else {
+ return context.selectedIDs().indexOf(d2.id) === -1;
+ }
+ });
+ };
}
- 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);
+ 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(d2) {
+ return [d2.id, d2.index];
}
+ );
+ markers.exit().remove();
+ markers = markers.enter().append("path").attr("class", pathclass).merge(markers).attr("marker-mid", marker).attr("d", function(d2) {
+ return d2.d;
});
- } else {
- extent = obj.extent(context.graph());
+ if (detected.ie) {
+ markers.each(function() {
+ this.parentNode.insertBefore(this, this);
+ });
+ }
}
- 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();
+ var getPath = svgPath(projection2, graph);
+ var ways = [];
+ var onewaydata = {};
+ var sideddata = {};
+ var oldMultiPolygonOuters = {};
+ for (var i3 = 0; i3 < entities.length; i3++) {
+ var entity = entities[i3];
+ 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();
});
- 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])
+ Object.keys(pathdata).forEach(function(k2) {
+ var v2 = pathdata[k2];
+ var onewayArr = v2.filter(function(d2) {
+ return d2.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";
+ }
);
- } 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])
+ onewaydata[k2] = utilArrayFlatten(onewayArr.map(onewaySegments));
+ var sidedArr = v2.filter(function(d2) {
+ return d2.isSided();
+ });
+ var sidedSegments = svgMarkerSegments(
+ projection2,
+ graph,
+ 30,
+ function shouldReverse() {
+ return false;
+ },
+ function bothDirections() {
+ return false;
+ }
);
- } 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;
+ sideddata[k2] = 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(d2) {
+ return "layergroup layer" + String(d2);
+ }).merge(layergroup);
+ layergroup.selectAll("g.linegroup").data(["shadow", "casing", "stroke", "shadow-highlighted", "casing-highlighted", "stroke-highlighted"]).enter().append("g").attr("class", function(d2) {
+ return "linegroup line-" + d2;
+ });
+ 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(d2) {
+ var category = graph.entity(d2.id).sidednessIdentifier();
+ return "url(#ideditor-sided-marker-" + category + ")";
+ }
+ );
+ });
+ touchLayer.call(drawTargets, graph, ways, filter2);
}
- 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]);
- dispatch14.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);
+ 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(d2) {
+ return filter2(d2.properties.entity);
+ }).data(data, function key(d2) {
+ return d2.id;
+ });
+ targets.exit().remove();
+ targets.enter().append("circle").attr("r", targetRadius).merge(targets).attr("class", function(d2) {
+ return "node midpoint target " + fillClass + d2.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;
}
- updateAreaFill();
- map2.pan([0, 0]);
- dispatch14.call("changeAreaFill", this);
- return map2;
- };
- map2.toggleWireframe = function() {
- var activeFill = map2.activeAreaFill();
- if (activeFill === "wireframe") {
- activeFill = corePreferences("area-fill-toggle") || "partial";
- } else {
- activeFill = "wireframe";
+ var poly = extent.polygon();
+ var midpoints = {};
+ for (var i3 = 0; i3 < entities.length; i3++) {
+ var entity = entities[i3];
+ 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 a2 = nodes[j2];
+ var b2 = nodes[j2 + 1];
+ var id2 = [a2.id, b2.id].sort().join("-");
+ if (midpoints[id2]) {
+ midpoints[id2].parents.push(entity);
+ } else if (geoVecLength(projection2(a2.loc), projection2(b2.loc)) > 40) {
+ var point2 = geoVecInterp(a2.loc, b2.loc, 0.5);
+ var loc = null;
+ if (extent.intersects(point2)) {
+ loc = point2;
+ } else {
+ for (var k2 = 0; k2 < 4; k2++) {
+ point2 = geoLineIntersection([a2.loc, b2.loc], [poly[k2], poly[k2 + 1]]);
+ if (point2 && geoVecLength(projection2(a2.loc), projection2(point2)) > 20 && geoVecLength(projection2(b2.loc), projection2(point2)) > 20) {
+ loc = point2;
+ break;
+ }
+ }
+ }
+ if (loc) {
+ midpoints[id2] = {
+ type: "midpoint",
+ id: id2,
+ loc,
+ edge: [a2.id, b2.id],
+ parents: [entity]
+ };
+ }
+ }
+ }
}
- map2.activeAreaFill(activeFill);
- };
- function updateAreaFill() {
- var activeFill = map2.activeAreaFill();
- map2.areaFillOptions.forEach(function(opt) {
- surface.classed("fill-" + opt, Boolean(opt === activeFill));
+ function midpointFilter(d2) {
+ if (midpoints[d2.id])
+ return true;
+ for (var i4 = 0; i4 < d2.parents.length; i4++) {
+ if (filter2(d2.parents[i4])) {
+ return true;
+ }
+ }
+ return false;
+ }
+ var groups = drawLayer.selectAll(".midpoint").filter(midpointFilter).data(Object.values(midpoints), function(d2) {
+ return d2.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(d2) {
+ var translate = svgPointTransform(projection2);
+ var a3 = graph.entity(d2.edge[0]);
+ var b3 = graph.entity(d2.edge[1]);
+ var angle2 = geoAngle(a3, b3, projection2) * (180 / Math.PI);
+ return translate(d2) + " rotate(" + angle2 + ")";
+ }).call(svgTagClasses().tags(
+ function(d2) {
+ return d2.parents[0].tags;
+ }
+ ));
+ groups.select("polygon.shadow");
+ groups.select("polygon.fill");
+ touchLayer.call(drawTargets, graph, Object.values(midpoints), midpointFilter);
}
- map2.layers = () => drawLayers;
- map2.doubleUpHandler = function() {
- return _doubleUpHandler;
- };
- return utilRebind(map2, dispatch14, "on");
+ return drawMidpoints;
}
- // modules/renderer/photos.js
- function rendererPhotos(context) {
- var dispatch14 = dispatch_default("change");
- var _layerIDs = ["streetside", "mapillary", "mapillary-map-features", "mapillary-signs", "kartaview", "mapilio", "vegbilder"];
- var _allPhotoTypes = ["flat", "panoramic"];
- var _shownPhotoTypes = _allPhotoTypes.slice();
- var _dateFilters = ["fromDate", "toDate"];
- var _fromDate;
- var _toDate;
- var _usernames;
- function photos() {
+ // 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 updateStorage() {
- if (window.mocha)
- return;
- var hash = utilStringQs(window.location.hash);
- var enabled = context.layers().all().filter(function(d2) {
- return _layerIDs.indexOf(d2.id) !== -1 && d2.layer && d2.layer.supported() && d2.layer.enabled();
- }).map(function(d2) {
+ function sortY(a2, b2) {
+ return b2.loc[1] - a2.loc[1];
+ }
+ function fastEntityKey(d2) {
+ var mode = context.mode();
+ var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
+ return isMoving ? d2.id : osmEntity.key(d2);
+ }
+ 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(d2) {
+ return filter2(d2.properties.entity);
+ }).data(data, function key(d2) {
return d2.id;
});
- if (enabled.length) {
- hash.photo_overlay = enabled.join(",");
- } else {
- delete hash.photo_overlay;
+ targets.exit().remove();
+ targets.enter().append("rect").attr("x", -10).attr("y", -26).attr("width", 20).attr("height", 30).merge(targets).attr("class", function(d2) {
+ return "node point target " + fillClass + d2.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);
}
- window.location.replace("#" + utilQsString(hash, true));
+ 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(d2) {
+ return "node point " + d2.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(d2) {
+ return !base.entities[d2.id];
+ }).classed("moved", function(d2) {
+ return base.entities[d2.id] && !(0, import_fast_deep_equal7.default)(graph.entities[d2.id].loc, base.entities[d2.id].loc);
+ }).classed("retagged", function(d2) {
+ return base.entities[d2.id] && !(0, import_fast_deep_equal7.default)(graph.entities[d2.id].tags, base.entities[d2.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);
}
- 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;
+ return drawPoints;
+ }
+
+ // modules/svg/turns.js
+ function svgTurns(projection2, context) {
+ function icon2(turn) {
+ var u2 = turn.u ? "-u" : "";
+ if (turn.no)
+ return "#iD-turn-no" + u2;
+ if (turn.only)
+ return "#iD-turn-only" + u2;
+ return "#iD-turn-yes" + u2;
+ }
+ function drawTurns(selection2, graph, turns) {
+ function turnTransform(d2) {
+ var pxRadius = 50;
+ var toWay = graph.entity(d2.to.way);
+ var toPoints = graph.childNodes(toWay).map(function(n3) {
+ return n3.loc;
+ }).map(projection2);
+ var toLength = geoPathLength(toPoints);
+ var mid = toLength / 2;
+ var toNode = graph.entity(d2.to.node);
+ var toVertex = graph.entity(d2.to.vertex);
+ var a2 = geoAngle(toVertex, toNode, projection2);
+ var o2 = projection2(toVertex.loc);
+ var r2 = d2.u ? 0 : !toWay.__via ? pxRadius : Math.min(mid, pxRadius);
+ return "translate(" + (r2 * Math.cos(a2) + o2[0]) + "," + (r2 * Math.sin(a2) + o2[1]) + ") rotate(" + a2 * 180 / Math.PI + ")";
}
- if (type2 === _dateFilters[0]) {
- _fromDate = val;
- if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
- _toDate = _fromDate;
- }
+ 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(d2) {
+ return d2.key;
+ });
+ groups.exit().remove();
+ var groupsEnter = groups.enter().append("g").attr("class", function(d2) {
+ return "turn " + d2.key;
+ });
+ var turnsEnter = groupsEnter.filter(function(d2) {
+ return !d2.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(d2) {
+ return d2.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(d2) {
+ return d2.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(d2) {
+ return d2.key;
+ });
+ groups.exit().remove();
+ groupsEnter = groups.enter().append("g").attr("class", function(d2) {
+ return "turn " + d2.key;
+ });
+ turnsEnter = groupsEnter.filter(function(d2) {
+ return !d2.u;
+ });
+ turnsEnter.append("rect").attr("class", "target " + fillClass).attr("transform", "translate(-22, -12)").attr("width", "44").attr("height", "24");
+ uEnter = groupsEnter.filter(function(d2) {
+ return d2.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(a2, b2) {
+ return b2.loc[1] - a2.loc[1];
+ }
+ function fastEntityKey(d2) {
+ var mode = context.mode();
+ var isMoving = mode && /^(add|draw|drag|move|rotate)/.test(mode.id);
+ return isMoving ? d2.id : osmEntity.key(d2);
+ }
+ 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 z2 = zoom < 17 ? 0 : zoom < 18 ? 1 : 2;
+ var activeID = context.activeID();
+ var base = context.history().base();
+ function getIcon(d2) {
+ var entity = graph.entity(d2.id);
+ if (entity.id in icons)
+ return icons[entity.id];
+ icons[entity.id] = entity.hasInterestingTags() && _mainPresetIndex.match(entity, graph).icon;
+ return icons[entity.id];
}
- if (type2 === _dateFilters[1]) {
- _toDate = val;
- if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
- _fromDate = _toDate;
- }
+ 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;
}
- dispatch14.call("change", this);
- if (updateUrl) {
- var rangeString;
- if (_fromDate || _toDate) {
- rangeString = (_fromDate || "") + "_" + (_toDate || "");
- }
- setUrlFilterValue("photo_dates", rangeString);
+ function updateAttributes(selection3) {
+ ["shadow", "stroke", "fill"].forEach(function(klass) {
+ var rads = radiuses[klass];
+ selection3.selectAll("." + klass).each(function(entity) {
+ var i3 = z2 && getIcon(entity);
+ var r2 = rads[i3 ? 3 : z2];
+ if (entity.id !== activeID && entity.isEndpoint(graph) && !entity.isConnected(graph)) {
+ r2 += 1.5;
+ }
+ if (klass === "shadow") {
+ _radii[entity.id] = r2;
+ }
+ select_default2(this).attr("r", r2).attr("visibility", i3 && klass === "fill" ? "hidden" : null);
+ });
+ });
}
- };
- photos.setUsernameFilter = function(val, updateUrl) {
- if (val && typeof val === "string")
- val = val.replace(/;/g, ",").split(",");
- if (val) {
- val = val.map((d2) => d2.trim()).filter(Boolean);
- if (!val.length) {
- val = 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(d2) {
+ return "node vertex " + d2.id;
+ }).order();
+ enter.append("circle").attr("class", "shadow");
+ enter.append("circle").attr("class", "stroke");
+ enter.filter(function(d2) {
+ return d2.hasInterestingTags();
+ }).append("circle").attr("class", "fill");
+ groups = groups.merge(enter).attr("transform", svgPointTransform(projection2)).classed("sibling", function(d2) {
+ return d2.id in sets2.selected;
+ }).classed("shared", function(d2) {
+ return graph.isShared(d2);
+ }).classed("endpoint", function(d2) {
+ return d2.isEndpoint(graph);
+ }).classed("added", function(d2) {
+ return !base.entities[d2.id];
+ }).classed("moved", function(d2) {
+ return base.entities[d2.id] && !(0, import_fast_deep_equal8.default)(graph.entities[d2.id].loc, base.entities[d2.id].loc);
+ }).classed("retagged", function(d2) {
+ return base.entities[d2.id] && !(0, import_fast_deep_equal8.default)(graph.entities[d2.id].tags, base.entities[d2.id].tags);
+ }).call(updateAttributes);
+ var iconUse = groups.selectAll(".icon").data(function data(d2) {
+ return zoom >= 17 && getIcon(d2) ? [d2] : [];
+ }, 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(d2) {
+ var picon = getIcon(d2);
+ return picon ? "#" + picon : "";
+ });
+ var dgroups = groups.selectAll(".viewfieldgroup").data(function data(d2) {
+ return zoom >= 18 && getDirections(d2) ? [d2] : [];
+ }, fastEntityKey);
+ dgroups.exit().remove();
+ dgroups = dgroups.enter().insert("g", ".shadow").attr("class", "viewfieldgroup").merge(dgroups);
+ var viewfields = dgroups.selectAll(".viewfield").data(getDirections, function key(d2) {
+ return osmEntity.key(d2);
+ });
+ 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(d2) {
+ return "rotate(" + d2 + ")";
+ });
+ }
+ 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()
+ });
}
- }
- _usernames = val;
- dispatch14.call("change", this);
- if (updateUrl) {
- var hashString;
- if (_usernames) {
- hashString = _usernames.join(",");
+ });
+ var targets = selection2.selectAll(".vertex.target-allowed").filter(function(d2) {
+ return filter2(d2.properties.entity);
+ }).data(data.targets, function key(d2) {
+ return d2.id;
+ });
+ targets.exit().remove();
+ targets.enter().append("circle").attr("r", function(d2) {
+ return _radii[d2.id] || radiuses.shadow[3];
+ }).merge(targets).attr("class", function(d2) {
+ return "node vertex target target-allowed " + targetClass + d2.id;
+ }).attr("transform", getTransform);
+ var nopes = selection2.selectAll(".vertex.target-nope").filter(function(d2) {
+ return filter2(d2.properties.entity);
+ }).data(data.nopes, function key(d2) {
+ return d2.id;
+ });
+ nopes.exit().remove();
+ nopes.enter().append("circle").attr("r", function(d2) {
+ return _radii[d2.properties.entity.id] || radiuses.shadow[3];
+ }).merge(nopes).attr("class", function(d2) {
+ return "node vertex target target-nope " + nopeClass + d2.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 i3;
+ if (entity.type === "way") {
+ for (i3 = 0; i3 < entity.nodes.length; i3++) {
+ var child = graph.hasEntity(entity.nodes[i3]);
+ if (child) {
+ addChildVertices(child);
+ }
+ }
+ } else if (entity.type === "relation") {
+ for (i3 = 0; i3 < entity.members.length; i3++) {
+ var member = graph.hasEntity(entity.members[i3].id);
+ if (member) {
+ addChildVertices(member);
+ }
+ }
+ } else if (renderAsVertex(entity, graph, wireframe, zoom)) {
+ results[entity.id] = entity;
+ }
}
- 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;
+ 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 {
- if (!(property in hash))
- return;
- delete hash[property];
+ addChildVertices(entity);
}
- window.location.replace("#" + utilQsString(hash, true));
- }
- }
- function showsLayer(id2) {
- var layer = context.layers().layer(id2);
- return layer && layer.supported() && layer.enabled();
+ });
+ return results;
}
- photos.shouldFilterByDate = function() {
- return showsLayer("mapillary") || showsLayer("kartaview") || showsLayer("streetside") || showsLayer("vegbilder");
- };
- photos.shouldFilterByPhotoType = function() {
- return showsLayer("mapillary") || showsLayer("streetside") && showsLayer("kartaview") || showsLayer("vegbilder");
- };
- 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);
- }
- dispatch14.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);
+ 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 = {};
}
- if (hash.photo_username) {
- this.setUsernameFilter(hash.photo_username, false);
+ for (var i3 = 0; i3 < entities.length; i3++) {
+ var entity = entities[i3];
+ 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];
+ }
}
- 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);
+ 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(d2) {
+ return d2.id in _currPersistent || d2.id in _currSelected || d2.id in _currHover || filter2(d2);
+ };
+ drawLayer.call(draw, graph, currentVisible(all), sets2, filterRendered);
+ var filterTouch = function(d2) {
+ return isMoving ? true : filterRendered(d2);
+ };
+ 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);
});
}
- 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);
- });
- });
+ }
+ 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);
}
- context.layers().on("change.rendererPhotos", updateStorage);
+ var filter2 = function(d2) {
+ return d2.id in _prevSelected;
+ };
+ drawVertices(selection2, graph, Object.values(_prevSelected), filter2, extent, false);
};
- return utilRebind(photos, dispatch14, "on");
- }
-
- // modules/ui/account.js
- function uiAccount(context) {
- const osm = context.connection();
- function updateUserDetails(selection2) {
- if (!osm)
+ drawVertices.drawHover = function(selection2, graph, target, extent) {
+ if (target === _currHoverTarget)
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", (e3) => {
- e3.preventDefault();
- osm.logout();
- tryLogout();
- });
+ 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 {
- userInfo.html("").classed("hide", true);
- loginLogout.classed("hide", false).select("a").text(_t("login")).on("click", (e3) => {
- e3.preventDefault();
- osm.authenticate();
- });
+ _currHover = {};
}
- }
- function tryLogout() {
- if (!osm)
- return;
- const url = osm.getUrlRoot() + "/logout?referer=%2Flogin";
- const w2 = 600;
- const h2 = 550;
- const settings = [
- ["width", w2],
- ["height", h2],
- ["left", window.screen.width / 2 - w2 / 2],
- ["top", window.screen.height / 2 - h2 / 2]
- ].map((x2) => x2.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);
+ var filter2 = function(d2) {
+ return d2.id in _prevHover;
+ };
+ drawVertices(selection2, graph, Object.values(_prevHover), filter2, extent, false);
};
+ return drawVertices;
}
- // 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, (d2) => d2.id);
- attributions.exit().remove();
- attributions = attributions.enter().append("span").attr("class", "attribution").each((d2, i3, nodes) => {
- let attribution = select_default2(nodes[i3]);
- if (d2.terms_html) {
- attribution.html(d2.terms_html);
- return;
- }
- if (d2.terms_url) {
- attribution = attribution.append("a").attr("href", d2.terms_url).attr("target", "_blank");
- }
- const sourceID = d2.id.replace(/\./g, "");
- const terms_text = _t(
- `imagery.${sourceID}.attribution.text`,
- { default: d2.terms_text || d2.id || d2.name() }
- );
- if (d2.icon && !d2.overlay) {
- attribution.append("img").attr("class", "source-image").attr("src", d2.icon);
- }
- attribution.append("span").attr("class", "attribution-text").text(terms_text);
- }).merge(attributions);
- let copyright = attributions.selectAll(".copyright-notice").data((d2) => {
- let notice = d2.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 z2 = context.map().zoom();
- let overlays = context.background().overlayLayerSources() || [];
- _selection.call(render, overlays.filter((s2) => s2.validZoom(z2)), "overlay-layer-attribution");
+ // 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);
}
- return function(selection2) {
- _selection = selection2;
- context.background().on("change.attribution", update);
- context.map().on("move.attribution", throttle_default(update, 400, { leading: false }));
- update();
- };
+ target.on(typeOnce, one2, capture);
+ return this;
}
- // 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 u2 = Object.keys(users), subset = u2.slice(0, u2.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(d2) {
- return osm.userURL(d2);
- }).attr("target", "_blank").text(String);
- if (u2.length > limit) {
- var count = select_default2(document.createElement("span"));
- var othersNum = u2.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 (!u2.length) {
- hidden = true;
- wrap2.transition().style("opacity", 0);
- } else if (hidden) {
- wrap2.transition().style("opacity", 1);
+ // modules/util/zoom_pan.js
+ function defaultFilter3(d3_event) {
+ return !d3_event.ctrlKey && !d3_event.button;
+ }
+ function defaultExtent2() {
+ var e3 = this;
+ if (e3 instanceof SVGElement) {
+ e3 = e3.ownerSVGElement || e3;
+ if (e3.hasAttribute("viewBox")) {
+ e3 = e3.viewBox.baseVal;
+ return [[e3.x, e3.y], [e3.x + e3.width, e3.y + e3.height]];
}
+ return [[0, 0], [e3.width.baseVal.value, e3.height.baseVal.value]];
}
- return function(selection2) {
- if (!osm)
- return;
- wrap2 = selection2;
- update();
- osm.on("loaded.contributors", debouncedUpdate);
- context.map().on("move.contributors", debouncedUpdate);
- };
+ return [[0, 0], [e3.clientWidth, e3.clientHeight]];
}
-
- // 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;
+ 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, dispatch14 = 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 {
- return _displayType;
+ selection2.interrupt().each(function() {
+ gesture(this, arguments).start(null).zoom(null, null, typeof transform2 === "function" ? transform2.apply(this, arguments) : transform2).end(null);
+ });
}
};
- popover.hasArrow = function(val) {
- if (arguments.length) {
- _hasArrow = utilFunctor(val);
- return popover;
- } else {
- return _hasArrow;
+ zoom.scaleBy = function(selection2, k2, p2) {
+ zoom.scaleTo(selection2, function() {
+ var k0 = _transform.k, k1 = typeof k2 === "function" ? k2.apply(this, arguments) : k2;
+ return k0 * k1;
+ }, p2);
+ };
+ zoom.scaleTo = function(selection2, k2, p2) {
+ zoom.transform(selection2, function() {
+ var e3 = extent.apply(this, arguments), t0 = _transform, p02 = !p2 ? centroid(e3) : typeof p2 === "function" ? p2.apply(this, arguments) : p2, p1 = t0.invert(p02), k1 = typeof k2 === "function" ? k2.apply(this, arguments) : k2;
+ return constrain(translate(scale(t0, k1), p02, p1), e3, translateExtent);
+ }, p2);
+ };
+ zoom.translateBy = function(selection2, x2, y2) {
+ zoom.transform(selection2, function() {
+ return constrain(_transform.translate(
+ typeof x2 === "function" ? x2.apply(this, arguments) : x2,
+ typeof y2 === "function" ? y2.apply(this, arguments) : y2
+ ), extent.apply(this, arguments), translateExtent);
+ });
+ };
+ zoom.translateTo = function(selection2, x2, y2, p2) {
+ zoom.transform(selection2, function() {
+ var e3 = extent.apply(this, arguments), t2 = _transform, p02 = !p2 ? centroid(e3) : typeof p2 === "function" ? p2.apply(this, arguments) : p2;
+ return constrain(identity2.translate(p02[0], p02[1]).scale(t2.k).translate(
+ typeof x2 === "function" ? -x2.apply(this, arguments) : -x2,
+ typeof y2 === "function" ? -y2.apply(this, arguments) : -y2
+ ), e3, translateExtent);
+ }, p2);
+ };
+ function scale(transform2, k2) {
+ k2 = Math.max(scaleExtent[0], Math.min(scaleExtent[1], k2));
+ return k2 === transform2.k ? transform2 : new Transform(k2, transform2.x, transform2.y);
+ }
+ function translate(transform2, p02, p1) {
+ var x2 = p02[0] - p1[0] * transform2.k, y2 = p02[1] - p1[1] * transform2.k;
+ return x2 === transform2.x && y2 === transform2.y ? transform2 : new Transform(transform2.k, x2, y2);
+ }
+ 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, g3 = gesture(that, args), e3 = extent.apply(that, args), p2 = !point2 ? centroid(e3) : typeof point2 === "function" ? point2.apply(that, args) : point2, w2 = Math.max(e3[1][0] - e3[0][0], e3[1][1] - e3[0][1]), a2 = _transform, b2 = typeof transform2 === "function" ? transform2.apply(that, args) : transform2, i3 = interpolate(a2.invert(p2).concat(w2 / a2.k), b2.invert(p2).concat(w2 / b2.k));
+ return function(t2) {
+ if (t2 === 1) {
+ t2 = b2;
+ } else {
+ var l2 = i3(t2);
+ var k2 = w2 / l2[2];
+ t2 = new Transform(k2, p2[0] - l2[0] * k2, p2[1] - l2[1] * k2);
+ }
+ g3.zoom(null, null, t2);
+ };
+ });
+ }
+ 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;
+ dispatch14.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;
+ dispatch14.call("zoom", this, d3_event, key, transform2);
+ return this;
+ },
+ end: function(d3_event) {
+ if (--this.active === 0) {
+ _activeGesture = null;
+ dispatch14.call("end", this, d3_event);
+ }
+ return this;
}
};
- popover.placement = function(val) {
- if (arguments.length) {
- _placement = utilFunctor(val);
- return popover;
+ function wheeled(d3_event) {
+ if (!filter2.apply(this, arguments))
+ return;
+ var g3 = gesture(this, arguments), t2 = _transform, k2 = Math.max(scaleExtent[0], Math.min(scaleExtent[1], t2.k * Math.pow(2, wheelDelta.apply(this, arguments)))), p2 = utilFastMouse(this)(d3_event);
+ if (g3.wheel) {
+ if (g3.mouse[0][0] !== p2[0] || g3.mouse[0][1] !== p2[1]) {
+ g3.mouse[1] = t2.invert(g3.mouse[0] = p2);
+ }
+ clearTimeout(g3.wheel);
} else {
- return _placement;
+ g3.mouse = [p2, t2.invert(p2)];
+ interrupt_default(this);
+ g3.start(d3_event);
}
- };
- popover.alignment = function(val) {
- if (arguments.length) {
- _alignment = utilFunctor(val);
- return popover;
- } else {
- return _alignment;
+ d3_event.preventDefault();
+ d3_event.stopImmediatePropagation();
+ g3.wheel = setTimeout(wheelidled, _wheelDelay);
+ g3.zoom(d3_event, "mouse", constrain(translate(scale(t2, k2), g3.mouse[0], g3.mouse[1]), g3.extent, translateExtent));
+ function wheelidled() {
+ g3.wheel = null;
+ g3.end(d3_event);
}
- };
- popover.scrollContainer = function(val) {
- if (arguments.length) {
- _scrollContainer = utilFunctor(val);
- return popover;
+ }
+ var _downPointerIDs = /* @__PURE__ */ new Set();
+ var _pointerLocGetter;
+ function pointerdown(d3_event) {
+ _downPointerIDs.add(d3_event.pointerId);
+ if (!filter2.apply(this, arguments))
+ return;
+ var g3 = gesture(this, arguments, _downPointerIDs.size === 1);
+ var started;
+ d3_event.stopImmediatePropagation();
+ _pointerLocGetter = utilFastMouse(this);
+ var loc = _pointerLocGetter(d3_event);
+ var p2 = [loc, _transform.invert(loc), d3_event.pointerId];
+ if (!g3.pointer0) {
+ g3.pointer0 = p2;
+ started = true;
+ } else if (!g3.pointer1 && g3.pointer0[2] !== p2[2]) {
+ g3.pointer1 = p2;
+ }
+ if (started) {
+ interrupt_default(this);
+ g3.start(d3_event);
+ }
+ }
+ function pointermove(d3_event) {
+ if (!_downPointerIDs.has(d3_event.pointerId))
+ return;
+ if (!_activeGesture || !_pointerLocGetter)
+ return;
+ var g3 = gesture(this, arguments);
+ var isPointer0 = g3.pointer0 && g3.pointer0[2] === d3_event.pointerId;
+ var isPointer1 = !isPointer0 && g3.pointer1 && g3.pointer1[2] === d3_event.pointerId;
+ if ((isPointer0 || isPointer1) && "buttons" in d3_event && !d3_event.buttons) {
+ if (g3.pointer0)
+ _downPointerIDs.delete(g3.pointer0[2]);
+ if (g3.pointer1)
+ _downPointerIDs.delete(g3.pointer1[2]);
+ g3.end(d3_event);
+ return;
+ }
+ d3_event.preventDefault();
+ d3_event.stopImmediatePropagation();
+ var loc = _pointerLocGetter(d3_event);
+ var t2, p2, l2;
+ if (isPointer0)
+ g3.pointer0[0] = loc;
+ else if (isPointer1)
+ g3.pointer1[0] = loc;
+ t2 = _transform;
+ if (g3.pointer1) {
+ var p02 = g3.pointer0[0], l0 = g3.pointer0[1], p1 = g3.pointer1[0], l1 = g3.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;
+ t2 = scale(t2, Math.sqrt(dp / dl));
+ p2 = [(p02[0] + p1[0]) / 2, (p02[1] + p1[1]) / 2];
+ l2 = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];
+ } else if (g3.pointer0) {
+ p2 = g3.pointer0[0];
+ l2 = g3.pointer0[1];
} else {
- return _scrollContainer;
+ return;
}
- };
- popover.content = function(val) {
- if (arguments.length) {
- _content = val;
- return popover;
+ g3.zoom(d3_event, "touch", constrain(translate(t2, p2, l2), g3.extent, translateExtent));
+ }
+ function pointerup(d3_event) {
+ if (!_downPointerIDs.has(d3_event.pointerId))
+ return;
+ _downPointerIDs.delete(d3_event.pointerId);
+ if (!_activeGesture)
+ return;
+ var g3 = gesture(this, arguments);
+ d3_event.stopImmediatePropagation();
+ if (g3.pointer0 && g3.pointer0[2] === d3_event.pointerId)
+ delete g3.pointer0;
+ else if (g3.pointer1 && g3.pointer1[2] === d3_event.pointerId)
+ delete g3.pointer1;
+ if (g3.pointer1 && !g3.pointer0) {
+ g3.pointer0 = g3.pointer1;
+ delete g3.pointer1;
+ }
+ if (g3.pointer0) {
+ g3.pointer0[1] = _transform.invert(g3.pointer0[0]);
} else {
- return _content;
+ g3.end(d3_event);
}
+ }
+ zoom.wheelDelta = function(_2) {
+ return arguments.length ? (wheelDelta = utilFunctor(+_2), zoom) : wheelDelta;
};
- popover.isShown = function() {
- var popoverSelection = _anchorSelection.select(".popover-" + _id);
- return !popoverSelection.empty() && popoverSelection.classed("in");
+ zoom.filter = function(_2) {
+ return arguments.length ? (filter2 = utilFunctor(!!_2), zoom) : filter2;
};
- popover.show = function() {
- _anchorSelection.each(show);
+ zoom.extent = function(_2) {
+ return arguments.length ? (extent = utilFunctor([[+_2[0][0], +_2[0][1]], [+_2[1][0], +_2[1][1]]]), zoom) : extent;
};
- popover.updateContent = function() {
- _anchorSelection.each(updateContent);
+ zoom.scaleExtent = function(_2) {
+ return arguments.length ? (scaleExtent[0] = +_2[0], scaleExtent[1] = +_2[1], zoom) : [scaleExtent[0], scaleExtent[1]];
};
- popover.hide = function() {
- _anchorSelection.each(hide);
+ zoom.translateExtent = function(_2) {
+ return arguments.length ? (translateExtent[0][0] = +_2[0][0], translateExtent[1][0] = +_2[1][0], translateExtent[0][1] = +_2[0][1], translateExtent[1][1] = +_2[1][1], zoom) : [[translateExtent[0][0], translateExtent[0][1]], [translateExtent[1][0], translateExtent[1][1]]];
};
- popover.toggle = function() {
- _anchorSelection.each(toggle);
+ zoom.constrain = function(_2) {
+ return arguments.length ? (constrain = _2, zoom) : constrain;
};
- 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();
+ zoom.interpolate = function(_2) {
+ return arguments.length ? (interpolate = _2, zoom) : interpolate;
};
- popover.destroyAny = function(selection2) {
- selection2.call(popover.destroy, ".popover");
+ zoom._transform = function(_2) {
+ return arguments.length ? (_transform = _2, zoom) : _transform;
};
- 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);
+ return utilRebind(zoom, dispatch14, "on");
+ }
+
+ // modules/util/double_up.js
+ function utilDoubleUp() {
+ var dispatch14 = dispatch_default("doubleUp");
+ var _maxTimespan = 500;
+ var _maxDistance = 20;
+ var _pointer;
+ function pointerIsValidFor(loc) {
+ return (/* @__PURE__ */ 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;
}
- 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);
- });
- });
+ if (!_pointer) {
+ _pointer = {
+ startLoc: loc,
+ startTime: (/* @__PURE__ */ new Date()).getTime(),
+ upCount: 0,
+ pointerId: d3_event.pointerId
+ };
+ } else {
+ _pointer.pointerId = d3_event.pointerId;
}
}
- 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();
+ 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);
+ dispatch14.call("doubleUp", this, d3_event, locInThis);
+ }
+ _pointer = void 0;
}
- anchor.each(updateContent);
}
- function updateContent() {
- var anchor = select_default2(this);
- if (_content) {
- anchor.selectAll(".popover-" + _id + " > .popover-inner").call(_content.apply(this, arguments));
+ function doubleUp(selection2) {
+ if ("PointerEvent" in window) {
+ selection2.on("pointerdown.doubleUp", pointerdown).on("pointerup.doubleUp", pointerup);
+ } else {
+ selection2.on("dblclick.doubleUp", function(d3_event) {
+ dispatch14.call("doubleUp", this, d3_event, utilFastMouse(this)(d3_event));
+ });
}
- 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;
+ doubleUp.off = function(selection2) {
+ selection2.on("pointerdown.doubleUp", null).on("pointerup.doubleUp", null).on("dblclick.doubleUp", null);
+ };
+ return utilRebind(doubleUp, dispatch14, "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 clamp2(num, min3, max3) {
+ return Math.max(min3, Math.min(num, max3));
+ }
+ function rendererMap(context) {
+ var dispatch14 = 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", zoomPan2).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);
}
- 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;
+ function didUndoOrRedo(targetTransform) {
+ var mode = context.mode().id;
+ if (mode !== "browse" && mode !== "select")
+ return;
+ if (targetTransform) {
+ map2.transformEase(targetTransform);
+ }
}
- 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;
+ 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());
+ dispatch14.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());
+ dispatch14.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 t2 = projection2.transform();
+ var p1 = t2.invert(p02);
+ t2 = t2.scale(zoomOut2 ? 0.5 : 2);
+ t2.x = p02[0] - p1[0] * t2.k;
+ t2.y = p02[1] - p1[1] * t2.k;
+ map2.transformEase(t2);
+ });
+ 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(d2) {
+ return d2.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());
+ dispatch14.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 i3 = 0; i3 < listeners.length; i3++) {
+ var listener = listeners[i3];
+ 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);
}
- 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");
+ event.view = window;
+ window.dispatchEvent(event);
}
- 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
+ return d3_event.button !== 2;
+ }
+ function pxCenter() {
+ return [_dimensions[0] / 2, _dimensions[1] / 2];
+ }
+ function drawEditable(difference2, 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 set4;
+ 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 (difference2) {
+ var complete = difference2.complete(map2.extent());
+ data = Object.values(complete).filter(Boolean);
+ set4 = new Set(Object.keys(complete));
+ filter2 = function(d2) {
+ return set4.has(d2.id);
+ };
+ features.clear(data);
+ } else {
+ if (features.gatherStats(all, graph, _dimensions)) {
+ extent = void 0;
+ }
+ if (extent) {
+ data = context.history().intersects(map2.extent().intersection(extent));
+ set4 = new Set(data.map(function(entity) {
+ return entity.id;
+ }));
+ filter2 = function(d2) {
+ return set4.has(d2.id);
};
} else {
- return {
- x: 0,
- y: 0,
- w: node.offsetWidth,
- h: node.offsetHeight
- };
+ data = all;
+ fullRedraw = true;
+ filter2 = utilFunctor(true);
}
}
- }
- 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);
+ if (applyFeatureLayerFilters) {
+ data = features.filter(data, graph);
} else {
- show.apply(this, arguments);
+ context.features().resetStats();
}
- }
- 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);
+ if (mode && mode.id === "select") {
+ surface.call(drawVertices.drawSelected, graph, map2.extent());
}
- 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;
+ 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);
+ dispatch14.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);
};
- tooltip.content(function() {
- var heading = _heading.apply(this, arguments);
- var text2 = _title.apply(this, arguments);
- var keys2 = _keys.apply(this, arguments);
- var headingCallback = typeof heading === "function" ? heading : (s2) => s2.text(heading);
- var textCallback = typeof text2 === "function" ? text2 : (s2) => s2.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(keys2 && keys2.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(keys2 && keys2.length ? keys2 : []).enter().append("kbd").attr("class", "shortcut").text(function(d2) {
- return d2;
- });
+ 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
};
- });
- return tooltip;
- }
-
- // modules/ui/edit_menu.js
- function uiEditMenu(context) {
- var dispatch14 = 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;
+ var mode = context.mode();
+ if (mode && !allowed[mode.id]) {
+ context.enter(modeBrowse(context));
}
- _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(d2) {
- return "edit-menu-item edit-menu-item-" + d2.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, d2) {
- if (!d2.relatedEntityIds || select_default2(this).classed("disabled"))
- return;
- utilHighlightEntities(d2.relatedEntityIds(), true, context);
- }).on("mouseleave.highlight", function(d3_event, d2) {
- if (!d2.relatedEntityIds)
+ dispatch14.call("drawn", this, { full: true });
+ }
+ function gestureChange(d3_event) {
+ var e3 = d3_event;
+ e3.preventDefault();
+ var props = {
+ deltaMode: 0,
+ // dummy values to ignore in zoomPan
+ deltaY: 1,
+ // dummy values to ignore in zoomPan
+ clientX: e3.clientX,
+ clientY: e3.clientY,
+ screenX: e3.screenX,
+ screenY: e3.screenY,
+ x: e3.x,
+ y: e3.y
+ };
+ var e22 = new WheelEvent("wheel", props);
+ e22._scale = e3.scale;
+ e22._rotation = e3.rotation;
+ _selection.node().dispatchEvent(e22);
+ }
+ function zoomPan2(event, key, transform2) {
+ var source = event && event.sourceEvent || event;
+ var eventTransform = transform2 || event && event.transform;
+ var x2 = eventTransform.x;
+ var y2 = eventTransform.y;
+ var k2 = eventTransform.k;
+ if (source && source.type === "wheel") {
+ if (_pointerDown)
return;
- utilHighlightEntities(d2.relatedEntityIds(), false, context);
- });
- buttonsEnter.each(function(d2) {
- var tooltip = uiTooltip().heading(() => d2.title).title(d2.tooltip).keys([d2.keys[0]]);
- _tooltips.push(tooltip);
- select_default2(this).call(tooltip).append("div").attr("class", "icon-wrap").call(svgIcon(d2.icon && d2.icon() || "#iD-operation-" + d2.id, "operation"));
- });
- if (showLabels) {
- buttonsEnter.append("span").attr("class", "label").each(function(d2) {
- select_default2(this).call(d2.title);
- });
- }
- buttonsEnter.merge(buttons).classed("disabled", function(d2) {
- return d2.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())();
+ var detected = utilDetect();
+ var dX = source.deltaX;
+ var dY = source.deltaY;
+ var x22 = x2;
+ var y22 = y2;
+ var k22 = k2;
+ var t0, p02, p1;
+ if (source.deltaMode === 1) {
+ var lines = Math.abs(source.deltaY);
+ var sign2 = source.deltaY > 0 ? 1 : -1;
+ dY = sign2 * clamp2(
+ Math.exp((lines - 1) * 0.75) * 4.000244140625,
+ 4.000244140625,
+ // min
+ 350.000244140625
+ // max
+ );
+ if (detected.os !== "mac") {
+ dY *= 5;
}
- } else {
- if (lastPointerUpType === "touch" || lastPointerUpType === "pen") {
- context.ui().flash.duration(2e3).iconName("#iD-operation-" + operation.id).iconClass("operation").label(operation.annotation() || operation.title)();
+ t0 = _isTransformed ? _transformLast : _transformStart;
+ p02 = _getMouseCoords(source);
+ p1 = t0.invert(p02);
+ k22 = t0.k * Math.pow(2, -dY / 500);
+ k22 = clamp2(k22, kMin, kMax);
+ x22 = p02[0] - p1[0] * k22;
+ y22 = p02[1] - p1[1] * k22;
+ } else if (source._scale) {
+ t0 = _gestureTransformStart;
+ p02 = _getMouseCoords(source);
+ p1 = t0.invert(p02);
+ k22 = t0.k * source._scale;
+ k22 = clamp2(k22, kMin, kMax);
+ x22 = p02[0] - p1[0] * k22;
+ y22 = p02[1] - p1[1] * k22;
+ } else if (source.ctrlKey && !isInteger(dY)) {
+ dY *= 6;
+ t0 = _isTransformed ? _transformLast : _transformStart;
+ p02 = _getMouseCoords(source);
+ p1 = t0.invert(p02);
+ k22 = t0.k * Math.pow(2, -dY / 500);
+ k22 = clamp2(k22, kMin, kMax);
+ x22 = p02[0] - p1[0] * k22;
+ y22 = p02[1] - p1[1] * k22;
+ } else if ((source.altKey || source.shiftKey) && isInteger(dY)) {
+ t0 = _isTransformed ? _transformLast : _transformStart;
+ p02 = _getMouseCoords(source);
+ p1 = t0.invert(p02);
+ k22 = t0.k * Math.pow(2, -dY / 500);
+ k22 = clamp2(k22, kMin, kMax);
+ x22 = p02[0] - p1[0] * k22;
+ y22 = p02[1] - p1[1] * k22;
+ } else if (detected.os === "mac" && detected.browser !== "Firefox" && !source.ctrlKey && isInteger(dX) && isInteger(dY)) {
+ p1 = projection2.translate();
+ x22 = p1[0] - dX;
+ y22 = p1[1] - dY;
+ k22 = projection2.scale();
+ k22 = clamp2(k22, kMin, kMax);
+ }
+ if (x22 !== x2 || y22 !== y2 || k22 !== k2) {
+ x2 = x22;
+ y2 = y22;
+ k2 = k22;
+ eventTransform = identity2.translate(x22, y22).scale(k22);
+ if (_zoomerPanner._transform) {
+ _zoomerPanner._transform(eventTransform);
+ } else {
+ _selection.node().__zoom = eventTransform;
}
- operation();
- editMenu.close();
}
- lastPointerUpType = null;
}
- dispatch14.call("toggled", this, true);
- };
- function updatePosition() {
- if (!_menu || _menu.empty())
+ if (_transformStart.x === x2 && _transformStart.y === y2 && _transformStart.k === k2) {
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();
+ }
+ if (geoScaleToZoom(k2, TILESIZE) < _minzoom) {
+ surface.interrupt();
+ dispatch14.call("hitMinZoom", this, map2);
+ setCenterZoom(map2.center(), context.minEditableZoom(), 0, true);
+ scheduleRedraw();
+ dispatch14.call("move", this, map2);
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;
+ projection2.transform(eventTransform);
+ var withinEditableZoom = map2.withinEditableZoom();
+ if (_lastWithinEditableZoom !== withinEditableZoom) {
+ if (_lastWithinEditableZoom !== void 0) {
+ dispatch14.call("crossEditableZoom", this, withinEditableZoom);
}
+ _lastWithinEditableZoom = withinEditableZoom;
}
- 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;
- }
+ var scale = k2 / _transformStart.k;
+ var tX = (x2 / scale - _transformStart.x) * scale;
+ var tY = (y2 / scale - _transformStart.y) * scale;
+ if (context.inIntro()) {
+ curtainProjection.transform({
+ x: x2 - tX,
+ y: y2 - tY,
+ k: k2
+ });
}
- 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";
- }
+ if (source) {
+ _lastPointerEvent = event;
+ }
+ _isTransformed = true;
+ _transformLast = eventTransform;
+ utilSetTransform(supersurface, tX, tY, scale);
+ scheduleRedraw();
+ dispatch14.call("move", this, map2);
+ function isInteger(val) {
+ return typeof val === "number" && isFinite(val) && Math.floor(val) === val;
}
}
- editMenu.close = function() {
- context.map().on("move.edit-menu", null).on("drawn.edit-menu", null);
- _menu.remove();
- _tooltips = [];
- dispatch14.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, dispatch14, "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(k2) {
- if (stats[k2]) {
- count += stats[k2];
- return _t.append("inspector.title_count", {
- title: _t("feature." + k2 + ".description"),
- count: stats[k2]
- });
- }
- 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"));
- });
+ function resetTransform() {
+ if (!_isTransformed)
+ return false;
+ utilSetTransform(supersurface, 0, 0);
+ _isTransformed = false;
+ if (context.inIntro()) {
+ curtainProjection.transform(projection2.transform());
}
- selection2.classed("hide", !hiddenList.length);
+ return true;
}
- 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 = (s2) => s2.text("");
- function flash() {
- if (_flashTimer) {
- _flashTimer.stop();
+ function redraw(difference2, extent) {
+ if (surface.empty() || !_redrawEnabled)
+ return;
+ if (resetTransform()) {
+ difference2 = extent = void 0;
+ }
+ var zoom = map2.zoom();
+ var z2 = String(~~zoom);
+ if (surface.attr("data-zoom") !== z2) {
+ surface.attr("data-zoom", z2);
}
- 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;
+ 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 (!difference2) {
+ supersurface.call(context.background());
+ wrapper.call(drawLayers);
+ }
+ if (map2.editableDataEnabled() || map2.isInWideSelection()) {
+ context.loadTiles(projection2);
+ drawEditable(difference2, extent);
+ } else {
+ editOff();
+ }
+ _transformStart = projection2.transform();
+ return map2;
}
- flash.duration = function(_2) {
- if (!arguments.length)
- return _duration;
- _duration = _2;
- return flash;
+ var immediateRedraw = function(difference2, extent) {
+ if (!difference2 && !extent)
+ cancelPendingRedraw();
+ redraw(difference2, extent);
};
- flash.label = function(_2) {
- if (!arguments.length)
- return _label;
- if (typeof _2 !== "function") {
- _label = (selection2) => selection2.text(_2);
- } else {
- _label = (selection2) => selection2.text("").call(_2);
+ map2.lastPointerEvent = function() {
+ return _lastPointerEvent;
+ };
+ map2.mouse = function(d3_event) {
+ var event = d3_event || _lastPointerEvent;
+ if (event) {
+ var s2;
+ while (s2 = event.sourceEvent) {
+ event = s2;
+ }
+ return _getMouseCoords(event);
}
- return flash;
+ return null;
};
- flash.iconName = function(_2) {
+ map2.mouseCoordinates = function() {
+ var coord2 = map2.mouse() || pxCenter();
+ return projection2.invert(coord2);
+ };
+ map2.dblclickZoomEnable = function(val) {
if (!arguments.length)
- return _iconName;
- _iconName = _2;
- return flash;
+ return _dblClickZoomEnabled;
+ _dblClickZoomEnabled = val;
+ return map2;
};
- flash.iconClass = function(_2) {
+ map2.redrawEnable = function(val) {
if (!arguments.length)
- return _iconClass;
- _iconClass = _2;
- return flash;
+ return _redrawEnabled;
+ _redrawEnabled = val;
+ return map2;
};
- 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;
+ map2.isTransformed = function() {
+ return _isTransformed;
+ };
+ function setTransform(t2, duration, force) {
+ var t3 = projection2.transform();
+ if (!force && t2.k === t3.k && t2.x === t3.x && t2.y === t3.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 isFullScreen() {
- return document.fullscreenElement || document.mozFullScreenElement || document.webkitFullscreenElement || document.msFullscreenElement;
- }
- function isSupported() {
- return !!getFullScreenFn();
+ function setCenterZoom(loc2, z2, duration, force) {
+ var c2 = map2.center();
+ var z3 = map2.zoom();
+ if (loc2[0] === c2[0] && loc2[1] === c2[1] && z2 === z3 && !force)
+ return false;
+ var proj = geoRawMercator().transform(projection2.transform());
+ var k2 = clamp2(geoZoomToScale(z2, TILESIZE), kMin, kMax);
+ proj.scale(k2);
+ var t2 = proj.translate();
+ var point2 = proj(loc2);
+ var center = pxCenter();
+ t2[0] += center[0] - point2[0];
+ t2[1] += center[1] - point2[1];
+ return setTransform(identity2.translate(t2[0], t2[1]).scale(k2), duration, force);
}
- function fullScreen(d3_event) {
- d3_event.preventDefault();
- if (!isFullScreen()) {
- getFullScreenFn().apply(element);
+ map2.pan = function(delta, duration) {
+ var t2 = projection2.translate();
+ var k2 = projection2.scale();
+ t2[0] += delta[0];
+ t2[1] += delta[1];
+ if (duration) {
+ _selection.transition().duration(duration).on("start", function() {
+ map2.startEase();
+ }).call(_zoomerPanner.transform, identity2.translate(t2[0], t2[1]).scale(k2));
} else {
- getExitFullScreenFn().apply(document);
+ projection2.translate(t2);
+ _transformStart = projection2.transform();
+ _selection.call(_zoomerPanner.transform, _transformStart);
+ dispatch14.call("move", this, map2);
+ immediateRedraw();
}
- }
- return function() {
- if (!isSupported())
- return;
- var detected = utilDetect();
- var keys2 = detected.os === "mac" ? [uiCmd("\u2303\u2318F"), "f11"] : ["f11"];
- context.keybinding().on(keys2, fullScreen);
+ return map2;
};
- }
-
- // 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
+ 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;
};
- 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 zoomIn(delta) {
+ setCenterZoom(map2.center(), ~~map2.zoom() + delta, 250, true);
}
- function success(geolocation) {
- _position = geolocation;
- var coords = _position.coords;
- _extent = geoExtent([coords.longitude, coords.latitude]).padByMeters(coords.accuracy);
- zoomTo();
- finish();
+ function zoomOut(delta) {
+ setCenterZoom(map2.center(), ~~map2.zoom() - delta, 250, true);
}
- function error() {
- if (_position) {
- zoomTo();
- } else {
- context.ui().flash.label(_t.append("geolocate.location_unavailable")).iconName("#iD-icon-geolocate")();
+ 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());
}
- finish();
- }
- function finish() {
- _locating.close();
- if (_timeoutID) {
- clearTimeout(_timeoutID);
+ if (setCenterZoom(loc2, map2.zoom())) {
+ dispatch14.call("move", this, map2);
}
- _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);
+ scheduleRedraw();
+ return map2;
};
- }
-
- // 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 = {};
+ 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];
}
- selection2.text("");
- var list = selection2.append("ul").attr("class", "background-info");
- list.append("li").call(_currSourceName);
- _metadataKeys.forEach(function(k2) {
- if (isDG && k2 === "vintage")
- return;
- list.append("li").attr("class", "background-info-list-" + k2).classed("hide", !_metadata[k2]).call(_t.append("info_panels.background." + k2, { suffix: ":" })).append("span").attr("class", "background-info-span-" + k2).text(_metadata[k2]);
- });
- 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);
- });
+ return [0, 0];
+ };
+ map2.zoom = function(z2) {
+ if (!arguments.length) {
+ return Math.max(geoScaleToZoom(projection2.scale(), TILESIZE), 0);
}
- ["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);
+ if (z2 < _minzoom) {
+ surface.interrupt();
+ dispatch14.call("hitMinZoom", this, map2);
+ z2 = context.minEditableZoom();
+ }
+ if (setCenterZoom(map2.center(), z2)) {
+ dispatch14.call("move", this, map2);
+ }
+ scheduleRedraw();
+ return map2;
+ };
+ map2.centerZoom = function(loc2, z2) {
+ if (setCenterZoom(loc2, z2)) {
+ dispatch14.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 = clamp2(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 = clamp2(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 br2 = projection2([extent[1][0], extent[0][1]]);
+ var hFactor = (br2[0] - tl[0]) / dim[0];
+ var vFactor = (br2[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;
}
- 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 d2 = tile.datum();
- var zoom = d2 && d2.length >= 3 && d2[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 (!d2 || !d2.length >= 3)
- return;
- background.baseLayerSource().getMetadata(center, d2, 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(k2) {
- if (k2 === "zoom" || k2 === "vintage")
- return;
- var val = result[k2];
- _metadata[k2] = val;
- selection2.selectAll(".background-info-list-" + k2).classed("hide", !val).selectAll(".background-info-span-" + k2).text(val);
- });
+ 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]);
+ dispatch14.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]);
+ dispatch14.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));
});
}
- 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);
+ map2.layers = () => drawLayers;
+ map2.doubleUpHandler = function() {
+ return _doubleUpHandler;
};
- panel.id = "background";
- panel.label = _t.append("info_panels.background.title");
- panel.key = _t("info_panels.background.key");
- return panel;
+ return utilRebind(map2, dispatch14, "on");
}
- // 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 d2 = new Date(timestamp);
- if (isNaN(d2.getTime()))
- return _t("info_panels.history.unknown");
- return d2.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");
+ // modules/renderer/photos.js
+ function rendererPhotos(context) {
+ var dispatch14 = dispatch_default("change");
+ var _layerIDs = ["streetside", "mapillary", "mapillary-map-features", "mapillary-signs", "kartaview", "mapilio", "vegbilder"];
+ var _allPhotoTypes = ["flat", "panoramic"];
+ var _shownPhotoTypes = _allPhotoTypes.slice();
+ var _dateFilters = ["fromDate", "toDate"];
+ var _fromDate;
+ var _toDate;
+ var _usernames;
+ function photos() {
}
- function displayChangeset(selection2, changeset) {
- if (!changeset) {
- selection2.append("span").call(_t.append("info_panels.history.unknown"));
+ function updateStorage() {
+ if (window.mocha)
return;
+ var hash = utilStringQs(window.location.hash);
+ var enabled = context.layers().all().filter(function(d2) {
+ return _layerIDs.indexOf(d2.id) !== -1 && d2.layer && d2.layer.supported() && d2.layer.enabled();
+ }).map(function(d2) {
+ return d2.id;
+ });
+ if (enabled.length) {
+ hash.photo_overlay = enabled.join(",");
+ } else {
+ delete hash.photo_overlay;
}
- 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");
+ window.location.replace("#" + utilQsString(hash, true));
}
- 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);
+ 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 {
- selected = context.selectedIDs().filter(function(e3) {
- return context.hasEntity(e3);
- });
- if (selected.length) {
- entity = context.entity(selected[0]);
+ val = null;
+ }
+ if (type2 === _dateFilters[0]) {
+ _fromDate = val;
+ if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
+ _toDate = _fromDate;
}
}
- 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 (type2 === _dateFilters[1]) {
+ _toDate = val;
+ if (_fromDate && _toDate && new Date(_toDate) < new Date(_fromDate)) {
+ _fromDate = _toDate;
+ }
}
- if (!singular)
- return;
- if (entity) {
- selection2.call(redrawEntity, entity);
- } else if (note) {
- selection2.call(redrawNote, note);
+ dispatch14.call("change", this);
+ if (updateUrl) {
+ var rangeString;
+ if (_fromDate || _toDate) {
+ rangeString = (_fromDate || "") + "_" + (_toDate || "");
+ }
+ setUrlFilterValue("photo_dates", rangeString);
}
- }
- function redrawNote(selection2, note) {
- if (!note || note.isNew()) {
- selection2.append("div").call(_t.append("info_panels.history.note_no_history"));
- return;
+ };
+ photos.setUsernameFilter = function(val, updateUrl) {
+ if (val && typeof val === "string")
+ val = val.replace(/;/g, ",").split(",");
+ if (val) {
+ val = val.map((d2) => d2.trim()).filter(Boolean);
+ if (!val.length) {
+ val = null;
+ }
}
- 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);
+ _usernames = val;
+ dispatch14.call("change", this);
+ if (updateUrl) {
+ var hashString;
+ if (_usernames) {
+ hashString = _usernames.join(",");
+ }
+ setUrlFilterValue("photo_username", hashString);
}
- 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 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 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);
+ function showsLayer(id2) {
+ var layer = context.layers().layer(id2);
+ return layer && layer.supported() && layer.enabled();
}
- 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);
- });
+ photos.shouldFilterByDate = function() {
+ return showsLayer("mapillary") || showsLayer("kartaview") || showsLayer("streetside") || showsLayer("vegbilder");
};
- panel.off = function() {
- context.map().on("drawn.info-history", null);
- context.on("enter.info-history", null);
+ photos.shouldFilterByPhotoType = function() {
+ return showsLayer("mapillary") || showsLayer("streetside") && showsLayer("kartaview") || showsLayer("vegbilder");
};
- 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(m2, isImperial) {
- var d2 = m2 * (isImperial ? 3.28084 : 1);
- var unit2;
- if (isImperial) {
- if (d2 >= 5280) {
- d2 /= 5280;
- unit2 = "miles";
+ 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 {
- unit2 = "feet";
+ _shownPhotoTypes.push(val);
}
- } else {
- if (d2 >= 1e3) {
- d2 /= 1e3;
- unit2 = "kilometers";
- } else {
- unit2 = "meters";
+ dispatch14.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);
}
- }
- return _t("units." + unit2, {
- quantity: d2.toLocaleString(_mainLocalizer.localeCode(), {
- maximumSignificantDigits: 4
- })
- });
- }
- function displayArea(m2, isImperial) {
- var locale2 = _mainLocalizer.localeCode();
- var d2 = m2 * (isImperial ? 10.7639111056 : 1);
- var d1, d22, area;
- var unit1 = "";
- var unit2 = "";
- if (isImperial) {
- if (d2 >= 6969600) {
- d1 = d2 / 27878400;
- unit1 = "square_miles";
- } else {
- d1 = d2;
- unit1 = "square_feet";
+ if (hash.photo_username) {
+ this.setUsernameFilter(hash.photo_username, false);
}
- if (d2 > 4356 && d2 < 4356e4) {
- d22 = d2 / 43560;
- unit2 = "acres";
+ 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);
+ });
}
- } else {
- if (d2 >= 25e4) {
- d1 = d2 / 1e6;
- unit1 = "square_kilometers";
+ 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, dispatch14, "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 {
- d1 = d2;
- unit1 = "square_meters";
+ osm.userDetails((err, user) => render(selection2, user));
}
- if (d2 > 1e3 && d2 < 1e7) {
- d22 = d2 / 1e4;
- unit2 = "hectares";
+ }
+ 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").text(user.display_name);
+ loginLogout.classed("hide", false).select("a").text(_t("logout")).on("click", (e3) => {
+ e3.preventDefault();
+ osm.logout();
+ tryLogout();
+ });
+ } else {
+ userInfo.html("").classed("hide", true);
+ loginLogout.classed("hide", false).select("a").text(_t("login")).on("click", (e3) => {
+ e3.preventDefault();
+ osm.authenticate();
+ });
}
}
- area = _t("units." + unit1, {
- quantity: d1.toLocaleString(locale2, {
- maximumSignificantDigits: 4
- })
- });
- if (d22) {
- return _t("units.area_pair", {
- area1: area,
- area2: _t("units." + unit2, {
- quantity: d22.toLocaleString(locale2, {
- maximumSignificantDigits: 2
- })
- })
- });
- } else {
- return area;
+ function tryLogout() {
+ if (!osm)
+ return;
+ const url = osm.getUrlRoot() + "/logout?referer=%2Flogin";
+ const w2 = 600;
+ const h2 = 550;
+ const settings = [
+ ["width", w2],
+ ["height", h2],
+ ["left", window.screen.width / 2 - w2 / 2],
+ ["top", window.screen.height / 2 - h2 / 2]
+ ].map((x2) => x2.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);
+ };
}
- function wrap(x2, min3, max3) {
- var d2 = max3 - min3;
- return ((x2 - min3) % d2 + d2) % d2 + min3;
- }
- function clamp2(x2, min3, max3) {
- return Math.max(min3, Math.min(x2, 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))
+
+ // modules/ui/attribution.js
+ function uiAttribution(context) {
+ let _selection = select_default2(null);
+ function render(selection2, data, klass) {
+ let div = selection2.selectAll(".".concat(klass)).data([0]);
+ div = div.enter().append("div").attr("class", klass).merge(div);
+ let attributions = div.selectAll(".attribution").data(data, (d2) => d2.id);
+ attributions.exit().remove();
+ attributions = attributions.enter().append("span").attr("class", "attribution").each((d2, i3, nodes) => {
+ let attribution = select_default2(nodes[i3]);
+ if (d2.terms_html) {
+ attribution.html(d2.terms_html);
+ return;
+ }
+ if (d2.terms_url) {
+ attribution = attribution.append("a").attr("href", d2.terms_url).attr("target", "_blank");
+ }
+ const sourceID = d2.id.replace(/\./g, "");
+ const terms_text = _t(
+ "imagery.".concat(sourceID, ".attribution.text"),
+ { default: d2.terms_text || d2.id || d2.name() }
+ );
+ if (d2.icon && !d2.overlay) {
+ attribution.append("img").attr("class", "source-image").attr("src", d2.icon);
+ }
+ attribution.append("span").attr("class", "attribution-text").text(terms_text);
+ }).merge(attributions);
+ let copyright = attributions.selectAll(".copyright-notice").data((d2) => {
+ let notice = d2.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 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)
- });
+ function update() {
+ let baselayer = context.background().baseLayerSource();
+ _selection.call(render, baselayer ? [baselayer] : [], "base-layer-attribution");
+ const z2 = context.map().zoom();
+ let overlays = context.background().overlayLayerSources() || [];
+ _selection.call(render, overlays.filter((s2) => s2.validZoom(z2)), "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/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);
+ // 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 u2 = Object.keys(users), subset = u2.slice(0, u2.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(d2) {
+ return osm.userURL(d2);
+ }).attr("target", "_blank").text(String);
+ if (u2.length > limit) {
+ var count = select_default2(document.createElement("span"));
+ var othersNum = u2.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 {
- services.geocoder.reverse(coord2, function(err, result) {
- currLocation = result ? result.display_name : _t("info_panels.location.unknown_location");
- selection2.selectAll(".location-info").text(currLocation);
- });
+ wrap2.append("span").html(_t.html("contributors.list", { users: { html: userList.html() } }));
+ }
+ if (!u2.length) {
+ hidden = true;
+ wrap2.transition().style("opacity", 0);
+ } else if (hidden) {
+ wrap2.transition().style("opacity", 1);
}
}
- 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);
+ return function(selection2) {
+ if (!osm)
+ return;
+ wrap2 = selection2;
+ update();
+ osm.on("loaded.contributors", debouncedUpdate);
+ context.map().on("move.contributors", debouncedUpdate);
};
- 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(r2) {
- return r2 * 63710071809e-4;
- }
- function steradiansToSqmeters(r2) {
- return r2 / (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";
+ // 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 {
- 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 i3 in selected) {
- var entity = selected[i3];
- 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));
+ return _displayType;
}
- if (typeof distance === "number") {
- list.append("li").call(_t.append("info_panels.measurement.distance", { suffix: ":" })).append("span").text(displayLength(distance, _isImperial));
+ };
+ popover.hasArrow = function(val) {
+ if (arguments.length) {
+ _hasArrow = utilFunctor(val);
+ return popover;
+ } else {
+ return _hasArrow;
}
- 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));
+ };
+ popover.placement = function(val) {
+ if (arguments.length) {
+ _placement = utilFunctor(val);
+ return popover;
+ } else {
+ return _placement;
}
- 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));
+ };
+ popover.alignment = function(val) {
+ if (arguments.length) {
+ _alignment = utilFunctor(val);
+ return popover;
+ } else {
+ return _alignment;
}
- 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));
+ };
+ popover.scrollContainer = function(val) {
+ if (arguments.length) {
+ _scrollContainer = utilFunctor(val);
+ return popover;
+ } else {
+ return _scrollContainer;
}
- 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);
- });
+ };
+ popover.content = function(val) {
+ if (arguments.length) {
+ _content = val;
+ return popover;
+ } else {
+ return _content;
}
- }
- 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);
+ popover.isShown = function() {
+ var popoverSelection = _anchorSelection.select(".popover-" + _id);
+ return !popoverSelection.empty() && popoverSelection.classed("in");
};
- 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(k2) {
- if (!panels[k2]) {
- panels[k2] = uiInfoPanels[k2](context);
- active[k2] = false;
- }
- });
- function info(selection2) {
- function redraw() {
- var activeids = ids.filter(function(k2) {
- return active[k2];
- }).sort();
- var containers = infoPanels.selectAll(".panel-container").data(activeids, function(k2) {
- return k2;
- });
- containers.exit().style("opacity", 1).transition().duration(200).style("opacity", 0).on("end", function(d2) {
- select_default2(this).call(panels[d2].off).remove();
- });
- var enter = containers.enter().append("div").attr("class", function(d2) {
- return "fillD2 panel-container panel-container-" + d2;
- });
- 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(d2) {
- return panels[d2].label(select_default2(this));
- });
- title.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", function(d3_event, d2) {
- d3_event.stopImmediatePropagation();
- d3_event.preventDefault();
- info.toggle(d2);
- }).call(svgIcon("#iD-icon-close"));
- enter.append("div").attr("class", function(d2) {
- return "panel-content panel-content-" + d2;
- });
- infoPanels.selectAll(".panel-content").each(function(d2) {
- select_default2(this).call(panels[d2]);
- });
- }
- info.toggle = function(which) {
- var activeids = ids.filter(function(k2) {
- return active[k2];
- });
- 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(k2) {
- active[k2] = false;
- });
- } else {
- wasActive.forEach(function(k2) {
- active[k2] = 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(k2) {
- var key = _t("info_panels." + k2 + ".key", { default: null });
- if (!key)
- return;
- context.keybinding().on(uiCmd("\u2318\u21E7" + key), function(d3_event) {
- d3_event.stopImmediatePropagation();
- d3_event.preventDefault();
- info.toggle(k2);
- });
- });
- }
- 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
+ popover.show = function() {
+ _anchorSelection.each(show);
};
- }
- 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
+ popover.updateContent = function() {
+ _anchorSelection.each(updateContent);
};
- }
- function icon(name, svgklass, useklass) {
- return '";
- }
- 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] };
+ 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 reps;
- if (replacements) {
- reps = Object.assign(replacements, helpStringReplacements);
- } else {
- reps = helpStringReplacements;
- }
- return _t.html(id2, reps).replace(/\`(.*?)\`/g, "$1");
- }
- 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(k2) {
- var key2 = "intro.graph." + k2;
- var tag = "addr:" + k2;
- 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;
+ 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;
+ }
}
- }
- });
- }
- 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 i3 = 0; i3 < points.length; i3++) {
- var a2 = points[(i3 - 1 + points.length) % points.length];
- var origin = points[i3];
- var b2 = points[(i3 + 1) % points.length];
- var dotp = geoVecNormalizedDot(a2, b2, origin);
- var mag = Math.abs(dotp);
- if (mag > lowerBound && mag < upperBound) {
- return false;
+ 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);
+ });
+ });
}
}
- 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());
+ 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);
}
- curtain.reveal = function(box, html2, options2) {
- options2 = options2 || {};
- if (typeof box === "string") {
- box = select_default2(box).node();
+ function updateContent() {
+ var anchor = select_default2(this);
+ if (_content) {
+ anchor.selectAll(".popover-" + _id + " > .popover-inner").call(_content.apply(this, arguments));
}
- if (box && box.getBoundingClientRect) {
- box = copyBox(box.getBoundingClientRect());
- var containerRect = containerNode.getBoundingClientRect();
- box.top -= containerRect.top;
- box.left -= containerRect.left;
+ 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;
}
- 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 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;
}
- 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());
+ 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 {
- tooltipBox = box;
+ popoverSelection.style("left", null).style("top", null);
}
- if (tooltipBox && html2) {
- if (html2.indexOf("**") !== -1) {
- if (html2.indexOf(")(.+?)(\*\*)/, "$1$2$3");
- } else {
- html2 = html2.replace(/^(.+?)(\*\*)/, "$1$2");
- }
- html2 = html2.replace(/\*\*(.*?)\*\*/g, '$1');
- }
- html2 = html2.replace(/\*(.*?)\*/g, "$1");
- html2 = html2.replace(/\{br\}/g, "
");
- if (options2.buttonText && options2.buttonCallback) {
- html2 += '";
- }
- 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()), w2 = containerNode.clientWidth, h2 = containerNode.clientHeight, tooltipWidth = 200, tooltipArrow = 5, side, pos;
- if (options2.tooltipClass === "intro-mouse") {
- tip.height += 80;
- }
- if (tooltipBox.top + tooltipBox.height > h2) {
- tooltipBox.height -= tooltipBox.top + tooltipBox.height - h2;
- }
- if (tooltipBox.left + tooltipBox.width > w2) {
- tooltipBox.width -= tooltipBox.left + tooltipBox.width - w2;
- }
- 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 > h2 - 140) {
- side = "top";
- pos = [
- tooltipBox.left + tooltipBox.width / 2 - tip.width / 2,
- tooltipBox.top - tip.height
- ];
+ 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 {
- 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 > w2 - 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 > h2 - 100) {
- shiftY = h2 - pos[1] - tip.height - 100;
- }
+ return {
+ x: 0,
+ y: 0,
+ w: node.offsetWidth,
+ h: node.offsetHeight
+ };
}
- 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;
+ }
+ 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 {
- selection2 = darkness.transition().duration(duration || 600).ease(linear2);
+ show.apply(this, arguments);
}
- selection2.attr("d", function(d2) {
- var containerWidth = containerNode.clientWidth;
- var containerHeight = containerNode.clientHeight;
- var string = "M 0,0 L 0," + containerHeight + " L " + containerWidth + "," + containerHeight + "L" + containerWidth + ",0 Z";
- if (!d2)
- return string;
- return string + "M" + d2.left + "," + d2.top + "L" + d2.left + "," + (d2.top + d2.height) + "L" + (d2.left + d2.width) + "," + (d2.top + d2.height) + "L" + (d2.left + d2.width) + "," + d2.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;
+ return popover;
}
- // modules/ui/intro/welcome.js
- function uiIntroWelcome(context, reveal) {
- var dispatch14 = dispatch_default("done");
- var chapter = {
- title: "intro.welcome.title"
+ // 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;
};
- 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() {
- dispatch14.call("done");
- reveal(
- ".intro-nav-wrap .chapter-navigation",
- helpHtml("intro.welcome.chapters", { next: _t("intro.navigation.title") })
- );
- }
- chapter.enter = function() {
- welcome();
+ var _heading = utilFunctor(null);
+ var _keys = utilFunctor(null);
+ tooltip.title = function(val) {
+ if (!arguments.length)
+ return _title;
+ _title = utilFunctor(val);
+ return tooltip;
};
- chapter.exit = function() {
- context.container().select(".curtain-tooltip.intro-mouse").selectAll(".counter").remove();
+ tooltip.heading = function(val) {
+ if (!arguments.length)
+ return _heading;
+ _heading = utilFunctor(val);
+ return tooltip;
};
- chapter.restart = function() {
- chapter.exit();
- chapter.enter();
+ tooltip.keys = function(val) {
+ if (!arguments.length)
+ return _keys;
+ _keys = utilFunctor(val);
+ return tooltip;
};
- return utilRebind(chapter, dispatch14, "on");
+ tooltip.content(function() {
+ var heading2 = _heading.apply(this, arguments);
+ var text = _title.apply(this, arguments);
+ var keys2 = _keys.apply(this, arguments);
+ var headingCallback = typeof heading2 === "function" ? heading2 : (s2) => s2.text(heading2);
+ var textCallback = typeof text === "function" ? text : (s2) => s2.text(text);
+ return function(selection2) {
+ var headingSelect = selection2.selectAll(".tooltip-heading").data(heading2 ? [heading2] : []);
+ headingSelect.exit().remove();
+ headingSelect.enter().append("div").attr("class", "tooltip-heading").merge(headingSelect).text("").call(headingCallback);
+ var textSelect = selection2.selectAll(".tooltip-text").data(text ? [text] : []);
+ textSelect.exit().remove();
+ textSelect.enter().append("div").attr("class", "tooltip-text").merge(textSelect).text("").call(textCallback);
+ var keyhintWrap = selection2.selectAll(".keyhint-wrap").data(keys2 && keys2.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(keys2 && keys2.length ? keys2 : []).enter().append("kbd").attr("class", "shortcut").text(function(d2) {
+ return d2;
+ });
+ };
+ });
+ return tooltip;
}
- // modules/ui/intro/navigation.js
- function uiIntroNavigation(context, reveal) {
- var dispatch14 = 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(f3, t2) {
- timeouts.push(window.setTimeout(f3, t2));
- }
- 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 });
+ // modules/ui/edit_menu.js
+ function uiEditMenu(context) {
+ var dispatch14 = 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;
}
- 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);
- }
+ _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(d2) {
+ return "edit-menu-item edit-menu-item-" + d2.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, d2) {
+ if (!d2.relatedEntityIds || select_default2(this).classed("disabled"))
+ return;
+ utilHighlightEntities(d2.relatedEntityIds(), true, context);
+ }).on("mouseleave.highlight", function(d3_event, d2) {
+ if (!d2.relatedEntityIds)
+ return;
+ utilHighlightEntities(d2.relatedEntityIds(), false, context);
+ });
+ buttonsEnter.each(function(d2) {
+ var tooltip = uiTooltip().heading(() => d2.title).title(d2.tooltip).keys([d2.keys[0]]);
+ _tooltips.push(tooltip);
+ select_default2(this).call(tooltip).append("div").attr("class", "icon-wrap").call(svgIcon(d2.icon && d2.icon() || "#iD-operation-" + d2.id, "operation"));
+ });
+ if (showLabels) {
+ buttonsEnter.append("span").attr("class", "label").each(function(d2) {
+ select_default2(this).call(d2.title);
});
- }, 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 });
+ buttonsEnter.merge(buttons).classed("disabled", function(d2) {
+ return d2.disabled();
});
- context.map().on("move.intro", function() {
- if (context.map().zoom() !== zoomStart) {
- context.map().on("move.intro", null);
- timeout2(function() {
- continueTo(features);
- }, 3e3);
+ 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();
});
- function continueTo(nextStep) {
- context.map().on("move.intro drawn.intro", null);
- nextStep();
+ var lastPointerUpType;
+ function pointerup(d3_event) {
+ lastPointerUpType = d3_event.pointerType;
}
- }
- 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 click(d3_event, operation2) {
+ d3_event.stopPropagation();
+ if (operation2.relatedEntityIds) {
+ utilHighlightEntities(operation2.relatedEntityIds(), false, context);
+ }
+ if (operation2.disabled()) {
+ if (lastPointerUpType === "touch" || lastPointerUpType === "pen") {
+ context.ui().flash.duration(4e3).iconName("#iD-operation-" + operation2.id).iconClass("operation disabled").label(operation2.tooltip())();
+ }
+ } else {
+ if (lastPointerUpType === "touch" || lastPointerUpType === "pen") {
+ context.ui().flash.duration(2e3).iconName("#iD-operation-" + operation2.id).iconClass("operation").label(operation2.annotation() || operation2.title)();
+ }
+ operation2();
+ editMenu.close();
+ }
+ lastPointerUpType = null;
}
- }
- 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();
+ dispatch14.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;
}
- }
- 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 }
- );
+ 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 continueTo(nextStep) {
- context.map().on("drawn.intro", null);
- nextStep();
+ 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";
+ }
}
}
- 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 });
+ editMenu.close = function() {
+ context.map().on("move.edit-menu", null).on("drawn.edit-menu", null);
+ _menu.remove();
+ _tooltips = [];
+ dispatch14.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, dispatch14, "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(k2) {
+ if (stats[k2]) {
+ count += stats[k2];
+ return _t.append("inspector.title_count", {
+ title: _t("feature." + k2 + ".description"),
+ count: stats[k2]
+ });
+ }
+ 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);
+ });
+ };
});
- context.on("enter.intro", function() {
- if (isTownHallSelected())
- continueTo(selectedTownHall);
+ 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"));
});
- }, 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();
}
+ selection2.classed("hide", !hiddenList.length);
}
- 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);
- }
+ return function(selection2) {
+ update(selection2);
+ context.features().on("change.feature_info", function() {
+ update(selection2);
});
- function continueTo(nextStep) {
- context.map().on("move.intro drawn.intro", null);
- context.history().on("change.intro", null);
- nextStep();
+ };
+ }
+
+ // modules/ui/flash.js
+ function uiFlash(context) {
+ var _flashTimer;
+ var _duration = 2e3;
+ var _iconName = "#iD-icon-no";
+ var _iconClass = "disabled";
+ var _label = (s2) => s2.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;
}
- 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();
+ flash.duration = function(_2) {
+ if (!arguments.length)
+ return _duration;
+ _duration = _2;
+ return flash;
+ };
+ flash.label = function(_2) {
+ if (!arguments.length)
+ return _label;
+ if (typeof _2 !== "function") {
+ _label = (selection2) => selection2.text(_2);
+ } else {
+ _label = (selection2) => selection2.text("").call(_2);
+ }
+ return flash;
+ };
+ flash.iconName = function(_2) {
+ if (!arguments.length)
+ return _iconName;
+ _iconName = _2;
+ return flash;
+ };
+ flash.iconClass = function(_2) {
+ if (!arguments.length)
+ return _iconClass;
+ _iconClass = _2;
+ 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 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 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 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 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);
}
}
- 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 }
+ return function() {
+ if (!isSupported())
+ return;
+ var detected = utilDetect();
+ var keys2 = detected.os === "mac" ? [uiCmd("\u2303\u2318F"), "f11"] : ["f11"];
+ context.keybinding().on(keys2, 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 */
);
- });
- function continueTo(nextStep) {
- context.on("exit.intro", null);
- context.history().on("change.intro", null);
- nextStep();
+ context.container().call(_locating);
+ navigator.geolocation.getCurrentPosition(success, error, _geolocationOptions);
+ } else {
+ _locating.close();
+ _layer.enabled(null, false);
+ updateButtonState();
}
}
- function searchStreet() {
+ function zoomTo() {
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);
+ var map2 = context.map();
+ _layer.enabled(_position, true);
+ updateButtonState();
+ map2.centerZoomEase(_extent.center(), Math.min(20, map2.extentZoom(_extent)));
}
- 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 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 selectedStreet() {
- if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
- return searchStreet();
+ function finish() {
+ _locating.close();
+ if (_timeoutID) {
+ clearTimeout(_timeoutID);
}
- 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 }
+ _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")])
);
- 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();
+ 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 = {};
}
- }
- 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);
+ selection2.text("");
+ var list2 = selection2.append("ul").attr("class", "background-info");
+ list2.append("li").call(_currSourceName);
+ _metadataKeys.forEach(function(k2) {
+ if (isDG && k2 === "vintage")
+ return;
+ list2.append("li").attr("class", "background-info-list-" + k2).classed("hide", !_metadata[k2]).call(_t.append("info_panels.background." + k2, { suffix: ":" })).append("span").attr("class", "background-info-span-" + k2).text(_metadata[k2]);
});
- 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 }
- );
+ 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);
});
- function continueTo(nextStep) {
- context.on("exit.intro", null);
- context.history().on("change.intro", null);
- nextStep();
+ 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);
+ });
}
- }
- function play() {
- dispatch14.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");
+ ["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);
}
}
- );
+ });
}
- 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);
+ 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 d2 = tile.datum();
+ var zoom = d2 && d2.length >= 3 && d2[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 (!d2 || !d2.length >= 3)
+ return;
+ background.baseLayerSource().getMetadata(center, d2, 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(k2) {
+ if (k2 === "zoom" || k2 === "vintage")
+ return;
+ var val = result[k2];
+ _metadata[k2] = val;
+ selection2.selectAll(".background-info-list-" + k2).classed("hide", !val).selectAll(".background-info-span-" + k2).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);
+ });
};
- chapter.restart = function() {
- chapter.exit();
- chapter.enter();
+ panel.off = function() {
+ context.map().on("drawn.info-background", null).on("move.info-background", null);
};
- return utilRebind(chapter, dispatch14, "on");
+ panel.id = "background";
+ panel.label = _t.append("info_panels.background.title");
+ panel.key = _t("info_panels.background.key");
+ return panel;
}
- // modules/ui/intro/point.js
- function uiIntroPoint(context, reveal) {
- var dispatch14 = 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(f3, t2) {
- timeouts.push(window.setTimeout(f3, t2));
- }
- function eventCancel(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
+ // 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 d2 = new Date(timestamp);
+ if (isNaN(d2.getTime()))
+ return _t("info_panels.history.unknown");
+ return d2.toLocaleString(_mainLocalizer.localeCode(), options2);
}
- function addPoint() {
- context.enter(modeBrowse(context));
- context.history().reset("initial");
- var msec = transitionTime(intersection, context.map().center());
- if (msec) {
- reveal(null, null, { duration: 0 });
+ function displayUser(selection2, userName) {
+ if (!userName) {
+ selection2.append("span").call(_t.append("info_panels.history.unknown"));
+ return;
}
- 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();
+ 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 placePoint() {
- if (context.mode().id !== "add-point") {
- return chapter.restart();
+ function displayChangeset(selection2, changeset) {
+ if (!changeset) {
+ selection2.append("span").call(_t.append("info_panels.history.unknown"));
+ return;
}
- 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();
+ 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 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 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(e3) {
+ return context.hasEntity(e3);
+ });
+ if (selected.length) {
+ entity = context.entity(selected[0]);
}
}
- 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();
+ 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 }));
}
- 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();
+ if (!singular)
+ return;
+ if (entity) {
+ selection2.call(redrawEntity, entity);
+ } else if (note) {
+ selection2.call(redrawNote, note);
}
}
- function addName() {
- if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
- return addPoint();
+ function redrawNote(selection2, note) {
+ if (!note || note.isNew()) {
+ selection2.append("div").call(_t.append("info_panels.history.note_no_history"));
+ return;
}
- 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();
+ var list2 = selection2.append("ul");
+ list2.append("li").call(_t.append("info_panels.history.note_comments", { suffix: ":" })).append("span").text(note.comments.length);
+ if (note.comments.length) {
+ list2.append("li").call(_t.append("info_panels.history.note_created_date", { suffix: ":" })).append("span").text(displayTimestamp(note.comments[0].date));
+ list2.append("li").call(_t.append("info_panels.history.note_created_user", { suffix: ":" })).call(displayUser, note.comments[0].user);
}
- }
- 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();
+ 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 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 });
+ function redrawEntity(selection2, entity) {
+ if (!entity || entity.isNew()) {
+ selection2.append("div").call(_t.append("info_panels.history.no_history"));
+ return;
}
- 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();
+ 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 list2 = selection2.append("ul");
+ list2.append("li").call(_t.append("info_panels.history.version", { suffix: ":" })).append("span").text(entity.version);
+ list2.append("li").call(_t.append("info_panels.history.last_edit", { suffix: ":" })).append("span").text(displayTimestamp(entity.timestamp));
+ list2.append("li").call(_t.append("info_panels.history.edited_by", { suffix: ":" })).call(displayUser, entity.user);
+ list2.append("li").call(_t.append("info_panels.history.changeset", { suffix: ":" })).call(displayChangeset, entity.changeset);
}
- 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);
+ var panel = function(selection2) {
+ selection2.call(redraw);
+ context.map().on("drawn.info-history", function() {
+ selection2.call(redraw);
});
- context.history().on("change.intro", function() {
- continueTo(updateCloseEditor);
+ context.on("enter.info-history", function() {
+ selection2.call(redraw);
});
- 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();
+ };
+ 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/ui/panels/location.js
+ function uiPanelLocation(context) {
+ var currLocation = "";
+ function redraw(selection2) {
+ selection2.html("");
+ var list2 = selection2.append("ul");
+ var coord2 = context.map().mouseCoordinates();
+ if (coord2.some(isNaN)) {
+ coord2 = context.map().center();
}
+ list2.append("li").text(dmsCoordinatePair(coord2)).append("li").text(decimalCoordinatePair(coord2));
+ selection2.append("div").attr("class", "location-info").text(currLocation || " ");
+ debouncedGetLocation(selection2, coord2);
}
- function updateCloseEditor() {
- if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
- return continueTo(reselectPoint);
+ 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);
+ });
}
- context.container().select(".inspector-wrap .panewrap").style("right", "0%");
- context.on("exit.intro", function() {
- continueTo(rightClickPoint);
+ }
+ var panel = function(selection2) {
+ selection2.call(redraw);
+ context.surface().on(("PointerEvent" in window ? "pointer" : "mouse") + "move.info-location", function() {
+ selection2.call(redraw);
});
- 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();
+ };
+ 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(r2) {
+ return r2 * 63710071809e-4;
+ }
+ function steradiansToSqmeters(r2) {
+ return r2 / (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;
}
- 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 });
+ var _isImperial = !_mainLocalizer.usesMetric();
+ function redraw(selection2) {
+ var graph = context.graph();
+ var selectedNoteID = context.selectedNoteID();
+ var osm = services.osm;
+ var localeCode = _mainLocalizer.localeCode();
+ var heading2;
+ var center, location, centroid;
+ var closed, geometry;
+ var totalNodeCount, length2 = 0, area = 0, distance;
+ if (selectedNoteID && osm) {
+ var note = osm.getNote(selectedNoteID);
+ heading2 = _t.html("note.note") + " " + selectedNoteID;
+ location = note.loc;
+ geometry = "note";
+ } else {
+ var selectedIDs = context.selectedIDs().filter(function(id2) {
+ return context.hasEntity(id2);
});
- }, 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();
+ var selected = selectedIDs.map(function(id2) {
+ return context.entity(id2);
+ });
+ heading2 = selected.length === 1 ? selected[0].id : _t.html("info_panels.selected", { n: selected.length });
+ if (selected.length) {
+ var extent = geoExtent();
+ for (var i3 in selected) {
+ var entity = selected[i3];
+ 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);
+ length2 += 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();
+ }
+ }
}
- }
- 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);
+ selection2.html("");
+ if (heading2) {
+ selection2.append("h4").attr("class", "measurement-heading").html(heading2);
}
- 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 }
- );
+ var list2 = selection2.append("ul");
+ var coordItem;
+ if (geometry) {
+ list2.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) {
+ list2.append("li").call(_t.append("info_panels.measurement.node_count", { suffix: ":" })).append("span").text(totalNodeCount.toLocaleString(localeCode));
+ }
+ if (area) {
+ list2.append("li").call(_t.append("info_panels.measurement.area", { suffix: ":" })).append("span").text(displayArea(area, _isImperial));
+ }
+ if (length2) {
+ list2.append("li").call(_t.append("info_panels.measurement." + (closed ? "perimeter" : "length"), { suffix: ":" })).append("span").text(displayLength(length2, _isImperial));
+ }
+ if (typeof distance === "number") {
+ list2.append("li").call(_t.append("info_panels.measurement.distance", { suffix: ":" })).append("span").text(displayLength(distance, _isImperial));
+ }
+ if (location) {
+ coordItem = list2.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 = list2.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 = list2.append("li").call(_t.append("info_panels.measurement.center", { suffix: ":" }));
+ coordItem.append("span").text(dmsCoordinatePair(center));
+ coordItem.append("span").text(decimalCoordinatePair(center));
+ }
+ if (length2 || 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);
});
- }, 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);
+ var panel = function(selection2) {
+ selection2.call(redraw);
+ context.map().on("drawn.info-measurement", function() {
+ selection2.call(redraw);
});
- reveal(
- ".top-toolbar button.undo-button",
- helpHtml("intro.points.undo")
- );
- function continueTo(nextStep) {
- context.history().on("change.intro", null);
- nextStep();
+ 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(k2) {
+ if (!panels[k2]) {
+ panels[k2] = uiInfoPanels[k2](context);
+ active[k2] = false;
}
- }
- function play() {
- dispatch14.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");
+ });
+ function info(selection2) {
+ function redraw() {
+ var activeids = ids.filter(function(k2) {
+ return active[k2];
+ }).sort();
+ var containers = infoPanels.selectAll(".panel-container").data(activeids, function(k2) {
+ return k2;
+ });
+ containers.exit().style("opacity", 1).transition().duration(200).style("opacity", 0).on("end", function(d2) {
+ select_default2(this).call(panels[d2].off).remove();
+ });
+ var enter = containers.enter().append("div").attr("class", function(d2) {
+ return "fillD2 panel-container panel-container-" + d2;
+ });
+ 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(d2) {
+ return panels[d2].label(select_default2(this));
+ });
+ title.append("button").attr("class", "close").attr("title", _t("icons.close")).on("click", function(d3_event, d2) {
+ d3_event.stopImmediatePropagation();
+ d3_event.preventDefault();
+ info.toggle(d2);
+ }).call(svgIcon("#iD-icon-close"));
+ enter.append("div").attr("class", function(d2) {
+ return "panel-content panel-content-" + d2;
+ });
+ infoPanels.selectAll(".panel-content").each(function(d2) {
+ select_default2(this).call(panels[d2]);
+ });
+ }
+ info.toggle = function(which) {
+ var activeids = ids.filter(function(k2) {
+ return active[k2];
+ });
+ 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(k2) {
+ active[k2] = false;
+ });
+ } else {
+ wasActive.forEach(function(k2) {
+ active[k2] = 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(k2) {
+ var key = _t("info_panels." + k2 + ".key", { default: null });
+ if (!key)
+ return;
+ context.keybinding().on(uiCmd("\u2318\u21E7" + key), function(d3_event) {
+ d3_event.stopImmediatePropagation();
+ d3_event.preventDefault();
+ info.toggle(k2);
+ });
+ });
}
- 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, dispatch14, "on");
+ return info;
}
- // modules/ui/intro/area.js
- function uiIntroArea(context, reveal) {
- var dispatch14 = 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"
+ // 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 timeout2(f3, t2) {
- timeouts.push(window.setTimeout(f3, t2));
+ }
+ 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;
}
- function eventCancel(d3_event) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
+ 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 '";
+ }
+ 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] };
+ }
}
- 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);
+ var reps;
+ if (replacements) {
+ reps = Object.assign(replacements, helpStringReplacements);
+ } else {
+ reps = helpStringReplacements;
}
- 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();
- }
+ return _t.html(id2, reps).replace(/\`(.*?)\`/g, "$1");
+ }
+ function slugify(text) {
+ return text.toString().toLowerCase().replace(/\s+/g, "-").replace(/[^\w\-]+/g, "").replace(/\-\-+/g, "-").replace(/^-+/, "").replace(/-+$/, "");
+ }
+ var missingStrings = {};
+ function checkKey(key, text) {
+ if (_t(key, { default: void 0 }) === void 0) {
+ if (missingStrings.hasOwnProperty(key))
+ return;
+ missingStrings[key] = text;
+ var missing = key + ": " + text;
+ if (typeof console !== "undefined")
+ console.log(missing);
}
- 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 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);
}
- 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);
+ 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(k2) {
+ var key2 = "intro.graph." + k2;
+ var tag2 = "addr:" + k2;
+ var val = obj.tags && obj.tags[tag2];
+ var str = _t(key2, { default: val });
+ if (str) {
+ if (str.match(/^<.*>$/) !== null) {
+ delete obj.tags[tag2];
} else {
- return;
+ obj.tags[tag2] = str;
}
- } 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();
+ 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 i3 = 0; i3 < points.length; i3++) {
+ var a2 = points[(i3 - 1 + points.length) % points.length];
+ var origin = points[i3];
+ var b2 = points[(i3 + 1) % points.length];
+ var dotp = geoVecNormalizedDot(a2, b2, origin);
+ var mag = Math.abs(dotp);
+ if (mag > lowerBound && mag < upperBound) {
+ return false;
}
}
- 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);
- }
+ return true;
+ }
+ function selectMenuItem(context, operation2) {
+ return context.container().select(".edit-menu .edit-menu-item-" + operation2);
+ }
+ 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);
});
- 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();
+ };
+ }
+
+ // 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());
}
}
- function clickAddField() {
- if (!_areaID || !context.hasEntity(_areaID)) {
- return addArea();
+ curtain.reveal = function(box, html3, options2) {
+ options2 = options2 || {};
+ if (typeof box === "string") {
+ box = select_default2(box).node();
}
- var ids = context.selectedIDs();
- if (context.mode().id !== "select" || !ids.length || ids[0] !== _areaID) {
- return searchPresets();
+ if (box && box.getBoundingClientRect) {
+ box = copyBox(box.getBoundingClientRect());
+ var containerRect = containerNode.getBoundingClientRect();
+ box.top -= containerRect.top;
+ box.left -= containerRect.left;
}
- if (!context.container().select(".form-field-description").empty()) {
- return continueTo(describePlayground);
+ 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;
}
- 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 tooltipBox;
+ if (options2.tooltipBox) {
+ tooltipBox = options2.tooltipBox;
+ if (typeof tooltipBox === "string") {
+ tooltipBox = select_default2(tooltipBox).node();
}
- 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 i3 = number_default(start2, end);
- return function(t2) {
- node.scrollTop = i3(t2);
- };
- });
+ if (tooltipBox && tooltipBox.getBoundingClientRect) {
+ tooltipBox = copyBox(tooltipBox.getBoundingClientRect());
}
- 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);
+ } else {
+ tooltipBox = box;
}
- 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);
+ if (tooltipBox && html3) {
+ if (html3.indexOf("**") !== -1) {
+ if (html3.indexOf(")(.+?)(\*\*)/, "$1$2$3");
+ } else {
+ html3 = html3.replace(/^(.+?)(\*\*)/, "$1$2");
+ }
+ html3 = html3.replace(/\*\*(.*?)\*\*/g, '$1');
+ }
+ html3 = html3.replace(/\*(.*?)\*/g, "$1");
+ html3 = html3.replace(/\{br\}/g, "
");
+ if (options2.buttonText && options2.buttonCallback) {
+ html3 += '";
+ }
+ var classes = "curtain-tooltip popover tooltip arrowed in " + (options2.tooltipClass || "");
+ tooltip.classed(classes, true).selectAll(".popover-inner").html(html3);
+ 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()), w2 = containerNode.clientWidth, h2 = containerNode.clientHeight, tooltipWidth = 200, tooltipArrow = 5, side, pos;
+ if (options2.tooltipClass === "intro-mouse") {
+ tip.height += 80;
+ }
+ if (tooltipBox.top + tooltipBox.height > h2) {
+ tooltipBox.height -= tooltipBox.top + tooltipBox.height - h2;
+ }
+ if (tooltipBox.left + tooltipBox.width > w2) {
+ tooltipBox.width -= tooltipBox.left + tooltipBox.width - w2;
+ }
+ 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 > h2 - 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 {
- continueTo(describePlayground);
+ side = "left";
+ pos = [tooltipBox.left - tooltipWidth - tooltipArrow, tipY];
}
- }, 300);
+ } else {
+ if (tooltipBox.left + tooltipBox.width + tooltipArrow + tooltipWidth > w2 - 70) {
+ side = "left";
+ pos = [tooltipBox.left - tooltipWidth - tooltipArrow, tipY];
+ } else {
+ side = "right";
+ pos = [tooltipBox.left + tooltipBox.width + tooltipArrow, tipY];
+ }
+ }
}
- }, 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();
+ 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 > h2 - 100) {
+ shiftY = h2 - pos[1] - tip.height - 100;
+ }
+ }
+ tooltip.selectAll(".popover-inner").style("top", shiftY + "px");
+ } else {
+ tooltip.classed("in", false).call(uiToggle(false));
}
- context.container().select(".inspector-wrap .panewrap").style("right", "0%");
- if (context.container().select(".form-field-description").empty()) {
- return continueTo(retryChooseDescription);
+ 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);
}
- context.on("exit.intro", function() {
- continueTo(play);
+ selection2.attr("d", function(d2) {
+ var containerWidth = containerNode.clientWidth;
+ var containerHeight = containerNode.clientHeight;
+ var string = "M 0,0 L 0," + containerHeight + " L " + containerWidth + "," + containerHeight + "L" + containerWidth + ",0 Z";
+ if (!d2)
+ return string;
+ return string + "M" + d2.left + "," + d2.top + "L" + d2.left + "," + (d2.top + d2.height) + "L" + (d2.left + d2.width) + "," + (d2.top + d2.height) + "L" + (d2.left + d2.width) + "," + d2.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 dispatch14 = dispatch_default("done");
+ var chapter = {
+ title: "intro.welcome.title"
+ };
+ function welcome() {
+ context.map().centerZoom([-85.63591, 41.94285], 19);
reveal(
- ".entity-editor-pane",
- helpHtml("intro.areas.describe_playground", { button: { html: icon("#iD-icon-close", "inline") } }),
- { duration: 300 }
+ ".intro-nav-wrap .chapter-welcome",
+ helpHtml("intro.welcome.welcome"),
+ { buttonText: _t.html("intro.ok"), buttonCallback: practice }
);
- 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%");
+ function practice() {
reveal(
- ".entity-editor-pane",
- helpHtml("intro.areas.retry_add_field", { field: descriptionField.title() }),
- {
- buttonText: _t.html("intro.ok"),
- buttonCallback: function() {
- continueTo(clickAddField);
- }
- }
+ ".intro-nav-wrap .chapter-welcome",
+ helpHtml("intro.welcome.practice"),
+ { buttonText: _t.html("intro.ok"), buttonCallback: words }
);
- context.on("exit.intro", function() {
- return continueTo(searchPresets);
- });
- function continueTo(nextStep) {
- context.on("exit.intro", null);
- nextStep();
- }
}
- function play() {
+ function words() {
+ reveal(
+ ".intro-nav-wrap .chapter-welcome",
+ helpHtml("intro.welcome.words"),
+ { buttonText: _t.html("intro.ok"), buttonCallback: chapters }
+ );
+ }
+ function chapters() {
dispatch14.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");
- }
- }
+ ".intro-nav-wrap .chapter-navigation",
+ helpHtml("intro.welcome.chapters", { next: _t("intro.navigation.title") })
);
}
chapter.enter = function() {
- addArea();
+ welcome();
};
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);
+ context.container().select(".curtain-tooltip.intro-mouse").selectAll(".counter").remove();
};
chapter.restart = function() {
chapter.exit();
@@ -58805,320 +60073,55 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return utilRebind(chapter, dispatch14, "on");
}
- // modules/ui/intro/line.js
- function uiIntroLine(context, reveal) {
+ // modules/ui/intro/navigation.js
+ function uiIntroNavigation(context, reveal) {
var dispatch14 = 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 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.lines.title"
+ title: "intro.navigation.title"
};
- function timeout2(f3, t2) {
- timeouts.push(window.setTimeout(f3, t2));
+ function timeout2(f2, t2) {
+ timeouts.push(window.setTimeout(f2, t2));
}
function eventCancel(d3_event) {
d3_event.stopPropagation();
d3_event.preventDefault();
}
- function addLine() {
+ 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(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());
+ var msec = transitionTime(townHall, context.map().center());
if (msec) {
reveal(null, null, { duration: 0 });
}
- context.map().centerZoomEase(woodRoadDragMidpoint, 19, msec);
+ context.map().centerZoomEase(townHall, 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 }
- );
+ 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) {
@@ -59126,373 +60129,358 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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 });
+ 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.history().on("change.intro", function(changed) {
- if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
- return continueTo(updateLine);
- }
- if (changed.created().length === 1) {
+ context.map().on("move.intro", function() {
+ if (context.map().zoom() !== zoomStart) {
+ context.map().on("move.intro", null);
timeout2(function() {
- continueTo(startDragEndpoint);
- }, 500);
- }
- });
- context.on("enter.intro", function(mode) {
- if (mode.id !== "select") {
- continueTo(updateLine);
+ continueTo(features);
+ }, 3e3);
}
});
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 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("move.intro drawn.intro", null);
+ context.map().on("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 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("move.intro drawn.intro", null);
- context.on("enter.intro", null);
+ context.map().on("drawn.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);
- }
+ 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 }
+ );
});
- context.on("enter.intro", function(mode) {
- if (mode.id !== "select") {
- context.enter(modeSelect(context, [woodRoadID]));
+ 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);
- 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);
+ 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.lines.continue_drag_midpoint"),
- { buttonText: _t.html("intro.ok"), buttonCallback: advance }
+ helpHtml("intro.navigation.selected_townhall"),
+ { buttonText: _t.html("intro.ok"), buttonCallback: onClick }
);
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;
+ var entity2 = context.hasEntity(hallId);
+ if (!entity2)
+ return;
+ var box2 = pointBox(entity2.loc, context);
reveal(
box2,
- helpHtml("intro.lines.continue_drag_midpoint"),
- { duration: 0, buttonText: _t.html("intro.ok"), buttonCallback: advance }
+ 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 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 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.map().on("move.intro drawn.intro", null);
+ context.on("exit.intro", null);
context.history().on("change.intro", null);
+ context.container().select(".inspector-wrap").on("wheel.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 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.map().on("move.intro drawn.intro", null);
- context.on("enter.intro", null);
+ context.on("exit.intro", null);
context.history().on("change.intro", null);
+ context.container().select(".inspector-wrap").on("wheel.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;
+ 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(
- ".edit-menu",
- helpHtml(
- "intro.lines.split_intersection",
- { street: _t("intro.graph.name.washington-street") }
- ),
- { padding: 50 }
+ ".entity-editor-pane .section-preset-fields",
+ helpHtml("intro.navigation.fields_townhall"),
+ { buttonText: _t.html("intro.ok"), buttonCallback: onClick }
);
- context.map().on("move.intro drawn.intro", function() {
- var node2 = selectMenuItem(context, "split").node();
- if (!wasChanged && !node2) {
- return continueTo(rightClickIntersection);
+ 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(
- ".edit-menu",
- helpHtml(
- "intro.lines.split_intersection",
- { street: _t("intro.graph.name.washington-street") }
- ),
- { duration: 0, padding: 50 }
+ ".entity-editor-pane",
+ helpHtml("intro.navigation.close_townhall", { button: { html: icon(href2, "inline") } }),
+ { duration: 0 }
);
});
- 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.on("exit.intro", null);
context.history().on("change.intro", null);
nextStep();
}
}
- function retrySplit() {
+ function searchStreet() {
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);
+ 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(
- box2,
- helpHtml("intro.lines.retry_split"),
- { duration: 0, buttonText: _t.html("intro.ok"), buttonCallback: advance }
+ 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.map().on("move.intro drawn.intro", null);
+ context.on("exit.intro", null);
+ context.container().select(".search-header input").on("keydown.intro", null).on("keyup.intro", null);
nextStep();
}
}
- function didSplit() {
- if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
- return continueTo(rightClickIntersection);
+ function selectedStreet() {
+ if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
+ return searchStreet();
}
- 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;
+ var onClick = function() {
+ continueTo(editorStreet);
+ };
+ var entity = context.entity(springStreetEndId);
+ var box = pointBox(entity.loc, context);
+ box.height = 500;
reveal(
box,
- helpHtml(string, { street1: street, street2: street }),
- { duration: 500 }
+ 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().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;
+ var entity2 = context.hasEntity(springStreetEndId);
+ if (!entity2)
+ return;
+ var box2 = pointBox(entity2.loc, context);
+ box2.height = 500;
reveal(
box2,
- helpHtml(string, { street1: street, street2: street }),
- { duration: 0 }
+ 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() {
- var ids2 = context.selectedIDs();
- if (ids2.length === 1 && ids2[0] === _washingtonSegmentID) {
- continueTo(multiSelect);
+ 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 (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
- return continueTo(rightClickIntersection);
+ if (!context.hasEntity(springStreetEndId) || !context.hasEntity(springStreetId)) {
+ timeout2(function() {
+ continueTo(searchStreet);
+ }, 300);
}
});
function continueTo(nextStep) {
@@ -59502,187 +60490,43 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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);
+ 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() {
- 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() {
+ var selector2 = ".entity-editor-pane button.close svg use";
+ var href2 = select_default2(selector2).attr("href") || "#iD-icon-close";
reveal(
- ".edit-menu",
- helpHtml("intro.lines.multi_delete"),
- { duration: 0, padding: 50 }
+ ".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 }
);
});
- 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() {
dispatch14.call("done");
reveal(
".ideditor",
- helpHtml("intro.lines.play", { next: _t("intro.buildings.title") }),
+ helpHtml("intro.navigation.play", { next: _t("intro.points.title") }),
{
- tooltipBox: ".intro-nav-wrap .chapter-building",
+ tooltipBox: ".intro-nav-wrap .chapter-point",
buttonText: _t.html("intro.ok"),
buttonCallback: function() {
reveal(".ideditor");
@@ -59691,16 +60535,15 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
);
}
chapter.enter = function() {
- addLine();
+ dragMap();
};
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);
+ context.container().select(".search-header input").on("keydown.intro keyup.intro", null);
};
chapter.restart = function() {
chapter.exit();
@@ -59709,56 +60552,43 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return utilRebind(chapter, dispatch14, "on");
}
- // modules/ui/intro/building.js
- function uiIntroBuilding(context, reveal) {
+ // modules/ui/intro/point.js
+ function uiIntroPoint(context, reveal) {
var dispatch14 = 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 intersection2 = [-85.63279, 41.94394];
+ var building = [-85.632422, 41.944045];
+ var cafePreset = _mainPresetIndex.item("amenity/cafe");
+ var _pointID = null;
var chapter = {
- title: "intro.buildings.title"
+ title: "intro.points.title"
};
- function timeout2(f3, t2) {
- timeouts.push(window.setTimeout(f3, t2));
+ function timeout2(f2, t2) {
+ timeouts.push(window.setTimeout(f2, t2));
}
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() {
+ function addPoint() {
context.enter(modeBrowse(context));
context.history().reset("initial");
- _houseID = null;
- var msec = transitionTime(house, context.map().center());
+ var msec = transitionTime(intersection2, context.map().center());
if (msec) {
reveal(null, null, { duration: 0 });
}
- context.map().centerZoomEase(house, 19, msec);
+ context.map().centerZoomEase(intersection2, 19, msec);
timeout2(function() {
var tooltip = reveal(
- "button.add-area",
- helpHtml("intro.buildings.add_building")
+ "button.add-point",
+ helpHtml("intro.points.points_info") + "{br}" + helpHtml("intro.points.add_point")
);
- tooltip.selectAll(".popover-inner").insert("svg", "span").attr("class", "tooltip-illustration").append("use").attr("xlink:href", "#iD-graphic-buildings");
+ _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-area")
+ if (mode.id !== "add-point")
return;
- continueTo(startHouse);
+ continueTo(placePoint);
});
}, msec + 100);
function continueTo(nextStep) {
@@ -59766,178 +60596,225 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
nextStep();
}
}
- function startHouse() {
- if (context.mode().id !== "add-area") {
- return continueTo(addHouse);
+ function placePoint() {
+ if (context.mode().id !== "add-point") {
+ return chapter.restart();
}
- _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);
+ 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 continueHouse() {
- if (context.mode().id !== "draw-area") {
- return continueTo(addHouse);
+ function searchPreset() {
+ if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
+ return addPoint();
}
- _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.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 (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(n3) {
- return context.projection(n3.loc);
+ 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);
});
- if (isMostlySquare(points)) {
- _houseID = way.id;
- return continueTo(chooseCategoryBuilding);
- } else {
- return continueTo(retryHouse);
+ }
+ }
+ 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 {
- return chapter.restart();
+ 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.map().on("move.intro drawn.intro", null);
- context.on("enter.intro", null);
+ context.on("exit.intro", null);
+ context.history().on("change.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 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.map().on("move.intro drawn.intro", null);
+ context.on("exit.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]));
+ 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.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);
+ 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);
});
- }, 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);
- }
- });
+ }, msec + 100);
function continueTo(nextStep) {
- context.container().select(".inspector-wrap").on("wheel.intro", null);
- context.container().select(".preset-list-button").on("click.intro", null);
+ context.map().on("move.intro drawn.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]));
+ function updatePoint() {
+ if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
+ return continueTo(reselectPoint);
}
- context.container().select(".inspector-wrap").on("wheel.intro", eventCancel);
+ 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() {
- 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 }
+ ".entity-editor-pane",
+ helpHtml("intro.points.update"),
+ { tooltipClass: "intro-points-describe" }
);
- 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);
+ context.on("exit.intro", null);
+ context.history().on("change.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]));
+ function updateCloseEditor() {
+ if (context.mode().id !== "select" || !_pointID || !context.hasEntity(_pointID)) {
+ return continueTo(reselectPoint);
}
- context.history().checkpoint("hasHouse");
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
context.on("exit.intro", function() {
- continueTo(rightClickHouse);
+ continueTo(rightClickPoint);
});
timeout2(function() {
reveal(
".entity-editor-pane",
- helpHtml("intro.buildings.close", { button: { html: icon("#iD-icon-close", "inline") } })
+ helpHtml("intro.points.update_close", { button: { html: icon("#iD-icon-close", "inline") } })
);
}, 500);
function continueTo(nextStep) {
@@ -59945,137 +60822,175 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
nextStep();
}
}
- function rightClickHouse() {
- if (!_houseID)
+ function rightClickPoint() {
+ if (!_pointID)
+ return chapter.restart();
+ var entity = context.hasEntity(_pointID);
+ if (!entity)
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);
+ 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] !== _houseID)
+ if (ids.length !== 1 || ids[0] !== _pointID)
return;
timeout2(function() {
- var node = selectMenuItem(context, "orthogonalize").node();
+ var node = selectMenuItem(context, "delete").node();
if (!node)
return;
- continueTo(clickSquare);
+ continueTo(enterDelete);
}, 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);
+ context.map().on("move.intro", null);
nextStep();
}
}
- function clickSquare() {
- if (!_houseID)
+ function enterDelete() {
+ if (!_pointID)
return chapter.restart();
- var entity = context.hasEntity(_houseID);
+ var entity = context.hasEntity(_pointID);
if (!entity)
- return continueTo(rightClickHouse);
- var node = selectMenuItem(context, "orthogonalize").node();
+ return chapter.restart();
+ var node = selectMenuItem(context, "delete").node();
if (!node) {
- return continueTo(rightClickHouse);
+ return continueTo(rightClickPoint);
}
- var wasChanged = false;
reveal(
".edit-menu",
- helpHtml("intro.buildings.square_building"),
+ helpHtml("intro.points.delete"),
{ padding: 50 }
);
- context.on("enter.intro", function(mode) {
- if (mode.id === "browse") {
- continueTo(rightClickHouse);
- } else if (mode.id === "move" || mode.id === "rotate") {
- continueTo(retryClickSquare);
- }
+ 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.map().on("move.intro", function() {
- var node2 = selectMenuItem(context, "orthogonalize").node();
- if (!wasChanged && !node2) {
- return continueTo(rightClickHouse);
+ context.history().on("change.intro", function(changed) {
+ if (changed.deleted().length) {
+ continueTo(undo);
}
- 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);
+ context.on("exit.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 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 doneSquare() {
- context.history().checkpoint("doneSquare");
- revealHouse(house, helpHtml("intro.buildings.done_square"), {
- buttonText: _t.html("intro.ok"),
- buttonCallback: function() {
- continueTo(addTank);
+ function play() {
+ dispatch14.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");
+ }
}
- });
- function continueTo(nextStep) {
- nextStep();
- }
+ );
}
- function addTank() {
+ 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, dispatch14, "on");
+ }
+
+ // modules/ui/intro/area.js
+ function uiIntroArea(context, reveal) {
+ var dispatch14 = 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, t2) {
+ timeouts.push(window.setTimeout(f2, t2));
+ }
+ function eventCancel(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ }
+ function revealPlayground(center, text, options2) {
+ var padding = 180 * Math.pow(2, context.map().zoom() - 19.5);
+ var box = pad(center, padding, context);
+ reveal(box, text, options2);
+ }
+ function addArea() {
context.enter(modeBrowse(context));
- context.history().reset("doneSquare");
- _tankID = null;
- var msec = transitionTime(tank, context.map().center());
+ context.history().reset("initial");
+ _areaID = null;
+ var msec = transitionTime(playground, context.map().center());
if (msec) {
reveal(null, null, { duration: 0 });
}
- context.map().centerZoomEase(tank, 19.5, msec);
+ context.map().centerZoomEase(playground, 19, msec);
timeout2(function() {
- reveal(
+ var tooltip = reveal(
"button.add-area",
- helpHtml("intro.buildings.add_tank")
+ 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(startTank);
+ continueTo(startPlayground);
});
}, msec + 100);
function continueTo(nextStep) {
@@ -60083,22 +60998,34 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
nextStep();
}
}
- function startTank() {
+ function startPlayground() {
if (context.mode().id !== "add-area") {
- return continueTo(addTank);
+ return chapter.restart();
}
- _tankID = null;
+ _areaID = null;
+ context.map().zoomEase(19.5, 500);
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);
- });
+ 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);
@@ -60106,24 +61033,74 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
nextStep();
}
}
- function continueTank() {
+ function continuePlayground() {
if (context.mode().id !== "draw-area") {
- return continueTo(addTank);
+ return chapter.restart();
}
- _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 });
+ _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") {
- _tankID = context.selectedIDs()[0];
- return continueTo(searchPresetTank);
+ _areaID = context.selectedIDs()[0];
+ return continueTo(searchPresets);
} else {
- return continueTo(addTank);
+ return chapter.restart();
}
});
function continueTo(nextStep) {
@@ -60132,13 +61109,13 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
nextStep();
}
}
- function searchPresetTank() {
- if (!_tankID || !context.hasEntity(_tankID)) {
- return addTank();
+ function searchPresets() {
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return addArea();
}
var ids = context.selectedIDs();
- if (context.mode().id !== "select" || !ids.length || ids[0] !== _tankID) {
- context.enter(modeSelect(context, [_tankID]));
+ 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() {
@@ -60146,168 +61123,200 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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() })
+ helpHtml("intro.areas.search_playground", { preset: playgroundPreset.name() })
);
}, 400);
context.on("enter.intro", function(mode) {
- if (!_tankID || !context.hasEntity(_tankID)) {
- return continueTo(addTank);
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return continueTo(addArea);
}
var ids2 = context.selectedIDs();
- if (mode.id !== "select" || !ids2.length || ids2[0] !== _tankID) {
- context.enter(modeSelect(context, [_tankID]));
+ 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.buildings.search_tank", { preset: tankPreset.name() })
+ 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-man_made-storage_tank")) {
+ if (first.classed("preset-leisure-playground")) {
reveal(
first.select(".preset-list-button").node(),
- helpHtml("intro.buildings.choose_tank", { preset: tankPreset.name() }),
+ 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(closeEditorTank);
+ 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 i3 = number_default(start2, end);
+ return function(t2) {
+ node.scrollTop = i3(t2);
+ };
+ });
+ }
+ 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) {
- 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);
+ if (watcher)
+ window.clearInterval(watcher);
+ context.on("exit.intro", null);
nextStep();
}
}
- function closeEditorTank() {
- if (!_tankID || !context.hasEntity(_tankID)) {
- return addTank();
+ function describePlayground() {
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return addArea();
}
var ids = context.selectedIDs();
- if (context.mode().id !== "select" || !ids.length || ids[0] !== _tankID) {
- context.enter(modeSelect(context, [_tankID]));
+ 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.history().checkpoint("hasTank");
context.on("exit.intro", function() {
- continueTo(rightClickTank);
+ continueTo(play);
});
- timeout2(function() {
- reveal(
- ".entity-editor-pane",
- helpHtml("intro.buildings.close", { button: { html: icon("#iD-icon-close", "inline") } })
- );
- }, 500);
+ 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 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 retryChooseDescription() {
+ if (!_areaID || !context.hasEntity(_areaID)) {
+ return addArea();
}
- }
- 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 ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _areaID) {
+ return searchPresets();
}
- var wasChanged = false;
+ context.container().select(".inspector-wrap .panewrap").style("right", "0%");
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);
+ ".entity-editor-pane",
+ helpHtml("intro.areas.retry_add_field", { field: descriptionField.title() }),
+ {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ continueTo(clickAddField);
}
- }, 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);
}
+ );
+ context.on("exit.intro", function() {
+ return continueTo(searchPresets);
});
function continueTo(nextStep) {
+ context.on("exit.intro", null);
nextStep();
}
}
@@ -60315,9 +61324,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
dispatch14.call("done");
reveal(
".ideditor",
- helpHtml("intro.buildings.play", { next: _t("intro.startediting.title") }),
+ helpHtml("intro.areas.play", { next: _t("intro.lines.title") }),
{
- tooltipBox: ".intro-nav-wrap .chapter-startEditing",
+ tooltipBox: ".intro-nav-wrap .chapter-line",
buttonText: _t.html("intro.ok"),
buttonCallback: function() {
reveal(".ideditor");
@@ -60326,7 +61335,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
);
}
chapter.enter = function() {
- addHouse();
+ addArea();
};
chapter.exit = function() {
timeouts.forEach(window.clearTimeout);
@@ -60344,9001 +61353,9583 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return utilRebind(chapter, dispatch14, "on");
}
- // modules/ui/intro/start_editing.js
- function uiIntroStartEditing(context, reveal) {
- var dispatch14 = dispatch_default("done", "startEditing");
- var modalSelection = select_default2(null);
+ // modules/ui/intro/line.js
+ function uiIntroLine(context, reveal) {
+ var dispatch14 = 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.startediting.title"
+ title: "intro.lines.title"
};
- function showHelp() {
- reveal(
- ".map-control.help-control",
- helpHtml("intro.startediting.help"),
- {
- buttonText: _t.html("intro.ok"),
- buttonCallback: function() {
- shortcuts();
- }
- }
- );
+ function timeout2(f2, t2) {
+ timeouts.push(window.setTimeout(f2, t2));
}
- function shortcuts() {
- reveal(
- ".map-control.help-control",
- helpHtml("intro.startediting.shortcuts"),
- {
- buttonText: _t.html("intro.ok"),
- buttonCallback: function() {
- showSave();
- }
+ 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 showSave() {
- context.container().selectAll(".shaded").remove();
+ 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(
- ".top-toolbar button.save",
- helpHtml("intro.startediting.save"),
- {
- buttonText: _t.html("intro.ok"),
- buttonCallback: function() {
- showStart();
- }
- }
+ box,
+ helpHtml("intro.lines.retry_intersect", { name: _t("intro.graph.name.flower-street") })
);
+ timeout2(chapter.restart, 3e3);
}
- 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();
+ 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();
+ }
});
- startbutton.append("svg").attr("class", "illustration").append("use").attr("xlink:href", "#iD-logo-walkthrough");
- startbutton.append("h2").call(_t.append("intro.startediting.start"));
- dispatch14.call("startEditing");
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ nextStep();
+ }
}
- chapter.enter = function() {
- showHelp();
- };
- chapter.exit = function() {
- modalSelection.remove();
- context.container().selectAll(".shaded").remove();
- };
- return utilRebind(chapter, dispatch14, "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 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 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();
+ 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();
}
- 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();
+ }
+ 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();
}
- overlays.forEach((d2) => context.background().toggleOverlayLayer(d2));
- let layers = context.layers();
- layers.all().forEach((item) => {
- if (typeof item.layer.enabled === "function") {
- item.layer.enabled(item.id === "osm");
- }
+ }
+ function nameRoad() {
+ context.on("exit.intro", function() {
+ continueTo(didNameRoad);
});
- 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, i3) => {
- let s2 = chapterUi[chapter](context, curtain.reveal).on("done", () => {
- buttons.filter((d2) => d2.title === s2.title).classed("finished", true);
- if (i3 < chapterFlow.length - 1) {
- const next = chapterFlow[i3 + 1];
- context.container().select(`button.chapter-${next}`).classed("next", true);
+ 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);
}
- progress.push(chapter);
- corePreferences("walkthrough_progress", utilArrayUniq(progress).join(";"));
});
- return s2;
+ }, 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 });
});
- 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().on("change.intro", function(changed) {
+ if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+ return continueTo(updateLine);
}
- context.history().reset().merge(Object.values(baseEntities));
- context.background().baseLayerSource(background);
- overlays.forEach((d2) => context.background().toggleOverlayLayer(d2));
- if (history) {
- context.history().fromJSON(history, false);
+ if (changed.created().length === 1) {
+ timeout2(function() {
+ continueTo(startDragEndpoint);
+ }, 500);
}
- 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", (d2, i3) => `chapter chapter-${chapterFlow[i3]}`).on("click", enterChapter);
- buttons.append("span").html((d2) => _t.html(d2.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.on("enter.intro", function(mode) {
+ if (mode.id !== "select") {
+ continueTo(updateLine);
}
- context.enter(modeBrowse(context));
- _currChapter = newChapter;
- _currChapter.enter();
- buttons.classed("next", false).classed("active", (d2) => d2.title === _currChapter.title);
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
}
}
- 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);
+ function startDragEndpoint() {
+ if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+ return continueTo(updateLine);
}
- if (corePreferences("validate-what") === "all") {
- var resolvedIssues = context.validator().getResolvedIssues();
- if (resolvedIssues.length) {
- resolvedItem.count = resolvedIssues.length;
- shownItems.push(resolvedItem);
+ 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);
}
- }
- var chips = selection2.selectAll(".chip").data(shownItems, function(d2) {
- return d2.id;
});
- chips.exit().remove();
- var enter = chips.enter().append("a").attr("class", function(d2) {
- return "chip " + d2.id + "-count";
- }).attr("href", "#").each(function(d2) {
- var chipSelection = select_default2(this);
- var tooltipBehavior = uiTooltip().placement("top").title(() => _t.append(d2.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("#" + d2.iconID));
+ 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);
+ }
});
- enter.append("span").attr("class", "count");
- enter.merge(chips).selectAll("span.count").text(function(d2) {
- return d2.count.toString();
+ context.on("enter.intro", function() {
+ continueTo(startDragMidpoint);
});
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
+ }
}
- 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 startDragMidpoint() {
+ if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+ return continueTo(updateLine);
}
- function zoomed(d3_event) {
- if (_skipEvents)
- return;
- var x2 = d3_event.transform.x;
- var y2 = d3_event.transform.y;
- var k2 = d3_event.transform.k;
- var isZooming = k2 !== _tStart.k;
- var isPanning = x2 !== _tStart.x || y2 !== _tStart.y;
- if (!isZooming && !isPanning) {
- return;
+ 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);
}
- if (!_gesture) {
- _gesture = isZooming ? "zoom" : "pan";
+ 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);
}
- var tMini = projection2.transform();
- var tX, tY, scale;
- if (_gesture === "zoom") {
- scale = k2 / tMini.k;
- tX = (_cMini[0] / scale - _cMini[0]) * scale;
- tY = (_cMini[1] / scale - _cMini[1]) * scale;
- } else {
- k2 = tMini.k;
- scale = 1;
- tX = x2 - tMini.x;
- tY = y2 - tMini.y;
+ });
+ context.on("enter.intro", function(mode) {
+ if (mode.id !== "select") {
+ context.enter(modeSelect(context, [woodRoadID]));
}
- utilSetTransform(tiles, tX, tY, scale);
- utilSetTransform(viewport, 0, 0, scale);
- _isTransformed = true;
- _tCurr = identity2.translate(x2, y2).scale(k2);
- var zMain = geoScaleToZoom(context.projection.scale());
- var zMini = geoScaleToZoom(k2);
- _zDiff = zMain - zMini;
- queueRedraw();
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.history().on("change.intro", null);
+ context.on("enter.intro", null);
+ nextStep();
}
- function zoomEnded() {
- if (_skipEvents)
- return;
- if (_gesture !== "pan")
- return;
- updateProjection();
- _gesture = null;
- context.map().center(projection2.invert(_cMini));
+ }
+ function continueDragMidpoint() {
+ if (!context.hasEntity(woodRoadID) || !context.hasEntity(woodRoadEndID)) {
+ return continueTo(updateLine);
}
- 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;
+ 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);
}
- zoom.scaleExtent([geoZoomToScale(0.5), geoZoomToScale(zMain - 3)]);
- _skipEvents = true;
- wrap2.call(zoom.transform, _tCurr);
- _skipEvents = false;
+ 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 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 i3 = 0; i3 < overlaySources.length; i3++) {
- if (overlaySources[i3].validZoom(zMini)) {
- if (!overlayLayers[i3])
- overlayLayers[i3] = rendererTileLayer(context);
- activeOverlayLayers.push(overlayLayers[i3].source(overlaySources[i3]).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(d2) {
- return d2.source().name();
+ }
+ 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 }
+ );
});
- overlays.exit().remove();
- overlays = overlays.enter().append("div").merge(overlays).each(function(layer) {
- select_default2(this).call(layer);
+ 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);
});
- 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(d2) {
- return getPath.area(d2) < 30;
- });
- }
+ }, 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 queueRedraw() {
- clearTimeout(_timeoutID);
- _timeoutID = setTimeout(function() {
- redraw();
- }, 750);
+ }
+ function splitIntersection() {
+ if (!context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(deleteLines);
}
- 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();
- });
- }
+ var node = selectMenuItem(context, "split").node();
+ if (!node) {
+ return continueTo(rightClickIntersection);
}
- 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();
+ 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 }
+ );
});
- redraw();
- context.keybinding().on(_t("background.minimap.key"), toggle);
+ 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();
+ }
}
- 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);
+ 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 }
+ );
});
- 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");
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ nextStep();
}
- context.map().on("move.notice", debounce_default(disableTooHigh, 500));
- disableTooHigh();
- };
- }
-
- // modules/ui/photoviewer.js
- function uiPhotoviewer(context) {
- var dispatch14 = 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);
- }
- if (services.mapilio) {
- services.mapilio.hideViewer(context);
- }
- if (services.vegbilder) {
- services.vegbilder.hideViewer(context);
- }
- }).append("div").call(svgIcon("#iD-icon-close"));
- function preventDefault(d3_event) {
- d3_event.preventDefault();
+ }
+ function didSplit() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
}
- selection2.append("button").attr("class", "resize-handle-xy").on("touchstart touchdown touchend", preventDefault).on(
- _pointerPrefix + "down",
- buildResizeListener(selection2, "resize", dispatch14, { resizeOnX: true, resizeOnY: true })
- );
- selection2.append("button").attr("class", "resize-handle-x").on("touchstart touchdown touchend", preventDefault).on(
- _pointerPrefix + "down",
- buildResizeListener(selection2, "resize", dispatch14, { resizeOnX: true })
- );
- selection2.append("button").attr("class", "resize-handle-y").on("touchstart touchdown touchend", preventDefault).on(
- _pointerPrefix + "down",
- buildResizeListener(selection2, "resize", dispatch14, { resizeOnY: true })
+ var ids = context.selectedIDs();
+ var string = "intro.lines.did_split_" + (ids.length > 1 ? "multi" : "single");
+ var street = _t("intro.graph.name.washington-street");
+ var padding = 200 * Math.pow(2, context.map().zoom() - 18);
+ var box = pad(twelfthAvenue, padding, context);
+ box.width = box.width / 2;
+ reveal(
+ box,
+ helpHtml(string, { street1: street, street2: street }),
+ { duration: 500 }
);
- function buildResizeListener(target, eventName, dispatch15, 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");
- }
- dispatch15.call(eventName, target, subtractPadding(utilGetDimensions(target, true), target));
- }
- function clamp3(num, min3, max3) {
- return Math.max(min3, Math.min(num, max3));
+ 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);
}
- function stopResize(d3_event) {
- if (pointerId !== (d3_event.pointerId || "mouse"))
- return;
- d3_event.preventDefault();
- d3_event.stopPropagation();
- select_default2(window).on("." + eventName, null);
+ });
+ context.history().on("change.intro", function() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
}
- 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);
- }
- };
+ });
+ function continueTo(nextStep) {
+ context.map().on("move.intro drawn.intro", null);
+ context.on("enter.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
}
}
- 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");
- dispatch14.call("resize", photoviewer2, subtractPadding(setPhotoDimensions, photoviewer2));
+ function multiSelect() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
}
- };
- function subtractPadding(dimensions, selection2) {
- return [
- dimensions[0] - parseFloat(selection2.style("padding-left")) - parseFloat(selection2.style("padding-right")),
- dimensions[1] - parseFloat(selection2.style("padding-top")) - parseFloat(selection2.style("padding-bottom"))
- ];
- }
- return utilRebind(photoviewer, dispatch14, "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, i3, val, dLon;
- if (isImperial) {
- buckets = [528e4, 528e3, 52800, 5280, 500, 50, 5, 1];
- } else {
- buckets = [5e6, 5e5, 5e4, 5e3, 500, 50, 5, 1];
+ 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);
}
- for (i3 = 0; i3 < buckets.length; i3++) {
- val = buckets[i3];
- if (dist >= val) {
- scale.dist = Math.floor(dist / val) * val;
- break;
+ 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 {
- scale.dist = +dist.toFixed(2);
+ 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();
}
- 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);
+ function multiRightClick() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
}
- 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, d2) {
- d3_event.preventDefault();
- var i3 = _dataShortcuts.indexOf(d2);
- _activeTab = i3;
- render(selection2);
- });
- tabsEnter.append("span").html(function(d2) {
- return _t.html(d2.text);
- });
- wrapper.selectAll(".tab").classed("active", function(d2, i3) {
- return i3 === _activeTab;
- });
- var shortcuts = shortcutsList.selectAll(".shortcut-tab").data(_dataShortcuts);
- var shortcutsEnter = shortcuts.enter().append("div").attr("class", function(d2) {
- return "shortcut-tab shortcut-tab-" + d2.tab;
- });
- var columnsEnter = shortcutsEnter.selectAll(".shortcut-column").data(function(d2) {
- return d2.columns;
- }).enter().append("table").attr("class", "shortcut-column");
- var rowsEnter = columnsEnter.selectAll(".shortcut-row").data(function(d2) {
- return d2.rows;
- }).enter().append("tr").attr("class", "shortcut-row");
- var sectionRows = rowsEnter.filter(function(d2) {
- return !d2.shortcuts;
- });
- sectionRows.append("td");
- sectionRows.append("td").attr("class", "shortcut-section").append("h3").html(function(d2) {
- return _t.html(d2.text);
- });
- var shortcutRows = rowsEnter.filter(function(d2) {
- return d2.shortcuts;
+ 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 });
});
- var shortcutKeys = shortcutRows.append("td").attr("class", "shortcut-keys");
- var modifierKeys = shortcutKeys.filter(function(d2) {
- return d2.modifiers;
+ 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);
});
- modifierKeys.selectAll("kbd.modifier").data(function(d2) {
- if (detected.os === "win" && d2.text === "shortcuts.editing.commands.redo") {
- return ["\u2318"];
- } else if (detected.os !== "mac" && d2.text === "shortcuts.browsing.display_options.fullscreen") {
- return [];
- } else {
- return d2.modifiers;
+ context.history().on("change.intro", function() {
+ if (!_washingtonSegmentID || !context.hasEntity(_washingtonSegmentID) || !context.hasEntity(washingtonStreetID) || !context.hasEntity(twelfthAvenueID) || !context.hasEntity(eleventhAvenueEndID)) {
+ return continueTo(rightClickIntersection);
}
- }).enter().each(function() {
- var selection3 = select_default2(this);
- selection3.append("kbd").attr("class", "modifier").text(function(d2) {
- return uiCmd.display(d2);
- });
- selection3.append("span").text("+");
});
- shortcutKeys.selectAll("kbd.shortcut").data(function(d2) {
- var arr = d2.shortcuts;
- if (detected.os === "win" && d2.text === "shortcuts.editing.commands.redo") {
- arr = ["Y"];
- } else if (detected.os !== "mac" && d2.text === "shortcuts.browsing.display_options.fullscreen") {
- arr = ["F11"];
+ 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);
}
- arr = arr.map(function(s2) {
- return uiCmd.display(s2.indexOf(".") !== -1 ? _t(s2) : s2);
- });
- return utilArrayUniq(arr).map(function(s2) {
- return {
- shortcut: s2,
- separator: d2.separator,
- suffix: d2.suffix
- };
- });
- }).enter().each(function(d2, i3, nodes) {
- var selection3 = select_default2(this);
- var click = d2.shortcut.toLowerCase().match(/(.*).click/);
- if (click && click[1]) {
- selection3.call(svgIcon("#iD-walkthrough-mouse-" + click[1], "operation"));
- } else if (d2.shortcut.toLowerCase() === "long-press") {
- selection3.call(svgIcon("#iD-walkthrough-longpress", "longpress operation"));
- } else if (d2.shortcut.toLowerCase() === "tap") {
- selection3.call(svgIcon("#iD-walkthrough-tap", "tap operation"));
+ });
+ context.history().on("change.intro", function() {
+ if (context.hasEntity(_washingtonSegmentID) || context.hasEntity(twelfthAvenueID)) {
+ continueTo(retryDelete);
} else {
- selection3.append("kbd").attr("class", "shortcut").text(function(d4) {
- return d4.shortcut;
- });
- }
- if (i3 < nodes.length - 1) {
- selection3.append("span").html(d2.separator || "\xA0" + _t.html("shortcuts.or") + "\xA0");
- } else if (i3 === nodes.length - 1 && d2.suffix) {
- selection3.append("span").text(d2.suffix);
+ continueTo(play);
}
});
- shortcutKeys.filter(function(d2) {
- return d2.gesture;
- }).each(function() {
- var selection3 = select_default2(this);
- selection3.append("span").text("+");
- selection3.append("span").attr("class", "gesture").html(function(d2) {
- return _t.html(d2.gesture);
- });
- });
- shortcutRows.append("td").attr("class", "shortcut-desc").html(function(d2) {
- return d2.text ? _t.html(d2.text) : "\xA0";
- });
- wrapper.selectAll(".shortcut-tab").style("display", function(d2, i3) {
- return i3 === _activeTab ? "flex" : "none";
+ 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();
+ }
}
- 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);
+ function play() {
+ dispatch14.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");
}
- });
- }
- };
- }
-
- // modules/ui/data_header.js
- function uiDataHeader() {
- var _datum;
- function dataHeader(selection2) {
- var header = selection2.selectAll(".data-header").data(
- _datum ? [_datum] : [],
- function(d2) {
- return d2.__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;
+ chapter.enter = function() {
+ addLine();
};
- return dataHeader;
+ 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, dispatch14, "on");
}
- // modules/ui/combobox.js
- var _comboHideTimerID;
- function uiCombobox(context, klass) {
- var dispatch14 = 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(d2) {
- var terms = d2.terms || [];
- terms.push(d2.value);
- if (d2.key) {
- terms.push(d2.key);
- }
- return terms.some(function(term) {
- return term.toString().toLowerCase().indexOf(val.toLowerCase()) !== -1;
- });
- }));
+ // modules/ui/intro/building.js
+ function uiIntroBuilding(context, reveal) {
+ var dispatch14 = 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"
};
- 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(d2) {
- return d2 === 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 = +/* @__PURE__ */ 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 timeout2(f2, t2) {
+ timeouts.push(window.setTimeout(f2, t2));
+ }
+ function eventCancel(d3_event) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ }
+ function revealHouse(center, text, options2) {
+ var padding = 160 * Math.pow(2, context.map().zoom() - 20);
+ var box = pad(center, padding, context);
+ reveal(box, text, options2);
+ }
+ function revealTank(center, text, options2) {
+ var padding = 190 * Math.pow(2, context.map().zoom() - 19.5);
+ var box = pad(center, padding, context);
+ reveal(box, text, 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 });
}
- function focus() {
- fetchComboData("");
+ 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 blur() {
- _comboHideTimerID = window.setTimeout(hide, 75);
+ }
+ function startHouse() {
+ if (context.mode().id !== "add-area") {
+ return continueTo(addHouse);
}
- 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();
+ _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 });
});
- container.on("scroll.combo-scroll", render, true);
+ 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 hide() {
- if (_comboHideTimerID) {
- window.clearTimeout(_comboHideTimerID);
- _comboHideTimerID = void 0;
- }
- container.selectAll(".combobox").remove();
- container.on("scroll.combo-scroll", null);
+ }
+ function continueHouse() {
+ if (context.mode().id !== "draw-area") {
+ return continueTo(addHouse);
}
- 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;
+ _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(n3) {
+ return context.projection(n3.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 keyup(d3_event) {
- switch (d3_event.keyCode) {
- case 27:
- cancel();
- break;
- }
+ }
+ 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 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 chooseCategoryBuilding() {
+ if (!_houseID || !context.hasEntity(_houseID)) {
+ return addHouse();
}
- function nav(dir) {
- if (_suggestions.length) {
- var index = -1;
- for (var i3 = 0; i3 < _suggestions.length; i3++) {
- if (_selected && _suggestions[i3].value === _selected) {
- index = i3;
- break;
- }
- }
- index = Math.max(Math.min(index + dir, _suggestions.length - 1), 0);
- _selected = _suggestions[index].value;
- utilGetSetValue(input, _selected);
- dispatch14.call("update");
- }
- render();
- ensureVisible();
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _houseID) {
+ context.enter(modeSelect(context, [_houseID]));
}
- 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();
+ 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 selected = combo.selectAll(".combobox-option.selected").node();
- if (selected) {
- selected.scrollIntoView({ behavior: "smooth", block: "nearest" });
+ 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 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 choosePresetHouse() {
+ if (!_houseID || !context.hasEntity(_houseID)) {
+ return addHouse();
}
- function fetchComboData(v2, cb) {
- _cancelFetch = false;
- _fetcher.call(input, v2, function(results) {
- if (_cancelFetch)
- return;
- _suggestions = results;
- results.forEach(function(d2) {
- _fetched[d2.value] = d2;
- });
- if (cb) {
- cb();
- }
- });
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _houseID) {
+ context.enter(modeSelect(context, [_houseID]));
}
- function tryAutocomplete() {
- if (!_canAutocomplete)
- return;
- var val = _caseSensitive ? value() : value().toLowerCase();
- if (!val)
- return;
- if (isFinite(val))
- return;
- const suggestionValues = [];
- _suggestions.forEach((s2) => {
- suggestionValues.push(s2.value);
- if (s2.key && s2.key !== s2.value) {
- suggestionValues.push(s2.key);
- }
+ 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);
});
- var bestIndex = -1;
- for (var i3 = 0; i3 < suggestionValues.length; i3++) {
- var suggestion = suggestionValues[i3];
- var compare = _caseSensitive ? suggestion : suggestion.toLowerCase();
- if (compare === val) {
- bestIndex = i3;
- break;
- } else if (bestIndex === -1 && compare.indexOf(val) === 0) {
- bestIndex = i3;
- }
+ }, 400);
+ context.on("enter.intro", function(mode) {
+ if (!_houseID || !context.hasEntity(_houseID)) {
+ return continueTo(addHouse);
}
- if (bestIndex !== -1) {
- var bestVal = suggestionValues[bestIndex];
- input.property("value", bestVal);
- input.node().setSelectionRange(val.length, bestVal.length);
- dispatch14.call("update");
- return bestVal;
+ 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 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(d2) {
- return d2.value;
- });
- options2.exit().remove();
- options2.enter().append("a").attr("class", function(d2) {
- return "combobox-option " + (d2.klass || "");
- }).attr("title", function(d2) {
- return d2.title;
- }).each(function(d2) {
- if (d2.display) {
- d2.display(select_default2(this));
- } else {
- select_default2(this).text(d2.value);
- }
- }).on("mouseenter", _mouseEnterHandler).on("mouseleave", _mouseLeaveHandler).merge(options2).classed("selected", function(d2) {
- return d2.value === _selected || d2.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 closeEditorHouse() {
+ if (!_houseID || !context.hasEntity(_houseID)) {
+ return addHouse();
}
- function accept(d3_event, d2) {
- _cancelFetch = true;
- var thiz = input.node();
- if (d2) {
- utilGetSetValue(input, d2.value);
- utilTriggerEvent(input, "change");
- }
- var val = utilGetSetValue(input);
- thiz.setSelectionRange(val.length, val.length);
- d2 = _fetched[val];
- dispatch14.call("accept", thiz, d2, val);
- hide();
+ var ids = context.selectedIDs();
+ if (context.mode().id !== "select" || !ids.length || ids[0] !== _houseID) {
+ context.enter(modeSelect(context, [_houseID]));
}
- 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);
- dispatch14.call("cancel", thiz);
- hide();
+ 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();
}
- };
- 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, dispatch14, "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 dispatch14 = 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";
+ }
+ 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;
}
- 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);
+ 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();
}
- 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 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);
}
- function toggle(d3_event) {
- d3_event.preventDefault();
- _expanded = !_expanded;
- if (_updatePreference) {
- corePreferences("disclosure." + key + ".expanded", _expanded);
+ 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);
}
- 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);
+ });
+ context.map().on("move.intro", function() {
+ var node2 = selectMenuItem(context, "orthogonalize").node();
+ if (!wasChanged && !node2) {
+ return continueTo(rightClickHouse);
}
- dispatch14.call("toggled", this, _expanded);
+ 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();
}
- };
- 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, dispatch14, "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;
+ }
+ 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();
}
- 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;
+ }
+ function doneSquare() {
+ context.history().checkpoint("doneSquare");
+ revealHouse(house, helpHtml("intro.buildings.done_square"), {
+ buttonText: _t.html("intro.ok"),
+ buttonCallback: function() {
+ continueTo(addTank);
}
- selection2.call(_disclosure);
- return;
- }
- if (_content) {
- selection2.call(_content);
+ });
+ function continueTo(nextStep) {
+ nextStep();
}
}
- 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;
+ 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 });
}
- 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();
+ 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);
});
- } 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"));
+ }, msec + 100);
+ function continueTo(nextStep) {
+ context.on("enter.intro", null);
+ nextStep();
}
- 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));
+ }
+ function startTank() {
+ if (context.mode().id !== "add-area") {
+ return continueTo(addTank);
}
- 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"));
+ _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 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);
+ 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 });
});
- _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");
+ 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();
+ }
}
- 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();
+ 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);
}
});
- };
- tagReference.body = function(selection2) {
- var itemID = what.qid || what.key + "-" + (what.value || "");
- _body = selection2.selectAll(".tag-reference-body").data([itemID], function(d2) {
- return d2;
- });
- _body.exit().remove();
- _body = _body.enter().append("div").attr("class", "tag-reference-body").style("max-height", "0").style("opacity", "0").merge(_body);
- if (_showing === false) {
- hide();
+ function 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();
}
- };
- 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 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 clickHelp(index) {
- var d2 = docs[index];
- var tkeys = fieldHelpKeys[fieldName][index][1];
- _body.selectAll(".field-help-nav-item").classed("active", function(d4, i3) {
- return i3 === index;
- });
- var content = _body.selectAll(".field-help-content").html(d2.html);
- content.selectAll("p").attr("class", function(d4, i3) {
- return tkeys[i3];
- });
- if (d2.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 (d2.key === "help.field.restrictions.modifying") {
- content.insert("img", "p.allow_turn").attr("class", "field-help-image cf").attr("src", context.imagePath("tr_modify.gif"));
+ 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();
}
}
- 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 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);
}
});
- };
- 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(d2) {
- return d2.title;
+ 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 }
+ );
});
- navEnter.selectAll(".field-help-nav-item").data(titles).enter().append("div").attr("class", "field-help-nav-item").html(function(d2) {
- return d2;
- }).on("click", function(d3_event, d2) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- clickHelp(titles.indexOf(d2));
+ 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);
});
- 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 dispatch14 = 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 i3 in options2) {
- var v2 = options2[i3];
- values.push(v2 === "undefined" ? void 0 : v2);
- texts.push(stringsField.t.html("options." + v2, { "default": v2 }));
- }
- } 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 continueTo(nextStep) {
+ context.on("enter.intro", null);
+ context.map().on("move.intro", null);
+ context.history().on("change.intro", null);
+ nextStep();
}
}
- 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 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 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 t2 = {};
- if (Array.isArray(_tags[field.key])) {
- if (values.indexOf("yes") !== -1) {
- t2[field.key] = "yes";
- } else {
- t2[field.key] = values[0];
+ function play() {
+ dispatch14.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");
}
- } else {
- t2[field.key] = values[(values.indexOf(_value) + 1) % values.length];
- }
- if (t2[field.key] === "reversible" || t2[field.key] === "alternating") {
- t2[field.key] = values[0];
}
- dispatch14.call("change", this, t2);
- });
- 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 i4 in _entityIDs) {
- graph = actionReverse(_entityIDs[i4])(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;
+ );
+ }
+ chapter.enter = function() {
+ addHouse();
};
- 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);
- }
+ 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);
};
- check.focus = function() {
- input.node().focus();
+ chapter.restart = function() {
+ chapter.exit();
+ chapter.enter();
};
- return utilRebind(check, dispatch14, "on");
+ return utilRebind(chapter, dispatch14, "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);
+ // modules/ui/intro/start_editing.js
+ function uiIntroStartEditing(context, reveal) {
+ var dispatch14 = dispatch_default("done", "startEditing");
+ var modalSelection = select_default2(null);
+ var chapter = {
+ title: "intro.startediting.title"
};
- 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", (d2) => d2 > maxChars).style("border-right-width", (d2) => `${Math.abs(maxChars - d2) * 2}px`).style("margin-right", (d2) => d2 > maxChars ? `${(maxChars - d2) * 2}px` : 0).style("opacity", (d2) => d2 > maxChars * 0.8 ? Math.min(1, (d2 / maxChars - 0.8) / (1 - 0.8)) : 0).style("pointer-events", (d2) => d2 > maxChars * 0.8 ? null : "none");
- if (_silent)
- return;
- if (strLen > maxChars) {
- _tooltip.show();
- } else {
- _tooltip.hide();
- }
+ 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"));
+ dispatch14.call("startEditing");
+ }
+ chapter.enter = function() {
+ showHelp();
};
- lengthIndicator.silent = function(val) {
- if (!arguments.length)
- return _silent;
- _silent = val;
- return lengthIndicator;
+ chapter.exit = function() {
+ modalSelection.remove();
+ context.container().selectAll(".shaded").remove();
};
- return lengthIndicator;
+ return utilRebind(chapter, dispatch14, "on");
}
- // modules/ui/fields/combo.js
- function uiFieldCombo(field, context) {
- var dispatch14 = 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(d2) {
- _dataDeprecated = d2;
- }).catch(function() {
- });
- if (_isMulti && field.key && /[^:]$/.test(field.key)) {
- field.key += ":";
- }
- function snake(s2) {
- return s2.replace(/\s+/g, "_");
- }
- function clean2(s2) {
- return s2.split(";").map(function(s3) {
- return s3.trim();
- }).join(";");
- }
- function tagValue(dval) {
- dval = clean2(dval || "");
- var found = getOptions(true).find(function(o2) {
- return o2.key && clean2(o2.value) === dval;
+ // 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() {
});
- 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();
+ 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();
}
- return dval;
- }
- function getLabelId(field2, v2) {
- return field2.hasTextForStringId(`options.${v2}.title`) ? `options.${v2}.title` : `options.${v2}`;
- }
- 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 });
+ 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();
}
- if (field.type === "typeCombo" && tval.toLowerCase() === "yes") {
- return "";
+ overlays.forEach((d2) => context.background().toggleOverlayLayer(d2));
+ 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, i3) => {
+ let s2 = chapterUi[chapter](context, curtain.reveal).on("done", () => {
+ buttons.filter((d2) => d2.title === s2.title).classed("finished", true);
+ if (i3 < chapterFlow.length - 1) {
+ const next = chapterFlow[i3 + 1];
+ context.container().select("button.chapter-".concat(next)).classed("next", true);
+ }
+ progress.push(chapter);
+ corePreferences("walkthrough_progress", utilArrayUniq(progress).join(";"));
+ });
+ return s2;
+ });
+ 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((d2) => context.background().toggleOverlayLayer(d2));
+ 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", (d2, i3) => "chapter chapter-".concat(chapterFlow[i3])).on("click", enterChapter);
+ buttons.append("span").html((d2) => _t.html(d2.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", (d2) => d2.title === _currChapter.title);
}
- 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 });
+ 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 (field.type === "typeCombo" && tval.toLowerCase() === "yes") {
- tval = "";
+ if (corePreferences("validate-what") === "all") {
+ var resolvedIssues = context.validator().getResolvedIssues();
+ if (resolvedIssues.length) {
+ resolvedItem.count = resolvedIssues.length;
+ shownItems.push(resolvedItem);
+ }
}
- return (selection2) => selection2.text(tval);
- }
- function objectDifference(a2, b2) {
- return a2.filter(function(d1) {
- return !b2.some(function(d2) {
- return d1.value === d2.value;
+ var chips = selection2.selectAll(".chip").data(shownItems, function(d2) {
+ return d2.id;
+ });
+ chips.exit().remove();
+ var enter = chips.enter().append("a").attr("class", function(d2) {
+ return "chip " + d2.id + "-count";
+ }).attr("href", "#").each(function(d2) {
+ var chipSelection = select_default2(this);
+ var tooltipBehavior = uiTooltip().placement("top").title(() => _t.append(d2.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("#" + d2.iconID));
});
- }
- 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);
- setTimeout(() => setStaticValues(setPlaceholder), 0);
- }
- }
- function getOptions(allOptions) {
- var stringsField = field.resolveReference("stringsCrossReference");
- if (!(field.options || stringsField.options))
- return [];
- let options2;
- if (allOptions !== true) {
- options2 = field.options || stringsField.options;
- } else {
- options2 = [].concat(field.options, stringsField.options).filter(Boolean);
- }
- return options2.map(function(v2) {
- const labelId = getLabelId(stringsField, v2);
- return {
- key: v2,
- value: stringsField.t(labelId, { default: v2 }),
- title: stringsField.t(`options.${v2}.description`, { default: v2 }),
- display: addComboboxIcons(stringsField.t.append(labelId, { default: v2 }), v2),
- klass: stringsField.hasTextForStringId(labelId) ? "" : "raw-option"
- };
+ enter.append("span").attr("class", "count");
+ enter.merge(chips).selectAll("span.count").text(function(d2) {
+ return d2.count.toString();
});
}
- function setStaticValues(callback, filter2) {
- _comboData = getOptions();
- if (filter2 !== void 0) {
- _comboData = _comboData.filter(filter2);
+ 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 x2 = d3_event.transform.x;
+ var y2 = d3_event.transform.y;
+ var k2 = d3_event.transform.k;
+ var isZooming = k2 !== _tStart.k;
+ var isPanning = x2 !== _tStart.x || y2 !== _tStart.y;
+ if (!isZooming && !isPanning) {
+ return;
+ }
+ if (!_gesture) {
+ _gesture = isZooming ? "zoom" : "pan";
+ }
+ var tMini = projection2.transform();
+ var tX, tY, scale;
+ if (_gesture === "zoom") {
+ scale = k2 / tMini.k;
+ tX = (_cMini[0] / scale - _cMini[0]) * scale;
+ tY = (_cMini[1] / scale - _cMini[1]) * scale;
+ } else {
+ k2 = tMini.k;
+ scale = 1;
+ tX = x2 - tMini.x;
+ tY = y2 - tMini.y;
+ }
+ utilSetTransform(tiles, tX, tY, scale);
+ utilSetTransform(viewport, 0, 0, scale);
+ _isTransformed = true;
+ _tCurr = identity2.translate(x2, y2).scale(k2);
+ var zMain = geoScaleToZoom(context.projection.scale());
+ var zMini = geoScaleToZoom(k2);
+ _zDiff = zMain - zMini;
+ queueRedraw();
}
- _comboData = objectDifference(_comboData, _multiData);
- _combobox.data(_comboData);
- if (callback)
- callback(_comboData);
- }
- function setTaginfoValues(q2, callback) {
- var queryFilter = (d2) => d2.value.toLowerCase().includes(q2.toLowerCase()) || d2.key.toLowerCase().includes(q2.toLowerCase());
- setStaticValues(callback, queryFilter);
- var stringsField = field.resolveReference("stringsCrossReference");
- var fn = _isMulti ? "multikeys" : "values";
- var query = (_isMulti ? field.key : "") + q2;
- var hasCountryPrefix = _isNetwork && _countryCode && _countryCode.indexOf(q2.toLowerCase()) === 0;
- if (hasCountryPrefix) {
- query = _countryCode + ":";
+ function zoomEnded() {
+ if (_skipEvents)
+ return;
+ if (_gesture !== "pan")
+ return;
+ updateProjection();
+ _gesture = null;
+ context.map().center(projection2.invert(_cMini));
}
- var params = {
- debounce: q2 !== "",
- key: field.key,
- query
- };
- if (_entityIDs.length) {
- params.geometry = context.graph().geometry(_entityIDs[0]);
+ 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;
}
- services.taginfo[fn](params, function(err, data) {
- if (err)
+ function redraw() {
+ clearTimeout(_timeoutID);
+ if (_isHidden)
return;
- data = data.filter((d2) => field.type !== "typeCombo" || d2.value !== "yes");
- data = data.filter((d2) => {
- var value = d2.value;
- if (_isMulti) {
- value = value.slice(field.key.length);
+ 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 i3 = 0; i3 < overlaySources.length; i3++) {
+ if (overlaySources[i3].validZoom(zMini)) {
+ if (!overlayLayers[i3])
+ overlayLayers[i3] = rendererTileLayer(context);
+ activeOverlayLayers.push(overlayLayers[i3].source(overlaySources[i3]).projection(projection2).dimensions(_dMini));
}
- return value === restrictTagValueSpelling(value);
- });
- var deprecatedValues = osmEntity.deprecatedTagValuesByKey(_dataDeprecated)[field.key];
- if (deprecatedValues) {
- data = data.filter((d2) => !deprecatedValues.includes(d2.value));
- }
- if (hasCountryPrefix) {
- data = data.filter((d2) => d2.value.toLowerCase().indexOf(_countryCode + ":") === 0);
}
- const additionalOptions = (field.options || stringsField.options || []).filter((v2) => !data.some((dv) => dv.value === (_isMulti ? field.key + v2 : v2))).map((v2) => ({ value: v2 }));
- _container.classed("empty-combobox", data.length === 0);
- _comboData = data.concat(additionalOptions).map(function(d2) {
- var v2 = d2.value;
- if (_isMulti)
- v2 = v2.replace(field.key, "");
- const labelId = getLabelId(stringsField, v2);
- var isLocalizable = stringsField.hasTextForStringId(labelId);
- var label = stringsField.t(labelId, { default: v2 });
- return {
- key: v2,
- value: label,
- title: stringsField.t(`options.${v2}.description`, { default: isLocalizable ? v2 : d2.title !== label ? d2.title : "" }),
- display: addComboboxIcons(stringsField.t.append(labelId, { default: v2 }), v2),
- klass: isLocalizable ? "" : "raw-option"
- };
- });
- _comboData = _comboData.filter(queryFilter);
- _comboData = objectDifference(_comboData, _multiData);
- if (callback)
- callback(_comboData);
- });
- }
- function addComboboxIcons(disp, value) {
- const iconsField = field.resolveReference("iconsCrossReference");
- if (iconsField.icons) {
- return function(selection2) {
- var span = selection2.insert("span", ":first-child").attr("class", "tag-value-icon");
- if (iconsField.icons[value]) {
- span.call(svgIcon(`#${iconsField.icons[value]}`));
- }
- disp.call(this, selection2);
- };
- }
- return disp;
- }
- function setPlaceholder(values) {
- if (_isMulti || _isSemi) {
- _staticPlaceholder = field.placeholder() || _t("inspector.add");
- } else {
- var vals = values.map(function(d2) {
- return d2.value;
- }).filter(function(s2) {
- return s2.length < 20;
+ 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(d2) {
+ return d2.source().name();
});
- var placeholders = vals.length > 1 ? vals : values.map(function(d2) {
- return d2.key;
+ overlays.exit().remove();
+ overlays = overlays.enter().append("div").merge(overlays).each(function(layer) {
+ select_default2(this).call(layer);
});
- _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);
- var hideAdd = !_allowCustomValues && !values.length;
- _container.selectAll(".chiplist .input-wrap").style("display", hideAdd ? "none" : null);
- }
- function change() {
- var t2 = {};
- 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(v2) {
- var key = (field.key || "") + v2;
- if (_tags) {
- var old = _tags[key];
- if (typeof old === "string" && old.toLowerCase() !== "no")
- return;
- }
- key = context.cleanTagKey(key);
- field.keys.push(key);
- t2[key] = "yes";
- });
- } else if (_isSemi) {
- var arr = _multiData.map(function(d2) {
- return d2.key;
+ 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(d2) {
+ return getPath.area(d2) < 30;
});
- arr = arr.concat(vals);
- t2[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));
- t2[field.key] = val || void 0;
- }
- dispatch14.call("change", this, t2);
- }
- function removeMultikey(d3_event, d2) {
- d3_event.preventDefault();
- d3_event.stopPropagation();
- var t2 = {};
- if (_isMulti) {
- t2[d2.key] = void 0;
- } else if (_isSemi) {
- var arr = _multiData.map(function(md) {
- return md.key === d2.key ? null : md.key;
- }).filter(Boolean);
- arr = utilArrayUniq(arr);
- t2[field.key] = arr.length ? arr.join(";") : void 0;
- _lengthIndicator.update(t2[field.key]);
}
- dispatch14.call("change", this, t2);
- }
- function invertMultikey(d3_event, d2) {
- d3_event.preventDefault();
- d3_event.stopPropagation();
- var t2 = {};
- if (_isMulti) {
- t2[d2.key] = _tags[d2.key] === "yes" ? "no" : "yes";
+ function queueRedraw() {
+ clearTimeout(_timeoutID);
+ _timeoutID = setTimeout(function() {
+ redraw();
+ }, 750);
}
- dispatch14.call("change", this, t2);
- }
- 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";
+ 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();
+ });
}
- _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);
- var hideAdd = !_allowCustomValues && !_comboData.length;
- _inputWrap.style("display", hideAdd ? "none" : null);
- _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, _container).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;
+ 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();
}
});
- 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));
- });
+ redraw();
+ context.keybinding().on(_t("background.minimap.key"), toggle);
}
- function updateIcon(value) {
- value = tagValue(value);
- let container = _container;
- if (field.type === "multiCombo" || field.type === "semiCombo") {
- container = _container.select(".input-wrap");
+ 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");
}
- const iconsField = field.resolveReference("iconsCrossReference");
- if (iconsField.icons) {
- container.selectAll(".tag-value-icon").remove();
- if (iconsField.icons[value]) {
- container.selectAll(".tag-value-icon").data([value]).enter().insert("div", "input").attr("class", "tag-value-icon").call(svgIcon(`#${iconsField.icons[value]}`));
+ context.map().on("move.notice", debounce_default(disableTooHigh, 500));
+ disableTooHigh();
+ };
+ }
+
+ // modules/ui/photoviewer.js
+ function uiPhotoviewer(context) {
+ var dispatch14 = 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);
}
- }
- }
- combo.tags = function(tags) {
- _tags = tags;
- var stringsField = field.resolveReference("stringsCrossReference");
- var isMixed = Array.isArray(tags[field.key]);
- var showsValue = (value) => !isMixed && value && !(field.type === "typeCombo" && value === "yes");
- var isRawValue = (value) => showsValue(value) && !stringsField.hasTextForStringId(`options.${value}`) && !stringsField.hasTextForStringId(`options.${value}.title`);
- var isKnownValue = (value) => showsValue(value) && !isRawValue(value);
- var isReadOnly = !_allowCustomValues;
- if (_isMulti || _isSemi) {
- _multiData = [];
- var maxLength;
- if (_isMulti) {
- for (var k2 in tags) {
- if (field.key && k2.indexOf(field.key) !== 0)
- continue;
- if (!field.key && field.keys.indexOf(k2) === -1)
- continue;
- var v2 = tags[k2];
- var suffix = field.key ? k2.slice(field.key.length) : k2;
- _multiData.push({
- key: k2,
- value: displayValue(suffix),
- display: addComboboxIcons(renderValue(suffix), suffix),
- state: typeof v2 === "string" ? v2.toLowerCase() : "",
- isMixed: Array.isArray(v2)
- });
- }
- if (field.key) {
- field.keys = _multiData.map(function(d2) {
- return d2.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(v3) {
- return {
- key: v3,
- value: displayValue(v3),
- display: addComboboxIcons(renderValue(v3), v3),
- isMixed: !commonValues.includes(v3)
- };
- });
- var currLength = utilUnicodeCharsCount(commonValues.join(";"));
- maxLength = context.maxCharsForTagValue() - currLength;
- if (currLength > 0) {
- maxLength -= 1;
- }
+ if (services.mapillary) {
+ services.mapillary.hideViewer(context);
}
- maxLength = Math.max(0, maxLength);
- var hideAdd = maxLength <= 0 || !_allowCustomValues && !_comboData.length;
- _container.selectAll(".chiplist .input-wrap").style("display", hideAdd ? "none" : null);
- var allowDragAndDrop = _isSemi && !Array.isArray(tags[field.key]);
- 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(d2) {
- var k3 = d2.key;
- if (_isMulti)
- k3 = k3.replace(field.key, "");
- return !stringsField.hasTextForStringId("options." + k3);
- }).classed("draggable", allowDragAndDrop).classed("mixed", function(d2) {
- return d2.isMixed;
- }).attr("title", function(d2) {
- if (d2.isMixed) {
- return _t("inspector.unshared_value_tooltip");
- }
- if (!["yes", "no"].includes(d2.state)) {
- return d2.state;
- }
- return null;
- }).classed("negated", (d2) => d2.state === "no");
- if (!_isSemi) {
- chips.selectAll("input[type=checkbox]").remove();
- chips.insert("input", "span").attr("type", "checkbox").property("checked", (d2) => d2.state === "yes").property("indeterminate", (d2) => d2.isMixed || !["yes", "no"].includes(d2.state)).on("click", invertMultikey);
+ if (services.kartaview) {
+ services.kartaview.hideViewer(context);
}
- if (allowDragAndDrop) {
- registerDragAndDrop(chips);
+ if (services.mapilio) {
+ services.mapilio.hideViewer(context);
}
- chips.select("span").each(function(d2) {
- const selection2 = select_default2(this);
- if (d2.display) {
- selection2.text("");
- d2.display(selection2);
- } else {
- selection2.text(d2.value);
- }
- });
- chips.select("a").attr("href", "#").on("click", removeMultikey).attr("class", "remove").text("\xD7");
- updateIcon("");
- } else {
- var mixedValues = isMixed && tags[field.key].map(function(val) {
- return displayValue(val);
- }).filter(Boolean);
- utilGetSetValue(_input, !isMixed ? displayValue(tags[field.key]) : "").data([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(tags[field.key]) && (d3_event.keyCode === utilKeybinding.keyCodes["\u232B"] || d3_event.keyCode === utilKeybinding.keyCodes["\u2326"])) {
- d3_event.preventDefault();
- d3_event.stopPropagation();
- var t2 = {};
- t2[field.key] = void 0;
- dispatch14.call("change", this, t2);
+ if (services.vegbilder) {
+ services.vegbilder.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", dispatch14, { resizeOnX: true, resizeOnY: true })
+ );
+ selection2.append("button").attr("class", "resize-handle-x").on("touchstart touchdown touchend", preventDefault).on(
+ _pointerPrefix + "down",
+ buildResizeListener(selection2, "resize", dispatch14, { resizeOnX: true })
+ );
+ selection2.append("button").attr("class", "resize-handle-y").on("touchstart touchdown touchend", preventDefault).on(
+ _pointerPrefix + "down",
+ buildResizeListener(selection2, "resize", dispatch14, { resizeOnY: true })
+ );
+ function buildResizeListener(target, eventName, dispatch15, 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 (!Array.isArray(tags[field.key])) {
- updateIcon(tags[field.key]);
+ if (resizeOnY) {
+ var maxHeight = mapSize[1] - 90;
+ var newHeight = clamp3(startHeight + startY - d3_event.clientY, minHeight, maxHeight);
+ target.style("height", newHeight + "px");
+ }
+ dispatch15.call(eventName, target, subtractPadding(utilGetDimensions(target, true), target));
}
- if (!isMixed) {
- _lengthIndicator.update(tags[field.key]);
+ function clamp3(num, min3, max3) {
+ return Math.max(min3, Math.min(num, max3));
}
- }
- const refreshStyles = () => {
- _input.data([tagValue(utilGetSetValue(_input))]).classed("raw-value", isRawValue).classed("known-value", isKnownValue);
- };
- _input.on("input.refreshStyles", refreshStyles);
- _combobox.on("update.refreshStyles", refreshStyles);
- refreshStyles();
- };
- 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 x2 = d3_event.x - dragOrigin.x, y2 = 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(x2, 2) + Math.pow(y2, 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(" + x2 + "px, " + y2 + "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(" + x2 + "px, " + y2 + "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")) {
+ 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);
}
- 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 t2 = {};
- if (_multiData.length) {
- t2[field.key] = _multiData.map(function(element2) {
- return element2.key;
- }).join(";");
- } else {
- t2[field.key] = void 0;
- }
- dispatch14.call("change", this, t2);
- }
- 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;
+ 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");
+ dispatch14.call("resize", photoviewer2, subtractPadding(setPhotoDimensions, photoviewer2));
+ }
};
- function combinedEntityExtent() {
- return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ function subtractPadding(dimensions, selection2) {
+ return [
+ dimensions[0] - parseFloat(selection2.style("padding-left")) - parseFloat(selection2.style("padding-right")),
+ dimensions[1] - parseFloat(selection2.style("padding-top")) - parseFloat(selection2.style("padding-bottom"))
+ ];
}
- return utilRebind(combo, dispatch14, "on");
+ return utilRebind(photoviewer, dispatch14, "on");
}
- // modules/ui/fields/input.js
- var likelyRawNumberFormat = /^-?(0\.\d*|\d*\.\d{0,2}(\d{4,})?|\d{4,}\.\d{3})$/;
- function uiFieldText(field, context) {
- var dispatch14 = 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");
- const formatFloat = _mainLocalizer.floatFormatter(_mainLocalizer.languageCode());
- const parseLocaleFloat = _mainLocalizer.floatParser(_mainLocalizer.languageCode());
- const countDecimalPlaces = _mainLocalizer.decimalPlaceCounter(_mainLocalizer.languageCode());
- if (field.type === "tel") {
- _mainFileFetcher.get("phone_formats").then(function(d2) {
- _phoneFormats = d2;
- updatePhonePlaceholder();
- }).catch(function() {
+ // 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();
});
- }
- 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"];
+ 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();
});
- field.locked(isLocked);
- }
- function i3(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(d2) {
- var which = d2 > 0 ? "increment" : "decrement";
- return "form-field-button " + which;
- }).attr("title", function(d2) {
- var which = d2 > 0 ? "increment" : "decrement";
- return _t(`inspector.${which}`);
- }).merge(buttons).on("click", function(d3_event, d2) {
- 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(v2) {
- v2 = v2.trim();
- const isRawNumber = likelyRawNumberFormat.test(v2);
- var num = isRawNumber ? parseFloat(v2) : parseLocaleFloat(v2);
- if (isDirectionField) {
- const compassDir = cardinal[v2.toLowerCase()];
- if (compassDir !== void 0) {
- num = compassDir;
- }
- }
- if (!isFinite(num))
- return v2;
- num = parseFloat(num);
- if (!isFinite(num))
- return v2;
- num += d2;
- if (isDirectionField) {
- num = (num % 360 + 360) % 360;
- }
- return formatFloat(clamped(num), isRawNumber ? v2.includes(".") ? v2.split(".")[1].length : 0 : countDecimalPlaces(v2));
- });
- 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");
- }
- }).classed("disabled", () => !validIdentifierValueForLink()).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();
- } else if (field.type === "date") {
- input.attr("type", "text");
- updateDateField();
+ 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, i3, val, dLon;
+ if (isImperial) {
+ buckets = [528e4, 528e3, 52800, 5280, 500, 50, 5, 1];
+ } else {
+ buckets = [5e6, 5e5, 5e4, 5e3, 500, 50, 5, 1];
}
- }
- function updateColourPreview() {
- function isColourValid(colour2) {
- if (!colour2.match(/^(#([0-9a-fA-F]{3}){1,2}|\w+)$/)) {
- return false;
- } else if (!CSS.supports("color", colour2) || ["unset", "inherit", "initial", "revert"].includes(colour2)) {
- return false;
+ for (i3 = 0; i3 < buckets.length; i3++) {
+ val = buckets[i3];
+ if (dist >= val) {
+ scale.dist = Math.floor(dist / val) * val;
+ break;
+ } else {
+ scale.dist = +dist.toFixed(2);
}
- return true;
}
- wrap2.selectAll(".colour-preview").remove();
- const colour = utilGetSetValue(input);
- if (!isColourValid(colour) && colour !== "") {
- wrap2.selectAll("input.colour-selector").remove();
- wrap2.selectAll(".form-field-button").remove();
- return;
- }
- var colourSelector = wrap2.selectAll(".colour-selector").data([0]);
- colourSelector.enter().append("input").attr("type", "color").attr("class", "colour-selector").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));
- wrap2.selectAll("input.colour-selector").attr("value", colour);
- var chooserButton = wrap2.selectAll(".colour-preview").data([colour]);
- chooserButton = chooserButton.enter().append("div").attr("class", "form-field-button colour-preview").append("div").style("background-color", (d2) => d2).attr("class", "colour-box");
- if (colour === "") {
- chooserButton = chooserButton.call(svgIcon("#iD-icon-edit"));
- }
- chooserButton.on("click", () => wrap2.select(".colour-selector").node().showPicker());
+ 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 updateDateField() {
- function isDateValid(date2) {
- return date2.match(/^[0-9]{4}(-[0-9]{2}(-[0-9]{2})?)?$/);
- }
- const date = utilGetSetValue(input);
- const now3 = /* @__PURE__ */ new Date();
- const today = new Date(now3.getTime() - now3.getTimezoneOffset() * 6e4).toISOString().split("T")[0];
- if ((field.key === "check_date" || field.key === "survey:date") && date !== today) {
- wrap2.selectAll(".date-set-today").data([0]).enter().append("button").attr("class", "form-field-button date-set-today").call(svgIcon("#fas-rotate")).call(uiTooltip().title(() => _t.append("inspector.set_today"))).on("click", () => {
- utilGetSetValue(input, today);
- change()();
- updateDateField();
- });
- } else {
- wrap2.selectAll(".date-set-today").remove();
- }
- if (!isDateValid(date) && date !== "") {
- wrap2.selectAll("input.date-selector").remove();
- wrap2.selectAll(".date-calendar").remove();
- return;
- }
- if (utilDetect().browser !== "Safari") {
- var dateSelector = wrap2.selectAll(".date-selector").data([0]);
- dateSelector.enter().append("input").attr("type", "date").attr("class", "date-selector").on("input", debounce_default(function(d3_event) {
- d3_event.preventDefault();
- var date2 = this.value;
- if (!isDateValid(date2))
- return;
- utilGetSetValue(input, this.value);
- change()();
- updateDateField();
- }, 100));
- wrap2.selectAll("input.date-selector").attr("value", date);
- var calendarButton = wrap2.selectAll(".date-calendar").data([date]);
- calendarButton = calendarButton.enter().append("button").attr("class", "form-field-button date-calendar").call(svgIcon("#fas-calendar-days"));
- calendarButton.on("click", () => wrap2.select(".date-selector").node().showPicker());
+ 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 updatePhonePlaceholder() {
- if (input.empty() || !Object.keys(_phoneFormats).length)
+ function render(selection2) {
+ if (!_dataShortcuts)
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 (e3) {
- return null;
+ 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, d2) {
+ d3_event.preventDefault();
+ var i3 = _dataShortcuts.indexOf(d2);
+ _activeTab = i3;
+ render(selection2);
+ });
+ tabsEnter.append("span").html(function(d2) {
+ return _t.html(d2.text);
+ });
+ wrapper.selectAll(".tab").classed("active", function(d2, i3) {
+ return i3 === _activeTab;
+ });
+ var shortcuts = shortcutsList.selectAll(".shortcut-tab").data(_dataShortcuts);
+ var shortcutsEnter = shortcuts.enter().append("div").attr("class", function(d2) {
+ return "shortcut-tab shortcut-tab-" + d2.tab;
+ });
+ var columnsEnter = shortcutsEnter.selectAll(".shortcut-column").data(function(d2) {
+ return d2.columns;
+ }).enter().append("table").attr("class", "shortcut-column");
+ var rowsEnter = columnsEnter.selectAll(".shortcut-row").data(function(d2) {
+ return d2.rows;
+ }).enter().append("tr").attr("class", "shortcut-row");
+ var sectionRows = rowsEnter.filter(function(d2) {
+ return !d2.shortcuts;
+ });
+ sectionRows.append("td");
+ sectionRows.append("td").attr("class", "shortcut-section").append("h3").html(function(d2) {
+ return _t.html(d2.text);
+ });
+ var shortcutRows = rowsEnter.filter(function(d2) {
+ return d2.shortcuts;
+ });
+ var shortcutKeys = shortcutRows.append("td").attr("class", "shortcut-keys");
+ var modifierKeys = shortcutKeys.filter(function(d2) {
+ return d2.modifiers;
+ });
+ modifierKeys.selectAll("kbd.modifier").data(function(d2) {
+ if (detected.os === "win" && d2.text === "shortcuts.editing.commands.redo") {
+ return ["\u2318"];
+ } else if (detected.os !== "mac" && d2.text === "shortcuts.browsing.display_options.fullscreen") {
+ return [];
+ } else {
+ return d2.modifiers;
+ }
+ }).enter().each(function() {
+ var selection3 = select_default2(this);
+ selection3.append("kbd").attr("class", "modifier").text(function(d2) {
+ return uiCmd.display(d2);
+ });
+ selection3.append("span").text("+");
+ });
+ shortcutKeys.selectAll("kbd.shortcut").data(function(d2) {
+ var arr = d2.shortcuts;
+ if (detected.os === "win" && d2.text === "shortcuts.editing.commands.redo") {
+ arr = ["Y"];
+ } else if (detected.os !== "mac" && d2.text === "shortcuts.browsing.display_options.fullscreen") {
+ arr = ["F11"];
+ }
+ arr = arr.map(function(s2) {
+ return uiCmd.display(s2.indexOf(".") !== -1 ? _t(s2) : s2);
+ });
+ return utilArrayUniq(arr).map(function(s2) {
+ return {
+ shortcut: s2,
+ separator: d2.separator,
+ suffix: d2.suffix
+ };
+ });
+ }).enter().each(function(d2, i3, nodes) {
+ var selection3 = select_default2(this);
+ var click = d2.shortcut.toLowerCase().match(/(.*).click/);
+ if (click && click[1]) {
+ selection3.call(svgIcon("#iD-walkthrough-mouse-" + click[1], "operation"));
+ } else if (d2.shortcut.toLowerCase() === "long-press") {
+ selection3.call(svgIcon("#iD-walkthrough-longpress", "longpress operation"));
+ } else if (d2.shortcut.toLowerCase() === "tap") {
+ selection3.call(svgIcon("#iD-walkthrough-tap", "tap operation"));
+ } else {
+ selection3.append("kbd").attr("class", "shortcut").text(function(d4) {
+ return d4.shortcut;
+ });
}
- }
- 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;
+ if (i3 < nodes.length - 1) {
+ selection3.append("span").html(d2.separator || "\xA0" + _t.html("shortcuts.or") + "\xA0");
+ } else if (i3 === nodes.length - 1 && d2.suffix) {
+ selection3.append("span").text(d2.suffix);
+ }
+ });
+ shortcutKeys.filter(function(d2) {
+ return d2.gesture;
+ }).each(function() {
+ var selection3 = select_default2(this);
+ selection3.append("span").text("+");
+ selection3.append("span").attr("class", "gesture").html(function(d2) {
+ return _t.html(d2.gesture);
+ });
+ });
+ shortcutRows.append("td").attr("class", "shortcut-desc").html(function(d2) {
+ return d2.text ? _t.html(d2.text) : "\xA0";
+ });
+ wrapper.selectAll(".shortcut-tab").style("display", function(d2, i3) {
+ return i3 === _activeTab ? "flex" : "none";
+ });
}
- function getVals(tags) {
- if (field.keys) {
- const multiSelection = context.selectedIDs();
- tags = multiSelection.length > 1 ? context.selectedIDs().map((id2) => context.graph().entity(id2)).map((entity) => entity.tags) : [tags];
- return tags.map((tags2) => new Set(field.keys.reduce((acc, key) => acc.concat(tags2[key]), []).filter(Boolean))).map((vals) => vals.size === 0 ? /* @__PURE__ */ new Set([void 0]) : vals).reduce((a2, b2) => /* @__PURE__ */ new Set([...a2, ...b2]));
+ return function(selection2, show) {
+ _selection = selection2;
+ if (show) {
+ _modalSelection = uiModal(selection2);
+ _modalSelection.call(shortcutsModal);
} else {
- return new Set([].concat(tags[field.key]));
- }
- }
- function change(onInput) {
- return function() {
- var t2 = {};
- var val = utilGetSetValue(input);
- if (!onInput)
- val = context.cleanTagValue(val);
- if (!val && getVals(_tags).size > 1)
- return;
- var displayVal = val;
- if (field.type === "number" && val) {
- var numbers2 = val.split(";");
- numbers2 = numbers2.map(function(v2) {
- if (likelyRawNumberFormat.test(v2)) {
- return v2;
- }
- var num = parseLocaleFloat(v2);
- const fractionDigits = countDecimalPlaces(v2);
- return isFinite(num) ? clamped(num).toFixed(fractionDigits) : v2;
- });
- val = numbers2.join(";");
- }
- if (!onInput)
- utilGetSetValue(input, displayVal);
- t2[field.key] = val || void 0;
- if (field.keys) {
- dispatch14.call("change", this, (tags) => {
- if (field.keys.some((key) => tags[key])) {
- field.keys.filter((key) => tags[key]).forEach((key) => {
- tags[key] = val || void 0;
- });
- } else {
- tags[field.key] = val || void 0;
+ context.keybinding().on([_t("shortcuts.toggle.key"), "?"], function() {
+ if (context.container().selectAll(".modal-shortcuts").size()) {
+ if (_modalSelection) {
+ _modalSelection.close();
+ _modalSelection = null;
}
- return tags;
- }, onInput);
- } else {
- dispatch14.call("change", this, t2, onInput);
+ } 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(d2) {
+ return d2.__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"));
}
- i3.entityIDs = function(val) {
+ dataHeader.datum = function(val) {
if (!arguments.length)
- return _entityIDs;
- _entityIDs = val;
- return i3;
+ return _datum;
+ _datum = val;
+ return this;
};
- i3.tags = function(tags) {
- _tags = tags;
- const vals = getVals(tags);
- const isMixed = vals.size > 1;
- var val = vals.size === 1 ? [...vals][0] ?? "" : "";
- var shouldUpdate;
- if (field.type === "number" && val) {
- var numbers2 = val.split(";");
- var oriNumbers = utilGetSetValue(input).split(";");
- if (numbers2.length !== oriNumbers.length)
- shouldUpdate = true;
- numbers2 = numbers2.map(function(v2) {
- v2 = v2.trim();
- var num = Number(v2);
- if (!isFinite(num) || v2 === "")
- return v2;
- const fractionDigits = v2.includes(".") ? v2.split(".")[1].length : 0;
- return formatFloat(num, fractionDigits);
+ return dataHeader;
+ }
+
+ // modules/ui/combobox.js
+ var _comboHideTimerID;
+ function uiCombobox(context, klass) {
+ var dispatch14 = 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(d2) {
+ var terms = d2.terms || [];
+ terms.push(d2.value);
+ if (d2.key) {
+ terms.push(d2.key);
+ }
+ return terms.some(function(term) {
+ return term.toString().toLowerCase().indexOf(val.toLowerCase()) !== -1;
});
- val = numbers2.join(";");
- shouldUpdate = (inputValue, setValue) => {
- const inputNums = inputValue.split(";").map(
- (setVal) => likelyRawNumberFormat.test(setVal) ? parseFloat(setVal) : parseLocaleFloat(setVal)
- );
- const setNums = setValue.split(";").map(parseLocaleFloat);
- return !isEqual_default(inputNums, setNums);
- };
+ }));
+ };
+ 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(d2) {
+ return d2 === 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 = +/* @__PURE__ */ 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);
}
- utilGetSetValue(input, val, shouldUpdate).attr("title", isMixed ? [...vals].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);
+ 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 {
- var raw_vals = tags[field.key] || "0";
- const canIncDec = raw_vals.split(";").some((val2) => isFinite(Number(val2)) || isDirectionField && cardinal[val2.trim().toLowerCase()]);
- buttons.attr("disabled", canIncDec ? null : "disabled").classed("disabled", !canIncDec);
+ hide();
}
}
- if (field.type === "tel")
- updatePhonePlaceholder();
- if (field.type === "colour")
- updateColourPreview();
- if (field.type === "date")
- updateDateField();
- if (outlinkButton && !outlinkButton.empty()) {
- var disabled = !validIdentifierValueForLink();
- outlinkButton.classed("disabled", disabled);
- }
- if (!isMixed) {
- _lengthIndicator.update(tags[field.key]);
+ function focus() {
+ fetchComboData("");
}
- };
- i3.focus = function() {
- var node = input.node();
- if (node)
- node.focus();
- };
- function combinedEntityExtent() {
- return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
- }
- return utilRebind(i3, dispatch14, "on");
- }
-
- // modules/ui/fields/access.js
- function uiFieldAccess(field, context) {
- var dispatch14 = 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(d2) {
- return "labeled-input preset-access-" + d2;
- });
- enter.append("span").attr("class", "label preset-label-access").attr("for", function(d2) {
- return "preset-input-access-" + d2;
- }).html(function(d2) {
- return field.t.html("types." + d2);
- });
- enter.append("div").attr("class", "preset-input-access-wrap").append("input").attr("type", "text").attr("class", function(d2) {
- return "preset-input-access preset-input-access-" + d2;
- }).call(utilNoAuto).each(function(d2) {
- select_default2(this).call(
- uiCombobox(context, "access-" + d2).data(access.options(d2))
- );
- });
- items = items.merge(enter);
- wrap2.selectAll(".preset-input-access").on("change", change).on("blur", change);
- }
- function change(d3_event, d2) {
- var tag = {};
- var value = context.cleanTagValue(utilGetSetValue(select_default2(this)));
- if (!value && typeof _tags[d2] !== "string")
- return;
- tag[d2] = value || void 0;
- dispatch14.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((v2) => v2 !== "yes" && v2 !== "designated");
+ function blur() {
+ _comboHideTimerID = window.setTimeout(hide, 75);
}
- if (type2 === "bicycle") {
- options2.splice(options2.length - 4, 0, "dismount");
+ 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);
}
- 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"
+ function hide() {
+ if (_comboHideTimerID) {
+ window.clearTimeout(_comboHideTimerID);
+ _comboHideTimerID = void 0;
}
- },
- 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"
+ 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;
}
}
- };
- access.tags = function(tags) {
- _tags = tags;
- utilGetSetValue(items.selectAll(".preset-input-access"), function(d2) {
- return typeof tags[d2] === "string" ? tags[d2] : "";
- }).classed("mixed", function(d2) {
- return tags[d2] && Array.isArray(tags[d2]);
- }).attr("title", function(d2) {
- return tags[d2] && Array.isArray(tags[d2]) && tags[d2].filter(Boolean).join("\n");
- }).attr("placeholder", function(d2) {
- if (tags[d2] && Array.isArray(tags[d2])) {
- return _t("inspector.multiple_values");
+ function keyup(d3_event) {
+ switch (d3_event.keyCode) {
+ case 27:
+ cancel();
+ break;
}
- if (d2 === "bicycle" || d2 === "motor_vehicle") {
- if (tags.vehicle && typeof tags.vehicle === "string") {
- return tags.vehicle;
+ }
+ function change(doAutoComplete) {
+ if (doAutoComplete === void 0)
+ doAutoComplete = true;
+ fetchComboData(value(), function(skipAutosuggest) {
+ _selected = null;
+ var val = input.property("value");
+ if (_suggestions.length) {
+ if (doAutoComplete && !skipAutosuggest && input.property("selectionEnd") === val.length) {
+ _selected = tryAutocomplete();
+ }
+ if (!_selected) {
+ _selected = val;
+ }
}
- }
- 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]][d2]) {
- return placeholdersByKey[tags[key]][d2];
+ if (val.length) {
+ var combo = container.selectAll(".combobox");
+ if (combo.empty()) {
+ show();
}
} else {
- var impliedAccesses = tags[key].filter(Boolean).map(function(val) {
- return placeholdersByKey[val] && placeholdersByKey[val][d2];
- }).filter(Boolean);
- if (impliedAccesses.length === tags[key].length && new Set(impliedAccesses).size === 1) {
- return impliedAccesses[0];
+ hide();
+ }
+ render();
+ });
+ }
+ function nav(dir) {
+ if (_suggestions.length) {
+ var index = -1;
+ for (var i3 = 0; i3 < _suggestions.length; i3++) {
+ if (_selected && _suggestions[i3].value === _selected) {
+ index = i3;
+ break;
}
}
+ index = Math.max(Math.min(index + dir, _suggestions.length - 1), 0);
+ _selected = _suggestions[index].value;
+ utilGetSetValue(input, _selected);
+ dispatch14.call("update");
}
- for (const key in placeholdersByTag) {
- if (tags[key]) {
- const impliedAccess = getPlaceholdersByTag(key, placeholdersByTag[key]);
- if (impliedAccess) {
- return impliedAccess;
- }
+ 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(v2, cb) {
+ _cancelFetch = false;
+ _fetcher.call(input, v2, function(results, skipAutosuggest) {
+ if (_cancelFetch)
+ return;
+ _suggestions = results;
+ results.forEach(function(d2) {
+ _fetched[d2.value] = d2;
+ });
+ if (cb) {
+ cb(skipAutosuggest);
+ }
+ });
+ }
+ function tryAutocomplete() {
+ if (!_canAutocomplete)
+ return;
+ var val = _caseSensitive ? value() : value().toLowerCase();
+ if (!val)
+ return;
+ if (isFinite(val))
+ return;
+ const suggestionValues = [];
+ _suggestions.forEach((s2) => {
+ suggestionValues.push(s2.value);
+ if (s2.key && s2.key !== s2.value) {
+ suggestionValues.push(s2.key);
+ }
+ });
+ var bestIndex = -1;
+ for (var i3 = 0; i3 < suggestionValues.length; i3++) {
+ var suggestion = suggestionValues[i3];
+ var compare2 = _caseSensitive ? suggestion : suggestion.toLowerCase();
+ if (compare2 === val) {
+ bestIndex = i3;
+ break;
+ } else if (bestIndex === -1 && compare2.indexOf(val) === 0) {
+ bestIndex = i3;
}
}
- if (d2 === "access" && !tags.barrier) {
- return "yes";
+ if (bestIndex !== -1) {
+ var bestVal = suggestionValues[bestIndex];
+ input.property("value", bestVal);
+ input.node().setSelectionRange(val.length, bestVal.length);
+ dispatch14.call("update");
+ return bestVal;
}
- return field.placeholder();
- });
+ }
+ 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(d2) {
+ return d2.value;
+ });
+ options2.exit().remove();
+ options2.enter().append("a").attr("class", function(d2) {
+ return "combobox-option " + (d2.klass || "");
+ }).attr("title", function(d2) {
+ return d2.title;
+ }).each(function(d2) {
+ if (d2.display) {
+ d2.display(select_default2(this));
+ } else {
+ select_default2(this).text(d2.value);
+ }
+ }).on("mouseenter", _mouseEnterHandler).on("mouseleave", _mouseLeaveHandler).merge(options2).classed("selected", function(d2) {
+ return d2.value === _selected || d2.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, d2) {
+ _cancelFetch = true;
+ var thiz = input.node();
+ if (d2) {
+ utilGetSetValue(input, d2.value);
+ utilTriggerEvent(input, "change");
+ }
+ var val = utilGetSetValue(input);
+ thiz.setSelectionRange(val.length, val.length);
+ d2 = _fetched[val];
+ dispatch14.call("accept", thiz, d2, 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);
+ dispatch14.call("cancel", thiz);
+ hide();
+ }
};
- access.focus = function() {
- items.selectAll(".preset-input-access").node().focus();
+ combobox.canAutocomplete = function(val) {
+ if (!arguments.length)
+ return _canAutocomplete;
+ _canAutocomplete = val;
+ return combobox;
};
- return utilRebind(access, dispatch14, "on");
+ 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, dispatch14, "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/fields/address.js
- function uiFieldAddress(field, context) {
- var dispatch14 = 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(d2) {
- _addressFormats = d2;
- if (!_selection.empty()) {
- _selection.call(address);
+ // modules/ui/disclosure.js
+ function uiDisclosure(context, key, expandedDefault) {
+ var dispatch14 = 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";
}
- }).catch(function() {
- });
- function getNear(isAddressable, type2, searchRadius, resultProp) {
- var extent = combinedEntityExtent();
- var l2 = extent.center();
- var box = geoExtent(l2).padByMeters(searchRadius);
- var features = context.history().intersects(box).filter(isAddressable).map((d2) => {
- let dist = geoSphericalDistance(d2.extent(context.graph()).center(), l2);
- if (d2.geometry(context.graph()) === "line") {
- var loc = context.projection([
- (extent[0][0] + extent[1][0]) / 2,
- (extent[0][1] + extent[1][1]) / 2
- ]);
- var choice = geoChooseEdge(context.graph().childNodes(d2), loc, context.projection);
- dist = geoSphericalDistance(choice.loc, l2);
+ 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.".concat(_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);
}
- const value = resultProp && d2.tags[resultProp] ? d2.tags[resultProp] : d2.tags.name;
- let title = value;
- if (type2 === "street") {
- title = `${addrField.t("placeholders.street")}: ${title}`;
- } else if (type2 === "place") {
- title = `${addrField.t("placeholders.place")}: ${title}`;
+ hideToggle.classed("expanded", _expanded).attr("aria-expanded", _expanded).attr("title", _t("icons.".concat(_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);
}
- return {
- title,
- value,
- dist,
- type: type2,
- klass: `address-${type2}`
- };
- }).sort(function(a2, b2) {
- return a2.dist - b2.dist;
- });
- return utilArrayUniqBy(features, "value");
- }
- function getNearStreets() {
- function isAddressable(d2) {
- return d2.tags.highway && d2.tags.name && d2.type === "way";
+ dispatch14.call("toggled", this, _expanded);
}
- return getNear(isAddressable, "street", 200);
- }
- function getNearPlaces() {
- function isAddressable(d2) {
- if (d2.tags.name) {
- if (d2.tags.place)
- return true;
- if (d2.tags.boundary === "administrative" && d2.tags.admin_level > 8)
- return true;
+ };
+ 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, dispatch14, "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;
}
- return false;
}
- return getNear(isAddressable, "place", 200);
- }
- function getNearCities() {
- function isAddressable(d2) {
- if (d2.tags.name) {
- if (d2.tags.boundary === "administrative" && d2.tags.admin_level === "8")
- return true;
- if (d2.tags.border_type === "city")
- return true;
- if (d2.tags.place === "city" || d2.tags.place === "town" || d2.tags.place === "village")
- return true;
+ if (_disclosureContent) {
+ if (!_disclosure) {
+ _disclosure = uiDisclosure(context, id2.replace(/-/g, "_"), _expandedByDefault()).label(_label || "").content(_disclosureContent);
}
- if (d2.tags[`${field.key}:city`])
- return true;
- return false;
- }
- return getNear(isAddressable, "city", 200, `${field.key}:city`);
- }
- function getNearPostcodes() {
- return [...new Set([].concat(getNearValues("postcode")).concat(getNear((d2) => d2.tags.postal_code, "postcode", 200, "postal_code")))];
- }
- function getNearValues(key) {
- const tagKey = `${field.key}:${key}`;
- function hasTag(d2) {
- return _entityIDs.indexOf(d2.id) === -1 && d2.tags[tagKey];
+ if (_disclosureExpanded !== void 0) {
+ _disclosure.expanded(_disclosureExpanded);
+ _disclosureExpanded = void 0;
+ }
+ selection2.call(_disclosure);
+ return;
+ }
+ if (_content) {
+ selection2.call(_content);
}
- return getNear(hasTag, key, 200, tagKey);
}
- function updateForCountryCode() {
- if (!_countryCode)
+ 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;
- var addressFormat;
- for (var i3 = 0; i3 < _addressFormats.length; i3++) {
- var format2 = _addressFormats[i3];
- 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",
- "street+place",
- "subdistrict",
- "suburb"
- ];
- var widths = addressFormat.widths || {
- housenumber: 1 / 5,
- unit: 1 / 5,
- street: 1 / 2,
- place: 1 / 2,
- city: 2 / 3,
- state: 1 / 4,
- postcode: 1 / 3
- };
- function row(r2) {
- var total = r2.reduce(function(sum, key) {
- return sum + (widths[key] || 0.5);
- }, 0);
- return r2.map(function(key) {
- return {
- id: key,
- width: (widths[key] || 0.5) / total
- };
+ 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 rows = _wrap.selectAll(".addr-row").data(addressFormat.format, function(d2) {
- return d2.toString();
+ 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");
+ }
});
- 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(d2) {
- return "addr-" + d2.id;
- }).call(utilNoAuto).each(addDropdown).style("width", function(d2) {
- return d2.width * 100 + "%";
+ }
+ function hide() {
+ _body.transition().duration(200).style("max-height", "0px").style("opacity", "0").on("end", function() {
+ _body.classed("expanded", false);
});
- function addDropdown(d2) {
- if (dropdowns.indexOf(d2.id) === -1)
- return;
- var nearValues;
- switch (d2.id) {
- case "street":
- nearValues = getNearStreets;
- break;
- case "place":
- nearValues = getNearPlaces;
- break;
- case "street+place":
- nearValues = () => [].concat(getNearStreets()).concat(getNearPlaces());
- d2.isAutoStreetPlace = true;
- d2.id = _tags[`${field.key}:place`] ? "place" : "street";
- break;
- case "city":
- nearValues = getNearCities;
- break;
- case "postcode":
- nearValues = getNearPostcodes;
- break;
- default:
- nearValues = getNearValues;
+ _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");
}
- select_default2(this).call(
- uiCombobox(context, `address-${d2.isAutoStreetPlace ? "street-place" : d2.id}`).minItems(1).caseSensitive(true).fetcher(function(typedValue, callback) {
- typedValue = typedValue.toLowerCase();
- callback(nearValues(d2.id).filter((v2) => v2.value.toLowerCase().indexOf(typedValue) !== -1));
- }).on("accept", function(selected) {
- if (d2.isAutoStreetPlace) {
- d2.id = selected ? selected.type : "street";
- }
- })
- );
- }
- _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");
+ 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 {
- var center = extent.center();
- countryCode = iso1A2Code(center);
- }
- if (countryCode) {
- _countryCode = countryCode.toLowerCase();
- updateForCountryCode();
+ load();
}
+ });
+ };
+ tagReference.body = function(selection2) {
+ var itemID = what.qid || what.key + "-" + (what.value || "");
+ _body = selection2.selectAll(".tag-reference-body").data([itemID], function(d2) {
+ return d2;
+ });
+ _body.exit().remove();
+ _body = _body.enter().append("div").attr("class", "tag-reference-body").style("max-height", "0").style("opacity", "0").merge(_body);
+ if (_showing === false) {
+ hide();
}
- }
- function change(onInput) {
- return function() {
- setTimeout(() => {
- 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;
- if (subfield.isAutoStreetPlace) {
- if (subfield.id === "street") {
- tags[`${field.key}:place`] = void 0;
- } else if (subfield.id === "place") {
- tags[`${field.key}:street`] = void 0;
- }
- }
- tags[key] = value || void 0;
- });
- Object.keys(tags).filter((k2) => tags[k2]).forEach((k2) => _tags[k2] = tags[k2]);
- dispatch14.call("change", this, tags, onInput);
- }, 0);
+ };
+ 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 text = 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(text.trim())
};
+ });
+ function show() {
+ updatePosition();
+ _body.classed("hide", false).style("opacity", "0").transition().duration(200).style("opacity", "1");
}
- function updatePlaceholder(inputSelection) {
- return inputSelection.attr("placeholder", function(subfield) {
- if (_tags && Array.isArray(_tags[field.key + ":" + subfield.id])) {
- return _t("inspector.multiple_values");
- }
- if (subfield.isAutoStreetPlace) {
- return `${getLocalPlaceholder("street")} / ${getLocalPlaceholder("place")}`;
- }
- return getLocalPlaceholder(subfield.id);
+ function hide() {
+ _body.classed("hide", true).transition().duration(200).style("opacity", "0").on("end", function() {
+ _body.classed("hide", true);
});
}
- function getLocalPlaceholder(key) {
- if (_countryCode) {
- var localkey = key + "!" + _countryCode;
- var tkey = addrField.hasTextForStringId("placeholders." + localkey) ? localkey : key;
- return addrField.t("placeholders." + tkey);
+ function clickHelp(index) {
+ var d2 = docs[index];
+ var tkeys = fieldHelpKeys[fieldName][index][1];
+ _body.selectAll(".field-help-nav-item").classed("active", function(d4, i3) {
+ return i3 === index;
+ });
+ var content = _body.selectAll(".field-help-content").html(d2.html);
+ content.selectAll("p").attr("class", function(d4, i3) {
+ return tkeys[i3];
+ });
+ if (d2.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 (d2.key === "help.field.restrictions.modifying") {
+ content.insert("img", "p.allow_turn").attr("class", "field-help-image cf").attr("src", context.imagePath("tr_modify.gif"));
}
}
- function updateTags(tags) {
- utilGetSetValue(_wrap.selectAll("input"), (subfield) => {
- var val;
- if (subfield.isAutoStreetPlace) {
- const streetKey = `${field.key}:street`;
- const placeKey = `${field.key}:place`;
- if (tags[streetKey] !== void 0 || tags[placeKey] === void 0) {
- val = tags[streetKey];
- subfield.id = "street";
- } else {
- val = tags[placeKey];
- subfield.id = "place";
- }
+ 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 {
- val = tags[`${field.key}:${subfield.id}`];
+ hide();
}
- 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();
+ 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(d2) {
+ return d2.title;
+ });
+ navEnter.selectAll(".field-help-nav-item").data(titles).enter().append("div").attr("class", "field-help-nav-item").html(function(d2) {
+ return d2;
+ }).on("click", function(d3_event, d2) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ clickHelp(titles.indexOf(d2));
+ });
+ enter.append("div").attr("class", "field-help-content");
+ _body = _body.merge(enter);
+ clickHelp(0);
};
- return utilRebind(address, dispatch14, "on");
+ return fieldHelp;
}
- // modules/ui/fields/directional_combo.js
- function uiFieldDirectionalCombo(field, context) {
+ // modules/ui/fields/check.js
+ function uiFieldCheck(field, context) {
var dispatch14 = dispatch_default("change");
- var items = select_default2(null);
- var wrap2 = select_default2(null);
+ var options2 = field.options;
+ var values = [];
+ var texts = [];
var _tags;
- var _combos = {};
- if (field.type === "cycleway") {
- field = {
- ...field,
- key: field.keys[0],
- keys: field.keys.slice(1)
- };
+ var input = select_default2(null);
+ var text = 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;
}
- function directionalCombo(selection2) {
- function stripcolon(s2) {
- return s2.replace(":", "");
+ if (options2) {
+ for (var i3 in options2) {
+ var v2 = options2[i3];
+ values.push(v2 === "undefined" ? void 0 : v2);
+ texts.push(stringsField.t.html("options." + v2, { "default": v2 }));
+ }
+ } 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"));
}
- 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);
- items = div.selectAll("li").data(field.keys);
- var enter = items.enter().append("li").attr("class", function(d2) {
- return "labeled-input preset-directionalcombo-" + stripcolon(d2);
- });
- enter.append("span").attr("class", "label preset-label-directionalcombo").attr("for", function(d2) {
- return "preset-input-directionalcombo-" + stripcolon(d2);
- }).html(function(d2) {
- return field.t.html("types." + d2);
- });
- 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", (t2) => change(key, t2[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.key;
- const otherKey = key === field.keys[0] ? field.keys[1] : field.keys[0];
- dispatch14.call("change", this, (tags) => {
- const otherValue = tags[otherKey] || tags[commonKey];
- if (newValue === otherValue) {
- tags[commonKey] = newValue;
- delete tags[key];
- delete tags[otherKey];
+ 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");
+ text = label.selectAll("span.value");
+ input.on("click", function(d3_event) {
+ d3_event.stopPropagation();
+ var t2 = {};
+ if (Array.isArray(_tags[field.key])) {
+ if (values.indexOf("yes") !== -1) {
+ t2[field.key] = "yes";
+ } else {
+ t2[field.key] = values[0];
+ }
} else {
- tags[key] = newValue;
- delete tags[commonKey];
- tags[otherKey] = otherValue;
+ t2[field.key] = values[(values.indexOf(_value) + 1) % values.length];
}
- return tags;
+ if (t2[field.key] === "reversible" || t2[field.key] === "alternating") {
+ t2[field.key] = values[0];
+ }
+ dispatch14.call("change", this, t2);
});
- }
- directionalCombo.tags = function(tags) {
+ 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 i4 in _entityIDs) {
+ graph = actionReverse(_entityIDs[i4])(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;
- const commonKey = field.key;
- 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] });
+ 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));
+ 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);
}
};
- directionalCombo.focus = function() {
- var node = wrap2.selectAll("input").node();
- if (node)
- node.focus();
+ check.focus = function() {
+ input.node().focus();
};
- return utilRebind(directionalCombo, dispatch14, "on");
+ return utilRebind(check, dispatch14, "on");
}
- // modules/ui/fields/lanes.js
- function uiFieldLanes(field, context) {
- var dispatch14 = 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);
+ // 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", (d2) => d2 > maxChars).style("border-right-width", (d2) => "".concat(Math.abs(maxChars - d2) * 2, "px")).style("margin-right", (d2) => d2 > maxChars ? "".concat((maxChars - d2) * 2, "px") : 0).style("opacity", (d2) => d2 > maxChars * 0.8 ? Math.min(1, (d2 / maxChars - 0.8) / (1 - 0.8)) : 0).style("pointer-events", (d2) => d2 > maxChars * 0.8 ? null : "none");
+ if (_silent)
return;
+ if (strLen > maxChars) {
+ _tooltip.show();
+ } else {
+ _tooltip.hide();
}
- 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 d2 = utilGetDimensions(wrap2);
- var freeSpace = d2[0] - lanesData.lanes.length * LANE_WIDTH * 1.5 + LANE_WIDTH * 0.5;
- surface = surface.enter().append("svg").attr("width", d2[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(d4) {
- return "translate(" + LANE_WIDTH * d4.index * 1.5 + ", 0)";
- });
- lane.select(".forward").style("visibility", function(d4) {
- return d4.direction === "forward" ? "visible" : "hidden";
- });
- lane.select(".bothways").style("visibility", function(d4) {
- return d4.direction === "bothways" ? "visible" : "hidden";
- });
- lane.select(".backward").style("visibility", function(d4) {
- return d4.direction === "backward" ? "visible" : "hidden";
- });
- }
- lanes.entityIDs = function(val) {
- _entityIDs = val;
};
- lanes.tags = function() {
- };
- lanes.focus = function() {
- };
- lanes.off = function() {
+ lengthIndicator.silent = function(val) {
+ if (!arguments.length)
+ return _silent;
+ _silent = val;
+ return lengthIndicator;
};
- return utilRebind(lanes, dispatch14, "on");
+ return lengthIndicator;
}
- uiFieldLanes.supportsMultiselection = false;
- // modules/ui/fields/localized.js
- var _languagesArray = [];
- function uiFieldLocalized(field, context) {
- var dispatch14 = dispatch_default("change", "input");
- var wikipedia = services.wikipedia;
- var input = select_default2(null);
- var localizedInputs = select_default2(null);
+ // modules/ui/fields/combo.js
+ function uiFieldCombo(field, context) {
+ var dispatch14 = 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 _countryCode;
+ var _comboData = [];
+ var _multiData = [];
+ var _entityIDs = [];
var _tags;
- _mainFileFetcher.get("languages").then(loadLanguagesArray).catch(function() {
- });
- var _territoryLanguages = {};
- _mainFileFetcher.get("territory_languages").then(function(d2) {
- _territoryLanguages = d2;
+ var _countryCode;
+ var _staticPlaceholder;
+ var _dataDeprecated = [];
+ _mainFileFetcher.get("deprecated").then(function(d2) {
+ _dataDeprecated = d2;
}).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)
- });
+ if (_isMulti && field.key && /[^:]$/.test(field.key)) {
+ field.key += ":";
+ }
+ function snake(s2) {
+ return s2.replace(/\s+/g, "_");
+ }
+ function clean2(s2) {
+ return s2.split(";").map(function(s3) {
+ return s3.trim();
+ }).join(";");
+ }
+ function tagValue(dval) {
+ dval = clean2(dval || "");
+ var found = getOptions(true).find(function(o2) {
+ return o2.key && clean2(o2.value) === dval;
+ });
+ if (found)
+ return found.key;
+ if (field.type === "typeCombo" && !dval) {
+ return "yes";
}
+ return restrictTagValueSpelling(dval) || void 0;
}
- 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(entity.extent(context.graph()).center());
- var showsBrandField = fields.some(function(d2) {
- return d2.id === "brand";
- });
- var showsOperatorField = fields.some(function(d2) {
- return d2.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;
+ function restrictTagValueSpelling(dval) {
+ if (_snake_case) {
+ dval = snake(dval);
+ }
+ if (!field.caseSensitive) {
+ dval = dval.toLowerCase();
+ }
+ return dval;
+ }
+ function getLabelId(field2, v2) {
+ return field2.hasTextForStringId("options.".concat(v2, ".title")) ? "options.".concat(v2, ".title") : "options.".concat(v2);
+ }
+ 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(a2, b2) {
+ return a2.filter(function(d1) {
+ return !b2.some(function(d2) {
+ return d1.value === d2.value;
+ });
});
- field.locked(isLocked);
}
- function calcMultilingual(tags) {
- var existingLangsOrdered = _multilingual.map(function(item2) {
- return item2.lang;
+ 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);
+ setTimeout(() => setStaticValues(setPlaceholder), 0);
+ }
+ }
+ function getOptions(allOptions) {
+ var stringsField = field.resolveReference("stringsCrossReference");
+ if (!(field.options || stringsField.options))
+ return [];
+ let options2;
+ if (allOptions !== true) {
+ options2 = field.options || stringsField.options;
+ } else {
+ options2 = [].concat(field.options, stringsField.options).filter(Boolean);
+ }
+ return options2.map(function(v2) {
+ const labelId = getLabelId(stringsField, v2);
+ return {
+ key: v2,
+ value: stringsField.t(labelId, { default: v2 }),
+ title: stringsField.t("options.".concat(v2, ".description"), { default: v2 }),
+ display: addComboboxIcons(stringsField.t.append(labelId, { default: v2 }), v2),
+ klass: stringsField.hasTextForStringId(labelId) ? "" : "raw-option"
+ };
});
- var existingLangs = new Set(existingLangsOrdered.filter(Boolean));
- for (var k2 in tags) {
- var m2 = k2.match(/^(.*):([a-z]{2,3}(?:-[A-Z][a-z]{3})?(?:-[A-Z]{2})?)$/);
- if (m2 && m2[1] === field.key && m2[2]) {
- var item = { lang: m2[2], value: tags[k2] };
- if (existingLangs.has(item.lang)) {
- _multilingual[existingLangsOrdered.indexOf(item.lang)].value = item.value;
- existingLangs.delete(item.lang);
- } else {
- _multilingual.push(item);
+ }
+ function hasStaticValues() {
+ return getOptions().length > 0;
+ }
+ 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(q2, callback) {
+ var queryFilter = (d2) => d2.value.toLowerCase().includes(q2.toLowerCase()) || d2.key.toLowerCase().includes(q2.toLowerCase());
+ if (hasStaticValues()) {
+ setStaticValues(callback, queryFilter);
+ }
+ var stringsField = field.resolveReference("stringsCrossReference");
+ var fn = _isMulti ? "multikeys" : "values";
+ var query = (_isMulti ? field.key : "") + q2;
+ var hasCountryPrefix = _isNetwork && _countryCode && _countryCode.indexOf(q2.toLowerCase()) === 0;
+ if (hasCountryPrefix) {
+ query = _countryCode + ":";
+ }
+ var params = {
+ debounce: q2 !== "",
+ 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((d2) => field.type !== "typeCombo" || d2.value !== "yes");
+ data = data.filter((d2) => {
+ var value = d2.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((d2) => !deprecatedValues.includes(d2.value));
}
- }
- _multilingual.forEach(function(item2) {
- if (item2.lang && existingLangs.has(item2.lang)) {
- item2.value = "";
+ if (hasCountryPrefix) {
+ data = data.filter((d2) => d2.value.toLowerCase().indexOf(_countryCode + ":") === 0);
}
+ const additionalOptions = (field.options || stringsField.options || []).filter((v2) => !data.some((dv) => dv.value === (_isMulti ? field.key + v2 : v2))).map((v2) => ({ value: v2 }));
+ _container.classed("empty-combobox", data.length === 0);
+ _comboData = data.concat(additionalOptions).map(function(d2) {
+ var v2 = d2.value;
+ if (_isMulti)
+ v2 = v2.replace(field.key, "");
+ const labelId = getLabelId(stringsField, v2);
+ var isLocalizable = stringsField.hasTextForStringId(labelId);
+ var label = stringsField.t(labelId, { default: v2 });
+ return {
+ key: v2,
+ value: label,
+ title: stringsField.t("options.".concat(v2, ".description"), { default: isLocalizable ? v2 : d2.title !== label ? d2.title : "" }),
+ display: addComboboxIcons(stringsField.t.append(labelId, { default: v2 }), v2),
+ klass: isLocalizable ? "" : "raw-option"
+ };
+ });
+ _comboData = _comboData.filter(queryFilter);
+ _comboData = objectDifference(_comboData, _multiData);
+ if (callback)
+ callback(_comboData, hasStaticValues());
});
}
- 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);
+ function addComboboxIcons(disp, value) {
+ const iconsField = field.resolveReference("iconsCrossReference");
+ if (iconsField.icons) {
+ return function(selection2) {
+ var span = selection2.insert("span", ":first-child").attr("class", "tag-value-icon");
+ if (iconsField.icons[value]) {
+ span.call(svgIcon("#".concat(iconsField.icons[value])));
+ }
+ disp.call(this, selection2);
+ };
}
- 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;
+ return disp;
+ }
+ function setPlaceholder(values) {
+ if (_isMulti || _isSemi) {
+ _staticPlaceholder = field.placeholder() || _t("inspector.add");
+ } else {
+ var vals = values.map(function(d2) {
+ return d2.value;
+ }).filter(function(s2) {
+ return s2.length < 20;
});
- var isLangEn = defaultLang.indexOf("en") > -1;
- if (isLangEn || langExists) {
- defaultLang = "";
- langExists = _multilingual.find(function(datum2) {
- return datum2.lang === defaultLang;
+ var placeholders = vals.length > 1 ? vals : values.map(function(d2) {
+ return d2.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);
+ var hideAdd = !_allowCustomValues && !values.length;
+ _container.selectAll(".chiplist .input-wrap").style("display", hideAdd ? "none" : null);
+ }
+ function change() {
+ var t2 = {};
+ 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(v2) {
+ var key = (field.key || "") + v2;
+ if (_tags) {
+ var old = _tags[key];
+ if (typeof old === "string" && old.toLowerCase() !== "no")
+ return;
+ }
+ key = context.cleanTagKey(key);
+ field.keys.push(key);
+ t2[key] = "yes";
});
+ } else if (_isSemi) {
+ var arr = _multiData.map(function(d2) {
+ return d2.key;
+ });
+ arr = arr.concat(vals);
+ t2[field.key] = context.cleanTagValue(utilArrayUniq(arr).filter(Boolean).join(";"));
}
- 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 t2 = {};
- t2[field.key] = val || void 0;
- dispatch14.call("change", this, t2, onInput);
- };
+ 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));
+ t2[field.key] = val || void 0;
}
+ dispatch14.call("change", this, t2);
}
- function key(lang) {
- return field.key + ":" + lang;
- }
- function changeLang(d3_event, d2) {
- var tags = {};
- var lang = utilGetSetValue(select_default2(this)).toLowerCase();
- var language = _languagesArray.find(function(d4) {
- return d4.label.toLowerCase() === lang || d4.localName && d4.localName.toLowerCase() === lang || d4.nativeName && d4.nativeName.toLowerCase() === lang;
- });
- if (language)
- lang = language.code;
- if (d2.lang && d2.lang !== lang) {
- tags[key(d2.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[d2.lang]) {
- tags[newKey] = _wikiTitles[d2.lang];
+ function removeMultikey(d3_event, d2) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var t2 = {};
+ if (_isMulti) {
+ t2[d2.key] = void 0;
+ } else if (_isSemi) {
+ var arr = _multiData.map(function(md) {
+ return md.key === d2.key ? null : md.key;
+ }).filter(Boolean);
+ arr = utilArrayUniq(arr);
+ t2[field.key] = arr.length ? arr.join(";") : void 0;
+ _lengthIndicator.update(t2[field.key]);
}
- d2.lang = lang;
- dispatch14.call("change", this, tags);
+ dispatch14.call("change", this, t2);
}
- function changeValue(d3_event, d2) {
- if (!d2.lang)
- return;
- var value = context.cleanTagValue(utilGetSetValue(select_default2(this))) || void 0;
- if (!value && Array.isArray(d2.value))
- return;
+ function invertMultikey(d3_event, d2) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
var t2 = {};
- t2[key(d2.lang)] = value;
- d2.value = value;
+ if (_isMulti) {
+ t2[d2.key] = _tags[d2.key] === "yes" ? "no" : "yes";
+ }
dispatch14.call("change", this, t2);
}
- function fetchLanguages(value, cb) {
- var v2 = value.toLowerCase();
- var langCodes = [_mainLocalizer.localeCode(), _mainLocalizer.languageCode()];
- if (_countryCode && _territoryLanguages[_countryCode]) {
- langCodes = langCodes.concat(_territoryLanguages[_countryCode]);
+ 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);
+ var hideAdd = !_allowCustomValues && !_comboData.length;
+ _inputWrap.style("display", hideAdd ? "none" : null);
+ _input = _inputWrap.selectAll("input").data([0]);
+ } else {
+ _input = _container.selectAll("input").data([0]);
}
- 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(d2) {
- return d2.label.toLowerCase().indexOf(v2) >= 0 || d2.localName && d2.localName.toLowerCase().indexOf(v2) >= 0 || d2.nativeName && d2.nativeName.toLowerCase().indexOf(v2) >= 0 || d2.code.toLowerCase().indexOf(v2) >= 0;
- }).map(function(d2) {
- return { value: d2.label };
- }));
- }
- function renderMultilingual(selection2) {
- var entries = selection2.selectAll("div.entry").data(_multilingual, function(d2) {
- return d2.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(_2, 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, d2) {
- if (field.locked())
- return;
- d3_event.preventDefault();
- _multilingual.splice(_multilingual.indexOf(d2), 1);
- var langKey = d2.lang && key(d2.lang);
- if (langKey && langKey in _tags) {
- delete _tags[langKey];
- var t2 = {};
- t2[langKey] = void 0;
- dispatch14.call("change", this, t2);
- 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);
+ _input = _input.enter().append("input").attr("type", "text").attr("id", field.domId).call(utilNoAuto).call(initCombo, _container).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);
});
- 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");
+ _input.on("keydown.field", function(d3_event) {
+ switch (d3_event.keyCode) {
+ case 13:
+ _input.node().blur();
+ d3_event.stopPropagation();
+ break;
+ }
});
- entries = entries.merge(entriesEnter);
- entries.order();
- entries.classed("present", true);
- utilGetSetValue(entries.select(".localized-lang"), function(d2) {
- var langItem = _languagesArray.find(function(item) {
- return item.code === d2.lang;
+ if (_isMulti || _isSemi) {
+ _combobox.on("accept", function() {
+ _input.node().blur();
+ _input.node().focus();
+ });
+ _input.on("focus", function() {
+ _container.classed("active", true);
});
- if (langItem)
- return langItem.label;
- return d2.lang;
- });
- utilGetSetValue(entries.select(".localized-value"), function(d2) {
- return typeof d2.value === "string" ? d2.value : "";
- }).attr("title", function(d2) {
- return Array.isArray(d2.value) ? d2.value.filter(Boolean).join("\n") : null;
- }).attr("placeholder", function(d2) {
- return Array.isArray(d2.value) ? _t("inspector.multiple_values") : _t("translate.localized_translation_name");
- }).classed("mixed", function(d2) {
- return Array.isArray(d2.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, d2) {
- if (err || !d2)
- return;
- _wikiTitles = d2;
- });
- }
- }
- 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());
+ _combobox.on("cancel", function() {
+ _input.node().blur();
+ }).on("update", function() {
+ updateIcon(utilGetSetValue(_input));
+ });
}
- return utilRebind(localized, dispatch14, "on");
- }
-
- // modules/ui/fields/roadheight.js
- function uiFieldRoadheight(field, context) {
- var dispatch14 = 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 formatFloat = _mainLocalizer.floatFormatter(_mainLocalizer.languageCode());
- var parseLocaleFloat = _mainLocalizer.floatParser(_mainLocalizer.languageCode());
- var primaryUnits = [
- {
- value: "m",
- title: _t("inspector.roadheight.meter")
- },
- {
- value: "ft",
- title: _t("inspector.roadheight.foot")
+ function updateIcon(value) {
+ value = tagValue(value);
+ let container = _container;
+ if (field.type === "multiCombo" || field.type === "semiCombo") {
+ container = _container.select(".input-wrap");
}
- ];
- 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;
+ const iconsField = field.resolveReference("iconsCrossReference");
+ if (iconsField.icons) {
+ container.selectAll(".tag-value-icon").remove();
+ if (iconsField.icons[value]) {
+ container.selectAll(".tag-value-icon").data([value]).enter().insert("div", "input").attr("class", "tag-value-icon").call(svgIcon("#".concat(iconsField.icons[value])));
}
- 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 {
- var rawPrimaryValue = likelyRawNumberFormat.test(primaryValue) ? parseFloat(primaryValue) : parseLocaleFloat(primaryValue);
- if (isNaN(rawPrimaryValue))
- rawPrimaryValue = primaryValue;
- var rawSecondaryValue = likelyRawNumberFormat.test(secondaryValue) ? parseFloat(secondaryValue) : parseLocaleFloat(secondaryValue);
- if (isNaN(rawSecondaryValue))
- rawSecondaryValue = secondaryValue;
- if (isNaN(rawPrimaryValue) || isNaN(rawSecondaryValue) || !_isImperial) {
- tag[field.key] = context.cleanTagValue(rawPrimaryValue);
- } else {
- if (rawPrimaryValue !== "") {
- rawPrimaryValue = rawPrimaryValue + "'";
+ combo.tags = function(tags) {
+ _tags = tags;
+ var stringsField = field.resolveReference("stringsCrossReference");
+ var isMixed = Array.isArray(tags[field.key]);
+ var showsValue = (value) => !isMixed && value && !(field.type === "typeCombo" && value === "yes");
+ var isRawValue = (value) => showsValue(value) && !stringsField.hasTextForStringId("options.".concat(value)) && !stringsField.hasTextForStringId("options.".concat(value, ".title"));
+ var isKnownValue = (value) => showsValue(value) && !isRawValue(value);
+ var isReadOnly = !_allowCustomValues;
+ if (_isMulti || _isSemi) {
+ _multiData = [];
+ var maxLength;
+ if (_isMulti) {
+ for (var k2 in tags) {
+ if (field.key && k2.indexOf(field.key) !== 0)
+ continue;
+ if (!field.key && field.keys.indexOf(k2) === -1)
+ continue;
+ var v2 = tags[k2];
+ var suffix = field.key ? k2.slice(field.key.length) : k2;
+ _multiData.push({
+ key: k2,
+ value: displayValue(suffix),
+ display: addComboboxIcons(renderValue(suffix), suffix),
+ state: typeof v2 === "string" ? v2.toLowerCase() : "",
+ isMixed: Array.isArray(v2)
+ });
}
- if (rawSecondaryValue !== "") {
- rawSecondaryValue = rawSecondaryValue + '"';
+ if (field.key) {
+ field.keys = _multiData.map(function(d2) {
+ return d2.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(v3) {
+ return {
+ key: v3,
+ value: displayValue(v3),
+ display: addComboboxIcons(renderValue(v3), v3),
+ isMixed: !commonValues.includes(v3)
+ };
+ });
+ var currLength = utilUnicodeCharsCount(commonValues.join(";"));
+ maxLength = context.maxCharsForTagValue() - currLength;
+ if (currLength > 0) {
+ maxLength -= 1;
+ }
+ }
+ maxLength = Math.max(0, maxLength);
+ var hideAdd = maxLength <= 0 || !_allowCustomValues && !_comboData.length;
+ _container.selectAll(".chiplist .input-wrap").style("display", hideAdd ? "none" : null);
+ var allowDragAndDrop = _isSemi && !Array.isArray(tags[field.key]);
+ var chips = _container.selectAll(".chip").data(_multiData);
+ chips.exit().remove();
+ var enter = chips.enter().insert("li", ".input-wrap").attr("class", "chip");
+ enter.append("span");
+ const field_buttons = enter.append("div").attr("class", "field_buttons");
+ field_buttons.append("a").attr("class", "remove");
+ chips = chips.merge(enter).order().classed("raw-value", function(d2) {
+ var k3 = d2.key;
+ if (_isMulti)
+ k3 = k3.replace(field.key, "");
+ return !stringsField.hasTextForStringId("options." + k3);
+ }).classed("draggable", allowDragAndDrop).classed("mixed", function(d2) {
+ return d2.isMixed;
+ }).attr("title", function(d2) {
+ if (d2.isMixed) {
+ return _t("inspector.unshared_value_tooltip");
+ }
+ if (!["yes", "no"].includes(d2.state)) {
+ return d2.state;
+ }
+ return null;
+ }).classed("negated", (d2) => d2.state === "no");
+ if (!_isSemi) {
+ chips.selectAll("input[type=checkbox]").remove();
+ chips.insert("input", "span").attr("type", "checkbox").property("checked", (d2) => d2.state === "yes").property("indeterminate", (d2) => d2.isMixed || !["yes", "no"].includes(d2.state)).on("click", invertMultikey);
+ }
+ if (allowDragAndDrop) {
+ registerDragAndDrop(chips);
+ }
+ chips.each(function(d2) {
+ const selection2 = select_default2(this);
+ const text_span = selection2.select("span");
+ const field_buttons2 = selection2.select(".field_buttons");
+ const clean_value = d2.value.trim();
+ text_span.text("");
+ if (clean_value.startsWith("https://")) {
+ text_span.text(clean_value);
+ field_buttons2.select("button").remove();
+ field_buttons2.append("button").call(svgIcon("#iD-icon-out-link")).attr("class", "form-field-button foreign-id-permalink").attr("title", () => _t("icons.visit_website")).attr("aria-label", () => _t("icons.visit_website")).on("click", function(d3_event) {
+ d3_event.preventDefault();
+ window.open(clean_value, "_blank");
+ });
+ return;
+ }
+ if (d2.display) {
+ d2.display(text_span);
+ return;
+ }
+ text_span.text(d2.value);
+ });
+ chips.select("a.remove").attr("href", "#").on("click", removeMultikey).attr("class", "remove").text("\xD7");
+ updateIcon("");
+ } else {
+ var mixedValues = isMixed && tags[field.key].map(function(val) {
+ return displayValue(val);
+ }).filter(Boolean);
+ utilGetSetValue(_input, !isMixed ? displayValue(tags[field.key]) : "").data([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(tags[field.key]) && (d3_event.keyCode === utilKeybinding.keyCodes["\u232B"] || d3_event.keyCode === utilKeybinding.keyCodes["\u2326"])) {
+ d3_event.preventDefault();
+ d3_event.stopPropagation();
+ var t2 = {};
+ t2[field.key] = void 0;
+ dispatch14.call("change", this, t2);
}
- tag[field.key] = context.cleanTagValue(rawPrimaryValue + rawSecondaryValue);
+ });
+ if (!Array.isArray(tags[field.key])) {
+ updateIcon(tags[field.key]);
+ }
+ if (!isMixed) {
+ _lengthIndicator.update(tags[field.key]);
}
}
- dispatch14.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 = formatFloat(parseFloat(secondaryValue[1]));
+ const refreshStyles = () => {
+ _input.data([tagValue(utilGetSetValue(_input))]).classed("raw-value", isRawValue).classed("known-value", isKnownValue);
+ };
+ _input.on("input.refreshStyles", refreshStyles);
+ _combobox.on("update.refreshStyles", refreshStyles);
+ refreshStyles();
+ };
+ 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 x2 = d3_event.x - dragOrigin.x, y2 = 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(x2, 2) + Math.pow(y2, 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(" + x2 + "px, " + y2 + "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(" + x2 + "px, " + y2 + "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;
+ });
}
- primaryValue = primaryValue.match(/(-?[\d.]+)'/);
- if (primaryValue !== null) {
- primaryValue = formatFloat(parseFloat(primaryValue[1]));
+ }).on("end", function() {
+ if (!select_default2(this).classed("dragging")) {
+ return;
}
- _isImperial = true;
- } else if (primaryValue) {
- var rawValue = primaryValue;
- primaryValue = parseFloat(rawValue);
- if (isNaN(primaryValue)) {
- primaryValue = rawValue;
- } else {
- primaryValue = formatFloat(primaryValue);
+ 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 t2 = {};
+ if (_multiData.length) {
+ t2[field.key] = _multiData.map(function(element2) {
+ return element2.key;
+ }).join(";");
+ } else {
+ t2[field.key] = void 0;
+ }
+ dispatch14.call("change", this, t2);
}
- _isImperial = false;
- }
- }
- setUnitSuggestions();
- var inchesPlaceholder = formatFloat(0);
- 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 ? inchesPlaceholder : 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();
+ dragOrigin = void 0;
+ targetIndex = void 0;
+ })
+ );
+ }
+ combo.focus = function() {
+ _input.node().focus();
};
- roadheight.entityIDs = function(val) {
+ 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(roadheight, dispatch14, "on");
+ return utilRebind(combo, dispatch14, "on");
}
- // modules/ui/fields/roadspeed.js
- function uiFieldRoadspeed(field, context) {
+ // modules/ui/fields/input.js
+ var likelyRawNumberFormat = /^-?(0\.\d*|\d*\.\d{0,2}(\d{4,})?|\d{4,}\.\d{3})$/;
+ function uiFieldText(field, context) {
var dispatch14 = dispatch_default("change");
- var unitInput = select_default2(null);
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 _isImperial;
- var formatFloat = _mainLocalizer.floatFormatter(_mainLocalizer.languageCode());
- var parseLocaleFloat = _mainLocalizer.floatParser(_mainLocalizer.languageCode());
- 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();
- }
+ var _phoneFormats = {};
+ const isDirectionField = field.key.split(":").some((keyPart) => keyPart === "direction");
+ const formatFloat = _mainLocalizer.floatFormatter(_mainLocalizer.languageCode());
+ const parseLocaleFloat = _mainLocalizer.floatParser(_mainLocalizer.languageCode());
+ const countDecimalPlaces = _mainLocalizer.decimalPlaceCounter(_mainLocalizer.languageCode());
+ if (field.type === "tel") {
+ _mainFileFetcher.get("phone_formats").then(function(d2) {
+ _phoneFormats = d2;
+ updatePhonePlaceholder();
+ }).catch(function() {
+ });
}
- function setUnitSuggestions() {
- speedCombo.data((_isImperial ? imperialValues : metricValues).map(comboValues));
- utilGetSetValue(unitInput, _isImperial ? "mph" : "km/h");
+ 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 comboValues(d2) {
- return {
- value: formatFloat(d2),
- title: formatFloat(d2)
- };
+ function i3(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(d2) {
+ var which = d2 > 0 ? "increment" : "decrement";
+ return "form-field-button " + which;
+ }).attr("title", function(d2) {
+ var which = d2 > 0 ? "increment" : "decrement";
+ return _t("inspector.".concat(which));
+ }).merge(buttons).on("click", function(d3_event, d2) {
+ 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(v2) {
+ v2 = v2.trim();
+ const isRawNumber = likelyRawNumberFormat.test(v2);
+ var num = isRawNumber ? parseFloat(v2) : parseLocaleFloat(v2);
+ if (isDirectionField) {
+ const compassDir = cardinal[v2.toLowerCase()];
+ if (compassDir !== void 0) {
+ num = compassDir;
+ }
+ }
+ if (!isFinite(num))
+ return v2;
+ num = parseFloat(num);
+ if (!isFinite(num))
+ return v2;
+ num += d2;
+ if (isDirectionField) {
+ num = (num % 360 + 360) % 360;
+ }
+ return formatFloat(clamped(num), isRawNumber ? v2.includes(".") ? v2.split(".")[1].length : 0 : countDecimalPlaces(v2));
+ });
+ 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 = 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 });
+ }
+ return "";
+ }).merge(outlinkButton);
+ outlinkButton.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");
+ }
+ }).classed("disabled", () => !validIdentifierValueForLink()).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();
+ } else if (field.type === "date") {
+ input.attr("type", "text");
+ updateDateField();
+ }
}
- function change() {
- var tag = {};
- var value = utilGetSetValue(input).trim();
- if (!value && Array.isArray(_tags[field.key]))
+ function updateColourPreview() {
+ wrap2.selectAll(".colour-preview").remove();
+ const colour = utilGetSetValue(input);
+ if (!isColourValid(colour) && colour !== "") {
+ wrap2.selectAll("input.colour-selector").remove();
+ wrap2.selectAll(".form-field-button").remove();
return;
- if (!value) {
- tag[field.key] = void 0;
- } else {
- var rawValue = likelyRawNumberFormat.test(value) ? parseFloat(value) : parseLocaleFloat(value);
- if (isNaN(rawValue))
- rawValue = value;
- if (isNaN(rawValue) || !_isImperial) {
- tag[field.key] = context.cleanTagValue(rawValue);
- } else {
- tag[field.key] = context.cleanTagValue(rawValue + " mph");
- }
}
- dispatch14.call("change", this, tag);
- }
- roadspeed.tags = function(tags) {
- _tags = tags;
- var rawValue = tags[field.key];
- var value = rawValue;
- var isMixed = Array.isArray(value);
- if (!isMixed) {
- if (rawValue && rawValue.indexOf("mph") >= 0) {
- _isImperial = true;
- } else if (rawValue) {
- _isImperial = false;
- }
- value = parseInt(value, 10);
- if (isNaN(value)) {
- value = rawValue;
- } else {
- value = formatFloat(value);
- }
+ var colourSelector = wrap2.selectAll(".colour-selector").data([0]);
+ colourSelector.enter().append("input").attr("type", "color").attr("class", "colour-selector").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));
+ wrap2.selectAll("input.colour-selector").attr("value", colour);
+ var chooserButton = wrap2.selectAll(".colour-preview").data([colour]);
+ chooserButton = chooserButton.enter().append("div").attr("class", "form-field-button colour-preview").append("div").style("background-color", (d2) => d2).attr("class", "colour-box");
+ if (colour === "") {
+ chooserButton = chooserButton.call(svgIcon("#iD-icon-edit"));
}
- 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, dispatch14, "on");
- }
-
- // modules/ui/fields/radio.js
- function uiFieldRadio(field, context) {
- var dispatch14 = 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(d2) {
- return stringsField.t("options." + d2, { "default": d2 });
- }).attr("checked", false);
- enter.append("span").each(function(d2) {
- stringsField.t.append("options." + d2, { "default": d2 })(select_default2(this));
- });
- labels = labels.merge(enter);
- radios = labels.selectAll("input").on("change", changeRadio);
+ chooserButton.on("click", () => wrap2.select(".colour-selector").node().showPicker());
}
- 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(d2) {
- return d2.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);
+ function updateDateField() {
+ function isDateValid(date2) {
+ return date2.match(/^[0-9]{4}(-[0-9]{2}(-[0-9]{2})?)?$/);
}
- 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(k2) {
- return k2 !== "layer";
+ const date = utilGetSetValue(input);
+ const now3 = /* @__PURE__ */ new Date();
+ const today = new Date(now3.getTime() - now3.getTimezoneOffset() * 6e4).toISOString().split("T")[0];
+ if ((field.key === "check_date" || field.key === "survey:date") && date !== today) {
+ wrap2.selectAll(".date-set-today").data([0]).enter().append("button").attr("class", "form-field-button date-set-today").call(svgIcon("#fas-rotate")).call(uiTooltip().title(() => _t.append("inspector.set_today"))).on("click", () => {
+ utilGetSetValue(input, today);
+ change()();
+ updateDateField();
});
+ } else {
+ wrap2.selectAll(".date-set-today").remove();
}
- 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);
+ if (!isDateValid(date) && date !== "") {
+ wrap2.selectAll("input.date-selector").remove();
+ wrap2.selectAll(".date-calendar").remove();
+ return;
+ }
+ if (utilDetect().browser !== "Safari") {
+ var dateSelector = wrap2.selectAll(".date-selector").data([0]);
+ dateSelector.enter().append("input").attr("type", "date").attr("class", "date-selector").on("input", debounce_default(function(d3_event) {
+ d3_event.preventDefault();
+ var date2 = this.value;
+ if (!isDateValid(date2))
+ return;
+ utilGetSetValue(input, this.value);
+ change()();
+ updateDateField();
+ }, 100));
+ wrap2.selectAll("input.date-selector").attr("value", date);
+ var calendarButton = wrap2.selectAll(".date-calendar").data([date]);
+ calendarButton = calendarButton.enter().append("button").attr("class", "form-field-button date-calendar").call(svgIcon("#fas-calendar-days"));
+ calendarButton.on("click", () => wrap2.select(".date-selector").node().showPicker());
}
}
- function changeType(t2, onInput) {
- var key = selectedKey();
- if (!key)
+ function updatePhonePlaceholder() {
+ if (input.empty() || !Object.keys(_phoneFormats).length)
return;
- var val = t2[key];
- if (val !== "no") {
- _oldType[key] = val;
- }
- if (field.type === "structureRadio") {
- if (val === "no" || key !== "bridge" && key !== "tunnel" || key === "tunnel" && val === "building_passage") {
- t2.layer = void 0;
- }
- if (t2.layer === void 0) {
- if (key === "bridge" && val !== "no") {
- t2.layer = "1";
- }
- if (key === "tunnel" && val !== "no" && val !== "building_passage") {
- t2.layer = "-1";
- }
+ var extent = combinedEntityExtent();
+ var countryCode = extent && iso1A2Code(extent.center());
+ var format2 = countryCode && _phoneFormats[countryCode.toLowerCase()];
+ if (format2)
+ input.attr("placeholder", format2);
+ }
+ function validIdentifierValueForLink() {
+ var _a2;
+ const value = utilGetSetValue(input).trim();
+ if (field.type === "url" && value) {
+ try {
+ return new URL(value).href;
+ } catch (e3) {
+ return null;
}
}
- dispatch14.call("change", this, t2, onInput);
+ if (field.type === "identifier" && field.pattern) {
+ return value && ((_a2 = value.match(new RegExp(field.pattern))) == null ? void 0 : _a2[0]);
+ }
+ return null;
}
- function changeLayer(t2, onInput) {
- if (t2.layer === "0") {
- t2.layer = void 0;
+ function clamped(num) {
+ if (field.minValue !== void 0) {
+ num = Math.max(num, field.minValue);
}
- dispatch14.call("change", this, t2, onInput);
+ if (field.maxValue !== void 0) {
+ num = Math.min(num, field.maxValue);
+ }
+ return num;
}
- function changeRadio() {
- var t2 = {};
- var activeKey;
- if (field.key) {
- t2[field.key] = void 0;
+ function getVals(tags) {
+ if (field.keys) {
+ const multiSelection = context.selectedIDs();
+ tags = multiSelection.length > 1 ? context.selectedIDs().map((id2) => context.graph().entity(id2)).map((entity) => entity.tags) : [tags];
+ return tags.map((tags2) => new Set(field.keys.reduce((acc, key) => acc.concat(tags2[key]), []).filter(Boolean))).map((vals) => vals.size === 0 ? /* @__PURE__ */ new Set([void 0]) : vals).reduce((a2, b2) => /* @__PURE__ */ new Set([...a2, ...b2]));
+ } else {
+ return new Set([].concat(tags[field.key]));
}
- radios.each(function(d2) {
- var active = select_default2(this).property("checked");
- if (active)
- activeKey = d2;
- if (field.key) {
- if (active)
- t2[field.key] = d2;
- } else {
- var val = _oldType[activeKey] || "yes";
- t2[d2] = active ? val : void 0;
+ }
+ function change(onInput) {
+ return function() {
+ var t2 = {};
+ var val = utilGetSetValue(input);
+ if (!onInput)
+ val = context.cleanTagValue(val);
+ if (!val && getVals(_tags).size > 1)
+ return;
+ var displayVal = val;
+ if (field.type === "number" && val) {
+ var numbers2 = val.split(";");
+ numbers2 = numbers2.map(function(v2) {
+ if (likelyRawNumberFormat.test(v2)) {
+ return v2;
+ }
+ var num = parseLocaleFloat(v2);
+ const fractionDigits = countDecimalPlaces(v2);
+ return isFinite(num) ? clamped(num).toFixed(fractionDigits) : v2;
+ });
+ val = numbers2.join(";");
}
- });
- if (field.type === "structureRadio") {
- if (activeKey === "bridge") {
- t2.layer = "1";
- } else if (activeKey === "tunnel" && t2.tunnel !== "building_passage") {
- t2.layer = "-1";
+ if (!onInput)
+ utilGetSetValue(input, displayVal);
+ t2[field.key] = val || void 0;
+ if (field.keys) {
+ dispatch14.call("change", this, (tags) => {
+ if (field.keys.some((key) => tags[key])) {
+ field.keys.filter((key) => tags[key]).forEach((key) => {
+ tags[key] = val || void 0;
+ });
+ } else {
+ tags[field.key] = val || void 0;
+ }
+ return tags;
+ }, onInput);
} else {
- t2.layer = void 0;
+ dispatch14.call("change", this, t2, onInput);
}
- }
- dispatch14.call("change", this, t2);
+ };
}
- radio.tags = function(tags) {
- function isOptionChecked(d2) {
- if (field.key) {
- return tags[field.key] === d2;
- }
- return !!(typeof tags[d2] === "string" && tags[d2].toLowerCase() !== "no");
+ i3.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return i3;
+ };
+ i3.tags = function(tags) {
+ var _a2;
+ _tags = tags;
+ const vals = getVals(tags);
+ const isMixed = vals.size > 1;
+ var val = vals.size === 1 ? (_a2 = [...vals][0]) != null ? _a2 : "" : "";
+ var shouldUpdate;
+ if (field.type === "number" && val) {
+ var numbers2 = val.split(";");
+ var oriNumbers = utilGetSetValue(input).split(";");
+ if (numbers2.length !== oriNumbers.length)
+ shouldUpdate = true;
+ numbers2 = numbers2.map(function(v2) {
+ v2 = v2.trim();
+ var num = Number(v2);
+ if (!isFinite(num) || v2 === "")
+ return v2;
+ const fractionDigits = v2.includes(".") ? v2.split(".")[1].length : 0;
+ return formatFloat(num, fractionDigits);
+ });
+ val = numbers2.join(";");
+ shouldUpdate = (inputValue, setValue) => {
+ const inputNums = inputValue.split(";").map(
+ (setVal) => likelyRawNumberFormat.test(setVal) ? parseFloat(setVal) : parseLocaleFloat(setVal)
+ );
+ const setNums = setValue.split(";").map(parseLocaleFloat);
+ return !isEqual_default(inputNums, setNums);
+ };
}
- function isMixed(d2) {
- if (field.key) {
- return Array.isArray(tags[field.key]) && tags[field.key].includes(d2);
+ utilGetSetValue(input, val, shouldUpdate).attr("title", isMixed ? [...vals].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((val2) => isFinite(Number(val2)) || isDirectionField && cardinal[val2.trim().toLowerCase()]);
+ buttons.attr("disabled", canIncDec ? null : "disabled").classed("disabled", !canIncDec);
}
- return Array.isArray(tags[d2]);
}
- radios.property("checked", function(d2) {
- return isOptionChecked(d2) && (field.key || field.options.filter(isOptionChecked).length === 1);
- });
- labels.classed("active", function(d2) {
- if (field.key) {
- return Array.isArray(tags[field.key]) && tags[field.key].includes(d2) || tags[field.key] === d2;
- }
- return Array.isArray(tags[d2]) && tags[d2].some((v2) => typeof v2 === "string" && v2.toLowerCase() !== "no") || !!(typeof tags[d2] === "string" && tags[d2].toLowerCase() !== "no");
- }).classed("mixed", isMixed).attr("title", function(d2) {
- return isMixed(d2) ? _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 === "tel")
+ updatePhonePlaceholder();
+ if (field.type === "colour")
+ updateColourPreview();
+ if (field.type === "date")
+ updateDateField();
+ if (outlinkButton && !outlinkButton.empty()) {
+ var disabled = !validIdentifierValueForLink();
+ outlinkButton.classed("disabled", disabled);
}
- if (field.type === "structureRadio") {
- if (!!tags.waterway && !_oldType.tunnel) {
- _oldType.tunnel = "culvert";
- }
- wrap2.call(structureExtras, tags);
+ if (!isMixed) {
+ _lengthIndicator.update(tags[field.key]);
}
};
- 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;
+ i3.focus = function() {
+ var node = input.node();
+ if (node)
+ node.focus();
};
- return utilRebind(radio, dispatch14, "on");
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ return utilRebind(i3, dispatch14, "on");
}
- // modules/ui/fields/restrictions.js
- function uiFieldRestrictions(field, context) {
+ // modules/ui/fields/access.js
+ function uiFieldAccess(field, context) {
var dispatch14 = 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 _initialized3 = 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 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 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);
+ var list2 = wrap2.selectAll("ul").data([0]);
+ list2 = list2.enter().append("ul").attr("class", "rows").merge(list2);
+ items = list2.selectAll("li").data(field.keys);
+ var enter = items.enter().append("li").attr("class", function(d2) {
+ return "labeled-input preset-access-" + d2;
});
- 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);
+ enter.append("div").attr("class", "label preset-label-access").attr("for", function(d2) {
+ return "preset-input-access-" + d2;
+ }).html(function(d2) {
+ return field.t.html("types." + d2);
});
- selection2.selectAll(".restriction-via-way-text").call(displayMaxVia(_maxViaWay));
+ enter.append("div").attr("class", "preset-input-access-wrap").append("input").attr("type", "text").attr("class", function(d2) {
+ return "preset-input-access preset-input-access-" + d2;
+ }).call(utilNoAuto).each(function(d2) {
+ select_default2(this).call(
+ uiCombobox(context, "access-" + d2).data(access.options(d2))
+ );
+ });
+ items = items.merge(enter);
+ wrap2.selectAll(".preset-input-access").on("change", change).on("blur", change);
}
- function renderViewer(selection2) {
- if (!_intersection)
+ function change(d3_event, d2) {
+ var tag2 = {};
+ var value = context.cleanTagValue(utilGetSetValue(select_default2(this)));
+ if (!value && typeof _tags[d2] !== "string")
return;
- var vgraph = _intersection.graph;
- var filter2 = utilFunctor(true);
- var projection2 = geoRawMercator();
- var sdims = utilGetDimensions(context.container().select(".sidebar"));
- var d2 = [sdims[0] - 50, 370];
- var c2 = geoVecScale(d2, 0.5);
- var z2 = 22;
- projection2.scale(geoZoomToScale(z2));
- var extent = geoExtent();
- for (var i3 = 0; i3 < _intersection.vertices.length; i3++) {
- extent._extend(_intersection.vertices[i3].extent());
- }
- var padTop = 35;
- if (_intersection.vertices.length > 1) {
- var hPadding = Math.min(160, Math.max(110, d2[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]) / (d2[0] - hPadding);
- var vFactor = (br[1] - tl[1]) / (d2[1] - vPadding - padTop);
- var hZoomDiff = Math.log(Math.abs(hFactor)) / Math.LN2;
- var vZoomDiff = Math.log(Math.abs(vFactor)) / Math.LN2;
- z2 = z2 - Math.max(hZoomDiff, vZoomDiff);
- projection2.scale(geoZoomToScale(z2));
- }
- var extentCenter = projection2(extent.center());
- extentCenter[1] = extentCenter[1] - padTop / 2;
- projection2.translate(geoVecSubtract(c2, extentCenter)).clipExtent([[0, 0], d2]);
- var drawLayers = svgLayers(projection2, context).only(["osm", "touch"]).dimensions(d2);
- 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) {
- _initialized3 = true;
- surface.call(breathe);
- }
- if (_fromWayID && !vgraph.hasEntity(_fromWayID)) {
- _fromWayID = null;
- _oldTurns = null;
- }
- surface.call(utilSetDimensions, d2).call(drawVertices, vgraph, _intersection.vertices, filter2, extent, z2).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, i4;
- 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 (i4 = 0; i4 < turns.length; i4++) {
- var turn = turns[i4];
- if (seen[turn.restrictionID])
- continue;
- if (turn.direct && turn.path[1] === datum2.path[1]) {
- seen[turns[i4].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 (i4 = 0; i4 < turns.length; i4++) {
- if (turns[i4].key !== datum2.key) {
- extraActions.push(actionRestrictTurn(turns[i4], turns[i4].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 s2 = surface.selectAll("." + datum2.key);
- datum2 = s2.empty() ? null : s2.datum();
- updateHints(datum2);
- } else {
- _fromWayID = null;
- _oldTurns = null;
- redraw();
- }
+ tag2[d2] = value || void 0;
+ dispatch14.call("change", this, tag2);
+ }
+ access.options = function(type2) {
+ var options2 = [
+ "yes",
+ "no",
+ "designated",
+ "permissive",
+ "destination",
+ "customers",
+ "private",
+ "permit",
+ "unknown"
+ ];
+ if (type2 === "access") {
+ options2 = options2.filter((v2) => v2 !== "yes" && v2 !== "designated");
}
- function mouseover(d3_event) {
- var datum2 = d3_event.target.__data__;
- updateHints(datum2);
+ if (type2 === "bicycle") {
+ options2.splice(options2.length - 4, 0, "dismount");
}
- _lastXPos = _lastXPos || sdims[0];
- function redraw(minChange) {
- var xPos = -1;
- if (minChange) {
- xPos = utilGetDimensions(context.container().select(".sidebar"))[0];
+ 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"
+ },
+ busway: {
+ access: "no",
+ bus: "designated",
+ emergency: "yes"
}
- if (!minChange || minChange && Math.abs(xPos - _lastXPos) >= minChange) {
- if (context.hasEntity(_vertexID)) {
- _lastXPos = xPos;
- _container.call(renderViewer);
- }
+ },
+ 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"
}
}
- 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 i4 = 0; i4 < turns.length; i4++) {
- var turn = turns[i4];
- 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");
- }
+ };
+ access.tags = function(tags) {
+ _tags = tags;
+ utilGetSetValue(items.selectAll(".preset-input-access"), function(d2) {
+ return typeof tags[d2] === "string" ? tags[d2] : "";
+ }).classed("mixed", function(d2) {
+ return tags[d2] && Array.isArray(tags[d2]);
+ }).attr("title", function(d2) {
+ return tags[d2] && Array.isArray(tags[d2]) && tags[d2].filter(Boolean).join("\n");
+ }).attr("placeholder", function(d2) {
+ if (tags[d2] && Array.isArray(tags[d2])) {
+ return _t("inspector.multiple_values");
}
- }
- function updateHints(datum2) {
- var help = _container.selectAll(".restriction-help").html("");
- var placeholders = {};
- ["from", "via", "to"].forEach(function(k2) {
- placeholders[k2] = { html: '' + _t("restriction.help." + k2) + "" };
- });
- var entity = datum2 && datum2.properties && datum2.properties.entity;
- if (entity) {
- datum2 = entity;
+ if (d2 === "bicycle" || d2 === "motor_vehicle") {
+ if (tags.vehicle && typeof tags.vehicle === "string") {
+ return tags.vehicle;
+ }
}
- if (_fromWayID) {
- way = vgraph.entity(_fromWayID);
- surface.selectAll("." + _fromWayID).classed("selected", true).classed("related", true);
+ if (tags.access && typeof tags.access === "string") {
+ return tags.access;
}
- 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: "" });
+ function getPlaceholdersByTag(key, placeholdersByKey) {
+ if (typeof tags[key] === "string") {
+ if (placeholdersByKey[tags[key]] && placeholdersByKey[tags[key]][d2]) {
+ return placeholdersByKey[tags[key]][d2];
+ }
} 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 i4 = 0; i4 < datum2.via.ways.length; i4++) {
- var prev = names[names.length - 1];
- var curr = displayName(datum2.via.ways[i4], vgraph);
- if (!prev || curr !== prev) {
- names.push(curr);
- }
+ var impliedAccesses = tags[key].filter(Boolean).map(function(val) {
+ return placeholdersByKey[val] && placeholdersByKey[val][d2];
+ }).filter(Boolean);
+ if (impliedAccesses.length === tags[key].length && new Set(impliedAccesses).size === 1) {
+ return impliedAccesses[0];
}
- 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
- }));
+ }
+ for (const key in placeholdersByTag) {
+ if (tags[key]) {
+ const impliedAccess = getPlaceholdersByTag(key, placeholdersByTag[key]);
+ if (impliedAccess) {
+ return impliedAccess;
+ }
}
}
- }
- }
- 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 }) };
+ if (d2 === "access" && !tags.barrier) {
+ return "yes";
}
- 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 (!_initialized3)
- 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, dispatch14, "on");
- }
- uiFieldRestrictions.supportsMultiselection = false;
-
- // modules/ui/fields/textarea.js
- function uiFieldTextarea(field, context) {
- var dispatch14 = 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 t2 = {};
- t2[field.key] = val || void 0;
- dispatch14.call("change", this, t2, 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]);
- }
+ return field.placeholder();
+ });
};
- textarea.focus = function() {
- input.node().focus();
+ access.focus = function() {
+ items.selectAll(".preset-input-access").node().focus();
};
- return utilRebind(textarea, dispatch14, "on");
+ return utilRebind(access, dispatch14, "on");
}
- // modules/ui/fields/wikidata.js
- function uiFieldWikidata(field, context) {
- var wikidata = services.wikidata;
+ // modules/ui/fields/address.js
+ function uiFieldAddress(field, context) {
var dispatch14 = dispatch_default("change");
var _selection = select_default2(null);
- var _searchInput = select_default2(null);
- var _qid = null;
- var _wikidataEntity = null;
- var _wikiURL = "";
+ var _wrap = select_default2(null);
+ var addrField = _mainPresetIndex.field("address");
var _entityIDs = [];
- var _wikipediaKey = field.keys && field.keys.find(function(key) {
- return key.includes("wikipedia");
+ var _tags;
+ var _countryCode;
+ var _addressFormats = [{
+ format: [
+ ["housenumber", "street"],
+ ["city", "postcode"]
+ ]
+ }];
+ _mainFileFetcher.get("address_formats").then(function(d2) {
+ _addressFormats = d2;
+ if (!_selection.empty()) {
+ _selection.call(address);
+ }
+ }).catch(function() {
});
- 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(d2) {
- if (d2) {
- _qid = d2.id;
- change();
+ function getNear(isAddressable, type2, searchRadius, resultProp) {
+ var extent = combinedEntityExtent();
+ var l2 = extent.center();
+ var box = geoExtent(l2).padByMeters(searchRadius);
+ var features = context.history().intersects(box).filter(isAddressable).map((d2) => {
+ let dist = geoSphericalDistance(d2.extent(context.graph()).center(), l2);
+ if (d2.geometry(context.graph()) === "line") {
+ var loc = context.projection([
+ (extent[0][0] + extent[1][0]) / 2,
+ (extent[0][1] + extent[1][1]) / 2
+ ]);
+ var choice = geoChooseEdge(context.graph().childNodes(d2), loc, context.projection);
+ dist = geoSphericalDistance(choice.loc, l2);
}
- }).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(d2) {
- return "labeled-input preset-wikidata-" + d2;
- });
- enter.append("span").attr("class", "label").html(function(d2) {
- return _t.html("wikidata." + d2);
- });
- 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");
+ const value = resultProp && d2.tags[resultProp] ? d2.tags[resultProp] : d2.tags.name;
+ let title = value;
+ if (type2 === "street") {
+ title = "".concat(addrField.t("placeholders.street"), ": ").concat(title);
+ } else if (type2 === "place") {
+ title = "".concat(addrField.t("placeholders.place"), ": ").concat(title);
+ }
+ return {
+ title,
+ value,
+ dist,
+ type: type2,
+ klass: "address-".concat(type2)
+ };
+ }).sort(function(a2, b2) {
+ return a2.dist - b2.dist;
});
+ return utilArrayUniqBy(features, "value");
}
- function fetchWikidataItems(q2, callback) {
- if (!q2 && _hintKey) {
- for (var i3 in _entityIDs) {
- var entity = context.hasEntity(_entityIDs[i3]);
- if (entity.tags[_hintKey]) {
- q2 = entity.tags[_hintKey];
- break;
- }
+ function getNearStreets() {
+ function isAddressable(d2) {
+ return d2.tags.highway && d2.tags.name && d2.type === "way";
+ }
+ return getNear(isAddressable, "street", 200);
+ }
+ function getNearPlaces() {
+ function isAddressable(d2) {
+ if (d2.tags.name) {
+ if (d2.tags.place)
+ return true;
+ if (d2.tags.boundary === "administrative" && d2.tags.admin_level > 8)
+ return true;
}
+ return false;
}
- wikidata.itemsForSearchQuery(q2, function(err, data) {
- if (err) {
- if (err !== "No query")
- console.error(err);
- return;
+ return getNear(isAddressable, "place", 200);
+ }
+ function getNearCities() {
+ function isAddressable(d2) {
+ if (d2.tags.name) {
+ if (d2.tags.boundary === "administrative" && d2.tags.admin_level === "8")
+ return true;
+ if (d2.tags.border_type === "city")
+ return true;
+ if (d2.tags.place === "city" || d2.tags.place === "town" || d2.tags.place === "village")
+ return true;
}
- var result = data.map(function(item) {
+ if (d2.tags["".concat(field.key, ":city")])
+ return true;
+ return false;
+ }
+ return getNear(isAddressable, "city", 200, "".concat(field.key, ":city"));
+ }
+ function getNearPostcodes() {
+ return [...new Set([].concat(getNearValues("postcode")).concat(getNear((d2) => d2.tags.postal_code, "postcode", 200, "postal_code")))];
+ }
+ function getNearValues(key) {
+ const tagKey = "".concat(field.key, ":").concat(key);
+ function hasTag(d2) {
+ return _entityIDs.indexOf(d2.id) === -1 && d2.tags[tagKey];
+ }
+ return getNear(hasTag, key, 200, tagKey);
+ }
+ function updateForCountryCode() {
+ if (!_countryCode)
+ return;
+ var addressFormat;
+ for (var i3 = 0; i3 < _addressFormats.length; i3++) {
+ var format2 = _addressFormats[i3];
+ 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",
+ "street+place",
+ "subdistrict",
+ "suburb"
+ ];
+ var widths = addressFormat.widths || {
+ housenumber: 1 / 5,
+ unit: 1 / 5,
+ street: 1 / 2,
+ place: 1 / 2,
+ city: 2 / 3,
+ state: 1 / 4,
+ postcode: 1 / 3
+ };
+ function row(r2) {
+ var total = r2.reduce(function(sum, key) {
+ return sum + (widths[key] || 0.5);
+ }, 0);
+ return r2.map(function(key) {
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
+ id: key,
+ width: (widths[key] || 0.5) / total
};
});
- if (callback)
- callback(result);
+ }
+ var rows = _wrap.selectAll(".addr-row").data(addressFormat.format, function(d2) {
+ return d2.toString();
});
- }
- function change() {
- var syncTags = {};
- syncTags[field.key] = _qid;
- dispatch14.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)
+ 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(d2) {
+ return "addr-" + d2.id;
+ }).call(utilNoAuto).each(addDropdown).style("width", function(d2) {
+ return d2.width * 100 + "%";
+ });
+ function addDropdown(d2) {
+ if (dropdowns.indexOf(d2.id) === -1)
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 i3 in langs) {
- var lang = langs[i3];
- 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);
+ var nearValues;
+ switch (d2.id) {
+ case "street":
+ nearValues = getNearStreets;
+ break;
+ case "place":
+ nearValues = getNearPlaces;
+ break;
+ case "street+place":
+ nearValues = () => [].concat(getNearStreets()).concat(getNearPlaces());
+ d2.isAutoStreetPlace = true;
+ d2.id = _tags["".concat(field.key, ":place")] ? "place" : "street";
+ break;
+ case "city":
+ nearValues = getNearCities;
+ break;
+ case "postcode":
+ nearValues = getNearPostcodes;
+ break;
+ default:
+ nearValues = getNearValues;
}
- 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()
+ select_default2(this).call(
+ uiCombobox(context, "address-".concat(d2.isAutoStreetPlace ? "street-place" : d2.id)).minItems(1).caseSensitive(true).fetcher(function(typedValue, callback) {
+ typedValue = typedValue.toLowerCase();
+ callback(nearValues(d2.id).filter((v2) => v2.value.toLowerCase().indexOf(typedValue) !== -1));
+ }).on("accept", function(selected) {
+ if (d2.isAutoStreetPlace) {
+ d2.id = selected ? selected.type : "street";
+ }
+ })
);
- });
+ }
+ _wrap.selectAll("input").on("blur", change()).on("change", change());
+ _wrap.selectAll("input:not(.combobox-input)").on("input", change(true));
+ if (_tags)
+ updateTags(_tags);
}
- function setLabelForEntity() {
- var label = "";
- if (_wikidataEntity) {
- label = entityPropertyForDisplay(_wikidataEntity, "labels");
- if (label.length === 0) {
- label = _wikidataEntity.id.toString();
+ 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();
}
}
- 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;
+ function change(onInput) {
+ return function() {
+ setTimeout(() => {
+ 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;
+ if (subfield.isAutoStreetPlace) {
+ if (subfield.id === "street") {
+ tags["".concat(field.key, ":place")] = void 0;
+ } else if (subfield.id === "place") {
+ tags["".concat(field.key, ":street")] = void 0;
+ }
+ }
+ tags[key] = value || void 0;
+ });
+ Object.keys(tags).filter((k2) => tags[k2]).forEach((k2) => _tags[k2] = tags[k2]);
+ dispatch14.call("change", this, tags, onInput);
+ }, 0);
+ };
+ }
+ function updatePlaceholder(inputSelection) {
+ return inputSelection.attr("placeholder", function(subfield) {
+ if (_tags && Array.isArray(_tags[field.key + ":" + subfield.id])) {
+ return _t("inspector.multiple_values");
}
- _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);
+ if (subfield.isAutoStreetPlace) {
+ return "".concat(getLocalPlaceholder("street"), " / ").concat(getLocalPlaceholder("place"));
+ }
+ return getLocalPlaceholder(subfield.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;
+ }
+ function getLocalPlaceholder(key) {
+ if (_countryCode) {
+ var localkey = key + "!" + _countryCode;
+ var tkey = addrField.hasTextForStringId("placeholders." + localkey) ? localkey : key;
+ return addrField.t("placeholders." + tkey);
+ }
+ }
+ function updateTags(tags) {
+ utilGetSetValue(_wrap.selectAll("input"), (subfield) => {
+ var val;
+ if (subfield.isAutoStreetPlace) {
+ const streetKey = "".concat(field.key, ":street");
+ const placeKey = "".concat(field.key, ":place");
+ if (tags[streetKey] !== void 0 || tags[placeKey] === void 0) {
+ val = tags[streetKey];
+ subfield.id = "street";
+ } else {
+ val = tags[placeKey];
+ subfield.id = "place";
+ }
} else {
- _wikiURL = "";
+ val = tags["".concat(field.key, ":").concat(subfield.id)];
}
- }
- };
- 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 i3 in langs) {
- var lang = langs[i3];
- var valueObj = propObj[lang];
- if (valueObj && valueObj.value && valueObj.value.length > 0)
- return valueObj.value;
- }
- return propObj[langKeys[0]].value;
+ 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);
}
- wiki.entityIDs = function(val) {
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ address.entityIDs = function(val) {
if (!arguments.length)
return _entityIDs;
_entityIDs = val;
- return wiki;
+ return address;
};
- wiki.focus = function() {
- _searchInput.node().focus();
+ address.tags = function(tags) {
+ _tags = tags;
+ updateTags(tags);
};
- return utilRebind(wiki, dispatch14, "on");
+ address.focus = function() {
+ var node = _wrap.selectAll("input").node();
+ if (node)
+ node.focus();
+ };
+ return utilRebind(address, dispatch14, "on");
}
- // modules/ui/fields/wikipedia.js
- function uiFieldWikipedia(field, context) {
- const dispatch14 = 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((d2) => {
- _dataWikipedia = d2;
- if (_tags)
- updateForTags(_tags);
- }).catch(() => {
- });
- const langCombo = uiCombobox(context, "wikipedia-lang").fetcher((value, callback) => {
- const v2 = value.toLowerCase();
- callback(
- _dataWikipedia.filter((d2) => {
- return d2[0].toLowerCase().indexOf(v2) >= 0 || d2[1].toLowerCase().indexOf(v2) >= 0 || d2[2].toLowerCase().indexOf(v2) >= 0;
- }).map((d2) => ({ value: d2[1] }))
- );
- });
- const titleCombo = uiCombobox(context, "wikipedia-title").fetcher((value, callback) => {
- if (!value) {
- value = "";
- for (let i3 in _entityIDs) {
- let entity = context.hasEntity(_entityIDs[i3]);
- if (entity.tags.name) {
- value = entity.tags.name;
- break;
- }
- }
+ // modules/ui/fields/directional_combo.js
+ function uiFieldDirectionalCombo(field, context) {
+ var dispatch14 = dispatch_default("change");
+ var items = select_default2(null);
+ var wrap2 = select_default2(null);
+ var _tags;
+ var _combos = {};
+ if (field.type === "cycleway") {
+ field = {
+ ...field,
+ key: field.keys[0],
+ keys: field.keys.slice(1)
+ };
+ }
+ function directionalCombo(selection2) {
+ function stripcolon(s2) {
+ return s2.replace(":", "");
}
- const searchfn = value.length > 7 ? wikipedia.search : wikipedia.suggestions;
- searchfn(language()[2], value, (query, data) => {
- callback(data.map((d2) => ({ value: d2 })));
+ 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 rows-table").merge(div);
+ items = div.selectAll("li").data(field.keys);
+ var enter = items.enter().append("li").attr("class", function(d2) {
+ return "labeled-input preset-directionalcombo-" + stripcolon(d2);
});
- });
- 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);
+ enter.append("div").attr("class", "label preset-label-directionalcombo").attr("for", function(d2) {
+ return "preset-input-directionalcombo-" + stripcolon(d2);
+ }).html(function(d2) {
+ return field.t.html("types." + d2);
});
- 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");
+ 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", (t2) => change(key, t2[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 defaultLanguageInfo(skipEnglishFallback) {
- const langCode = _mainLocalizer.languageCode().toLowerCase();
- for (let i3 in _dataWikipedia) {
- let d2 = _dataWikipedia[i3];
- if (d2[2] === langCode)
- return d2;
- }
- return skipEnglishFallback ? ["", "", ""] : ["English", "English", "en"];
- }
- function language(skipEnglishFallback) {
- const value = utilGetSetValue(_langInput).toLowerCase();
- for (let i3 in _dataWikipedia) {
- let d2 = _dataWikipedia[i3];
- if (d2[0].toLowerCase() === value || d2[1].toLowerCase() === value || d2[2] === value)
- return d2;
- }
- return defaultLanguageInfo(skipEnglishFallback);
- }
- function changeLang() {
- utilGetSetValue(_langInput, language()[1]);
- change(true);
- }
- function change(skipWikidata) {
- let value = utilGetSetValue(_titleInput);
- const m2 = value.match(/https?:\/\/([-a-z]+)\.wikipedia\.org\/(?:wiki|\1-[-a-z]+)\/([^#]+)(?:#(.+))?/);
- const langInfo = m2 && _dataWikipedia.find((d2) => m2[1] === d2[2]);
- let syncTags = {};
- if (langInfo) {
- const nativeLangName = langInfo[1];
- value = decodeURIComponent(m2[2]).replace(/_/g, " ");
- if (m2[3]) {
- let anchor;
- anchor = decodeURIComponent(m2[3]);
- value += "#" + anchor.replace(/_/g, " ");
+ function change(key, newValue) {
+ const commonKey = field.key;
+ const otherKey = key === field.keys[0] ? field.keys[1] : field.keys[0];
+ dispatch14.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;
}
- 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;
- }
- dispatch14.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()
- );
+ return tags;
});
}
- wiki.tags = (tags) => {
+ directionalCombo.tags = function(tags) {
_tags = tags;
- updateForTags(tags);
- };
- function updateForTags(tags) {
- const value = typeof tags[field.key] === "string" ? tags[field.key] : "";
- const m2 = value.match(/([^:]+):([^#]+)(?:#(.+))?/);
- const tagLang = m2 && m2[1];
- const tagArticleTitle = m2 && m2[2];
- let anchor = m2 && m2[3];
- const tagLangInfo = tagLang && _dataWikipedia.find((d2) => tagLang === d2[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 (e3) {
- 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 = "";
- }
+ const commonKey = field.key;
+ 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] });
}
- }
- wiki.entityIDs = (val) => {
- if (!arguments.length)
- return _entityIDs;
- _entityIDs = val;
- return wiki;
};
- wiki.focus = () => {
- _titleInput.node().focus();
+ directionalCombo.focus = function() {
+ var node = wrap2.selectAll("input").node();
+ if (node)
+ node.focus();
};
- return utilRebind(wiki, dispatch14, "on");
+ return utilRebind(directionalCombo, dispatch14, "on");
}
- uiFieldWikipedia.supportsMultiselection = false;
-
- // modules/ui/fields/index.js
- var uiFields = {
- access: uiFieldAccess,
- address: uiFieldAddress,
- check: uiFieldCheck,
- colour: uiFieldText,
- combo: uiFieldCombo,
- cycleway: uiFieldDirectionalCombo,
- date: uiFieldText,
- 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 dispatch14 = 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");
- if (_show && !field.impl) {
- createField();
- }
- function createField() {
- field.impl = uiFields[field.type](field, context).on("change", function(t2, onInput) {
- dispatch14.call("change", field, t2, onInput);
+ // modules/ui/fields/lanes.js
+ function uiFieldLanes(field, context) {
+ var dispatch14 = 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 d2 = utilGetDimensions(wrap2);
+ var freeSpace = d2[0] - lanesData.lanes.length * LANE_WIDTH * 1.5 + LANE_WIDTH * 0.5;
+ surface = surface.enter().append("svg").attr("width", d2[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(d4) {
+ return "translate(" + LANE_WIDTH * d4.index * 1.5 + ", 0)";
+ });
+ lane.select(".forward").style("visibility", function(d4) {
+ return d4.direction === "forward" ? "visible" : "hidden";
+ });
+ lane.select(".bothways").style("visibility", function(d4) {
+ return d4.direction === "bothways" ? "visible" : "hidden";
+ });
+ lane.select(".backward").style("visibility", function(d4) {
+ return d4.direction === "backward" ? "visible" : "hidden";
});
- if (entityIDs) {
- field.entityIDs = entityIDs;
- if (field.impl.entityIDs) {
- field.impl.entityIDs(entityIDs);
- }
- }
}
- function allKeys() {
- let keys2 = field.keys || [field.key];
- if (field.type === "directionalCombo" && field.key) {
- keys2 = keys2.concat(field.key);
+ lanes.entityIDs = function(val) {
+ _entityIDs = val;
+ };
+ lanes.tags = function() {
+ };
+ lanes.focus = function() {
+ };
+ lanes.off = function() {
+ };
+ return utilRebind(lanes, dispatch14, "on");
+ }
+ uiFieldLanes.supportsMultiselection = false;
+
+ // modules/ui/fields/localized.js
+ var _languagesArray = [];
+ function uiFieldLocalized(field, context) {
+ var dispatch14 = 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(d2) {
+ _territoryLanguages = d2;
+ }).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)
+ });
}
- return keys2;
}
- function isModified() {
- if (!entityIDs || !entityIDs.length)
+ 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(entity.extent(context.graph()).center());
+ var showsBrandField = fields.some(function(d2) {
+ return d2.id === "brand";
+ });
+ var showsOperatorField = fields.some(function(d2) {
+ return d2.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;
- return entityIDs.some(function(entityID) {
- var original = context.graph().base().entities[entityID];
- var latest = context.graph().entity(entityID);
- return allKeys().some(function(key) {
- return original ? latest.tags[key] !== original.tags[key] : latest.tags[key];
- });
});
+ field.locked(isLocked);
}
- function tagsContainFieldKey() {
- return allKeys().some(function(key) {
- if (field.type === "multiCombo") {
- for (var tagKey in _tags) {
- if (tagKey.indexOf(key) === 0) {
- return true;
- }
+ function calcMultilingual(tags) {
+ var existingLangsOrdered = _multilingual.map(function(item2) {
+ return item2.lang;
+ });
+ var existingLangs = new Set(existingLangsOrdered.filter(Boolean));
+ for (var k2 in tags) {
+ var m2 = k2.match(/^(.*):([a-z]{2,3}(?:-[A-Z][a-z]{3})?(?:-[A-Z]{2})?)$/);
+ if (m2 && m2[1] === field.key && m2[2]) {
+ var item = { lang: m2[2], value: tags[k2] };
+ if (existingLangs.has(item.lang)) {
+ _multilingual[existingLangsOrdered.indexOf(item.lang)].value = item.value;
+ existingLangs.delete(item.lang);
+ } else {
+ _multilingual.push(item);
}
- return false;
}
- return _tags[key] !== void 0;
- });
- }
- function revert(d3_event, d2) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- if (!entityIDs || _locked)
- return;
- dispatch14.call("revert", d2, allKeys());
- }
- function remove2(d3_event, d2) {
- d3_event.stopPropagation();
- d3_event.preventDefault();
- if (_locked)
- return;
- var t2 = {};
- allKeys().forEach(function(key) {
- t2[key] = void 0;
+ }
+ _multilingual.forEach(function(item2) {
+ if (item2.lang && existingLangs.has(item2.lang)) {
+ item2.value = "";
+ }
});
- dispatch14.call("change", d2, t2);
}
- field.render = function(selection2) {
- var container = selection2.selectAll(".form-field").data([field]);
- var enter = container.enter().append("div").attr("class", function(d2) {
- return "form-field form-field-" + d2.safeid;
- }).classed("nowrap", !options2.wrap);
- if (options2.wrap) {
- var labelEnter = enter.append("label").attr("class", "field-label").attr("for", function(d2) {
- return d2.domId;
- });
- var textEnter = labelEnter.append("span").attr("class", "label-text");
- textEnter.append("span").attr("class", "label-textvalue").each(function(d2) {
- d2.label()(select_default2(this));
+ 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;
});
- 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"));
+ var isLangEn = defaultLang.indexOf("en") > -1;
+ if (isLangEn || langExists) {
+ defaultLang = "";
+ langExists = _multilingual.find(function(datum2) {
+ return datum2.lang === defaultLang;
+ });
}
- 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"));
+ if (!langExists) {
+ _multilingual.unshift({ lang: defaultLang, value: "" });
+ localizedInputs.call(renderMultilingual);
}
}
- 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(d2) {
- var selection3 = select_default2(this);
- if (!d2.impl) {
- createField();
- }
- var reference, help;
- if (options2.wrap && field.type === "restrictions") {
- help = uiFieldHelp(context, "restrictions");
- }
- if (options2.wrap && options2.info) {
- var referenceKey = d2.key || "";
- if (d2.type === "multiCombo") {
- referenceKey = referenceKey.replace(/:$/, "");
- }
- var referenceOptions = d2.reference || {
- key: referenceKey,
- value: _tags[referenceKey]
- };
- reference = uiTagReference(referenceOptions, context);
- if (_state === "hover") {
- reference.showing(false);
+ function change(onInput) {
+ return function(d3_event) {
+ if (field.locked()) {
+ d3_event.preventDefault();
+ return;
}
- }
- selection3.call(d2.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);
- }
- d2.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 t2 = {};
- t2[field.key] = field.default;
- dispatch14.call("change", this, t2);
+ var val = utilGetSetValue(select_default2(this));
+ if (!onInput)
+ val = context.cleanTagValue(val);
+ if (!val && Array.isArray(_tags[field.key]))
+ return;
+ var t2 = {};
+ t2[field.key] = val || void 0;
+ dispatch14.call("change", this, t2, onInput);
+ };
}
- };
- 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;
+ }
+ function key(lang) {
+ return field.key + ":" + lang;
+ }
+ function changeLang(d3_event, d2) {
+ var tags = {};
+ var lang = utilGetSetValue(select_default2(this)).toLowerCase();
+ var language = _languagesArray.find(function(d4) {
+ return d4.label.toLowerCase() === lang || d4.localName && d4.localName.toLowerCase() === lang || d4.nativeName && d4.nativeName.toLowerCase() === lang;
+ });
+ if (language)
+ lang = language.code;
+ if (d2.lang && d2.lang !== lang) {
+ tags[key(d2.lang)] = void 0;
}
- 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;
+ 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[d2.lang]) {
+ tags[newKey] = _wikiTitles[d2.lang];
}
- return true;
- };
- field.focus = function() {
- if (field.impl) {
- field.impl.focus();
+ d2.lang = lang;
+ dispatch14.call("change", this, tags);
+ }
+ function changeValue(d3_event, d2) {
+ if (!d2.lang)
+ return;
+ var value = context.cleanTagValue(utilGetSetValue(select_default2(this))) || void 0;
+ if (!value && Array.isArray(d2.value))
+ return;
+ var t2 = {};
+ t2[key(d2.lang)] = value;
+ d2.value = value;
+ dispatch14.call("change", this, t2);
+ }
+ function fetchLanguages(value, cb) {
+ var v2 = value.toLowerCase();
+ var langCodes = [_mainLocalizer.localeCode(), _mainLocalizer.languageCode()];
+ if (_countryCode && _territoryLanguages[_countryCode]) {
+ langCodes = langCodes.concat(_territoryLanguages[_countryCode]);
}
- };
- return utilRebind(field, dispatch14, "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 langItems = [];
+ langCodes.forEach(function(code) {
+ var langItem = _languagesArray.find(function(item) {
+ return item.code === code;
+ });
+ if (langItem)
+ langItems.push(langItem);
});
- 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(d2) {
- return d2.id + (d2.entityIDs ? d2.entityIDs.join() : "");
+ langItems = utilArrayUniq(langItems.concat(_languagesArray));
+ cb(langItems.filter(function(d2) {
+ return d2.label.toLowerCase().indexOf(v2) >= 0 || d2.localName && d2.localName.toLowerCase().indexOf(v2) >= 0 || d2.nativeName && d2.nativeName.toLowerCase().indexOf(v2) >= 0 || d2.code.toLowerCase().indexOf(v2) >= 0;
+ }).map(function(d2) {
+ return { value: d2.label };
+ }));
+ }
+ function renderMultilingual(selection2) {
+ var entries = selection2.selectAll("div.entry").data(_multilingual, function(d2) {
+ return d2.lang;
});
- fields.exit().remove();
- var enter = fields.enter().append("div").attr("class", function(d2) {
- return "wrap-form-field wrap-form-field-" + d2.safeid;
+ 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(_2, index) {
+ var wrap2 = select_default2(this);
+ var domId = utilUniqueDomId(index);
+ var label = wrap2.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").call(_t.append("translate.localized_translation_label"));
+ text.append("span").attr("class", "label-textannotation");
+ label.append("button").attr("class", "remove-icon-multilingual").attr("title", _t("icons.remove")).on("click", function(d3_event, d2) {
+ if (field.locked())
+ return;
+ d3_event.preventDefault();
+ _multilingual.splice(_multilingual.indexOf(d2), 1);
+ var langKey = d2.lang && key(d2.lang);
+ if (langKey && langKey in _tags) {
+ delete _tags[langKey];
+ var t2 = {};
+ t2[langKey] = void 0;
+ dispatch14.call("change", this, t2);
+ 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);
});
- fields = fields.merge(enter);
- fields.order().each(function(d2) {
- select_default2(this).call(d2.render);
+ 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");
});
- 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
- };
+ entries = entries.merge(entriesEnter);
+ entries.order();
+ entries.classed("present", true);
+ utilGetSetValue(entries.select(".localized-lang"), function(d2) {
+ var langItem = _languagesArray.find(function(item) {
+ return item.code === d2.lang;
+ });
+ if (langItem)
+ return langItem.label;
+ return d2.lang;
+ });
+ utilGetSetValue(entries.select(".localized-value"), function(d2) {
+ return typeof d2.value === "string" ? d2.value : "";
+ }).attr("title", function(d2) {
+ return Array.isArray(d2.value) ? d2.value.filter(Boolean).join("\n") : null;
+ }).attr("placeholder", function(d2) {
+ return Array.isArray(d2.value) ? _t("inspector.multiple_values") : _t("translate.localized_translation_name");
+ }).classed("mixed", function(d2) {
+ return Array.isArray(d2.value);
});
- 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(d2) {
- if (!d2)
- return;
- var field = d2.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;
+ 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, d2) {
+ if (err || !d2)
+ return;
+ _wikiTitles = d2;
+ });
+ }
+ }
+ 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]);
+ }
};
- formFields.state = function(val) {
- if (!arguments.length)
- return _state;
- _state = val;
- return formFields;
+ localized.focus = function() {
+ input.node().focus();
};
- formFields.klass = function(val) {
+ localized.entityIDs = function(val) {
if (!arguments.length)
- return _klass;
- _klass = val;
- return formFields;
+ return _entityIDs;
+ _entityIDs = val;
+ _multilingual = [];
+ loadCountryCode();
+ return localized;
};
- return formFields;
+ 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, dispatch14, "on");
}
- // modules/ui/changeset_editor.js
- function uiChangesetEditor(context) {
+ // modules/ui/fields/roadheight.js
+ function uiFieldRoadheight(field, context) {
var dispatch14 = dispatch_default("change");
- var formFields = uiFormFields(context);
- var commentCombo = uiCombobox(context, "comment").caseSensitive(true);
- var _fieldsArr;
+ 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 _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(t2, onInput) {
- dispatch14.call("change", field, void 0, t2, onInput);
- });
- });
+ var _isImperial;
+ var formatFloat = _mainLocalizer.floatFormatter(_mainLocalizer.languageCode());
+ var parseLocaleFloat = _mainLocalizer.floatParser(_mainLocalizer.languageCode());
+ var primaryUnits = [
+ {
+ value: "m",
+ title: _t("inspector.roadheight.meter")
+ },
+ {
+ value: "ft",
+ title: _t("inspector.roadheight.foot")
}
- _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"))
- );
- });
+ ];
+ 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();
}
- 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, (d2) => d2.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(d2) {
- let selection3 = select_default2(this);
- if (d2.link) {
- selection3 = selection3.append("a").attr("target", "_blank").attr("href", d2.link);
+ }
+ function setUnitSuggestions() {
+ utilGetSetValue(primaryUnitInput, _isImperial ? "ft" : "m");
+ }
+ function change() {
+ var tag2 = {};
+ var primaryValue = utilGetSetValue(primaryInput).trim();
+ var secondaryValue = utilGetSetValue(secondaryInput).trim();
+ if (!primaryValue && !secondaryValue && Array.isArray(_tags[field.key]))
+ return;
+ if (!primaryValue && !secondaryValue) {
+ tag2[field.key] = void 0;
+ } else {
+ var rawPrimaryValue = likelyRawNumberFormat.test(primaryValue) ? parseFloat(primaryValue) : parseLocaleFloat(primaryValue);
+ if (isNaN(rawPrimaryValue))
+ rawPrimaryValue = primaryValue;
+ var rawSecondaryValue = likelyRawNumberFormat.test(secondaryValue) ? parseFloat(secondaryValue) : parseLocaleFloat(secondaryValue);
+ if (isNaN(rawSecondaryValue))
+ rawSecondaryValue = secondaryValue;
+ if (isNaN(rawPrimaryValue) || isNaN(rawSecondaryValue) || !_isImperial) {
+ tag2[field.key] = context.cleanTagValue(rawPrimaryValue);
+ } else {
+ if (rawPrimaryValue !== "") {
+ rawPrimaryValue = rawPrimaryValue + "'";
+ }
+ if (rawSecondaryValue !== "") {
+ rawSecondaryValue = rawSecondaryValue + '"';
+ }
+ tag2[field.key] = context.cleanTagValue(rawPrimaryValue + rawSecondaryValue);
}
- selection3.call(d2.msg);
- });
+ }
+ dispatch14.call("change", this, tag2);
}
- changesetEditor.tags = function(_2) {
- if (!arguments.length)
- return _tags;
- _tags = _2;
- return changesetEditor;
+ 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 = formatFloat(parseFloat(secondaryValue[1]));
+ }
+ primaryValue = primaryValue.match(/(-?[\d.]+)'/);
+ if (primaryValue !== null) {
+ primaryValue = formatFloat(parseFloat(primaryValue[1]));
+ }
+ _isImperial = true;
+ } else if (primaryValue) {
+ var rawValue = primaryValue;
+ primaryValue = parseFloat(rawValue);
+ if (isNaN(primaryValue)) {
+ primaryValue = rawValue;
+ } else {
+ primaryValue = formatFloat(primaryValue);
+ }
+ _isImperial = false;
+ }
+ }
+ setUnitSuggestions();
+ var inchesPlaceholder = formatFloat(0);
+ 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 ? inchesPlaceholder : null).classed("mixed", isMixed).classed("disabled", !_isImperial).attr("readonly", _isImperial ? null : "readonly");
+ secondaryUnitInput.attr("value", _isImperial ? _t("inspector.roadheight.inch") : null);
};
- changesetEditor.changesetID = function(_2) {
- if (!arguments.length)
- return _changesetID;
- if (_changesetID === _2)
- return changesetEditor;
- _changesetID = _2;
- _fieldsArr = null;
- return changesetEditor;
+ roadheight.focus = function() {
+ primaryInput.node().focus();
};
- return utilRebind(changesetEditor, dispatch14, "on");
+ roadheight.entityIDs = function(val) {
+ _entityIDs = val;
+ };
+ function combinedEntityExtent() {
+ return _entityIDs && _entityIDs.length && utilTotalExtent(_entityIDs, context.graph());
+ }
+ return utilRebind(roadheight, dispatch14, "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);
+ // modules/ui/fields/roadspeed.js
+ function uiFieldRoadspeed(field, context) {
+ var dispatch14 = dispatch_default("change");
+ var unitInput = select_default2(null);
+ var input = select_default2(null);
+ var _entityIDs = [];
+ var _tags;
+ var _isImperial;
+ var formatFloat = _mainLocalizer.floatFormatter(_mainLocalizer.languageCode());
+ var parseLocaleFloat = _mainLocalizer.floatParser(_mainLocalizer.languageCode());
+ 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();
}
- return sValue;
}
- function EmptyTree() {
+ function setUnitSuggestions() {
+ speedCombo.data((_isImperial ? imperialValues : metricValues).map(comboValues));
+ utilGetSetValue(unitInput, _isImperial ? "mph" : "km/h");
}
- 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 comboValues(d2) {
+ return {
+ value: formatFloat(d2),
+ title: formatFloat(d2)
+ };
}
- 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);
- }
+ function change() {
+ var tag2 = {};
+ var value = utilGetSetValue(input).trim();
+ if (!value && Array.isArray(_tags[field.key]))
+ return;
+ if (!value) {
+ tag2[field.key] = void 0;
+ } else {
+ var rawValue = likelyRawNumberFormat.test(value) ? parseFloat(value) : parseLocaleFloat(value);
+ if (isNaN(rawValue))
+ rawValue = value;
+ if (isNaN(rawValue) || !_isImperial) {
+ tag2[field.key] = context.cleanTagValue(rawValue);
+ } else {
+ tag2[field.key] = context.cleanTagValue(rawValue + " mph");
}
}
- 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);
+ dispatch14.call("change", this, tag2);
+ }
+ roadspeed.tags = function(tags) {
+ _tags = tags;
+ var rawValue = tags[field.key];
+ var value = rawValue;
+ var isMixed = Array.isArray(value);
+ if (!isMixed) {
+ if (rawValue && rawValue.indexOf("mph") >= 0) {
+ _isImperial = true;
+ } else if (rawValue) {
+ _isImperial = false;
+ }
+ value = parseInt(value, 10);
+ if (isNaN(value)) {
+ value = rawValue;
} else {
- vResult[sProp] = vContent;
- nLength++;
+ value = formatFloat(value);
}
}
- 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;
+ 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, dispatch14, "on");
+ }
+
+ // modules/ui/fields/radio.js
+ function uiFieldRadio(field, context) {
+ var dispatch14 = 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(d2) {
+ return stringsField.t("options." + d2, { "default": d2 });
+ }).attr("checked", false);
+ enter.append("span").each(function(d2) {
+ stringsField.t.append("options." + d2, { "default": d2 })(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 list2 = extrasWrap.selectAll("ul").data([0]);
+ list2 = list2.enter().append("ul").attr("class", "rows").merge(list2);
+ if (type2) {
+ if (!typeField || typeField.id !== selected) {
+ typeField = uiField(context, type2, _entityIDs, { wrap: false }).on("change", changeType);
}
+ typeField.tags(tags);
+ } else {
+ typeField = null;
}
- if (nVerb === 3 || (nVerb === 2 || nVerb === 1 && nLength > 0) && sCollectedTxt) {
- vResult[sValueProp] = vBuiltVal;
- } else if (!bHighVerb && nLength === 0 && sCollectedTxt) {
- vResult = vBuiltVal;
+ var typeItem = list2.selectAll(".structure-type-item").data(typeField ? [typeField] : [], function(d2) {
+ return d2.id;
+ });
+ typeItem.exit().remove();
+ var typeEnter = typeItem.enter().insert("li", ":first-child").attr("class", "labeled-input structure-type-item");
+ typeEnter.append("div").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 (bFreeze && (bHighVerb || nLength > 0)) {
- Object.freeze(vResult);
+ 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(k2) {
+ return k2 !== "layer";
+ });
+ }
+ var layerItem = list2.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("div").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);
}
- 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()));
+ function changeType(t2, onInput) {
+ var key = selectedKey();
+ if (!key)
+ return;
+ var val = t2[key];
+ if (val !== "no") {
+ _oldType[key] = val;
}
- for (var sName in oParentObj) {
- vValue = oParentObj[sName];
- if (isFinite(sName) || vValue instanceof Function) {
- continue;
+ if (field.type === "structureRadio") {
+ if (val === "no" || key !== "bridge" && key !== "tunnel" || key === "tunnel" && val === "building_passage") {
+ t2.layer = void 0;
}
- 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);
+ if (t2.layer === void 0) {
+ if (key === "bridge" && val !== "no") {
+ t2.layer = "1";
}
- } 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()));
+ if (key === "tunnel" && val !== "no" && val !== "building_passage") {
+ t2.layer = "-1";
}
- oParentEl.appendChild(oChild);
}
}
+ dispatch14.call("change", this, t2, onInput);
}
- 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(d2) {
- _discardTags = d2;
- }).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(d2) {
- select_default2(this).call(svgIcon("#iD-icon-" + d2.entity.geometry(d2.graph), "pre-text " + d2.changeType));
- });
- buttons.append("span").attr("class", "change-type").html(function(d2) {
- return _t.html("commit." + d2.changeType) + " ";
- });
- buttons.append("strong").attr("class", "entity-type").text(function(d2) {
- var matched = _mainPresetIndex.match(d2.entity, d2.graph);
- return matched && matched.name() || utilDisplayType(d2.entity.id);
- });
- buttons.append("span").attr("class", "entity-name").text(function(d2) {
- var name = utilDisplayName(d2.entity) || "", string = "";
- if (name !== "") {
- string += ":";
+ function changeLayer(t2, onInput) {
+ if (t2.layer === "0") {
+ t2.layer = void 0;
+ }
+ dispatch14.call("change", this, t2, onInput);
+ }
+ function changeRadio() {
+ var t2 = {};
+ var activeKey;
+ if (field.key) {
+ t2[field.key] = void 0;
+ }
+ radios.each(function(d2) {
+ var active = select_default2(this).property("checked");
+ if (active)
+ activeKey = d2;
+ if (field.key) {
+ if (active)
+ t2[field.key] = d2;
+ } else {
+ var val = _oldType[activeKey] || "yes";
+ t2[d2] = active ? val : void 0;
}
- 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(d2) {
- if (d2.entity) {
- context.surface().selectAll(
- utilEntityOrMemberSelector([d2.entity.id], context.graph())
- ).classed("hover", true);
+ if (field.type === "structureRadio") {
+ if (activeKey === "bridge") {
+ t2.layer = "1";
+ } else if (activeKey === "tunnel" && t2.tunnel !== "building_passage") {
+ t2.layer = "-1";
+ } else {
+ t2.layer = void 0;
}
}
- function mouseout() {
- context.surface().selectAll(".hover").classed("hover", false);
+ dispatch14.call("change", this, t2);
+ }
+ radio.tags = function(tags) {
+ function isOptionChecked(d2) {
+ if (field.key) {
+ return tags[field.key] === d2;
+ }
+ return !!(typeof tags[d2] === "string" && tags[d2].toLowerCase() !== "no");
}
- 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);
+ function isMixed(d2) {
+ if (field.key) {
+ return Array.isArray(tags[field.key]) && tags[field.key].includes(d2);
}
+ return Array.isArray(tags[d2]);
}
- }
- 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";
- });
+ radios.property("checked", function(d2) {
+ return isOptionChecked(d2) && (field.key || field.options.filter(isOptionChecked).length === 1);
+ });
+ labels.classed("active", function(d2) {
+ if (field.key) {
+ return Array.isArray(tags[field.key]) && tags[field.key].includes(d2) || tags[field.key] === d2;
}
- 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(d2) {
- return d2.key;
- });
- items.exit().remove();
- var itemsEnter = items.enter().append("li").attr("class", issueItem);
- var buttons = itemsEnter.append("button").on("mouseover", function(d3_event, d2) {
- if (d2.entityIds) {
- context.surface().selectAll(
- utilEntityOrMemberSelector(
- d2.entityIds,
- context.graph()
- )
- ).classed("hover", true);
- }
- }).on("mouseout", function() {
- context.surface().selectAll(".hover").classed("hover", false);
- }).on("click", function(d3_event, d2) {
- context.validator().focusIssue(d2);
- });
- buttons.call(svgIcon("#iD-icon-alert", "pre-text"));
- buttons.append("strong").attr("class", "issue-message");
- buttons.filter(function(d2) {
- return d2.tooltip;
- }).call(
- uiTooltip().title(function(d2) {
- return d2.tooltip;
- }).placement("top")
- );
- items = itemsEnter.merge(items);
- items.selectAll(".issue-message").text("").each(function(d2) {
- return d2.message(context)(select_default2(this));
- });
+ return Array.isArray(tags[d2]) && tags[d2].some((v2) => typeof v2 === "string" && v2.toLowerCase() !== "no") || !!(typeof tags[d2] === "string" && tags[d2].toLowerCase() !== "no");
+ }).classed("mixed", isMixed).attr("title", function(d2) {
+ return isMixed(d2) ? _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()];
}
- }
- return commitWarnings;
+ 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, dispatch14, "on");
}
- // 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 dispatch14 = 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;
+ // modules/ui/fields/restrictions.js
+ function uiFieldRestrictions(field, context) {
+ var dispatch14 = 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 _initialized3 = 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 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(";"));
+ var isOK = _intersection && _intersection.vertices.length && // has vertices
+ _intersection.vertices.filter(function(vertex) {
+ return vertex.id === _vertexID;
+ }).length && _intersection.ways.length > 2;
+ 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;
}
- context.changeset = new osmChangeset({ tags });
+ 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 loadDerivedChangesetTags() {
- var osm = context.connection();
- if (!osm)
+ 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 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(";"));
+ var vgraph = _intersection.graph;
+ var filter2 = utilFunctor(true);
+ var projection2 = geoRawMercator();
+ var sdims = utilGetDimensions(context.container().select(".sidebar"));
+ var d2 = [sdims[0] - 50, 370];
+ var c2 = geoVecScale(d2, 0.5);
+ var z2 = 22;
+ projection2.scale(geoZoomToScale(z2));
+ var extent = geoExtent();
+ for (var i3 = 0; i3 < _intersection.vertices.length; i3++) {
+ extent._extend(_intersection.vertices[i3].extent());
}
- if (services.keepRight) {
- var krClosed = services.keepRight.getClosedIDs();
- if (krClosed.length) {
- tags["closed:keepright"] = context.cleanTagValue(krClosed.join(";"));
- }
+ var padTop = 35;
+ if (_intersection.vertices.length > 1) {
+ var hPadding = Math.min(160, Math.max(110, d2[0] * 0.4));
+ var vPadding = 160;
+ var tl = projection2([extent[0][0], extent[1][1]]);
+ var br2 = projection2([extent[1][0], extent[0][1]]);
+ var hFactor = (br2[0] - tl[0]) / (d2[0] - hPadding);
+ var vFactor = (br2[1] - tl[1]) / (d2[1] - vPadding - padTop);
+ var hZoomDiff = Math.log(Math.abs(hFactor)) / Math.LN2;
+ var vZoomDiff = Math.log(Math.abs(vFactor)) / Math.LN2;
+ z2 = z2 - Math.max(hZoomDiff, vZoomDiff);
+ projection2.scale(geoZoomToScale(z2));
}
- if (services.improveOSM) {
- var iOsmClosed = services.improveOSM.getClosedCounts();
- for (itemType in iOsmClosed) {
- tags["closed:improveosm:" + itemType] = context.cleanTagValue(iOsmClosed[itemType].toString());
- }
+ var extentCenter = projection2(extent.center());
+ extentCenter[1] = extentCenter[1] - padTop / 2;
+ projection2.translate(geoVecSubtract(c2, extentCenter)).clipExtent([[0, 0], d2]);
+ var drawLayers = svgLayers(projection2, context).only(["osm", "touch"]).dimensions(d2);
+ 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) {
+ _initialized3 = true;
+ surface.call(breathe);
}
- if (services.osmose) {
- var osmoseClosed = services.osmose.getClosedCounts();
- for (itemType in osmoseClosed) {
- tags["closed:osmose:" + itemType] = context.cleanTagValue(osmoseClosed[itemType].toString());
- }
+ if (_fromWayID && !vgraph.hasEntity(_fromWayID)) {
+ _fromWayID = null;
+ _oldTurns = null;
}
- for (var key in tags) {
- if (key.match(/(^warnings:)|(^resolved:)/)) {
- delete tags[key];
- }
+ surface.call(utilSetDimensions, d2).call(drawVertices, vgraph, _intersection.vertices, filter2, extent, z2).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);
}
- 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());
+ 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, i4;
+ 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 (i4 = 0; i4 < turns.length; i4++) {
+ var turn = turns[i4];
+ if (seen[turn.restrictionID])
+ continue;
+ if (turn.direct && turn.path[1] === datum2.path[1]) {
+ seen[turns[i4].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 (i4 = 0; i4 < turns.length; i4++) {
+ if (turns[i4].key !== datum2.key) {
+ extraActions.push(actionRestrictTurn(turns[i4], turns[i4].restrictionType));
+ }
}
+ _oldTurns = null;
+ actions = _intersection.actions.concat(extraActions, [
+ actionUnrestrictTurn(datum2),
+ _t("operations.restriction.annotation.delete")
+ ]);
} else {
- tags[prefix + ":" + issueType] = context.cleanTagValue(issuesOfType.length.toString());
+ actions = _intersection.actions.concat([
+ actionRestrictTurn(datum2, restrictionType),
+ _t("operations.restriction.annotation.create")
+ ]);
}
+ context.perform.apply(context, actions);
+ var s2 = surface.selectAll("." + datum2.key);
+ datum2 = s2.empty() ? null : s2.datum();
+ updateHints(datum2);
+ } else {
+ _fromWayID = null;
+ _oldTurns = null;
+ redraw();
}
}
- 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() {
- dispatch14.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;
+ function mouseover(d3_event) {
+ var datum2 = d3_event.target.__data__;
+ updateHints(datum2);
}
- 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");
+ _lastXPos = _lastXPos || sdims[0];
+ function redraw(minChange) {
+ var xPos = -1;
+ if (minChange) {
+ xPos = utilGetDimensions(context.container().select(".sidebar"))[0];
}
- 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() {
- dispatch14.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];
+ if (!minChange || minChange && Math.abs(xPos - _lastXPos) >= minChange) {
+ if (context.hasEntity(_vertexID)) {
+ _lastXPos = xPos;
+ _container.call(renderViewer);
}
- 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(_2, changed, onInput) {
- if (changed.hasOwnProperty("comment")) {
- if (changed.comment === void 0) {
- changed.comment = "";
- }
- if (!onInput) {
- corePreferences("comment", changed.comment);
- corePreferences("commentDate", Date.now());
+ 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 i4 = 0; i4 < turns.length; i4++) {
+ var turn = turns[i4];
+ 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");
+ }
}
}
- if (changed.hasOwnProperty("source")) {
- if (changed.source === void 0) {
- corePreferences("source", null);
- } else if (!onInput) {
- corePreferences("source", changed.source);
- corePreferences("commentDate", Date.now());
+ function updateHints(datum2) {
+ var help = _container.selectAll(".restriction-help").html("");
+ var placeholders = {};
+ ["from", "via", "to"].forEach(function(k2) {
+ placeholders[k2] = { html: '' + _t("restriction.help." + k2) + "" };
+ });
+ var entity = datum2 && datum2.properties && datum2.properties.entity;
+ if (entity) {
+ datum2 = entity;
}
- }
- 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;
+ if (_fromWayID) {
+ way = vgraph.entity(_fromWayID);
+ surface.selectAll("." + _fromWayID).classed("selected", true).classed("related", 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(s2) {
- if (s2[0] !== "#") {
- s2 = "#" + s2;
+ 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: "" });
}
- var matched = s2.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(k2) {
- var v2 = changed[k2];
- k2 = context.cleanTagKey(k2);
- if (readOnlyTags.indexOf(k2) !== -1)
- return;
- if (v2 === void 0) {
- delete tags[k2];
- } else if (onInput) {
- tags[k2] = v2;
- } else {
- tags[k2] = context.cleanTagValue(v2);
- }
- });
- 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 s2;
- s2 = corePreferences("walkthrough_completed");
- if (s2) {
- tags["ideditor:walkthrough_completed"] = s2;
+ 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 i4 = 0; i4 < datum2.via.ways.length; i4++) {
+ var prev = names[names.length - 1];
+ var curr = displayName(datum2.via.ways[i4], vgraph);
+ if (!prev || curr !== prev) {
+ names.push(curr);
+ }
+ }
+ help.append("div").html(_t.html("restriction.help.via_names", {
+ via: placeholders.via,
+ viaNames: names.join(", ")
+ }));
}
- s2 = corePreferences("walkthrough_progress");
- if (s2) {
- tags["ideditor:walkthrough_progress"] = s2;
+ if (!indirect) {
+ help.append("div").html(_t.html("restriction.help.toggle", { turn: { html: nextText.trim() } }));
}
- s2 = corePreferences("walkthrough_started");
- if (s2) {
- tags["ideditor:walkthrough_started"] = s2;
+ 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
+ }));
}
}
- } 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, dispatch14, "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 dispatch14 = 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();
- dispatch14.call("save");
- }
- function cancel() {
- keybindingOff();
- dispatch14.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(d2) {
- return d2.name;
- }).on("click", function(d3_event, d2) {
- d3_event.preventDefault();
- zoomToEntity(d2.id);
- });
- var details = conflictEnter.append("div").attr("class", "conflict-detail-container");
- details.append("ul").attr("class", "conflict-detail-list").selectAll("li").data(function(d2) {
- return d2.details || [];
- }).enter().append("li").attr("class", "conflict-detail-item").html(function(d2) {
- return d2;
- });
- 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(d2, i3) {
- return i3 === 0 && index === 0 || i3 === 1 && index === _conflictList.length - 1 || null;
- }).on("click", function(d3_event, d2) {
- d3_event.preventDefault();
- var container = parent.selectAll(".conflict-container");
- var sign2 = d2 === "previous" ? -1 : 1;
- container.selectAll(".conflict").remove();
- container.call(showConflict, index + sign2);
- }).each(function(d2) {
- _t.append("save.conflict." + d2)(select_default2(this));
- });
- }
- function addChoices(selection2) {
- var choices = selection2.append("ul").attr("class", "layer-list").selectAll("li").data(function(d2) {
- return d2.choices || [];
- });
- var choicesEnter = choices.enter().append("li").attr("class", "layer");
- var labelEnter = choicesEnter.append("label");
- labelEnter.append("input").attr("type", "radio").attr("name", function(d2) {
- return d2.id;
- }).on("change", function(d3_event, d2) {
- var ul = this.parentNode.parentNode.parentNode;
- ul.__data__.chosen = d2.id;
- choose(d3_event, ul, d2);
- });
- labelEnter.append("span").text(function(d2) {
- return d2.text;
- });
- choicesEnter.merge(choices).each(function(d2) {
- var ul = this.parentNode;
- if (ul.__data__.chosen === d2.id) {
- choose(null, ul, d2);
- }
- });
- }
- function choose(d3_event, ul, datum2) {
- if (d3_event)
- d3_event.preventDefault();
- select_default2(ul).selectAll("li").classed("active", function(d2) {
- return d2 === datum2;
- }).selectAll("input").property("checked", function(d2) {
- return d2 === 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);
+ 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 {
- context.map().zoomToEase(entity);
+ opts = { distance: _t("units.meters", { quantity: maxDist }) };
}
- context.surface().selectAll(utilEntityOrMemberSelector([entity.id], context.graph())).classed("hover", true);
- }
+ return selection2.html("").call(_t.append("restriction.controls.distance_up_to", opts));
+ };
}
- conflicts.conflictList = function(_2) {
- if (!arguments.length)
- return _conflictList;
- _conflictList = _2;
- return conflicts;
+ 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];
};
- conflicts.origChanges = function(_2) {
- if (!arguments.length)
- return _origChanges;
- _origChanges = _2;
- return conflicts;
+ restrictions.tags = function() {
};
- conflicts.shownEntityIds = function() {
- if (_conflictList && typeof _shownConflictIndex === "number") {
- return [_conflictList[_shownConflictIndex].id];
- }
- return [];
+ restrictions.focus = function() {
};
- return utilRebind(conflicts, dispatch14, "on");
+ restrictions.off = function(selection2) {
+ if (!_initialized3)
+ 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, dispatch14, "on");
}
+ uiFieldRestrictions.supportsMultiselection = false;
- // modules/ui/entity_editor.js
- var import_fast_deep_equal10 = __toESM(require_fast_deep_equal());
+ // modules/ui/fields/textarea.js
+ function uiFieldTextarea(field, context) {
+ var dispatch14 = 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 t2 = {};
+ t2[field.key] = val || void 0;
+ dispatch14.call("change", this, t2, 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, dispatch14, "on");
+ }
- // modules/ui/sections/entity_issues.js
- function uiSectionEntityIssues(context) {
- var preference = corePreferences("entity-issues.reference.expanded");
- var _expanded = preference === null ? true : preference === "true";
+ // modules/ui/fields/wikidata.js
+ function uiFieldWikidata(field, context) {
+ var wikidata = services.wikidata;
+ var dispatch14 = dispatch_default("change");
+ var _selection = select_default2(null);
+ var _searchInput = select_default2(null);
+ var _qid = null;
+ var _wikidataEntity = null;
+ var _wikiURL = "";
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);
+ var _wikipediaKey = field.keys && field.keys.find(function(key) {
+ return key.includes("wikipedia");
});
- function reloadIssues() {
- _issues = context.validator().getSharedEntityIssues(_entityIDs, { includeDisabledRules: true });
- }
- function makeActiveIssue(issueID) {
- _activeIssueID = issueID;
- section.selection().selectAll(".issue-container").classed("active", function(d2) {
- return d2.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(d2) {
- return d2.key;
- });
- containers.exit().remove();
- var containersEnter = containers.enter().append("div").attr("class", "issue-container");
- var itemsEnter = containersEnter.append("div").attr("class", function(d2) {
- return "issue severity-" + d2.severity;
- }).on("mouseover.highlight", function(d3_event, d2) {
- var ids = d2.entityIds.filter(function(e3) {
- return _entityIDs.indexOf(e3) === -1;
- });
- utilHighlightEntities(ids, true, context);
- }).on("mouseout.highlight", function(d3_event, d2) {
- var ids = d2.entityIds.filter(function(e3) {
- return _entityIDs.indexOf(e3) === -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, d2) {
- makeActiveIssue(d2.id);
- var extent = d2.extent(context.graph());
- if (extent) {
- var setZoom = Math.max(context.map().zoom(), 19);
- context.map().unobscuredCenterZoomEase(extent.center(), setZoom);
+ 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 list2 = wrap2.selectAll("ul").data([0]);
+ list2 = list2.enter().append("ul").attr("class", "rows").merge(list2);
+ var searchRow = list2.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(d2) {
+ if (d2) {
+ _qid = d2.id;
+ change();
}
+ }).on("cancel", function() {
+ setLabelForEntity();
});
- textEnter.each(function(d2) {
- var iconName = "#iD-icon-" + (d2.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();
+ 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();
- 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);
- });
- }
+ if (_wikiURL)
+ window.open(_wikiURL, "_blank");
});
- 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(d2) {
- if (typeof d2.reference === "function") {
- select_default2(this).call(d2.reference);
- } else {
- select_default2(this).call(_t.append("inspector.no_documentation_key"));
- }
+ searchRow = searchRow.merge(searchRowEnter);
+ _searchInput = searchRow.select("input");
+ var wikidataProperties = ["description", "identifier"];
+ var items = list2.selectAll("li.labeled-input").data(wikidataProperties);
+ var enter = items.enter().append("li").attr("class", function(d2) {
+ return "labeled-input preset-wikidata-" + d2;
});
- containers = containers.merge(containersEnter).classed("active", function(d2) {
- return d2.id === _activeIssueID;
+ enter.append("span").attr("class", "label").html(function(d2) {
+ return _t.html("wikidata." + d2);
});
- containers.selectAll(".issue-message").text("").each(function(d2) {
- return d2.message(context)(select_default2(this));
+ 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");
});
- var fixLists = containers.selectAll(".issue-fix-list");
- var fixes = fixLists.selectAll(".issue-fix-item").data(function(d2) {
- return d2.fixes ? d2.fixes(context) : [];
- }, function(fix) {
- return fix.id;
+ }
+ function fetchWikidataItems(q2, callback) {
+ if (!q2 && _hintKey) {
+ for (var i3 in _entityIDs) {
+ var entity = context.hasEntity(_entityIDs[i3]);
+ if (entity.tags[_hintKey]) {
+ q2 = entity.tags[_hintKey];
+ break;
+ }
+ }
+ }
+ wikidata.itemsForSearchQuery(q2, function(err, data) {
+ if (err) {
+ if (err !== "No query")
+ console.error(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);
});
- fixes.exit().remove();
- var fixesEnter = fixes.enter().append("li").attr("class", "issue-fix-item");
- var buttons = fixesEnter.append("button").on("click", function(d3_event, d2) {
- if (select_default2(this).attr("disabled") || !d2.onClick)
+ }
+ function change() {
+ var syncTags = {};
+ syncTags[field.key] = _qid;
+ dispatch14.call("change", this, syncTags);
+ var initGraph = context.graph();
+ var initEntityIDs = _entityIDs;
+ wikidata.entityByQID(_qid, function(err, entity) {
+ if (err)
return;
- if (d2.issue.dateLastRanFix && /* @__PURE__ */ new Date() - d2.issue.dateLastRanFix < 1e3)
+ if (context.graph() !== initGraph)
return;
- d2.issue.dateLastRanFix = /* @__PURE__ */ new Date();
- utilHighlightEntities(d2.issue.entityIds.concat(d2.entityIds), false, context);
- new Promise(function(resolve, reject) {
- d2.onClick(context, resolve, reject);
- if (d2.onClick.length <= 1) {
- resolve();
+ 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);
}
- }).then(function() {
- context.validator().validate();
});
- }).on("mouseover.highlight", function(d3_event, d2) {
- utilHighlightEntities(d2.entityIds, true, context);
- }).on("mouseout.highlight", function(d3_event, d2) {
- utilHighlightEntities(d2.entityIds, false, context);
- });
- buttons.each(function(d2) {
- var iconName = d2.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(d2) {
- return d2.title(select_default2(this));
- });
- fixesEnter.merge(fixes).selectAll("button").classed("actionable", function(d2) {
- return d2.onClick;
- }).attr("disabled", function(d2) {
- return d2.onClick ? null : "true";
- }).attr("title", function(d2) {
- if (d2.disabledReason) {
- return d2.disabledReason;
+ var newWikipediaValue;
+ if (_wikipediaKey) {
+ var foundPreferred;
+ for (var i3 in langs) {
+ var lang = langs[i3];
+ 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;
+ }
+ }
}
- 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(p2, geom) {
- if (p2.isFallback && p2.isFallback())
- return geom === "vertex" ? "" : "iD-icon-" + p2.id;
- if (p2.icon)
- return p2.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 w2 = 40;
- const h2 = 40;
- pointBorderEnter.append("svg").attr("class", "preset-icon-fill preset-icon-point-border").attr("width", w2).attr("height", h2).attr("viewBox", `0 0 ${w2} ${h2}`).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 d2 = 60;
- let svgEnter = categoryBorderEnter.append("svg").attr("class", "preset-icon-fill preset-icon-category-border").attr("width", d2).attr("height", d2).attr("viewBox", `0 0 ${d2} ${d2}`);
- 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 w2 = 60;
- const h2 = 60;
- const d2 = 40;
- vertexFillEnter.append("svg").attr("class", "preset-icon-fill preset-icon-fill-vertex").attr("width", w2).attr("height", h2).attr("viewBox", `0 0 ${w2} ${h2}`).append("circle").attr("cx", w2 / 2).attr("cy", h2 / 2).attr("r", d2 / 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 d2 = 60;
- const w2 = d2;
- const h2 = d2;
- const l2 = d2 * 2 / 3;
- const c1 = (w2 - l2) / 2;
- const c2 = c1 + l2;
- fillEnter = fillEnter.append("svg").attr("class", "preset-icon-fill preset-icon-fill-area").attr("width", w2).attr("height", h2).attr("viewBox", `0 0 ${w2} ${h2}`);
- ["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, w2 / 2], [c2, w2 / 2], [h2 / 2, c1], [h2 / 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 d2 = 60;
- const w2 = d2;
- const h2 = d2;
- const y2 = Math.round(d2 * 0.72);
- const l2 = Math.round(d2 * 0.6);
- const r2 = 2.5;
- const x12 = (w2 - l2) / 2;
- const x2 = x12 + l2;
- lineEnter = lineEnter.append("svg").attr("class", "preset-icon-line").attr("width", w2).attr("height", h2).attr("viewBox", `0 0 ${w2} ${h2}`);
- ["casing", "stroke"].forEach((klass) => {
- lineEnter.append("path").attr("d", `M${x12} ${y2} L${x2} ${y2}`).attr("class", `line ${klass}`);
- });
- [[x12 - 1, y2], [x2 + 1, y2]].forEach((point2) => {
- lineEnter.append("circle").attr("class", "vertex").attr("cx", point2[0]).attr("cy", point2[1]).attr("r", r2);
- });
- 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, p2) {
- let route = container.selectAll(".preset-icon-route").data(drawRoute ? [0] : []);
- route.exit().remove();
- let routeEnter = route.enter();
- const d2 = 60;
- const w2 = d2;
- const h2 = d2;
- const y12 = Math.round(d2 * 0.8);
- const y2 = Math.round(d2 * 0.68);
- const l2 = Math.round(d2 * 0.6);
- const r2 = 2;
- const x12 = (w2 - l2) / 2;
- const x2 = x12 + l2 / 3;
- const x3 = x2 + l2 / 3;
- const x4 = x3 + l2 / 3;
- routeEnter = routeEnter.append("svg").attr("class", "preset-icon-route").attr("width", w2).attr("height", h2).attr("viewBox", `0 0 ${w2} ${h2}`);
- ["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", r2);
- });
- route = routeEnter.merge(route);
- if (drawRoute) {
- let routeType = p2.tags.type === "waterway" ? "waterway" : p2.tags.route;
- const segmentPresetIDs = routeSegments[routeType];
- for (let i3 in segmentPresetIDs) {
- const segmentPreset = _mainPresetIndex.item(segmentPresetIDs[i3]);
- const segmentTagClasses = svgTagClasses().getClassesString(segmentPreset.tags, "");
- route.selectAll(`path.stroke.segment${i3}`).attr("class", `segment${i3} line stroke ${segmentTagClasses}`);
- route.selectAll(`path.casing.segment${i3}`).attr("class", `segment${i3} line casing ${segmentTagClasses}`);
+ if (newWikipediaValue) {
+ newWikipediaValue = context.cleanTagValue(newWikipediaValue);
}
- }
- }
- 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 isR\u00F6ntgen = picon && /^roentgen-/.test(picon);
- const isiDIcon = picon && !(isMaki || isTemaki || isFa || isR\u00F6ntgen);
- 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);
+ 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 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);
+ function setLabelForEntity() {
+ var label = "";
+ if (_wikidataEntity) {
+ label = entityPropertyForDisplay(_wikidataEntity, "labels");
+ if (label.length === 0) {
+ label = _wikidataEntity.id.toString();
+ }
+ }
+ utilGetSetValue(_searchInput, label);
}
- 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 p2 = _preset.apply(this, arguments);
- let geom = _geometry ? _geometry.apply(this, arguments) : null;
- if (geom === "relation" && p2.tags && (p2.tags.type === "route" && p2.tags.route && routeSegments[p2.tags.route] || p2.tags.type === "waterway")) {
- geom = "route";
+ 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;
}
- const showThirdPartyIcons = corePreferences("preferences.privacy.thirdpartyicons") || "true";
- const isFallback = p2.isFallback && p2.isFallback();
- const imageURL = showThirdPartyIcons === "true" && p2.imageURL;
- const picon = getIcon(p2, geom);
- const isCategory = !p2.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 ? p2.setTags({}, geom) : {};
- for (let k2 in tags) {
- if (tags[k2] === "*") {
- tags[k2] = "yes";
+ _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 = "";
}
}
- 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 && p2);
- renderPointBorder(container, drawPoint);
- renderCircleFill(container, drawVertex);
- renderSquareFill(container, drawArea, tagClasses);
- renderLine(container, drawLine, tagClasses);
- renderRoute(container, drawRoute, p2);
- renderSvgIcon(container, picon, geom, isFramed, isCategory, tagClasses);
- renderImageIcon(container, imageURL);
+ };
+ 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 i3 in langs) {
+ var lang = langs[i3];
+ var valueObj = propObj[lang];
+ if (valueObj && valueObj.value && valueObj.value.length > 0)
+ return valueObj.value;
+ }
+ return propObj[langKeys[0]].value;
}
- presetIcon.preset = function(val) {
+ wiki.entityIDs = function(val) {
if (!arguments.length)
- return _preset;
- _preset = utilFunctor(val);
- return presetIcon;
+ return _entityIDs;
+ _entityIDs = val;
+ return wiki;
};
- presetIcon.geometry = function(val) {
- if (!arguments.length)
- return _geometry;
- _geometry = utilFunctor(val);
- return presetIcon;
+ wiki.focus = function() {
+ _searchInput.node().focus();
};
- return presetIcon;
+ return utilRebind(wiki, dispatch14, "on");
}
- // modules/ui/sections/feature_type.js
- function uiSectionFeatureType(context) {
- var dispatch14 = 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")
+ // modules/ui/fields/wikipedia.js
+ function uiFieldWikipedia(field, context) {
+ const dispatch14 = 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((d2) => {
+ _dataWikipedia = d2;
+ if (_tags)
+ updateForTags(_tags);
+ }).catch(() => {
+ });
+ const langCombo = uiCombobox(context, "wikipedia-lang").fetcher((value, callback) => {
+ const v2 = value.toLowerCase();
+ callback(
+ _dataWikipedia.filter((d2) => {
+ return d2[0].toLowerCase().indexOf(v2) >= 0 || d2[1].toLowerCase().indexOf(v2) >= 0 || d2[2].toLowerCase().indexOf(v2) >= 0;
+ }).map((d2) => ({ value: d2[1] }))
);
- 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);
+ });
+ const titleCombo = uiCombobox(context, "wikipedia-title").fetcher((value, callback) => {
+ if (!value) {
+ value = "";
+ for (let i3 in _entityIDs) {
+ let entity = context.hasEntity(_entityIDs[i3]);
+ if (entity.tags.name) {
+ value = entity.tags.name;
+ break;
+ }
+ }
}
- selection2.selectAll(".preset-reset").on("click", function() {
- dispatch14.call("choose", this, _presets);
- }).on("pointerdown pointerup mousedown mouseup", function(d3_event) {
+ const searchfn = value.length > 7 ? wikipedia.search : wikipedia.suggestions;
+ searchfn(language()[2], value, (query, data) => {
+ callback(data.map((d2) => ({ value: d2 })));
+ });
+ });
+ 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-".concat(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 link3 = titleContainer.selectAll(".wiki-link").data([0]);
+ link3 = link3.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(link3);
+ link3.on("click", (d3_event) => {
d3_event.preventDefault();
- d3_event.stopPropagation();
+ if (_wikiURL)
+ window.open(_wikiURL, "_blank");
});
- 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, (d2) => d2.stringId);
- nameparts.exit().remove();
- nameparts.enter().append("div").attr("class", "namepart").text("").each(function(d2) {
- d2(select_default2(this));
+ }
+ function defaultLanguageInfo(skipEnglishFallback) {
+ const langCode = _mainLocalizer.languageCode().toLowerCase();
+ for (let i3 in _dataWikipedia) {
+ let d2 = _dataWikipedia[i3];
+ if (d2[2] === langCode)
+ return d2;
+ }
+ return skipEnglishFallback ? ["", "", ""] : ["English", "English", "en"];
+ }
+ function language(skipEnglishFallback) {
+ const value = utilGetSetValue(_langInput).toLowerCase();
+ for (let i3 in _dataWikipedia) {
+ let d2 = _dataWikipedia[i3];
+ if (d2[0].toLowerCase() === value || d2[1].toLowerCase() === value || d2[2] === value)
+ return d2;
+ }
+ return defaultLanguageInfo(skipEnglishFallback);
+ }
+ function changeLang() {
+ utilGetSetValue(_langInput, language()[1]);
+ change(true);
+ }
+ function change(skipWikidata) {
+ let value = utilGetSetValue(_titleInput);
+ const m2 = value.match(/https?:\/\/([-a-z]+)\.wikipedia\.org\/(?:wiki|\1-[-a-z]+)\/([^#]+)(?:#(.+))?/);
+ const langInfo = m2 && _dataWikipedia.find((d2) => m2[1] === d2[2]);
+ let syncTags = {};
+ if (langInfo) {
+ const nativeLangName = langInfo[1];
+ value = decodeURIComponent(m2[2]).replace(/_/g, " ");
+ if (m2[3]) {
+ let anchor;
+ anchor = decodeURIComponent(m2[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;
+ }
+ dispatch14.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()
+ );
});
}
- section.entityIDs = function(val) {
+ wiki.tags = (tags) => {
+ _tags = tags;
+ updateForTags(tags);
+ };
+ function updateForTags(tags) {
+ const value = typeof tags[field.key] === "string" ? tags[field.key] : "";
+ const m2 = value.match(/([^:]+):([^#]+)(?:#(.+))?/);
+ const tagLang = m2 && m2[1];
+ const tagArticleTitle = m2 && m2[2];
+ let anchor = m2 && m2[3];
+ const tagLangInfo = tagLang && _dataWikipedia.find((d2) => tagLang === d2[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 (e3) {
+ 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://".concat(defaultLangInfo[2], ".wikipedia.org/w/index.php?fulltext=1&search=").concat(value);
+ } else {
+ const shownOrDefaultLangInfo = language(
+ true
+ /* skipEnglishFallback */
+ );
+ utilGetSetValue(_langInput, shownOrDefaultLangInfo[1]);
+ _wikiURL = "";
+ }
+ }
+ }
+ wiki.entityIDs = (val) => {
if (!arguments.length)
return _entityIDs;
_entityIDs = val;
- return section;
+ return wiki;
};
- 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;
+ wiki.focus = () => {
+ _titleInput.node().focus();
};
- function entityGeometries() {
- var counts = {};
- for (var i3 in _entityIDs) {
- var geometry = context.graph().geometry(_entityIDs[i3]);
- 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, dispatch14, "on");
+ return utilRebind(wiki, dispatch14, "on");
}
+ uiFieldWikipedia.supportsMultiselection = false;
- // modules/ui/sections/preset_fields.js
- function uiSectionPresetFields(context) {
- var section = uiSection("preset-fields", context).label(() => _t.append("inspector.fields")).disclosureContent(renderDisclosureContent);
+ // modules/ui/fields/index.js
+ var uiFields = {
+ access: uiFieldAccess,
+ address: uiFieldAddress,
+ check: uiFieldCheck,
+ colour: uiFieldText,
+ combo: uiFieldCombo,
+ cycleway: uiFieldDirectionalCombo,
+ date: uiFieldText,
+ 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 dispatch14 = 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;
- }, {}));
- const loc = _entityIDs.reduce(function(extent, entityID) {
- var entity = context.graph().entity(entityID);
- return extent.extend(entity.extent(context.graph()));
- }, geoExtent()).center();
- var presetsManager = _mainPresetIndex;
- var allFields = [];
- var allMoreFields = [];
- var sharedTotalFields;
- _presets.forEach(function(preset) {
- var fields = preset.fields(loc);
- var moreFields = preset.moreFields(loc);
- 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 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");
+ if (_show && !field.impl) {
+ createField();
+ }
+ function createField() {
+ field.impl = uiFields[field.type](field, context).on("change", function(t2, onInput) {
+ dispatch14.call("change", field, t2, onInput);
+ });
+ if (entityIDs) {
+ field.entityIDs = entityIDs;
+ if (field.impl.entityIDs) {
+ field.impl.entityIDs(entityIDs);
}
- var additionalFields = utilArrayUnion(sharedMoreFields, presetsManager.universal());
- additionalFields.sort(function(field1, field2) {
- return field1.title().localeCompare(field2.title(), _mainLocalizer.localeCode());
+ }
+ }
+ function allKeys() {
+ let keys2 = field.keys || [field.key];
+ if (field.type === "directionalCombo" && field.key) {
+ keys2 = keys2.concat(field.key);
+ }
+ return keys2;
+ }
+ 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 allKeys().some(function(key) {
+ return original ? latest.tags[key] !== original.tags[key] : latest.tags[key];
});
- additionalFields.forEach(function(field) {
- if (sharedFields.indexOf(field) === -1 && field.matchAllGeometry(geometries)) {
- _fieldsArr.push(
- uiField(context, field, _entityIDs, { show: false })
- );
+ });
+ }
+ function tagsContainFieldKey() {
+ return allKeys().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, d2) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ if (!entityIDs || _locked)
+ return;
+ dispatch14.call("revert", d2, allKeys());
+ }
+ function remove2(d3_event, d2) {
+ d3_event.stopPropagation();
+ d3_event.preventDefault();
+ if (_locked)
+ return;
+ var t2 = {};
+ allKeys().forEach(function(key) {
+ t2[key] = void 0;
+ });
+ dispatch14.call("change", d2, t2);
+ }
+ field.render = function(selection2) {
+ var container = selection2.selectAll(".form-field").data([field]);
+ var enter = container.enter().append("div").attr("class", function(d2) {
+ return "form-field form-field-" + d2.safeid;
+ }).classed("nowrap", !options2.wrap);
+ if (options2.wrap) {
+ var labelEnter = enter.append("label").attr("class", "field-label").attr("for", function(d2) {
+ return d2.domId;
});
- _fieldsArr.forEach(function(field) {
- field.on("change", function(t2, onInput) {
- dispatch14.call("change", field, _entityIDs, t2, onInput);
- }).on("revert", function(keys2) {
- dispatch14.call("revert", field, keys2);
- });
+ var textEnter = labelEnter.append("span").attr("class", "label-text");
+ textEnter.append("span").attr("class", "label-textvalue").each(function(d2) {
+ d2.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"));
+ }
}
- _fieldsArr.forEach(function(field) {
- field.state(_state).tags(_tags);
+ 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(d2) {
+ var selection3 = select_default2(this);
+ if (!d2.impl) {
+ createField();
+ }
+ var reference, help;
+ if (options2.wrap && field.type === "restrictions") {
+ help = uiFieldHelp(context, "restrictions");
+ }
+ if (options2.wrap && options2.info) {
+ var referenceKey = d2.key || "";
+ if (d2.type === "multiCombo") {
+ referenceKey = referenceKey.replace(/:$/, "");
+ }
+ var referenceOptions = d2.reference || {
+ key: referenceKey,
+ value: _tags[referenceKey]
+ };
+ reference = uiTagReference(referenceOptions, context);
+ if (_state === "hover") {
+ reference.showing(false);
+ }
+ }
+ selection3.call(d2.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);
+ }
+ d2.impl.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;
+ 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);
};
- section.state = function(val) {
+ field.state = function(val) {
if (!arguments.length)
return _state;
_state = val;
- return section;
+ return field;
};
- section.tags = function(val) {
+ field.tags = function(val) {
if (!arguments.length)
return _tags;
_tags = val;
- return section;
+ if (tagsContainFieldKey() && !_show) {
+ _show = true;
+ if (!field.impl) {
+ createField();
+ }
+ }
+ return field;
};
- section.entityIDs = function(val) {
+ field.locked = function(val) {
if (!arguments.length)
- return _entityIDs;
- if (!val || !_entityIDs || !utilArrayIdentical(_entityIDs, val)) {
- _entityIDs = val;
- _fieldsArr = null;
+ 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 t2 = {};
+ t2[field.key] = field.default;
+ dispatch14.call("change", this, t2);
}
- return section;
};
- return utilRebind(section, dispatch14, "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)
+ field.isShown = function() {
+ return _show;
+ };
+ field.isAllowed = function() {
+ if (entityIDs && entityIDs.length > 1 && uiFields[field.type].supportsMultiselection === false)
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 gt2 = entity.members.length > _maxMembers ? ">" : "";
- var count = gt2 + 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, d2) {
- d3_event.preventDefault();
- select_default2(this.parentNode).classed("tag-reference-loading", true);
- context.loadEntity(d2.id, function() {
- section.reRender();
- });
- }
- function zoomToMember(d3_event, d2) {
- d3_event.preventDefault();
- var entity = context.entity(d2.id);
- context.map().zoomToEase(entity);
- utilHighlightEntities([d2.id], true, context);
- }
- function selectMember(d3_event, d2) {
- d3_event.preventDefault();
- utilHighlightEntities([d2.id], false, context);
- var entity = context.entity(d2.id);
- var mapExtent = context.map().extent();
- if (!entity.intersects(mapExtent, context.graph())) {
- context.map().zoomToEase(entity);
+ 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;
}
- context.enter(modeSelect(context, [d2.id]));
- }
- function changeRole(d3_event, d2) {
- var oldRole = d2.role;
- var newRole = context.cleanRelationRole(select_default2(this).property("value"));
- if (oldRole !== newRole) {
- var member = { id: d2.id, type: d2.type, role: newRole };
- context.perform(
- actionChangeMember(d2.relation.id, member, d2.index),
- _t("operations.change_role.annotation", {
- n: 1
- })
- );
- context.validator().validate();
+ 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;
}
- }
- function deleteMember(d3_event, d2) {
- utilHighlightEntities([d2.id], false, context);
- context.perform(
- actionDeleteMember(d2.relation.id, d2.index),
- _t("operations.delete_member.annotation", {
- n: 1
- })
- );
- if (!context.hasEntity(d2.relation.id)) {
- context.enter(modeBrowse(context));
- } else {
- context.validator().validate();
+ return true;
+ };
+ field.focus = function() {
+ if (field.impl) {
+ field.impl.focus();
}
- }
- 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)
- });
+ };
+ return utilRebind(field, dispatch14, "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 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(d2) {
- return osmEntity.key(d2.relation) + "," + d2.index + "," + (d2.member ? osmEntity.key(d2.member) : "incomplete");
+ var shown = allowedFields.filter(function(field) {
+ return field.isShown();
});
- items.exit().each(unbind).remove();
- var itemsEnter = items.enter().append("li").attr("class", "member-row form-field").classed("member-incomplete", function(d2) {
- return !d2.member;
+ var notShown = allowedFields.filter(function(field) {
+ return !field.isShown();
});
- itemsEnter.each(function(d2) {
- var item = select_default2(this);
- var label = item.append("label").attr("class", "field-label").attr("for", d2.domId);
- if (d2.member) {
- item.on("mouseover", function() {
- utilHighlightEntities([d2.id], true, context);
- }).on("mouseout", function() {
- utilHighlightEntities([d2.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(d4) {
- var matched = _mainPresetIndex.match(d4.member, context.graph());
- return matched && matched.name() || utilDisplayType(d4.member.id);
- });
- labelLink.append("span").attr("class", "member-entity-name").text(function(d4) {
- return utilDisplayName(d4.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." + d2.type, { id: d2.id }));
- labelText.append("span").attr("class", "member-entity-name").call(_t.append("inspector.incomplete", { id: d2.id }));
- label.append("button").attr("class", "member-download").attr("title", _t("icons.download")).call(svgIcon("#iD-icon-load")).on("click", downloadMember);
- }
+ 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(d2) {
+ return d2.id + (d2.entityIDs ? d2.entityIDs.join() : "");
});
- 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(d2) {
- return d2.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(d2) {
- return d2.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 x2 = d3_event.x - dragOrigin.x, y2 = 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(x2, 2) + Math.pow(y2, 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(" + x2 + "px, " + y2 + "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, d2) {
- if (!select_default2(this).classed("dragging"))
+ fields.exit().remove();
+ var enter = fields.enter().append("div").attr("class", function(d2) {
+ return "wrap-form-field wrap-form-field-" + d2.safeid;
+ });
+ fields = fields.merge(enter);
+ fields.order().each(function(d2) {
+ select_default2(this).call(d2.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(d2) {
+ if (!d2)
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(d2.relation.id, index, targetIndex),
- _t("operations.reorder_members.annotation")
- );
- context.validator().validate();
- }
+ var field = d2.field;
+ field.show();
+ selection2.call(formFields);
+ field.focus();
})
);
- function bindTypeahead(d2) {
- 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 i3 = 0; i3 < data.length; i3++) {
- if (data[i3].value.substring(0, value.length) === value) {
- sameletter.push(data[i3]);
- } else {
- other.push(data[i3]);
- }
- }
- return sameletter.concat(other);
+ 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 dispatch14 = 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: true, revert: false }),
+ uiField(context, presets.field("hashtags"), null, { show: false, revert: false })
+ ];
+ _fieldsArr.forEach(function(field) {
+ field.on("change", function(t2, onInput) {
+ dispatch14.call("change", field, void 0, t2, 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"))
+ );
+ });
}
- role.call(
- uiCombobox(context, "member-role").fetcher(function(role2, callback) {
- var geometry;
- if (d2.member) {
- geometry = context.graph().geometry(d2.member.id);
- } else if (d2.type === "relation") {
- geometry = "relation";
- } else if (d2.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);
+ 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, (d2) => d2.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(d2) {
+ let selection3 = select_default2(this);
+ if (d2.link) {
+ selection3 = selection3.append("a").attr("target", "_blank").attr("href", d2.link);
+ }
+ selection3.call(d2.msg);
+ });
}
- section.entityIDs = function(val) {
+ changesetEditor.tags = function(_2) {
if (!arguments.length)
- return _entityIDs;
- _entityIDs = val;
- return section;
+ return _tags;
+ _tags = _2;
+ return changesetEditor;
};
- return section;
- }
-
- // modules/actions/delete_members.js
- function actionDeleteMembers(relationId, memberIndexes) {
- return function(graph) {
- memberIndexes.sort((a2, b2) => b2 - a2);
- for (var i3 in memberIndexes) {
- graph = actionDeleteMember(relationId, memberIndexes[i3])(graph);
- }
- return graph;
+ changesetEditor.changesetID = function(_2) {
+ if (!arguments.length)
+ return _changesetID;
+ if (_changesetID === _2)
+ return changesetEditor;
+ _changesetID = _2;
+ _fieldsArr = null;
+ return changesetEditor;
};
+ return utilRebind(changesetEditor, dispatch14, "on");
}
- // 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 gt2 = parents.length > _maxMemberships ? ">" : "";
- var count = gt2 + 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, d2) {
- if (d2.relation)
- utilHighlightEntities([d2.relation.id], true, context);
- }).itemsMouseLeave(function(d3_event, d2) {
- if (d2.relation)
- utilHighlightEntities([d2.relation.id], false, context);
- });
- var _inChange = false;
- var _entityIDs = [];
- var _showBlank;
- var _maxMemberships = 1e3;
- function getSharedParentRelations() {
- var parents = [];
- for (var i3 = 0; i3 < _entityIDs.length; i3++) {
- var entity = context.graph().hasEntity(_entityIDs[i3]);
- if (!entity)
- continue;
- if (i3 === 0) {
- parents = context.graph().parentRelations(entity);
- } else {
- parents = utilArrayIntersection(parents, context.graph().parentRelations(entity));
- }
- if (!parents.length)
- break;
+ // 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;
}
- return parents;
- }
- function getMemberships() {
- var memberships = [];
- var relations = getSharedParentRelations().slice(0, _maxMemberships);
- var isMultiselect = _entityIDs.length > 1;
- var i3, relation, membership, index, member, indexedMember;
- for (i3 = 0; i3 < relations.length; i3++) {
- relation = relations[i3];
- 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);
+ if (rIsBool.test(sValue)) {
+ return sValue.toLowerCase() === "true";
}
- 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;
+ if (isFinite(sValue)) {
+ return parseFloat(sValue);
+ }
+ if (isFinite(Date.parse(sValue))) {
+ return new Date(sValue);
+ }
+ return sValue;
}
- function selectRelation(d3_event, d2) {
- d3_event.preventDefault();
- utilHighlightEntities([d2.relation.id], false, context);
- context.enter(modeSelect(context, [d2.relation.id]));
+ function EmptyTree() {
}
- function zoomToRelation(d3_event, d2) {
- d3_event.preventDefault();
- var entity = context.entity(d2.relation.id);
- context.map().zoomToEase(entity);
- utilHighlightEntities([d2.relation.id], true, context);
+ 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 changeRole(d3_event, d2) {
- if (d2 === 0)
- return;
- if (_inChange)
- return;
- var newRole = context.cleanRelationRole(select_default2(this).property("value"));
- if (!newRole.trim() && typeof d2.role !== "string")
- return;
- var membersToUpdate = d2.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(d2.relation.id, newMember, member.index)(graph);
- });
- return graph;
- },
- _t("operations.change_role.annotation", {
- n: membersToUpdate.length
- })
- );
- context.validator().validate();
+ 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);
+ }
+ }
}
- _inChange = false;
- }
- function addMembership(d2, role) {
- this.blur();
- _showBlank = false;
- function actionAddMembers(relationId, ids, role2) {
- return function(graph) {
- for (var i3 in ids) {
- var member = { id: ids[i3], type: graph.entity(ids[i3]).type, role: role2 };
- graph = actionAddMember(relationId, member)(graph);
+ 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]];
}
- return graph;
- };
+ vResult[sProp].push(vContent);
+ } else {
+ vResult[sProp] = vContent;
+ nLength++;
+ }
}
- if (d2.relation) {
- context.perform(
- actionAddMembers(d2.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));
+ 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 deleteMembership(d3_event, d2) {
- this.blur();
- if (d2 === 0)
- return;
- utilHighlightEntities([d2.relation.id], false, context);
- var indexes = d2.members.map(function(member) {
- return member.index;
- });
- context.perform(
- actionDeleteMembers(d2.relation.id, indexes),
- _t("operations.delete_member.annotation", {
- n: _entityIDs.length
- })
- );
- context.validator().validate();
- }
- function fetchNearbyRelations(q2, 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;
+ 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()));
}
- var explicitRelation = q2 && context.hasEntity(q2.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 (q2 && (value + " " + entity.id).toLowerCase().indexOf(q2.toLowerCase()) === -1)
- return;
- result.push({ relation: entity, value });
- });
- result.sort(function(a2, b2) {
- return osmRelation.creationOrder(a2.relation, b2.relation);
- });
- var dupeGroups = Object.values(utilArrayGroupBy(result, "value")).filter(function(v2) {
- return v2.length > 1;
- });
- dupeGroups.forEach(function(group) {
- group.forEach(function(obj) {
- obj.value += " " + obj.relation.id;
- });
- });
+ 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);
+ }
}
- result.forEach(function(obj) {
- obj.title = obj.value;
- });
- result.unshift(newRelation);
- callback(result);
}
+ 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(d2) {
+ _discardTags = d2;
+ }).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 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(d2) {
- return d2.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, d2) {
- utilHighlightEntities([d2.relation.id], true, context);
- }).on("mouseout", function(d3_event, d2) {
- utilHighlightEntities([d2.relation.id], false, context);
- });
- var labelEnter = itemsEnter.append("label").attr("class", "field-label").attr("for", function(d2) {
- return d2.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(d2) {
- var matched = _mainPresetIndex.match(d2.relation, context.graph());
- return matched && matched.name() || _t.html("inspector.relation");
- });
- labelLink.append("span").attr("class", "member-entity-name").text(function(d2) {
- return utilDisplayName(d2.relation);
+ 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(d2) {
+ select_default2(this).call(svgIcon("#iD-icon-" + d2.entity.geometry(d2.graph), "pre-text " + d2.changeType));
});
- 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(d2) {
- return d2.domId;
- }).property("type", "text").property("value", function(d2) {
- return typeof d2.role === "string" ? d2.role : "";
- }).attr("title", function(d2) {
- return Array.isArray(d2.role) ? d2.role.filter(Boolean).join("\n") : d2.role;
- }).attr("placeholder", function(d2) {
- return Array.isArray(d2.role) ? _t("inspector.multiple_roles") : _t("inspector.role");
- }).classed("mixed", function(d2) {
- return Array.isArray(d2.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();
+ buttons.append("span").attr("class", "change-type").html(function(d2) {
+ return _t.html("commit." + d2.changeType) + " ";
});
- 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();
+ buttons.append("strong").attr("class", "entity-type").text(function(d2) {
+ var matched = _mainPresetIndex.match(d2.entity, d2.graph);
+ return matched && matched.name() || utilDisplayType(d2.entity.id);
});
- function acceptEntity(d2) {
- if (!d2) {
- cancelEntity();
- return;
+ buttons.append("span").attr("class", "entity-name").text(function(d2) {
+ var name = utilDisplayName(d2.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(d2) {
+ if (d2.entity) {
+ context.surface().selectAll(
+ utilEntityOrMemberSelector([d2.entity.id], context.graph())
+ ).classed("hover", true);
}
- if (d2.relation)
- utilHighlightEntities([d2.relation.id], false, context);
- var role = context.cleanRelationRole(list.selectAll(".member-row-new .member-role").property("value"));
- addMembership(d2, role);
}
- function cancelEntity() {
- var input = newMembership.selectAll(".member-entity-input");
- input.property("value", "");
- context.surface().selectAll(".highlighted").classed("highlighted", false);
+ function mouseout() {
+ context.surface().selectAll(".hover").classed("hover", false);
}
- function bindTypeahead(d2) {
- 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 i3 = 0; i3 < data.length; i3++) {
- if (data[i3].value.substring(0, value.length) === value) {
- sameletter.push(data[i3]);
- } else {
- other.push(data[i3]);
- }
- }
- return sameletter.concat(other);
+ 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);
}
- role.call(
- uiCombobox(context, "member-role").fetcher(function(role2, callback) {
- var rtype = d2.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(d2) {
- select_default2(this).on("mouseover", function() {
- utilHighlightEntities([d2.id], true, context);
+ // 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(d2) {
+ return d2.key;
+ });
+ items.exit().remove();
+ var itemsEnter = items.enter().append("li").attr("class", issueItem);
+ var buttons = itemsEnter.append("button").on("mouseover", function(d3_event, d2) {
+ if (d2.entityIds) {
+ context.surface().selectAll(
+ utilEntityOrMemberSelector(
+ d2.entityIds,
+ context.graph()
+ )
+ ).classed("hover", true);
+ }
}).on("mouseout", function() {
- utilHighlightEntities([d2.id], false, context);
+ context.surface().selectAll(".hover").classed("hover", false);
+ }).on("click", function(d3_event, d2) {
+ context.validator().focusIssue(d2);
});
- });
- 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(d2) {
- var entity = context.entity(d2.id);
- return utilDisplayName(entity);
- });
+ buttons.call(svgIcon("#iD-icon-alert", "pre-text"));
+ buttons.append("strong").attr("class", "issue-message");
+ buttons.filter(function(d2) {
+ return d2.tooltip;
+ }).call(
+ uiTooltip().title(function(d2) {
+ return d2.tooltip;
+ }).placement("top")
+ );
+ items = itemsEnter.merge(items);
+ items.selectAll(".issue-message").text("").each(function(d2) {
+ return d2.message(context)(select_default2(this));
+ });
+ }
}
- return section;
+ return commitWarnings;
}
- // modules/ui/entity_editor.js
- function uiEntityEditor(context) {
- var dispatch14 = 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() {
- dispatch14.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) {
- dispatch14.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)
- ];
+ // 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 dispatch14 = 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);
}
- _sections.forEach(function(section) {
- if (section.entityIDs) {
- section.entityIDs(_entityIDs);
+ 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");
}
- if (section.presets) {
- section.presets(_activePresets);
+ 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 (section.tags) {
- section.tags(combinedTags);
+ }
+ if (services.improveOSM) {
+ var iOsmClosed = services.improveOSM.getClosedCounts();
+ for (itemType in iOsmClosed) {
+ tags["closed:improveosm:" + itemType] = context.cleanTagValue(iOsmClosed[itemType].toString());
}
- if (section.state) {
- section.state(_state);
+ }
+ if (services.osmose) {
+ var osmoseClosed = services.osmose.getClosedCounts();
+ for (itemType in osmoseClosed) {
+ tags["closed:osmose:" + itemType] = context.cleanTagValue(osmoseClosed[itemType].toString());
}
- body.call(section.render);
+ }
+ 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";
});
- 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)
+ 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() {
+ dispatch14.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;
- _entityIDs = _entityIDs.filter(context.hasEntity);
- if (!_entityIDs.length)
+ if (_userDetails2 === user)
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);
+ _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"));
}
- }
- function changeTags(entityIDs, changed, onInput) {
- var actions = [];
- for (var i3 in entityIDs) {
- var entityID = entityIDs[i3];
- var entity = context.entity(entityID);
- var tags = Object.assign({}, entity.tags);
- if (typeof changed === "function") {
- tags = changed(tags);
- } else {
- for (var k2 in changed) {
- if (!k2)
- continue;
- var v2 = changed[k2];
- if (typeof v2 === "object") {
- tags[k2] = tags[v2.oldKey];
- } else if (v2 !== void 0 || tags.hasOwnProperty(k2)) {
- tags[k2] = v2;
- }
+ 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() {
+ dispatch14.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);
}
- if (!onInput) {
- tags = utilCleanTags(tags);
+ });
+ 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");
}
- if (!(0, import_fast_deep_equal10.default)(entity.tags, tags)) {
- actions.push(actionChangeTags(entityID, tags));
+ }
+ return null;
+ }
+ function changeTags(_2, changed, onInput) {
+ if (changed.hasOwnProperty("comment")) {
+ if (changed.comment === void 0) {
+ changed.comment = "";
+ }
+ if (!onInput) {
+ corePreferences("comment", changed.comment);
+ corePreferences("commentDate", Date.now());
}
}
- 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 (changed.hasOwnProperty("source")) {
+ if (changed.source === void 0) {
+ corePreferences("source", null);
+ } else if (!onInput) {
+ corePreferences("source", changed.source);
+ corePreferences("commentDate", Date.now());
}
}
- if (!onInput) {
- context.validator().validate();
+ updateChangeset(changed, onInput);
+ if (_selection) {
+ _selection.call(render);
}
}
- function revertTags(keys2) {
- var actions = [];
- for (var i3 in _entityIDs) {
- var entityID = _entityIDs[i3];
- var original = context.graph().base().entities[entityID];
- var changed = {};
- for (var j3 in keys2) {
- var key = keys2[j3];
- changed[key] = original ? original.tags[key] : void 0;
+ 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;
}
- var entity = context.entity(entityID);
- var tags = Object.assign({}, entity.tags);
- for (var k2 in changed) {
- if (!k2)
- continue;
- var v2 = changed[k2];
- if (v2 !== void 0 || tags.hasOwnProperty(k2)) {
- tags[k2] = v2;
+ 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(s2) {
+ if (s2[0] !== "#") {
+ s2 = "#" + s2;
}
+ var matched = s2.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(k2) {
+ var v2 = changed[k2];
+ k2 = context.cleanTagKey(k2);
+ if (readOnlyTags.indexOf(k2) !== -1)
+ return;
+ if (v2 === void 0) {
+ delete tags[k2];
+ } else if (onInput) {
+ tags[k2] = v2;
+ } else {
+ tags[k2] = context.cleanTagValue(v2);
}
- tags = utilCleanTags(tags);
- if (!(0, import_fast_deep_equal10.default)(entity.tags, tags)) {
- actions.push(actionChangeTags(entityID, tags));
+ });
+ 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 (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;
+ if (_userDetails2 && _userDetails2.changesets_count !== void 0) {
+ var changesetsCount = parseInt(_userDetails2.changesets_count, 10) + 1;
+ tags.changesets_count = String(changesetsCount);
+ if (changesetsCount <= 100) {
+ var s2;
+ s2 = corePreferences("walkthrough_completed");
+ if (s2) {
+ tags["ideditor:walkthrough_completed"] = s2;
+ }
+ s2 = corePreferences("walkthrough_progress");
+ if (s2) {
+ tags["ideditor:walkthrough_progress"] = s2;
+ }
+ s2 = corePreferences("walkthrough_started");
+ if (s2) {
+ tags["ideditor:walkthrough_started"] = s2;
+ }
}
+ } else {
+ delete tags.changesets_count;
+ }
+ if (!(0, import_fast_deep_equal9.default)(context.changeset.tags, tags)) {
+ context.changeset = context.changeset.update({ tags });
}
- 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);
+ commit.reset = function() {
+ context.changeset = null;
};
- entityEditor.newFeature = function(val) {
- if (!arguments.length)
- return _newFeature;
- _newFeature = val;
- return entityEditor;
+ return utilRebind(commit, dispatch14, "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;
};
- function loadActivePresets(isForNewSelection) {
- var graph = context.graph();
- var counts = {};
- for (var i3 in _entityIDs) {
- var entity = graph.hasEntity(_entityIDs[i3]);
- if (!entity)
- return;
- var match = _mainPresetIndex.match(entity, graph);
- if (!counts[match.id])
- counts[match.id] = 0;
- counts[match.id] += 1;
+ return modalSelection;
+ }
+
+ // modules/ui/conflicts.js
+ function uiConflicts(context) {
+ var dispatch14 = 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();
+ dispatch14.call("save");
+ }
+ function cancel() {
+ keybindingOff();
+ dispatch14.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 matches = Object.keys(counts).sort(function(p1, p2) {
- return counts[p2] - counts[p1];
- }).map(function(pID) {
- return _mainPresetIndex.item(pID);
+ 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(d2) {
+ return d2.name;
+ }).on("click", function(d3_event, d2) {
+ d3_event.preventDefault();
+ zoomToEntity(d2.id);
});
- 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;
+ var details = conflictEnter.append("div").attr("class", "conflict-detail-container");
+ details.append("ul").attr("class", "conflict-detail-list").selectAll("li").data(function(d2) {
+ return d2.details || [];
+ }).enter().append("li").attr("class", "conflict-detail-item").html(function(d2) {
+ return d2;
+ });
+ 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(d2, i3) {
+ return i3 === 0 && index === 0 || i3 === 1 && index === _conflictList.length - 1 || null;
+ }).on("click", function(d3_event, d2) {
+ d3_event.preventDefault();
+ var container = parent.selectAll(".conflict-container");
+ var sign2 = d2 === "previous" ? -1 : 1;
+ container.selectAll(".conflict").remove();
+ container.call(showConflict, index + sign2);
+ }).each(function(d2) {
+ _t.append("save.conflict." + d2)(select_default2(this));
+ });
+ }
+ function addChoices(selection2) {
+ var choices = selection2.append("ul").attr("class", "layer-list").selectAll("li").data(function(d2) {
+ return d2.choices || [];
+ });
+ var choicesEnter = choices.enter().append("li").attr("class", "layer");
+ var labelEnter = choicesEnter.append("label");
+ labelEnter.append("input").attr("type", "radio").attr("name", function(d2) {
+ return d2.id;
+ }).on("change", function(d3_event, d2) {
+ var ul = this.parentNode.parentNode.parentNode;
+ ul.__data__.chosen = d2.id;
+ choose(d3_event, ul, d2);
+ });
+ labelEnter.append("span").text(function(d2) {
+ return d2.text;
+ });
+ choicesEnter.merge(choices).each(function(d2) {
+ var ul = this.parentNode;
+ if (ul.__data__.chosen === d2.id) {
+ choose(null, ul, d2);
+ }
+ });
+ }
+ function choose(d3_event, ul, datum2) {
+ if (d3_event)
+ d3_event.preventDefault();
+ select_default2(ul).selectAll("li").classed("active", function(d2) {
+ return d2 === datum2;
+ }).selectAll("input").property("checked", function(d2) {
+ return d2 === 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);
}
- entityEditor.presets(matches);
}
- entityEditor.presets = function(val) {
+ conflicts.conflictList = function(_2) {
if (!arguments.length)
- return _activePresets;
- if (!utilArrayIdentical(val, _activePresets)) {
- _activePresets = val;
+ return _conflictList;
+ _conflictList = _2;
+ return conflicts;
+ };
+ conflicts.origChanges = function(_2) {
+ if (!arguments.length)
+ return _origChanges;
+ _origChanges = _2;
+ return conflicts;
+ };
+ conflicts.shownEntityIds = function() {
+ if (_conflictList && typeof _shownConflictIndex === "number") {
+ return [_conflictList[_shownConflictIndex].id];
}
- return entityEditor;
+ return [];
};
- return utilRebind(entityEditor, dispatch14, "on");
+ return utilRebind(conflicts, dispatch14, "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 q2 = search.property("value"), items = list.selectAll(".feature-list-item");
- if (d3_event.keyCode === 13 && // â© Return
- q2.length && items.size()) {
- click(d3_event, items.datum());
- }
- }
- function inputevent() {
- _geocodeResults = void 0;
- drawList();
- }
- function clearSearch() {
- search.property("value", "");
- drawList();
- }
- function mapDrawn(e3) {
- if (e3.full) {
- drawList();
- }
- }
- function features() {
- var result = [];
- var graph = context.graph();
- var visibleCenter = context.map().extent().center();
- var q2 = search.property("value").toLowerCase();
- if (!q2)
- return result;
- var locationMatch = sexagesimal.pair(q2.toUpperCase()) || q2.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 && q2.match(/(?:^|\W)(node|way|relation|[nwr])\W{0,2}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(q2) < 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(a2, b2) {
- return a2.distance - b2.distance;
- });
- result = result.concat(localResults);
- (_geocodeResults || []).forEach(function(d2) {
- if (d2.osm_type && d2.osm_id) {
- var id3 = osmEntity.id.fromOSM(d2.osm_type, d2.osm_id);
- var tags = {};
- tags[d2.class] = d2.type;
- var attrs = { id: id3, type: d2.osm_type, tags };
- if (d2.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: d2.display_name,
- extent: new geoExtent(
- [Number(d2.boundingbox[3]), Number(d2.boundingbox[0])],
- [Number(d2.boundingbox[2]), Number(d2.boundingbox[1])]
- )
- });
- }
+ // 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(d2) {
+ return d2.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(d2) {
+ return d2.key;
+ });
+ containers.exit().remove();
+ var containersEnter = containers.enter().append("div").attr("class", "issue-container");
+ var itemsEnter = containersEnter.append("div").attr("class", function(d2) {
+ return "issue severity-" + d2.severity;
+ }).on("mouseover.highlight", function(d3_event, d2) {
+ var ids = d2.entityIds.filter(function(e3) {
+ return _entityIDs.indexOf(e3) === -1;
});
- if (q2.match(/^[0-9]+$/)) {
- result.push({
- id: "n" + q2,
- geometry: "point",
- type: _t("inspector.node"),
- name: q2
- });
- result.push({
- id: "w" + q2,
- geometry: "line",
- type: _t("inspector.way"),
- name: q2
+ utilHighlightEntities(ids, true, context);
+ }).on("mouseout.highlight", function(d3_event, d2) {
+ var ids = d2.entityIds.filter(function(e3) {
+ return _entityIDs.indexOf(e3) === -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, d2) {
+ makeActiveIssue(d2.id);
+ var extent = d2.extent(context.graph());
+ if (extent) {
+ var setZoom = Math.max(context.map().zoom(), 19);
+ context.map().unobscuredCenterZoomEase(extent.center(), setZoom);
+ }
+ });
+ textEnter.each(function(d2) {
+ var iconName = "#iD-icon-" + (d2.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);
});
- result.push({
- id: "r" + q2,
- geometry: "relation",
- type: _t("inspector.relation"),
- name: q2
+ } else {
+ info.classed("expanded", true).transition().duration(200).style("max-height", "200px").style("opacity", "1").on("end", function() {
+ info.style("max-height", null);
});
}
- 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"));
+ });
+ 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(d2) {
+ if (typeof d2.reference === "function") {
+ select_default2(this).call(d2.reference);
+ } else {
+ select_default2(this).call(_t.append("inspector.no_documentation_key"));
}
- 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(d2) {
- return d2.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(d2) {
- select_default2(this).call(svgIcon("#iD-icon-" + d2.geometry, "pre-text"));
- });
- label.append("span").attr("class", "entity-type").text(function(d2) {
- return d2.type;
- });
- label.append("span").attr("class", "entity-name").text(function(d2) {
- return d2.name;
- });
- enter.style("opacity", 0).transition().style("opacity", 1);
- items.order();
- items.exit().remove();
- }
- function mouseover(d3_event, d2) {
- if (d2.id === -1)
+ });
+ containers = containers.merge(containersEnter).classed("active", function(d2) {
+ return d2.id === _activeIssueID;
+ });
+ containers.selectAll(".issue-message").text("").each(function(d2) {
+ return d2.message(context)(select_default2(this));
+ });
+ var fixLists = containers.selectAll(".issue-fix-list");
+ var fixes = fixLists.selectAll(".issue-fix-item").data(function(d2) {
+ return d2.fixes ? d2.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, d2) {
+ if (select_default2(this).attr("disabled") || !d2.onClick)
return;
- utilHighlightEntities([d2.id], true, context);
- }
- function mouseout(d3_event, d2) {
- if (d2.id === -1)
+ if (d2.issue.dateLastRanFix && /* @__PURE__ */ new Date() - d2.issue.dateLastRanFix < 1e3)
return;
- utilHighlightEntities([d2.id], false, context);
- }
- function click(d3_event, d2) {
- d3_event.preventDefault();
- if (d2.location) {
- context.map().centerZoomEase([d2.location[1], d2.location[0]], 19);
- } else if (d2.entity) {
- utilHighlightEntities([d2.id], false, context);
- context.enter(modeSelect(context, [d2.entity.id]));
- context.map().zoomToEase(d2.entity);
- } else {
- context.zoomToEntity(d2.id);
- }
- }
- function geocoderSearch() {
- services.geocoder.search(search.property("value"), function(err, resp) {
- _geocodeResults = resp || [];
- drawList();
+ d2.issue.dateLastRanFix = /* @__PURE__ */ new Date();
+ utilHighlightEntities(d2.issue.entityIds.concat(d2.entityIds), false, context);
+ new Promise(function(resolve, reject) {
+ d2.onClick(context, resolve, reject);
+ if (d2.onClick.length <= 1) {
+ resolve();
+ }
+ }).then(function() {
+ context.validator().validate();
});
- }
+ }).on("mouseover.highlight", function(d3_event, d2) {
+ utilHighlightEntities(d2.entityIds, true, context);
+ }).on("mouseout.highlight", function(d3_event, d2) {
+ utilHighlightEntities(d2.entityIds, false, context);
+ });
+ buttons.each(function(d2) {
+ var iconName = d2.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(d2) {
+ return d2.title(select_default2(this));
+ });
+ fixesEnter.merge(fixes).selectAll("button").classed("actionable", function(d2) {
+ return d2.onClick;
+ }).attr("disabled", function(d2) {
+ return d2.onClick ? null : "true";
+ }).attr("title", function(d2) {
+ if (d2.disabledReason) {
+ return d2.disabledReason;
+ }
+ return null;
+ });
}
- return featureList;
+ 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/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((d2) => {
- if (!d2.comments)
- return;
- const commentEnter = comments.selectAll(".comment").data(d2.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(d4) {
- const osm = services.osm;
- let selection3 = select_default2(this);
- if (osm && d4.username) {
- selection3 = selection3.append("a").attr("class", "comment-author-link").attr("href", osm.userURL(d4.username)).attr("target", "_blank");
- }
- selection3.text((d5) => d5.username);
- });
- metadataEnter.append("div").attr("class", "comment-date").html((d4) => _t.html("note.status.commented", { when: localeDateString2(d4.timestamp) }));
- mainEnter.append("div").attr("class", "comment-text").append("p").text((d4) => d4.text);
- }).catch((err) => {
- console.log(err);
+ // modules/ui/preset_icon.js
+ function uiPresetIcon() {
+ let _preset;
+ let _geometry;
+ function presetIcon(selection2) {
+ selection2.each(render);
+ }
+ function getIcon(p2, geom) {
+ if (p2.isFallback && p2.isFallback())
+ return geom === "vertex" ? "" : "iD-icon-" + p2.id;
+ if (p2.icon)
+ return p2.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 w2 = 40;
+ const h2 = 40;
+ pointBorderEnter.append("svg").attr("class", "preset-icon-fill preset-icon-point-border").attr("width", w2).attr("height", h2).attr("viewBox", "0 0 ".concat(w2, " ").concat(h2)).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 d2 = 60;
+ let svgEnter = categoryBorderEnter.append("svg").attr("class", "preset-icon-fill preset-icon-category-border").attr("width", d2).attr("height", d2).attr("viewBox", "0 0 ".concat(d2, " ").concat(d2));
+ 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 ".concat(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 w2 = 60;
+ const h2 = 60;
+ const d2 = 40;
+ vertexFillEnter.append("svg").attr("class", "preset-icon-fill preset-icon-fill-vertex").attr("width", w2).attr("height", h2).attr("viewBox", "0 0 ".concat(w2, " ").concat(h2)).append("circle").attr("cx", w2 / 2).attr("cy", h2 / 2).attr("r", d2 / 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 d2 = 60;
+ const w2 = d2;
+ const h2 = d2;
+ const l2 = d2 * 2 / 3;
+ const c1 = (w2 - l2) / 2;
+ const c2 = c1 + l2;
+ fillEnter = fillEnter.append("svg").attr("class", "preset-icon-fill preset-icon-fill-area").attr("width", w2).attr("height", h2).attr("viewBox", "0 0 ".concat(w2, " ").concat(h2));
+ ["fill", "stroke"].forEach((klass) => {
+ fillEnter.append("path").attr("d", "M".concat(c1, " ").concat(c1, " L").concat(c1, " ").concat(c2, " L").concat(c2, " ").concat(c2, " L").concat(c2, " ").concat(c1, " Z")).attr("class", "area ".concat(klass));
+ });
+ 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, w2 / 2], [c2, w2 / 2], [h2 / 2, c1], [h2 / 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 ".concat(tagClasses));
+ fill.selectAll("path.fill").attr("class", "area fill ".concat(tagClasses));
}
- function localeDateString2(s2) {
- if (!s2)
- return null;
- const options2 = { day: "numeric", month: "short", year: "numeric" };
- const d2 = new Date(s2 * 1e3);
- if (isNaN(d2.getTime()))
- return null;
- return d2.toLocaleDateString(_mainLocalizer.localeCode(), options2);
+ function renderLine(container, drawLine, tagClasses) {
+ let line = container.selectAll(".preset-icon-line").data(drawLine ? [0] : []);
+ line.exit().remove();
+ let lineEnter = line.enter();
+ const d2 = 60;
+ const w2 = d2;
+ const h2 = d2;
+ const y2 = Math.round(d2 * 0.72);
+ const l2 = Math.round(d2 * 0.6);
+ const r2 = 2.5;
+ const x12 = (w2 - l2) / 2;
+ const x2 = x12 + l2;
+ lineEnter = lineEnter.append("svg").attr("class", "preset-icon-line").attr("width", w2).attr("height", h2).attr("viewBox", "0 0 ".concat(w2, " ").concat(h2));
+ ["casing", "stroke"].forEach((klass) => {
+ lineEnter.append("path").attr("d", "M".concat(x12, " ").concat(y2, " L").concat(x2, " ").concat(y2)).attr("class", "line ".concat(klass));
+ });
+ [[x12 - 1, y2], [x2 + 1, y2]].forEach((point2) => {
+ lineEnter.append("circle").attr("class", "vertex").attr("cx", point2[0]).attr("cy", point2[1]).attr("r", r2);
+ });
+ line = lineEnter.merge(line);
+ line.selectAll("path.stroke").attr("class", "line stroke ".concat(tagClasses));
+ line.selectAll("path.casing").attr("class", "line casing ".concat(tagClasses));
}
- issueComments.issue = function(val) {
+ function renderRoute(container, drawRoute, p2) {
+ let route = container.selectAll(".preset-icon-route").data(drawRoute ? [0] : []);
+ route.exit().remove();
+ let routeEnter = route.enter();
+ const d2 = 60;
+ const w2 = d2;
+ const h2 = d2;
+ const y12 = Math.round(d2 * 0.8);
+ const y2 = Math.round(d2 * 0.68);
+ const l2 = Math.round(d2 * 0.6);
+ const r2 = 2;
+ const x12 = (w2 - l2) / 2;
+ const x2 = x12 + l2 / 3;
+ const x3 = x2 + l2 / 3;
+ const x4 = x3 + l2 / 3;
+ routeEnter = routeEnter.append("svg").attr("class", "preset-icon-route").attr("width", w2).attr("height", h2).attr("viewBox", "0 0 ".concat(w2, " ").concat(h2));
+ ["casing", "stroke"].forEach((klass) => {
+ routeEnter.append("path").attr("d", "M".concat(x12, " ").concat(y12, " 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(y12)).attr("class", "segment1 line ".concat(klass));
+ routeEnter.append("path").attr("d", "M".concat(x3, " ").concat(y12, " L").concat(x4, " ").concat(y2)).attr("class", "segment2 line ".concat(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", r2);
+ });
+ route = routeEnter.merge(route);
+ if (drawRoute) {
+ let routeType = p2.tags.type === "waterway" ? "waterway" : p2.tags.route;
+ const segmentPresetIDs = routeSegments[routeType];
+ for (let i3 in segmentPresetIDs) {
+ const segmentPreset = _mainPresetIndex.item(segmentPresetIDs[i3]);
+ const segmentTagClasses = svgTagClasses().getClassesString(segmentPreset.tags, "");
+ route.selectAll("path.stroke.segment".concat(i3)).attr("class", "segment".concat(i3, " line stroke ").concat(segmentTagClasses));
+ route.selectAll("path.casing.segment".concat(i3)).attr("class", "segment".concat(i3, " line casing ").concat(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 isR\u00F6ntgen = picon && /^roentgen-/.test(picon);
+ const isiDIcon = picon && !(isMaki || isTemaki || isFa || isR\u00F6ntgen);
+ 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"],
+ railway: ["railway/rail", "railway/rail", "railway/rail"],
+ waterway: ["waterway/stream", "waterway/stream", "waterway/stream"]
+ };
+ function render() {
+ let p2 = _preset.apply(this, arguments);
+ let geom = _geometry ? _geometry.apply(this, arguments) : null;
+ if (geom === "relation" && p2.tags && (p2.tags.type === "route" && p2.tags.route && routeSegments[p2.tags.route] || p2.tags.type === "waterway")) {
+ geom = "route";
+ }
+ const showThirdPartyIcons = corePreferences("preferences.privacy.thirdpartyicons") || "true";
+ const isFallback = p2.isFallback && p2.isFallback();
+ const imageURL = showThirdPartyIcons === "true" && p2.imageURL;
+ const picon = getIcon(p2, geom);
+ const isCategory = !p2.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 ? p2.setTags({}, geom) : {};
+ for (let k2 in tags) {
+ if (tags[k2] === "*") {
+ tags[k2] = "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 && p2);
+ renderPointBorder(container, drawPoint);
+ renderCircleFill(container, drawVertex);
+ renderSquareFill(container, drawArea, tagClasses);
+ renderLine(container, drawLine, tagClasses);
+ renderRoute(container, drawRoute, p2);
+ renderSvgIcon(container, picon, geom, isFramed, isCategory, tagClasses);
+ renderImageIcon(container, imageURL);
+ }
+ presetIcon.preset = function(val) {
if (!arguments.length)
- return _qaItem;
- _qaItem = val;
- return issueComments;
+ return _preset;
+ _preset = utilFunctor(val);
+ return presetIcon;
};
- return issueComments;
+ presetIcon.geometry = function(val) {
+ if (!arguments.length)
+ return _geometry;
+ _geometry = utilFunctor(val);
+ return presetIcon;
+ };
+ return presetIcon;
}
- // modules/ui/improveOSM_details.js
- function uiImproveOsmDetails(context) {
- let _qaItem;
- function issueDetail(d2) {
- if (d2.desc)
- return d2.desc;
- const issueKey = d2.issueKey;
- d2.replacements = d2.replacements || {};
- d2.replacements.default = { html: _t.html("inspector.unknown") };
- return _t.html(`QA.improveOSM.error_types.${issueKey}.description`, d2.replacements);
- }
- function improveOsmDetails(selection2) {
- const details = selection2.selectAll(".error-details").data(
- _qaItem ? [_qaItem] : [],
- (d2) => `${d2.id}-${d2.status || 0}`
+ // modules/ui/sections/feature_type.js
+ function uiSectionFeatureType(context) {
+ var dispatch14 = 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")
);
- 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]));
+ 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() {
+ dispatch14.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, (d2) => d2.stringId);
+ nameparts.exit().remove();
+ nameparts.enter().append("div").attr("class", "namepart").text("").each(function(d2) {
+ d2(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 i3 in _entityIDs) {
+ var geometry = context.graph().geometry(_entityIDs[i3]);
+ 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, dispatch14, "on");
+ }
+
+ // modules/ui/sections/preset_fields.js
+ function uiSectionPresetFields(context) {
+ var section = uiSection("preset-fields", context).label(() => _t.append("inspector.fields")).disclosureContent(renderDisclosureContent);
+ var dispatch14 = 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;
+ }, {}));
+ const loc = _entityIDs.reduce(function(extent, entityID) {
+ var entity = context.graph().entity(entityID);
+ return extent.extend(entity.extent(context.graph()));
+ }, geoExtent()).center();
+ var presetsManager = _mainPresetIndex;
+ var allFields = [];
+ var allMoreFields = [];
+ var sharedTotalFields;
+ _presets.forEach(function(preset) {
+ var fields = preset.fields(loc);
+ var moreFields = preset.moreFields(loc);
+ allFields = utilArrayUnion(allFields, fields);
+ allMoreFields = utilArrayUnion(allMoreFields, moreFields);
+ if (!sharedTotalFields) {
+ sharedTotalFields = utilArrayUnion(fields, moreFields);
} else {
- context.loadEntity(entityID, (err, result) => {
- if (err)
- return;
- const entity2 = result.data.find((e3) => e3.id === entityID);
- if (entity2)
- context.enter(modeSelect(context, [entityID]));
+ sharedTotalFields = sharedTotalFields.filter(function(field) {
+ return fields.indexOf(field) !== -1 || moreFields.indexOf(field) !== -1;
});
}
});
- 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;
+ 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(t2, onInput) {
+ dispatch14.call("change", field, _entityIDs, t2, onInput);
+ }).on("revert", function(keys2) {
+ dispatch14.call("revert", field, keys2);
+ });
+ });
+ }
+ _fieldsArr.forEach(function(field) {
+ field.state(_state).tags(_tags);
});
- context.features().forceVisible(relatedEntities);
- context.map().pan([0, 0]);
+ selection2.call(
+ formFields.fieldsArr(_fieldsArr).state(_state).klass("grouped-items-area")
+ );
}
- improveOsmDetails.issue = function(val) {
+ section.presets = function(val) {
if (!arguments.length)
- return _qaItem;
- _qaItem = val;
- return improveOsmDetails;
+ return _presets;
+ if (!_presets || !val || !utilArrayIdentical(_presets, val)) {
+ _presets = val;
+ _fieldsArr = null;
+ }
+ return section;
};
- return improveOsmDetails;
- }
-
- // modules/ui/improveOSM_header.js
- function uiImproveOsmHeader() {
- let _qaItem;
- function issueTitle(d2) {
- const issueKey = d2.issueKey;
- d2.replacements = d2.replacements || {};
- d2.replacements.default = { html: _t.html("inspector.unknown") };
- return _t.html(`QA.improveOSM.error_types.${issueKey}.title`, d2.replacements);
- }
- function improveOsmHeader(selection2) {
- const header = selection2.selectAll(".qa-header").data(
- _qaItem ? [_qaItem] : [],
- (d2) => `${d2.id}-${d2.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", (d2) => d2.id < 0).append("svg").attr("width", "20px").attr("height", "30px").attr("viewbox", "0 0 20 30").attr("class", (d2) => `preset-icon-28 qaItem ${d2.service} itemId-${d2.id} itemType-${d2.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", (d2) => d2.icon ? "#" + d2.icon : "");
- headerEnter.append("div").attr("class", "qa-header-label").html(issueTitle);
- }
- improveOsmHeader.issue = function(val) {
+ section.state = function(val) {
if (!arguments.length)
- return _qaItem;
- _qaItem = val;
- return improveOsmHeader;
+ return _state;
+ _state = val;
+ return section;
};
- return improveOsmHeader;
+ 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, dispatch14, "on");
}
- // modules/ui/improveOSM_editor.js
- function uiImproveOsmEditor(context) {
- const dispatch14 = 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);
+ // 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 gt2 = entity.members.length > _maxMembers ? ">" : "";
+ var count = gt2 + 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, d2) {
+ d3_event.preventDefault();
+ select_default2(this.parentNode).classed("tag-reference-loading", true);
+ context.loadEntity(d2.id, function() {
+ section.reRender();
+ });
}
- 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] : [],
- (d2) => `${d2.id}-${d2.status || 0}`
+ function zoomToMember(d3_event, d2) {
+ d3_event.preventDefault();
+ var entity = context.entity(d2.id);
+ context.map().zoomToEase(entity);
+ utilHighlightEntities([d2.id], true, context);
+ }
+ function selectMember(d3_event, d2) {
+ d3_event.preventDefault();
+ utilHighlightEntities([d2.id], false, context);
+ var entity = context.entity(d2.id);
+ var mapExtent = context.map().extent();
+ if (!entity.intersects(mapExtent, context.graph())) {
+ context.map().zoomToEase(entity);
+ }
+ context.enter(modeSelect(context, [d2.id]));
+ }
+ function changeRole(d3_event, d2) {
+ var oldRole = d2.role;
+ var newRole = context.cleanRelationRole(select_default2(this).property("value"));
+ if (oldRole !== newRole) {
+ var member = { id: d2.id, type: d2.type, role: newRole };
+ context.perform(
+ actionChangeMember(d2.relation.id, member, d2.index),
+ _t("operations.change_role.annotation", {
+ n: 1
+ })
+ );
+ context.validator().validate();
+ }
+ }
+ function deleteMember(d3_event, d2) {
+ utilHighlightEntities([d2.id], false, context);
+ context.perform(
+ actionDeleteMember(d2.relation.id, d2.index),
+ _t("operations.delete_member.annotation", {
+ n: 1
+ })
);
- 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", (d2) => d2.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);
+ if (!context.hasEntity(d2.relation.id)) {
+ context.enter(modeBrowse(context));
+ } else {
+ context.validator().validate();
}
}
- function qaSaveButtons(selection2) {
- const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
- let buttonSection = selection2.selectAll(".buttons").data(isSelected ? [_qaItem] : [], (d2) => d2.status + d2.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", (d2) => d2.newComment ? null : true).on("click.comment", function(d3_event, d2) {
- this.blur();
- const qaService = services.improveOSM;
- if (qaService) {
- qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
- }
+ 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)
+ });
});
- buttonSection.select(".close-button").html((d2) => {
- const andComment = d2.newComment ? "_comment" : "";
- return _t.html(`QA.keepRight.close${andComment}`);
- }).on("click.close", function(d3_event, d2) {
- this.blur();
- const qaService = services.improveOSM;
- if (qaService) {
- d2.newStatus = "SOLVED";
- qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
- }
+ var list2 = selection2.selectAll(".member-list").data([0]);
+ list2 = list2.enter().append("ul").attr("class", "member-list").merge(list2);
+ var items = list2.selectAll("li").data(memberships, function(d2) {
+ return osmEntity.key(d2.relation) + "," + d2.index + "," + (d2.member ? osmEntity.key(d2.member) : "incomplete");
});
- buttonSection.select(".ignore-button").html((d2) => {
- const andComment = d2.newComment ? "_comment" : "";
- return _t.html(`QA.keepRight.ignore${andComment}`);
- }).on("click.ignore", function(d3_event, d2) {
- this.blur();
- const qaService = services.improveOSM;
- if (qaService) {
- d2.newStatus = "INVALID";
- qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
- }
+ items.exit().each(unbind).remove();
+ var itemsEnter = items.enter().append("li").attr("class", "member-row form-field").classed("member-incomplete", function(d2) {
+ return !d2.member;
});
- }
- improveOsmEditor.error = function(val) {
- if (!arguments.length)
- return _qaItem;
- _qaItem = val;
- return improveOsmEditor;
- };
- return utilRebind(improveOsmEditor, dispatch14, "on");
- }
-
- // modules/ui/preset_list.js
- function uiPresetList(context) {
- var dispatch14 = 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() {
- dispatch14.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(d2) {
- d2.choose.call(this);
+ itemsEnter.each(function(d2) {
+ var item = select_default2(this);
+ var label = item.append("label").attr("class", "field-label").attr("for", d2.domId);
+ if (d2.member) {
+ item.on("mouseover", function() {
+ utilHighlightEntities([d2.id], true, context);
+ }).on("mouseout", function() {
+ utilHighlightEntities([d2.id], false, context);
});
- }
- }
- 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
+ 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(d4) {
+ var matched = _mainPresetIndex.match(d4.member, context.graph());
+ return matched && matched.name() || utilDisplayType(d4.member.id);
+ });
+ labelLink.append("span").attr("class", "member-entity-name").classed("has-colour", (d4) => d4.member.type === "relation" && d4.member.tags.colour && isColourValid(d4.member.tags.colour)).style("border-color", (d4) => d4.member.type === "relation" && d4.member.tags.colour).text(function(d4) {
+ return utilDisplayName(d4.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 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));
+ var labelText = label.append("span").attr("class", "label-text");
+ labelText.append("span").attr("class", "member-entity-type").call(_t.append("inspector." + d2.type, { id: d2.id }));
+ labelText.append("span").attr("class", "member-entity-name").call(_t.append("inspector.incomplete", { id: d2.id }));
+ label.append("button").attr("class", "member-download").attr("title", _t("icons.download")).call(svgIcon("#iD-icon-load")).on("click", downloadMember);
}
- return collection2;
- }, []);
- var items = list.selectAll(".preset-list-item").data(collection, function(d2) {
- return d2.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);
+ 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(d2) {
+ return d2.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(d2) {
+ return d2.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 x2 = d3_event.x - dragOrigin.x, y2 = 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(x2, 2) + Math.pow(y2, 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(" + x2 + "px, " + y2 + "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, d2) {
+ 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(d2.relation.id, index, targetIndex),
+ _t("operations.reorder_members.annotation")
+ );
+ context.validator().validate();
}
- } 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;
+ })
+ );
+ function bindTypeahead(d2) {
+ 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 i3 = 0; i3 < data.length; i3++) {
+ if (data[i3].value.substring(0, value.length) === value) {
+ sameletter.push(data[i3]);
+ } else {
+ other.push(data[i3]);
+ }
}
- } else if (previousItem.select(".preset-list-button").classed("expanded")) {
- previousItem = previousItem.select(".subgrid .preset-list-item:last-child");
+ return sameletter.concat(other);
}
- if (!previousItem.empty()) {
- previousItem.select(".preset-list-button").node().focus();
+ role.call(
+ uiCombobox(context, "member-role").fetcher(function(role2, callback) {
+ var geometry;
+ if (d2.member) {
+ geometry = context.graph().geometry(d2.member.id);
+ } else if (d2.type === "relation") {
+ geometry = "relation";
+ } else if (d2.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((a2, b2) => b2 - a2);
+ for (var i3 in memberIndexes) {
+ graph = actionDeleteMember(relationId, memberIndexes[i3])(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 gt2 = parents.length > _maxMemberships ? ">" : "";
+ var count = gt2 + 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, d2) {
+ if (d2.relation)
+ utilHighlightEntities([d2.relation.id], true, context);
+ }).itemsMouseLeave(function(d3_event, d2) {
+ if (d2.relation)
+ utilHighlightEntities([d2.relation.id], false, context);
+ });
+ var _inChange = false;
+ var _entityIDs = [];
+ var _showBlank;
+ var _maxMemberships = 1e3;
+ function getSharedParentRelations() {
+ var parents = [];
+ for (var i3 = 0; i3 < _entityIDs.length; i3++) {
+ var entity = context.graph().hasEntity(_entityIDs[i3]);
+ if (!entity)
+ continue;
+ if (i3 === 0) {
+ parents = context.graph().parentRelations(entity);
} 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();
+ parents = utilArrayIntersection(parents, context.graph().parentRelations(entity));
}
- } 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());
+ if (!parents.length)
+ break;
}
+ return parents;
}
- 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);
+ function getMemberships() {
+ var memberships = [];
+ var relations = getSharedParentRelations().slice(0, _maxMemberships);
+ var isMultiselect = _entityIDs.length > 1;
+ var i3, relation, membership, index, member, indexedMember;
+ for (i3 = 0; i3 < relations.length; i3++) {
+ relation = relations[i3];
+ 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)
+ };
}
- } 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, (d2) => d2.stringId).enter().append("div").attr("class", "namepart").text("").each(function(d2) {
- d2(select_default2(this));
- });
- wrap2.call(item.reference.button);
- selection2.call(item.reference.body);
+ if (membership.members.length)
+ memberships.push(membership);
}
- item.choose = function() {
- if (select_default2(this).classed("disabled"))
- return;
- if (!context.inIntro()) {
- _mainPresetIndex.setMostRecent(preset, entityGeometries()[0]);
- }
+ 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, d2) {
+ d3_event.preventDefault();
+ utilHighlightEntities([d2.relation.id], false, context);
+ context.enter(modeSelect(context, [d2.relation.id]));
+ }
+ function zoomToRelation(d3_event, d2) {
+ d3_event.preventDefault();
+ var entity = context.entity(d2.relation.id);
+ context.map().zoomToEase(entity);
+ utilHighlightEntities([d2.relation.id], true, context);
+ }
+ function changeRole(d3_event, d2) {
+ if (d2 === 0)
+ return;
+ if (_inChange)
+ return;
+ var newRole = context.cleanRelationRole(select_default2(this).property("value"));
+ if (!newRole.trim() && typeof d2.role !== "string")
+ return;
+ var membersToUpdate = d2.members.filter(function(member) {
+ return member.role !== newRole;
+ });
+ if (membersToUpdate.length) {
+ _inChange = true;
context.perform(
- function(graph) {
- for (var i3 in _entityIDs) {
- var entityID = _entityIDs[i3];
- var oldPreset = _mainPresetIndex.match(graph.entity(entityID), graph);
- graph = actionChangePreset(entityID, oldPreset, preset)(graph);
- }
+ function actionChangeMemberRoles(graph) {
+ membersToUpdate.forEach(function(member) {
+ var newMember = Object.assign({}, member, { role: newRole });
+ delete newMember.index;
+ graph = actionChangeMember(d2.relation.id, newMember, member.index)(graph);
+ });
return graph;
},
- _t("operations.change_tags.annotation")
+ _t("operations.change_role.annotation", {
+ n: membersToUpdate.length
+ })
);
context.validator().validate();
- dispatch14.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;
+ }
+ _inChange = false;
}
- function updateForFeatureHiddenState() {
- if (!_entityIDs.every(context.hasEntity))
+ function addMembership(d2, role) {
+ this.blur();
+ _showBlank = false;
+ function actionAddMembers(relationId, ids, role2) {
+ return function(graph) {
+ for (var i3 in ids) {
+ var member = { id: ids[i3], type: graph.entity(ids[i3]).type, role: role2 };
+ graph = actionAddMember(relationId, member)(graph);
+ }
+ return graph;
+ };
+ }
+ if (d2.relation) {
+ context.perform(
+ actionAddMembers(d2.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, d2) {
+ this.blur();
+ if (d2 === 0)
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 i3 in geometries) {
- hiddenPresetFeaturesId = context.features().isHiddenPreset(item.preset, geometries[i3]);
- if (hiddenPresetFeaturesId)
- break;
+ utilHighlightEntities([d2.relation.id], false, context);
+ var indexes = d2.members.map(function(member) {
+ return member.index;
+ });
+ context.perform(
+ actionDeleteMembers(d2.relation.id, indexes),
+ _t("operations.delete_member.annotation", {
+ n: _entityIDs.length
+ })
+ );
+ context.validator().validate();
+ }
+ function fetchNearbyRelations(q2, 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 baseDisplayValue(entity) {
+ var matched = _mainPresetIndex.match(entity, graph);
+ var presetName = matched && matched.name() || _t("inspector.relation");
+ var entityName = utilDisplayName(entity) || "";
+ return presetName + " " + entityName;
+ }
+ function baseDisplayLabel(entity) {
+ var matched = _mainPresetIndex.match(entity, graph);
+ var presetName = matched && matched.name() || _t("inspector.relation");
+ var entityName = utilDisplayName(entity) || "";
+ return (selection2) => {
+ selection2.append("b").text(presetName + " ");
+ selection2.append("span").classed("has-colour", entity.tags.colour && isColourValid(entity.tags.colour)).style("border-color", entity.tags.colour).text(entityName);
+ };
+ }
+ var explicitRelation = q2 && context.hasEntity(q2.toLowerCase());
+ if (explicitRelation && explicitRelation.type === "relation" && explicitRelation.id !== entityID) {
+ result.push({
+ relation: explicitRelation,
+ value: baseDisplayValue(explicitRelation) + " " + explicitRelation.id,
+ display: baseDisplayLabel(explicitRelation)
+ });
+ } else {
+ context.history().intersects(context.map().extent()).forEach(function(entity) {
+ if (entity.type !== "relation" || entity.id === entityID)
+ return;
+ var value = baseDisplayValue(entity);
+ if (q2 && (value + " " + entity.id).toLowerCase().indexOf(q2.toLowerCase()) === -1)
+ return;
+ result.push({
+ relation: entity,
+ value,
+ display: baseDisplayLabel(entity)
+ });
+ });
+ result.sort(function(a2, b2) {
+ return osmRelation.creationOrder(a2.relation, b2.relation);
+ });
+ var dupeGroups = Object.values(utilArrayGroupBy(result, "value")).filter(function(v2) {
+ return v2.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 list2 = selection2.selectAll(".member-list").data([0]);
+ list2 = list2.enter().append("ul").attr("class", "member-list").merge(list2);
+ var items = list2.selectAll("li.member-row-normal").data(memberships, function(d2) {
+ return d2.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, d2) {
+ utilHighlightEntities([d2.relation.id], true, context);
+ }).on("mouseout", function(d3_event, d2) {
+ utilHighlightEntities([d2.relation.id], false, context);
+ });
+ var labelEnter = itemsEnter.append("label").attr("class", "field-label").attr("for", function(d2) {
+ return d2.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(d2) {
+ var matched = _mainPresetIndex.match(d2.relation, context.graph());
+ return matched && matched.name() || _t.html("inspector.relation");
+ });
+ labelLink.append("span").attr("class", "member-entity-name").classed("has-colour", (d2) => d2.relation.tags.colour && isColourValid(d2.relation.tags.colour)).style("border-color", (d2) => d2.relation.tags.colour).text(function(d2) {
+ return utilDisplayName(d2.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(d2) {
+ return d2.domId;
+ }).property("type", "text").property("value", function(d2) {
+ return typeof d2.role === "string" ? d2.role : "";
+ }).attr("title", function(d2) {
+ return Array.isArray(d2.role) ? d2.role.filter(Boolean).join("\n") : d2.role;
+ }).attr("placeholder", function(d2) {
+ return Array.isArray(d2.role) ? _t("inspector.multiple_roles") : _t("inspector.role");
+ }).classed("mixed", function(d2) {
+ return Array.isArray(d2.role);
+ }).call(utilNoAuto).on("blur", changeRole).on("change", changeRole);
+ if (taginfo) {
+ wrapEnter.each(bindTypeahead);
+ }
+ var newMembership = list2.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() {
+ list2.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();
+ list2.selectAll(".member-entity-input").node().focus();
+ });
+ function acceptEntity(d2) {
+ if (!d2) {
+ cancelEntity();
+ return;
}
- 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")
- );
+ if (d2.relation)
+ utilHighlightEntities([d2.relation.id], false, context);
+ var role = context.cleanRelationRole(list2.selectAll(".member-row-new .member-role").property("value"));
+ addMembership(d2, role);
+ }
+ function cancelEntity() {
+ var input = newMembership.selectAll(".member-entity-input");
+ input.property("value", "");
+ context.surface().selectAll(".highlighted").classed("highlighted", false);
+ }
+ function bindTypeahead(d2) {
+ 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 i3 = 0; i3 < data.length; i3++) {
+ if (data[i3].value.substring(0, value.length) === value) {
+ sameletter.push(data[i3]);
+ } else {
+ other.push(data[i3]);
+ }
+ }
+ return sameletter.concat(other);
}
- });
+ role.call(
+ uiCombobox(context, "member-role").fetcher(function(role2, callback) {
+ var rtype = d2.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);
+ }
}
- presetList.autofocus = function(val) {
- if (!arguments.length)
- return _autofocus;
- _autofocus = val;
- return presetList;
- };
- presetList.entityIDs = function(val) {
+ section.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;
+ _showBlank = false;
+ return section;
};
- presetList.presets = function(val) {
+ 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(difference2) {
+ if (difference2) {
+ section.reRender();
+ }
+ });
+ section.entityIDs = function(val) {
if (!arguments.length)
- return _currentPresets;
- _currentPresets = val;
- return presetList;
+ return _selectedIDs;
+ _selectedIDs = val;
+ return section;
};
- function entityGeometries() {
- var counts = {};
- for (var i3 in _entityIDs) {
- var entityID = _entityIDs[i3];
- 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];
- });
+ function selectEntity(d3_event, entity) {
+ context.enter(modeSelect(context, [entity.id]));
}
- return utilRebind(presetList, dispatch14, "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);
+ 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));
}
- var data = !_what || _what.isNew() ? [] : [_what];
- var link2 = selection2.selectAll(".view-on-osm").data(data, function(d2) {
- return d2.id;
+ }
+ function renderDisclosureContent(selection2) {
+ var list2 = selection2.selectAll(".feature-list").data([0]);
+ list2 = list2.enter().append("ul").attr("class", "feature-list").merge(list2);
+ var entities = _selectedIDs.map(function(id2) {
+ return context.hasEntity(id2);
+ }).filter(Boolean);
+ var items = list2.selectAll(".feature-list-item").data(entities, osmEntity.key);
+ items.exit().remove();
+ var enter = items.enter().append("li").attr("class", "feature-list-item").each(function(d2) {
+ select_default2(this).on("mouseover", function() {
+ utilHighlightEntities([d2.id], true, context);
+ }).on("mouseout", function() {
+ utilHighlightEntities([d2.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(d2) {
+ var entity = context.entity(d2.id);
+ return utilDisplayName(entity);
});
- 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(_2) {
- if (!arguments.length)
- return _what;
- _what = _2;
- return viewOnOSM;
- };
- return viewOnOSM;
+ return section;
}
- // 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);
+ // modules/ui/entity_editor.js
+ function uiEntityEditor(context) {
+ var dispatch14 = dispatch_default("choose");
var _state = "select";
+ var _coalesceChanges = false;
+ var _modified = false;
+ var _base;
var _entityIDs;
- var _newFeature = false;
- function inspector(selection2) {
- presetList.entityIDs(_entityIDs).autofocus(_newFeature).on("choose", inspector.setPreset).on("cancel", function() {
- inspector.setPreset();
+ 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-".concat(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() {
+ dispatch14.call("choose", this, _activePresets);
});
- 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;
+ 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) {
+ dispatch14.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)
+ ];
}
- 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);
+ _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(difference2) {
+ if (selection2.selectAll(".entity-editor").empty())
+ return;
+ if (_state === "hide")
+ return;
+ var significant = !difference2 || difference2.didChange.properties || difference2.didChange.addition || difference2.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);
+ }
}
- 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);
+ function changeTags(entityIDs, changed, onInput) {
+ var actions = [];
+ for (var i3 in entityIDs) {
+ var entityID = entityIDs[i3];
+ var entity = context.entity(entityID);
+ var tags = Object.assign({}, entity.tags);
+ if (typeof changed === "function") {
+ tags = changed(tags);
+ } else {
+ for (var k2 in changed) {
+ if (!k2)
+ continue;
+ var v2 = changed[k2];
+ if (typeof v2 === "object") {
+ tags[k2] = tags[v2.oldKey];
+ } else if (v2 !== void 0 || tags.hasOwnProperty(k2)) {
+ tags[k2] = v2;
+ }
+ }
+ }
+ if (!onInput) {
+ tags = utilCleanTags(tags);
+ }
+ if (!(0, import_fast_deep_equal10.default)(entity.tags, tags)) {
+ actions.push(actionChangeTags(entityID, tags));
+ }
}
- 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]);
+ 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(keys2) {
+ var actions = [];
+ for (var i3 in _entityIDs) {
+ var entityID = _entityIDs[i3];
+ var original = context.graph().base().entities[entityID];
+ var changed = {};
+ for (var j2 in keys2) {
+ var key = keys2[j2];
+ changed[key] = original ? original.tags[key] : void 0;
+ }
+ var entity = context.entity(entityID);
+ var tags = Object.assign({}, entity.tags);
+ for (var k2 in changed) {
+ if (!k2)
+ continue;
+ var v2 = changed[k2];
+ if (v2 !== void 0 || tags.hasOwnProperty(k2)) {
+ tags[k2] = v2;
+ }
+ }
+ tags = utilCleanTags(tags);
+ if (!(0, import_fast_deep_equal10.default)(entity.tags, tags)) {
+ actions.push(actionChangeTags(entityID, tags));
}
- editorPane.call(entityEditor);
}
+ 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;
};
- inspector.state = function(val) {
+ entityEditor.state = function(val) {
if (!arguments.length)
return _state;
_state = val;
- entityEditor.state(_state);
- context.container().selectAll(".field-help-body").remove();
- return inspector;
+ return entityEditor;
};
- inspector.entityIDs = function(val) {
+ entityEditor.entityIDs = function(val) {
if (!arguments.length)
return _entityIDs;
+ _base = context.graph();
+ _coalesceChanges = false;
+ if (val && _entityIDs && utilArrayIdentical(_entityIDs, val))
+ return entityEditor;
_entityIDs = val;
- return inspector;
+ loadActivePresets(true);
+ return entityEditor.modified(false);
};
- inspector.newFeature = function(val) {
+ entityEditor.newFeature = function(val) {
if (!arguments.length)
return _newFeature;
_newFeature = val;
- return inspector;
+ return entityEditor;
};
- return inspector;
- }
-
- // modules/ui/keepRight_details.js
- function uiKeepRightDetails(context) {
- let _qaItem;
- function issueDetail(d2) {
- const { itemType, parentIssueType } = d2;
- const unknown = { html: _t.html("inspector.unknown") };
- let replacements = d2.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 loadActivePresets(isForNewSelection) {
+ var graph = context.graph();
+ var counts = {};
+ for (var i3 in _entityIDs) {
+ var entity = graph.hasEntity(_entityIDs[i3]);
+ if (!entity)
+ return;
+ var match = _mainPresetIndex.match(entity, graph);
+ if (!counts[match.id])
+ counts[match.id] = 0;
+ counts[match.id] += 1;
}
- }
- function keepRightDetails(selection2) {
- const details = selection2.selectAll(".error-details").data(
- _qaItem ? [_qaItem] : [],
- (d2) => `${d2.id}-${d2.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((e3) => e3.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;
- }
- }
+ var matches = Object.keys(counts).sort(function(p1, p2) {
+ return counts[p2] - counts[p1];
+ }).map(function(pID) {
+ return _mainPresetIndex.item(pID);
});
- context.features().forceVisible(relatedEntities);
- context.map().pan([0, 0]);
+ 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);
}
- keepRightDetails.issue = function(val) {
+ entityEditor.presets = function(val) {
if (!arguments.length)
- return _qaItem;
- _qaItem = val;
- return keepRightDetails;
+ return _activePresets;
+ if (!utilArrayIdentical(val, _activePresets)) {
+ _activePresets = val;
+ }
+ return entityEditor;
+ };
+ return utilRebind(entityEditor, dispatch14, "on");
+ }
+
+ // modules/ui/feature_list.js
+ var sexagesimal = __toESM(require_sexagesimal());
+
+ // 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 keepRightDetails;
+ return mode;
}
- // modules/ui/keepRight_header.js
- function uiKeepRightHeader() {
- let _qaItem;
- function issueTitle(d2) {
- const { itemType, parentIssueType } = d2;
- const unknown = _t.html("inspector.unknown");
- let replacements = d2.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);
- }
+ // 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);
+ function defaultTags(loc) {
+ var defaultTags2 = { area: "yes" };
+ if (mode.preset)
+ defaultTags2 = mode.preset.setTags(defaultTags2, "area", false, loc);
+ return defaultTags2;
}
- function keepRightHeader(selection2) {
- const header = selection2.selectAll(".qa-header").data(
- _qaItem ? [_qaItem] : [],
- (d2) => `${d2.id}-${d2.status || 0}`
+ 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(loc) });
+ context.perform(
+ actionAddEntity(node),
+ actionAddEntity(way),
+ actionAddVertex(way.id, node.id),
+ actionClose(way.id)
);
- 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", (d2) => d2.id < 0);
- iconEnter.append("div").attr("class", (d2) => `preset-icon-28 qaItem ${d2.service} itemId-${d2.id} itemType-${d2.parentIssueType}`).call(svgIcon("#iD-icon-bolt", "qaItem-fill"));
- headerEnter.append("div").attr("class", "qa-header-label").html(issueTitle);
+ context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
}
- 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", (d2) => d2).call(svgIcon("#iD-icon-out-link", "inline"));
- linkEnter.append("span").call(_t.append("inspector.view_on_keepRight"));
+ function startFromWay(loc, edge) {
+ var startGraph = context.graph();
+ var node = osmNode({ loc });
+ var way = osmWay({ tags: defaultTags(loc) });
+ 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));
}
- viewOnKeepRight.what = function(val) {
- if (!arguments.length)
- return _qaItem;
- _qaItem = val;
- return viewOnKeepRight;
+ function startFromNode(node) {
+ var startGraph = context.graph();
+ var way = osmWay({ tags: defaultTags(node.loc) });
+ 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);
};
- return viewOnKeepRight;
+ mode.exit = function() {
+ context.uninstall(behavior);
+ };
+ return mode;
}
- // modules/ui/keepRight_editor.js
- function uiKeepRightEditor(context) {
- const dispatch14 = 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));
+ // 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);
+ function defaultTags(loc) {
+ var defaultTags2 = {};
+ if (mode.preset)
+ defaultTags2 = mode.preset.setTags(defaultTags2, "line", false, loc);
+ return defaultTags2;
}
- 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] : [],
- (d2) => `${d2.id}-${d2.status || 0}`
+ function start2(loc) {
+ var startGraph = context.graph();
+ var node = osmNode({ loc });
+ var way = osmWay({ tags: defaultTags(loc) });
+ context.perform(
+ actionAddEntity(node),
+ actionAddEntity(way),
+ actionAddVertex(way.id, node.id)
);
- 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", (d2) => d2.newComment || d2.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);
- }
+ context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
}
- function qaSaveButtons(selection2) {
- const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
- let buttonSection = selection2.selectAll(".buttons").data(isSelected ? [_qaItem] : [], (d2) => d2.status + d2.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", (d2) => d2.newComment ? null : true).on("click.comment", function(d3_event, d2) {
- this.blur();
- const qaService = services.keepRight;
- if (qaService) {
- qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
- }
- });
- buttonSection.select(".close-button").html((d2) => {
- const andComment = d2.newComment ? "_comment" : "";
- return _t.html(`QA.keepRight.close${andComment}`);
- }).on("click.close", function(d3_event, d2) {
- this.blur();
- const qaService = services.keepRight;
- if (qaService) {
- d2.newStatus = "ignore_t";
- qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
- }
- });
- buttonSection.select(".ignore-button").html((d2) => {
- const andComment = d2.newComment ? "_comment" : "";
- return _t.html(`QA.keepRight.ignore${andComment}`);
- }).on("click.ignore", function(d3_event, d2) {
- this.blur();
- const qaService = services.keepRight;
- if (qaService) {
- d2.newStatus = "ignore";
- qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
- }
- });
+ function startFromWay(loc, edge) {
+ var startGraph = context.graph();
+ var node = osmNode({ loc });
+ var way = osmWay({ tags: defaultTags(loc) });
+ context.perform(
+ actionAddEntity(node),
+ actionAddEntity(way),
+ actionAddVertex(way.id, node.id),
+ actionAddMidpoint({ loc, edge }, node)
+ );
+ context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
}
- keepRightEditor.error = function(val) {
- if (!arguments.length)
- return _qaItem;
- _qaItem = val;
- return keepRightEditor;
+ function startFromNode(node) {
+ var startGraph = context.graph();
+ var way = osmWay({ tags: defaultTags(node.loc) });
+ context.perform(
+ actionAddEntity(way),
+ actionAddVertex(way.id, node.id)
+ );
+ context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
+ }
+ mode.enter = function() {
+ context.install(behavior);
};
- return utilRebind(keepRightEditor, dispatch14, "on");
+ mode.exit = function() {
+ context.uninstall(behavior);
+ };
+ return mode;
}
- // 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));
+ // 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);
+ function defaultTags(loc) {
+ var defaultTags2 = {};
+ if (mode.preset)
+ defaultTags2 = mode.preset.setTags(defaultTags2, "point", false, loc);
+ return defaultTags2;
}
- function draw() {
- if (polygon2) {
- polygon2.data([lasso.coordinates]).attr("d", function(d2) {
- return "M" + d2.join(" L") + " Z";
- });
+ function add(loc) {
+ var node = osmNode({ loc, tags: defaultTags(loc) });
+ context.perform(
+ actionAddEntity(node),
+ _t("operations.add.annotation.point")
+ );
+ enterSelectMode(node);
+ }
+ function addWay(loc, edge) {
+ var node = osmNode({ tags: defaultTags(loc) });
+ 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) {
+ const _defaultTags = defaultTags(node.loc);
+ 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);
}
- lasso.extent = function() {
- return lasso.coordinates.reduce(function(extent, point2) {
- return extent.extend(geoExtent(point2));
- }, geoExtent());
- };
- lasso.p = function(_2) {
- if (!arguments.length)
- return lasso;
- lasso.coordinates.push(_2);
- draw();
- return lasso;
+ function cancel() {
+ context.enter(modeBrowse(context));
+ }
+ mode.enter = function() {
+ context.install(behavior);
};
- lasso.close = function() {
- if (group) {
- group.call(uiToggle(false, function() {
- select_default2(this).remove();
- }));
- }
- context.container().classed("lasso", false);
+ mode.exit = function() {
+ context.uninstall(behavior);
};
- return lasso;
+ return mode;
}
// modules/ui/note_comments.js
@@ -69465,9 +71056,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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(d2) {
+ var link3 = selection2.selectAll(".note-report").data(url ? [url] : []);
+ link3.exit().remove();
+ var linkEnter = link3.enter().append("a").attr("class", "note-report").attr("target", "_blank").attr("href", function(d2) {
return d2;
}).call(svgIcon("#iD-icon-out-link", "inline"));
linkEnter.append("span").call(_t.append("note.report"));
@@ -69481,6 +71072,33 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return noteReport;
}
+ // 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 link3 = selection2.selectAll(".view-on-osm").data(data, function(d2) {
+ return d2.id;
+ });
+ link3.exit().remove();
+ var linkEnter = link3.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(_2) {
+ if (!arguments.length)
+ return _what;
+ _what = _2;
+ return viewOnOSM;
+ };
+ return viewOnOSM;
+ }
+
// modules/ui/note_editor.js
function uiNoteEditor(context) {
var dispatch14 = dispatch_default("change");
@@ -69672,172 +71290,132 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return utilRebind(noteEditor, dispatch14, "on");
}
- // modules/ui/source_switch.js
- function uiSourceSwitch(context) {
- var keys2;
- function click(d3_event) {
- d3_event.preventDefault();
- var osm = context.connection();
- if (!osm)
+ // 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;
- if (context.inIntro())
+ 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;
- if (context.history().hasChanges() && !window.confirm(_t("source_switch.lose_changes")))
+ 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;
- 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 ? keys2[0] : keys2[1]);
}
- var sourceSwitch = function(selection2) {
- selection2.append("a").attr("href", "#").call(_t.append("source_switch.live")).attr("class", "live chip").on("click", click);
+ mode.zoomToSelected = function() {
+ if (!services.osm)
+ return;
+ var note = services.osm.getNote(selectedNoteID);
+ if (note) {
+ context.map().centerZoomEase(note.loc, 20);
+ }
};
- sourceSwitch.keys = function(_2) {
+ mode.newFeature = function(val) {
if (!arguments.length)
- return keys2;
- keys2 = _2;
- return sourceSwitch;
+ return _newFeature;
+ _newFeature = val;
+ return mode;
};
- 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);
- });
- }
+ 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/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, d2) => {
- d3_event.preventDefault();
- corePreferences("preferences.privacy.thirdpartyicons", d2 === "true" ? "false" : "true");
- });
- thirdPartyIconsEnter.append("span").call(_t.append("preferences.privacy.third_party_icons.description"));
- selection2.selectAll(".privacy-third-party-icons-item").classed("active", (d2) => d2 === "true").select("input").property("checked", (d2) => d2 === "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: '' + _t.html("splash.changelog") + "" },
- github: { html: 'github.com' }
- }));
- modalSection.append("p").html(_t.html("splash.privacy", {
- updateMessage,
- privacyLink: { html: '' + _t("splash.privacy_policy") + "" }
- }));
- 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/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")
};
- }
-
- // modules/ui/status.js
- function uiStatus(context) {
- var osm = context.connection();
- return function(selection2) {
+ var behavior = behaviorDraw(context).on("click", add).on("cancel", cancel).on("finish", cancel);
+ function add(loc) {
+ var osm = services.osm;
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();
+ 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;
}
// node_modules/osm-community-index/lib/simplify.js
var import_diacritics2 = __toESM(require_diacritics(), 1);
- function simplify(str2) {
- if (typeof str2 !== "string")
+ function simplify(str) {
+ if (typeof str !== "string")
return "";
return import_diacritics2.default.remove(
- str2.replace(/&/g, "and").replace(/(Ä°|iÌ)/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()
+ str.replace(/&/g, "and").replace(/(Ä°|iÌ)/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()
);
}
@@ -69849,13 +71427,13 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (localizerFn) {
if (itemStrings.community) {
const communityID = simplify(itemStrings.community);
- itemStrings.community = localizerFn(`_communities.${communityID}`);
+ itemStrings.community = localizerFn("_communities.".concat(communityID));
}
["name", "description", "extendedDescription"].forEach((prop) => {
if (defaultStrings[prop])
- defaultStrings[prop] = localizerFn(`_defaults.${item.type}.${prop}`);
+ defaultStrings[prop] = localizerFn("_defaults.".concat(item.type, ".").concat(prop));
if (itemStrings[prop])
- itemStrings[prop] = localizerFn(`${item.id}.${prop}`);
+ itemStrings[prop] = localizerFn("".concat(item.id, ".").concat(prop));
});
}
let replacements = {
@@ -69888,12 +71466,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return void 0;
let result = s2;
for (let key in replacements) {
- const token = `{${key}}`;
+ const token = "{".concat(key, "}");
const regex = new RegExp(token, "g");
if (regex.test(result)) {
let replacement = replacements[key];
if (!replacement) {
- throw new Error(`Cannot resolve token: ${token}`);
+ throw new Error("Cannot resolve token: ".concat(token));
} else {
if (addLinks && (key === "signupUrl" || key === "url")) {
replacement = linkify(replacement);
@@ -69904,18 +71482,18 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
const leftovers = result.match(anyToken);
if (leftovers) {
- throw new Error(`Cannot resolve tokens: ${leftovers}`);
+ throw new Error("Cannot resolve tokens: ".concat(leftovers));
}
if (addLinks && item.type === "reddit") {
result = result.replace(/(\/r\/\w+\/*)/i, (match) => linkify(resolved.url, match));
}
return result;
}
- function linkify(url, text2) {
+ function linkify(url, text) {
if (!url)
return void 0;
- text2 = text2 || url;
- return `${text2}`;
+ text = text || url;
+ return '').concat(text, "");
}
}
@@ -69988,8 +71566,20 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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: `${_changeset2.id}` }
+ changeset_id: { html: '').concat(_changeset2.id, "") }
}));
+ if (showDonationMessage !== false) {
+ const donationUrl = "https://supporting.openstreetmap.org/";
+ let supporting = body.append("div").attr("class", "save-supporting");
+ supporting.append("h3").call(_t.append("success.supporting.title"));
+ supporting.append("p").call(_t.append("success.supporting.details"));
+ table = supporting.append("table").attr("class", "supporting-table");
+ row = table.append("tr").attr("class", "supporting-row");
+ row.append("td").attr("class", "cell-icon supporting-icon").append("a").attr("target", "_blank").attr("href", donationUrl).append("svg").attr("class", "logo-small").append("use").attr("xlink:href", "#iD-donation");
+ let supportingDetail = row.append("td").attr("class", "cell-detail supporting-detail");
+ supportingDetail.append("a").attr("class", "cell-detail support-the-map").attr("target", "_blank").attr("href", donationUrl).call(_t.append("success.supporting.donation.title"));
+ supportingDetail.append("div").call(_t.append("success.supporting.donation.details"));
+ }
ensureOSMCommunityIndex().then((oci) => {
const loc = context.map().center();
const validHere = _sharedLocationManager.locationSetsAt(loc);
@@ -69998,7 +71588,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
let area = validHere[resource.locationSetID];
if (!area)
return;
- const localizer = (stringID) => _t.html(`community.${stringID}`);
+ const localizer = (stringID) => _t.html("community.".concat(stringID));
resource.resolved = resolveStrings(resource, oci.defaults, localizer);
communities.push({
area,
@@ -70016,7 +71606,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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", (d2) => d2.resolved.url).append("svg").attr("class", "logo-small").append("use").attr("xlink:href", (d2) => `#community-${d2.type}`);
+ rowEnter.append("td").attr("class", "cell-icon community-icon").append("a").attr("target", "_blank").attr("href", (d2) => d2.resolved.url).append("svg").attr("class", "logo-small").append("use").attr("xlink:href", (d2) => "#community-".concat(d2.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"));
@@ -70028,7 +71618,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
selection2.append("div").attr("class", "community-description").html(d2.resolved.descriptionHTML);
if (d2.resolved.extendedDescriptionHTML || d2.languageCodes && d2.languageCodes.length) {
selection2.append("div").call(
- uiDisclosure(context, `community-more-${d2.id}`, false).expanded(false).updatePreference(false).label(() => _t.append("success.more")).content(showMore)
+ uiDisclosure(context, "community-more-".concat(d2.id), false).expanded(false).updatePreference(false).label(() => _t.append("success.more")).content(showMore)
);
}
let nextEvents = (d2.events || []).map((event) => {
@@ -70043,7 +71633,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}).slice(0, MAXEVENTS);
if (nextEvents.length) {
selection2.append("div").call(
- uiDisclosure(context, `community-events-${d2.id}`, false).expanded(false).updatePreference(false).label(_t.html("success.events")).content(showNextEvents)
+ uiDisclosure(context, "community-events-".concat(d2.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) {
@@ -70064,7 +71654,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
itemEnter.append("div").attr("class", "community-event-name").append("a").attr("target", "_blank").attr("href", (d4) => d4.url).text((d4) => {
let name = d4.name;
if (d4.i18n && d4.id) {
- name = _t(`community.${communityID}.events.${d4.id}.name`, { default: name });
+ name = _t("community.".concat(communityID, ".events.").concat(d4.id, ".name"), { default: name });
}
return name;
});
@@ -70079,14 +71669,14 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
itemEnter.append("div").attr("class", "community-event-where").text((d4) => {
let where = d4.where;
if (d4.i18n && d4.id) {
- where = _t(`community.${communityID}.events.${d4.id}.where`, { default: where });
+ where = _t("community.".concat(communityID, ".events.").concat(d4.id, ".where"), { default: where });
}
return where;
});
itemEnter.append("div").attr("class", "community-event-description").text((d4) => {
let description = d4.description;
if (d4.i18n && d4.id) {
- description = _t(`community.${communityID}.events.${d4.id}.description`, { default: description });
+ description = _t("community.".concat(communityID, ".events.").concat(d4.id, ".description"), { default: description });
}
return description;
});
@@ -70107,572 +71697,600 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return utilRebind(success, dispatch14, "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();
+ // 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 zoomOut(d3_event) {
- if (d3_event.shiftKey)
- return;
- d3_event.preventDefault();
- context.map().zoomOut();
+ 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 zoomInFurther(d3_event) {
- if (d3_event.shiftKey)
- return;
- d3_event.preventDefault();
- context.map().zoomInFurther();
+ 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 zoomOutFurther(d3_event) {
- if (d3_event.shiftKey)
- return;
- d3_event.preventDefault();
- context.map().zoomOutFurther();
+ 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();
}
- return function(selection2) {
- var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === "rtl" ? "right" : "left").title(function(d2) {
- if (d2.disabled()) {
- return d2.disabledTitle;
- }
- return d2.title;
- }).keys(function(d2) {
- return [d2.key];
+ 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(d2) {
+ return d2.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 lastPointerUpType;
- var buttons = selection2.selectAll("button").data(zooms).enter().append("button").attr("class", function(d2) {
- return d2.id;
- }).on("pointerup.editor", function(d3_event) {
- lastPointerUpType = d3_event.pointerType;
- }).on("click.editor", function(d3_event, d2) {
- if (!d2.disabled()) {
- d2.action(d3_event);
- } else if (lastPointerUpType === "touch" || lastPointerUpType === "pen") {
- context.ui().flash.duration(2e3).iconName("#" + d2.icon).iconClass("disabled").label(d2.disabledTitle)();
- }
- lastPointerUpType = null;
- }).call(tooltipBehavior);
- buttons.each(function(d2) {
- select_default2(this).call(svgIcon("#" + d2.icon, "light"));
+ 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(d2) {
+ return d2.details || [];
+ }).enter().append("li").attr("class", "error-detail-item").text(function(d2) {
+ return d2;
});
- utilKeybinding.plusKeys.forEach(function(key) {
- context.keybinding().on([key], zoomIn);
- context.keybinding().on([uiCmd("\u2325" + key)], zoomInFurther);
+ items.exit().remove();
+ }
+ function showSuccess(changeset) {
+ commit.reset();
+ var ui = _success.changeset(changeset).location(_location).on("cancel", function() {
+ context.ui().sidebar.hide();
});
- utilKeybinding.minusKeys.forEach(function(key) {
- context.keybinding().on([key], zoomOut);
- context.keybinding().on([uiCmd("\u2325" + key)], zoomOutFurther);
+ 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 }
+ );
});
- function updateButtonStates() {
- buttons.classed("disabled", function(d2) {
- return d2.disabled();
- }).each(function() {
- var selection3 = select_default2(this);
- if (!selection3.select(".tooltip.in").empty()) {
- selection3.call(tooltipBehavior.updateContent);
+ }
+ 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();
}
});
}
- updateButtonStates();
- context.map().on("move.uiZoom", updateButtonStates);
};
+ mode.exit = function() {
+ keybindingOff();
+ context.container().selectAll(".main-content").classed("active", true).classed("inactive", false);
+ context.ui().sidebar.hide();
+ };
+ return mode;
}
- // 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(d2) {
- return d2;
- }).length;
- return _t.append("inspector.title_count", { title: _t("inspector.tags"), count });
- }).expandedByDefault(false).disclosureContent(renderDisclosureContent);
- var taginfo = services.taginfo;
- var dispatch14 = dispatch_default("change");
- var availableViews = [
- { id: "list", icon: "#fas-th-list" },
- { id: "text", icon: "#fas-i-cursor" }
- ];
- let _discardTags = {};
- _mainFileFetcher.get("discarded").then((d2) => {
- _discardTags = d2;
- }).catch(() => {
- });
- 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 i3 in missingKeys) {
- _orderedKeys.push(missingKeys[i3]);
- }
- var rowData = _orderedKeys.map(function(key, i4) {
- return { index: i4, 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(d2) {
- return d2.id;
- }).enter();
- optionEnter.append("button").attr("class", function(d2) {
- return "raw-tag-option raw-tag-option-" + d2.id + (_tagView === d2.id ? " selected" : "");
- }).attr("aria-selected", function(d2) {
- return _tagView === d2.id;
- }).attr("role", "tab").attr("title", function(d2) {
- return _t("icons." + d2.id);
- }).on("click", function(d3_event, d2) {
- _tagView = d2.id;
- corePreferences("raw-tag-editor-view", d2.id);
- wrap2.selectAll(".raw-tag-option").classed("selected", function(datum2) {
- return datum2 === d2;
- }).attr("aria-selected", function(datum2) {
- return datum2 === d2;
+ // 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((d2) => {
+ if (!d2.comments)
+ return;
+ const commentEnter = comments.selectAll(".comment").data(d2.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(d4) {
+ const osm = services.osm;
+ let selection3 = select_default2(this);
+ if (osm && d4.username) {
+ selection3 = selection3.append("a").attr("class", "comment-author-link").attr("href", osm.userURL(d4.username)).attr("target", "_blank");
+ }
+ selection3.text((d5) => d5.username);
});
- wrap2.selectAll(".tag-text").classed("hide", d2.id !== "text").each(setTextareaHeight);
- wrap2.selectAll(".tag-list, .add-row").classed("hide", d2.id !== "list");
- }).each(function(d2) {
- select_default2(this).call(svgIcon(d2.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(d2) {
- return d2.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(a2, b2) {
- return a2.index - b2.index;
- });
- items.each(function(d2) {
- 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: d2.key };
- if (typeof d2.value === "string") {
- referenceOptions.value = d2.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(d2) {
- return d2.key;
- }).call(utilGetSetValue, function(d2) {
- return d2.key;
- }).attr("readonly", function(d2) {
- return isReadOnly(d2) || null;
- });
- items.selectAll("input.value").attr("title", function(d2) {
- return Array.isArray(d2.value) ? d2.value.filter(Boolean).join("\n") : d2.value;
- }).classed("mixed", function(d2) {
- return Array.isArray(d2.value);
- }).attr("placeholder", function(d2) {
- return typeof d2.value === "string" ? null : _t("inspector.multiple_values");
- }).call(utilGetSetValue, function(d2) {
- return typeof d2.value === "string" ? d2.value : "";
- }).attr("readonly", function(d2) {
- return isReadOnly(d2) || null;
+ metadataEnter.append("div").attr("class", "comment-date").html((d4) => _t.html("note.status.commented", { when: localeDateString2(d4.timestamp) }));
+ mainEnter.append("div").attr("class", "comment-text").append("p").text((d4) => d4.text);
+ }).catch((err) => {
+ console.log(err);
});
- items.selectAll("button.remove").on(
- ("PointerEvent" in window ? "pointer" : "mouse") + "down",
- // 'click' fires too late - #5878
- (d3_event, d2) => {
- if (d3_event.button !== 0)
- return;
- removeTag(d3_event, d2);
- }
- );
}
- function isReadOnly(d2) {
- for (var i3 = 0; i3 < _readOnlyTags.length; i3++) {
- if (d2.key.match(_readOnlyTags[i3]) !== null) {
- return true;
+ function localeDateString2(s2) {
+ if (!s2)
+ return null;
+ const options2 = { day: "numeric", month: "short", year: "numeric" };
+ const d2 = new Date(s2 * 1e3);
+ if (isNaN(d2.getTime()))
+ return null;
+ return d2.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(d2) {
+ if (d2.desc)
+ return d2.desc;
+ const issueKey = d2.issueKey;
+ d2.replacements = d2.replacements || {};
+ d2.replacements.default = { html: _t.html("inspector.unknown") };
+ return _t.html("QA.improveOSM.error_types.".concat(issueKey, ".description"), d2.replacements);
+ }
+ function improveOsmDetails(selection2) {
+ const details = selection2.selectAll(".error-details").data(
+ _qaItem ? [_qaItem] : [],
+ (d2) => "".concat(d2.id, "-").concat(d2.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 link3 = select_default2(this);
+ const isObjectLink = link3.classed("error_object_link");
+ const entityID = isObjectLink ? utilEntityRoot(_qaItem.objectType) + _qaItem.objectId : this.textContent;
+ const entity = context.hasEntity(entityID);
+ relatedEntities.push(entityID);
+ link3.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((e3) => e3.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;
+ }
}
- }
- return false;
+ });
+ context.features().forceVisible(relatedEntities);
+ context.map().pan([0, 0]);
}
- 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");
+ 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(d2) {
+ const issueKey = d2.issueKey;
+ d2.replacements = d2.replacements || {};
+ d2.replacements.default = { html: _t.html("inspector.unknown") };
+ return _t.html("QA.improveOSM.error_types.".concat(issueKey, ".title"), d2.replacements);
}
- function stringify3(s2) {
- return JSON.stringify(s2).slice(1, -1);
+ function improveOsmHeader(selection2) {
+ const header = selection2.selectAll(".qa-header").data(
+ _qaItem ? [_qaItem] : [],
+ (d2) => "".concat(d2.id, "-").concat(d2.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", (d2) => d2.id < 0).append("svg").attr("width", "20px").attr("height", "30px").attr("viewbox", "0 0 20 30").attr("class", (d2) => "preset-icon-28 qaItem ".concat(d2.service, " itemId-").concat(d2.id, " itemType-").concat(d2.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", (d2) => d2.icon ? "#" + d2.icon : "");
+ headerEnter.append("div").attr("class", "qa-header-label").html(issueTitle);
}
- function unstringify(s2) {
- var leading = "";
- var trailing = "";
- if (s2.length < 1 || s2.charAt(0) !== '"') {
- leading = '"';
- }
- if (s2.length < 2 || s2.charAt(s2.length - 1) !== '"' || s2.charAt(s2.length - 1) === '"' && s2.charAt(s2.length - 2) === "\\") {
- trailing = '"';
- }
- return JSON.parse(leading + s2 + trailing);
+ improveOsmHeader.issue = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return improveOsmHeader;
+ };
+ return improveOsmHeader;
+ }
+
+ // modules/ui/improveOSM_editor.js
+ function uiImproveOsmEditor(context) {
+ const dispatch14 = 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 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";
+ 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] : [],
+ (d2) => "".concat(d2.id, "-").concat(d2.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", (d2) => d2.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);
}
- return str2;
}
- function textChanged() {
- var newText = this.value.trim();
- var newTags = {};
- newText.split("\n").forEach(function(row) {
- var m2 = row.match(/^\s*([^=]+)=(.*)$/);
- if (m2 !== null) {
- var k2 = context.cleanTagKey(unstringify(m2[1].trim()));
- var v2 = context.cleanTagValue(unstringify(m2[2].trim()));
- newTags[k2] = v2;
+ function qaSaveButtons(selection2) {
+ const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+ let buttonSection = selection2.selectAll(".buttons").data(isSelected ? [_qaItem] : [], (d2) => d2.status + d2.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", (d2) => d2.newComment ? null : true).on("click.comment", function(d3_event, d2) {
+ this.blur();
+ const qaService = services.improveOSM;
+ if (qaService) {
+ qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
}
});
- 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 || "";
+ buttonSection.select(".close-button").html((d2) => {
+ const andComment = d2.newComment ? "_comment" : "";
+ return _t.html("QA.keepRight.close".concat(andComment));
+ }).on("click.close", function(d3_event, d2) {
+ this.blur();
+ const qaService = services.improveOSM;
+ if (qaService) {
+ d2.newStatus = "SOLVED";
+ qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
+ }
+ });
+ buttonSection.select(".ignore-button").html((d2) => {
+ const andComment = d2.newComment ? "_comment" : "";
+ return _t.html("QA.keepRight.ignore".concat(andComment));
+ }).on("click.ignore", function(d3_event, d2) {
+ this.blur();
+ const qaService = services.improveOSM;
+ if (qaService) {
+ d2.newStatus = "INVALID";
+ qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
}
});
- 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();
+ improveOsmEditor.error = function(val) {
+ if (!arguments.length)
+ return _qaItem;
+ _qaItem = val;
+ return improveOsmEditor;
+ };
+ return utilRebind(improveOsmEditor, dispatch14, "on");
+ }
+
+ // modules/ui/keepRight_details.js
+ function uiKeepRightDetails(context) {
+ let _qaItem;
+ function issueDetail(d2) {
+ const { itemType, parentIssueType } = d2;
+ const unknown = { html: _t.html("inspector.unknown") };
+ let replacements = d2.replacements || {};
+ replacements.default = unknown;
+ if (_mainLocalizer.hasTextForStringId("QA.keepRight.errorTypes.".concat(itemType, ".title"))) {
+ return _t.html("QA.keepRight.errorTypes.".concat(itemType, ".description"), replacements);
+ } else {
+ return _t.html("QA.keepRight.errorTypes.".concat(parentIssueType, ".description"), replacements);
}
}
- 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) {
- const filtered = data.filter((d2) => _tags[d2.value] === void 0).filter((d2) => !(d2.value in _discardTags)).filter((d2) => d2.value.toLowerCase().includes(value2.toLowerCase()));
- callback(sort(value2, filtered));
+ function keepRightDetails(selection2) {
+ const details = selection2.selectAll(".error-details").data(
+ _qaItem ? [_qaItem] : [],
+ (d2) => "".concat(d2.id, "-").concat(d2.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 link3 = select_default2(this);
+ const isObjectLink = link3.classed("error_object_link");
+ const entityID = isObjectLink ? utilEntityRoot(_qaItem.objectType) + _qaItem.objectId : this.textContent;
+ const entity = context.hasEntity(entityID);
+ relatedEntities.push(entityID);
+ link3.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);
}
- });
- }));
- 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) {
- const filtered = data.filter((d2) => d2.value.toLowerCase().includes(value2.toLowerCase()));
- callback(sort(value2, filtered));
+ 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((e3) => e3.id === entityID);
+ if (entity2)
+ context.enter(modeSelect(context, [entityID]));
+ });
}
});
- }).caseSensitive(allowUpperCaseTagValues.test(utilGetSetValue(key))));
- function sort(value2, data) {
- var sameletter = [];
- var other = [];
- for (var i3 = 0; i3 < data.length; i3++) {
- if (data[i3].value.substring(0, value2.length) === value2) {
- sameletter.push(data[i3]);
- } else {
- other.push(data[i3]);
+ if (entity) {
+ let name = utilDisplayName(entity);
+ if (!name && !isObjectLink) {
+ const preset = _mainPresetIndex.match(entity, context.graph());
+ name = preset && !preset.isFallback() && preset.name();
}
- }
- 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, d2) {
- if (select_default2(this).attr("readonly"))
- return;
- var kOld = d2.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(d4) {
- if (d4.key === kNew) {
- var input = select_default2(this).node();
- input.focus();
- input.select();
+ if (name) {
+ this.innerText = name;
}
- });
- 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;
- d2.key = kNew;
- this.value = kNew;
- scheduleChange();
- }
- function valueChange(d3_event, d2) {
- if (isReadOnly(d2))
- return;
- if (typeof d2.value !== "string" && !this.value)
- return;
- if (_pendingChange && _pendingChange.hasOwnProperty(d2.key) && _pendingChange[d2.key] === void 0)
- return;
- _pendingChange = _pendingChange || {};
- _pendingChange[d2.key] = context.cleanTagValue(this.value);
- scheduleChange();
+ }
+ });
+ context.features().forceVisible(relatedEntities);
+ context.map().pan([0, 0]);
}
- function removeTag(d3_event, d2) {
- if (isReadOnly(d2))
- return;
- if (d2.key === "") {
- _showBlank = false;
- section.reRender();
+ 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(d2) {
+ const { itemType, parentIssueType } = d2;
+ const unknown = _t.html("inspector.unknown");
+ let replacements = d2.replacements || {};
+ replacements.default = { html: unknown };
+ if (_mainLocalizer.hasTextForStringId("QA.keepRight.errorTypes.".concat(itemType, ".title"))) {
+ return _t.html("QA.keepRight.errorTypes.".concat(itemType, ".title"), replacements);
} else {
- _orderedKeys = _orderedKeys.filter(function(key) {
- return key !== d2.key;
- });
- _pendingChange = _pendingChange || {};
- _pendingChange[d2.key] = void 0;
- scheduleChange();
+ return _t.html("QA.keepRight.errorTypes.".concat(parentIssueType, ".title"), replacements);
}
}
- 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;
- dispatch14.call("change", this, entityIDs, _pendingChange);
- _pendingChange = null;
- }, 10);
+ function keepRightHeader(selection2) {
+ const header = selection2.selectAll(".qa-header").data(
+ _qaItem ? [_qaItem] : [],
+ (d2) => "".concat(d2.id, "-").concat(d2.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", (d2) => d2.id < 0);
+ iconEnter.append("div").attr("class", (d2) => "preset-icon-28 qaItem ".concat(d2.service, " itemId-").concat(d2.id, " itemType-").concat(d2.parentIssueType)).call(svgIcon("#iD-icon-bolt", "qaItem-fill"));
+ headerEnter.append("div").attr("class", "qa-header-label").html(issueTitle);
}
- 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) {
+ keepRightHeader.issue = function(val) {
if (!arguments.length)
- return _tags;
- _tags = val;
- return section;
+ return _qaItem;
+ _qaItem = val;
+ return keepRightHeader;
};
- section.entityIDs = function(val) {
- if (!arguments.length)
- return _entityIDs;
- if (!_entityIDs || !val || !utilArrayIdentical(_entityIDs, val)) {
- _entityIDs = val;
- _orderedKeys = [];
+ 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);
}
- return section;
- };
- section.readOnlyTags = function(val) {
+ const link3 = selection2.selectAll(".view-on-keepRight").data(url ? [url] : []);
+ link3.exit().remove();
+ const linkEnter = link3.enter().append("a").attr("class", "view-on-keepRight").attr("target", "_blank").attr("rel", "noopener").attr("href", (d2) => d2).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 _readOnlyTags;
- _readOnlyTags = val;
- return section;
+ return _qaItem;
+ _qaItem = val;
+ return viewOnKeepRight;
};
- return utilRebind(section, dispatch14, "on");
+ return viewOnKeepRight;
}
- // 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]);
+ // modules/ui/keepRight_editor.js
+ function uiKeepRightEditor(context) {
+ const dispatch14 = 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);
- 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);
+ 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] : [],
+ (d2) => "".concat(d2.id, "-").concat(d2.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", (d2) => d2.newComment || d2.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);
+ }
}
- dataEditor.datum = function(val) {
+ function qaSaveButtons(selection2) {
+ const isSelected = _qaItem && _qaItem.id === context.selectedErrorID();
+ let buttonSection = selection2.selectAll(".buttons").data(isSelected ? [_qaItem] : [], (d2) => d2.status + d2.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", (d2) => d2.newComment ? null : true).on("click.comment", function(d3_event, d2) {
+ this.blur();
+ const qaService = services.keepRight;
+ if (qaService) {
+ qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
+ }
+ });
+ buttonSection.select(".close-button").html((d2) => {
+ const andComment = d2.newComment ? "_comment" : "";
+ return _t.html("QA.keepRight.close".concat(andComment));
+ }).on("click.close", function(d3_event, d2) {
+ this.blur();
+ const qaService = services.keepRight;
+ if (qaService) {
+ d2.newStatus = "ignore_t";
+ qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
+ }
+ });
+ buttonSection.select(".ignore-button").html((d2) => {
+ const andComment = d2.newComment ? "_comment" : "";
+ return _t.html("QA.keepRight.ignore".concat(andComment));
+ }).on("click.ignore", function(d3_event, d2) {
+ this.blur();
+ const qaService = services.keepRight;
+ if (qaService) {
+ d2.newStatus = "ignore";
+ qaService.postUpdate(d2, (err, item) => dispatch14.call("change", item));
+ }
+ });
+ }
+ keepRightEditor.error = function(val) {
if (!arguments.length)
- return _datum;
- _datum = val;
- return this;
+ return _qaItem;
+ _qaItem = val;
+ return keepRightEditor;
};
- return dataEditor;
+ return utilRebind(keepRightEditor, dispatch14, "on");
}
// modules/ui/osmose_details.js
@@ -70687,7 +72305,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
function osmoseDetails(selection2) {
const details = selection2.selectAll(".error-details").data(
_qaItem ? [_qaItem] : [],
- (d2) => `${d2.id}-${d2.status || 0}`
+ (d2) => "".concat(d2.id, "-").concat(d2.status || 0)
);
details.exit().remove();
const detailsEnter = details.enter().append("div").attr("class", "error-details qa-details-container");
@@ -70712,7 +72330,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
services.osmose.loadIssueDetail(_qaItem).then((d2) => {
if (!d2.elems || d2.elems.length === 0)
return;
- if (context.selectedErrorID() !== thisItem.id && context.container().selectAll(`.qaItem.osmose.hover.itemId-${thisItem.id}`).empty())
+ if (context.selectedErrorID() !== thisItem.id && context.container().selectAll(".qaItem.osmose.hover.itemId-".concat(thisItem.id)).empty())
return;
if (d2.detail) {
detailsDiv.append("h4").call(_t.append("QA.osmose.detail_title"));
@@ -70720,10 +72338,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
elemsDiv.append("h4").call(_t.append("QA.osmose.elems_title"));
elemsDiv.append("ul").selectAll("li").data(d2.elems).enter().append("li").append("a").attr("href", "#").attr("class", "error_entity_link").text((d4) => d4).each(function() {
- const link2 = select_default2(this);
+ const link3 = select_default2(this);
const entityID = this.textContent;
const entity = context.hasEntity(entityID);
- link2.on("mouseenter", () => {
+ link3.on("mouseenter", () => {
utilHighlightEntities([entityID], true, context);
}).on("mouseleave", () => {
utilHighlightEntities([entityID], false, context);
@@ -70786,11 +72404,11 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
function osmoseHeader(selection2) {
const header = selection2.selectAll(".qa-header").data(
_qaItem ? [_qaItem] : [],
- (d2) => `${d2.id}-${d2.status || 0}`
+ (d2) => "".concat(d2.id, "-").concat(d2.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", (d2) => d2.id < 0).append("svg").attr("width", "20px").attr("height", "30px").attr("viewbox", "0 0 20 30").attr("class", (d2) => `preset-icon-28 qaItem ${d2.service} itemId-${d2.id} itemType-${d2.itemType}`);
+ const svgEnter = headerEnter.append("div").attr("class", "qa-header-icon").classed("new", (d2) => d2.id < 0).append("svg").attr("width", "20px").attr("height", "30px").attr("viewbox", "0 0 20 30").attr("class", (d2) => "preset-icon-28 qaItem ".concat(d2.service, " itemId-").concat(d2.id, " itemType-").concat(d2.itemType));
svgEnter.append("polygon").attr("fill", (d2) => services.osmose.getColor(d2.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", (d2) => d2.icon ? "#" + d2.icon : "");
headerEnter.append("div").attr("class", "qa-header-label").text(issueTitle);
@@ -70812,9 +72430,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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", (d2) => d2).call(svgIcon("#iD-icon-out-link", "inline"));
+ const link3 = selection2.selectAll(".view-on-osmose").data(url ? [url] : []);
+ link3.exit().remove();
+ const linkEnter = link3.enter().append("a").attr("class", "view-on-osmose").attr("target", "_blank").attr("rel", "noopener").attr("href", (d2) => d2).call(svgIcon("#iD-icon-out-link", "inline"));
linkEnter.append("span").call(_t.append("inspector.view_on_osmose"));
}
viewOnOsmose.what = function(val) {
@@ -70849,7 +72467,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
const isShown = _qaItem && isSelected;
let saveSection = selection2.selectAll(".qa-save").data(
isShown ? [_qaItem] : [],
- (d2) => `${d2.id}-${d2.status || 0}`
+ (d2) => "".concat(d2.id, "-").concat(d2.status || 0)
);
saveSection.exit().remove();
const saveSectionEnter = saveSection.enter().append("div").attr("class", "qa-save save-section cf");
@@ -70889,825 +72507,1828 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return utilRebind(osmoseEditor, dispatch14, "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)
+ // 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;
- if ("button" in d3_event && d3_event.button !== 0)
+ 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;
- 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);
+ context.enter(modeBrowse(context));
}
- function pointermove(d3_event) {
- if (downPointerId !== (d3_event.pointerId || "mouse"))
+ };
+ 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/feature_list.js
+ 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 list2 = 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 q2 = search.property("value"), items = list2.selectAll(".feature-list-item");
+ if (d3_event.keyCode === 13 && // â© Return
+ q2.length && items.size()) {
+ click(d3_event, items.datum());
+ }
+ }
+ function inputevent() {
+ _geocodeResults = void 0;
+ drawList();
+ }
+ function clearSearch() {
+ search.property("value", "");
+ drawList();
+ }
+ function mapDrawn(e3) {
+ if (e3.full) {
+ drawList();
+ }
+ }
+ function features() {
+ var result = [];
+ var graph = context.graph();
+ var visibleCenter = context.map().extent().center();
+ var q2 = search.property("value").toLowerCase();
+ if (!q2)
+ return result;
+ var locationMatch = sexagesimal.pair(q2.toUpperCase()) || dmsMatcher(q2);
+ 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 && q2.match(/(?:^|\W)(node|way|relation|note|[nwr])\W{0,2}0*([1-9]\d*)(?:\W|$)/i);
+ if (idMatch) {
+ var elemType = idMatch[1] === "note" ? idMatch[1] : idMatch[1].charAt(0);
+ var elemId = idMatch[2];
+ result.push({
+ id: elemType + elemId,
+ geometry: elemType === "n" ? "point" : elemType === "w" ? "line" : elemType === "note" ? "note" : "relation",
+ type: elemType === "n" ? _t("inspector.node") : elemType === "w" ? _t("inspector.way") : elemType === "note" ? _t("note.note") : _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(q2) < 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(a2, b2) {
+ return a2.distance - b2.distance;
+ });
+ result = result.concat(localResults);
+ (_geocodeResults || []).forEach(function(d2) {
+ if (d2.osm_type && d2.osm_id) {
+ var id3 = osmEntity.id.fromOSM(d2.osm_type, d2.osm_id);
+ var tags = {};
+ tags[d2.class] = d2.type;
+ var attrs = { id: id3, type: d2.osm_type, tags };
+ if (d2.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: d2.display_name,
+ extent: new geoExtent(
+ [Number(d2.boundingbox[3]), Number(d2.boundingbox[0])],
+ [Number(d2.boundingbox[2]), Number(d2.boundingbox[1])]
+ )
+ });
+ }
+ });
+ if (q2.match(/^[0-9]+$/)) {
+ result.push({
+ id: "n" + q2,
+ geometry: "point",
+ type: _t("inspector.node"),
+ name: q2
+ });
+ result.push({
+ id: "w" + q2,
+ geometry: "line",
+ type: _t("inspector.way"),
+ name: q2
+ });
+ result.push({
+ id: "r" + q2,
+ geometry: "relation",
+ type: _t("inspector.relation"),
+ name: q2
+ });
+ result.push({
+ id: "note" + q2,
+ geometry: "note",
+ type: _t("note.note"),
+ name: q2
+ });
+ }
+ return result;
+ }
+ function drawList() {
+ var value = search.property("value");
+ var results = features();
+ list2.classed("filtered", value.length);
+ var resultsIndicator = list2.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");
+ list2.selectAll(".no-results-item .entity-name").html("").call(_t.append("geocoder.no_results_worldwide"));
+ if (services.geocoder) {
+ list2.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"));
+ }
+ list2.selectAll(".no-results-item").style("display", value.length && !results.length ? "block" : "none");
+ list2.selectAll(".geocode-item").style("display", value && _geocodeResults === void 0 ? "block" : "none");
+ list2.selectAll(".feature-list-item").data([-1]).remove();
+ var items = list2.selectAll(".feature-list-item").data(results, function(d2) {
+ return d2.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(d2) {
+ select_default2(this).call(svgIcon("#iD-icon-" + d2.geometry, "pre-text"));
+ });
+ label.append("span").attr("class", "entity-type").text(function(d2) {
+ return d2.type;
+ });
+ label.append("span").attr("class", "entity-name").classed("has-colour", (d2) => d2.entity && d2.entity.type === "relation" && d2.entity.tags.colour && isColourValid(d2.entity.tags.colour)).style("border-color", (d2) => d2.entity && d2.entity.type === "relation" && d2.entity.tags.colour).text(function(d2) {
+ return d2.name;
+ });
+ enter.style("opacity", 0).transition().style("opacity", 1);
+ items.order();
+ items.exit().remove();
+ }
+ function mouseover(d3_event, d2) {
+ if (d2.id === -1)
+ return;
+ utilHighlightEntities([d2.id], true, context);
+ }
+ function mouseout(d3_event, d2) {
+ if (d2.id === -1)
return;
+ utilHighlightEntities([d2.id], false, context);
+ }
+ function click(d3_event, d2) {
+ d3_event.preventDefault();
+ if (d2.location) {
+ context.map().centerZoomEase([d2.location[1], d2.location[0]], 19);
+ } else if (d2.entity) {
+ utilHighlightEntities([d2.id], false, context);
+ context.enter(modeSelect(context, [d2.entity.id]));
+ context.map().zoomToEase(d2.entity);
+ } else if (d2.geometry === "note") {
+ const noteId = d2.id.replace(/\D/g, "");
+ context.loadNote(noteId, (err, result) => {
+ if (err)
+ return;
+ const entity = result.data.find((e3) => e3.id === noteId);
+ if (entity) {
+ const note = services.osm.getNote(noteId);
+ context.map().centerZoom(note.loc, 15);
+ const noteLayer = context.layers().layer("notes");
+ noteLayer.enabled(true);
+ context.enter(modeSelectNote(context, noteId));
+ }
+ });
+ } else {
+ context.zoomToEntity(d2.id);
+ }
+ }
+ function geocoderSearch() {
+ services.geocoder.search(search.property("value"), function(err, resp) {
+ _geocodeResults = resp || [];
+ drawList();
+ });
+ }
+ }
+ return featureList;
+ }
+
+ // modules/ui/preset_list.js
+ function uiPresetList(context) {
+ var dispatch14 = 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() {
+ dispatch14.call("cancel", this);
+ }).call(svgIcon("#iD-icon-".concat(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 = list2.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) {
+ list2.selectAll(".preset-list-item:first-child").each(function(d2) {
+ d2.choose.call(this);
+ });
+ }
+ }
+ function inputevent() {
+ var value = search.property("value");
+ list2.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");
+ }
+ list2.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 list2 = 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(list2, 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 = list2.selectAll(".preset-list-item").data(collection, function(d2) {
+ return d2.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();
- 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 x2 = containerLocGetter(d3_event)[0] - dragOffset;
- sidebarWidth = isRTL ? containerWidth - x2 : x2;
- 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]);
+ d3_event.stopPropagation();
+ var nextItem = select_default2(item.node().nextElementSibling);
+ if (nextItem.empty()) {
+ if (!parentItem.empty()) {
+ nextItem = select_default2(parentItem.node().nextElementSibling);
}
- } 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]);
+ } 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 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);
- }
+ }
+ 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();
}
- };
- 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;
+ 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 {
- errEditor = improveOsmEditor;
- }
- context.container().selectAll(".qaItem." + datum2.service).classed("hover", function(d2) {
- return d2.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);
+ itemKeydown.call(this, d3_event);
}
- } 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();
- }
+ });
+ 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");
}
- 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);
+ 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 {
- inspector.state("hide");
+ 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");
}
};
- 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);
+ 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, (d2) => d2.stringId).enter().append("div").attr("class", "namepart").text("").each(function(d2) {
+ d2(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 i3 in _entityIDs) {
+ var entityID = _entityIDs[i3];
+ var oldPreset = _mainPresetIndex.match(graph.entity(entityID), graph);
+ graph = actionChangePreset(entityID, oldPreset, preset)(graph);
+ }
+ return graph;
+ },
+ _t("operations.change_tags.annotation")
+ );
+ context.validator().validate();
+ dispatch14.call("choose", this, preset);
};
- sidebar.collapse = function(moveMap) {
- if (!selection2.classed("collapsed")) {
- sidebar.toggle(moveMap);
- }
+ item.help = function(d3_event) {
+ d3_event.stopPropagation();
+ item.reference.toggle();
};
- 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;
+ 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 i3 in geometries) {
+ hiddenPresetFeaturesId = context.features().isHiddenPreset(item.preset, geometries[i3]);
+ if (hiddenPresetFeaturesId)
+ break;
}
- if (!isCollapsing) {
- selection2.classed("collapsed", isCollapsing);
+ 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")
+ );
}
- selection2.transition().style(xMarginProperty, endMargin + "px").tween("panner", function() {
- var i3 = number_default(startMargin, endMargin);
- return function(t2) {
- var dx = lastMargin - Math.round(i3(t2));
- 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 + "%");
- }
+ });
+ }
+ 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());
});
- };
- resizer.on("dblclick", function(d3_event) {
- d3_event.preventDefault();
- if (d3_event.sourceEvent) {
- d3_event.sourceEvent.preventDefault();
+ presetList.presets(presets);
+ }
+ return presetList;
+ };
+ presetList.presets = function(val) {
+ if (!arguments.length)
+ return _currentPresets;
+ _currentPresets = val;
+ return presetList;
+ };
+ function entityGeometries() {
+ var counts = {};
+ for (var i3 in _entityIDs) {
+ var entityID = _entityIDs[i3];
+ var entity = context.entity(entityID);
+ var geometry = entity.geometry(context.graph());
+ if (geometry === "vertex" && entity.isOnAddressLine(context.graph())) {
+ geometry = "point";
}
- sidebar.toggle();
+ if (!counts[geometry])
+ counts[geometry] = 0;
+ counts[geometry] += 1;
+ }
+ return Object.keys(counts).sort(function(geom1, geom2) {
+ return counts[geom2] - counts[geom1];
});
- context.map().on("crossEditableZoom.sidebar", function(within) {
- if (!within && !selection2.select(".inspector-hover").empty()) {
- hover([]);
- }
+ }
+ return utilRebind(presetList, dispatch14, "on");
+ }
+
+ // 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]))
+ );
}
- sidebar.showPresetList = function() {
- };
- sidebar.hover = function() {
- };
- sidebar.hover.cancel = function() {
+ 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));
};
- sidebar.intersects = function() {
+ 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);
+ }
};
- sidebar.select = function() {
+ inspector.state = function(val) {
+ if (!arguments.length)
+ return _state;
+ _state = val;
+ entityEditor.state(_state);
+ context.container().selectAll(".field-help-body").remove();
+ return inspector;
};
- sidebar.show = function() {
+ inspector.entityIDs = function(val) {
+ if (!arguments.length)
+ return _entityIDs;
+ _entityIDs = val;
+ return inspector;
};
- sidebar.hide = function() {
+ inspector.newFeature = function(val) {
+ if (!arguments.length)
+ return _newFeature;
+ _newFeature = val;
+ return inspector;
};
- sidebar.expand = function() {
+ return inspector;
+ }
+
+ // 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(d2) {
+ return "M" + d2.join(" L") + " Z";
+ });
+ }
+ }
+ lasso.extent = function() {
+ return lasso.coordinates.reduce(function(extent, point2) {
+ return extent.extend(geoExtent(point2));
+ }, geoExtent());
};
- sidebar.collapse = function() {
+ lasso.p = function(_2) {
+ if (!arguments.length)
+ return lasso;
+ lasso.coordinates.push(_2);
+ draw();
+ return lasso;
};
- sidebar.toggle = function() {
+ lasso.close = function() {
+ if (group) {
+ group.call(uiToggle(false, function() {
+ select_default2(this).remove();
+ }));
+ }
+ context.container().classed("lasso", false);
};
- return sidebar;
+ return lasso;
}
- // modules/modes/draw_area.js
- function modeDrawArea(context, wayID, startGraph, button) {
- var mode = {
- button,
- id: "draw-area"
+ // modules/ui/source_switch.js
+ function uiSourceSwitch(context) {
+ var keys2;
+ 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 ? keys2[0] : keys2[1]);
+ }
+ var sourceSwitch = function(selection2) {
+ selection2.append("a").attr("href", "#").call(_t.append("source_switch.live")).attr("class", "live chip").on("click", click);
};
- 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);
+ sourceSwitch.keys = function(_2) {
+ if (!arguments.length)
+ return keys2;
+ keys2 = _2;
+ return sourceSwitch;
};
- mode.exit = function() {
- context.uninstall(behavior);
+ 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);
+ });
+ }
};
- mode.selectedIDs = function() {
- return [wayID];
+ }
+
+ // 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, d2) => {
+ d3_event.preventDefault();
+ corePreferences("preferences.privacy.thirdpartyicons", d2 === "true" ? "false" : "true");
+ });
+ thirdPartyIconsEnter.append("span").call(_t.append("preferences.privacy.third_party_icons.description"));
+ selection2.selectAll(".privacy-third-party-icons-item").classed("active", (d2) => d2 === "true").select("input").property("checked", (d2) => d2 === "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: '' + _t.html("splash.changelog") + "" },
+ github: { html: 'github.com' }
+ }));
+ modalSection.append("p").html(_t.html("splash.privacy", {
+ updateMessage,
+ privacyLink: { html: '' + _t("splash.privacy_policy") + "" }
+ }));
+ 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");
};
- mode.activeID = function() {
- return behavior && behavior.activeID() || [];
+ }
+
+ // 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();
};
- 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);
- function defaultTags(loc) {
- var defaultTags2 = { area: "yes" };
- if (mode.preset)
- defaultTags2 = mode.preset.setTags(defaultTags2, "area", false, loc);
- return defaultTags2;
+ // 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;
}
- function actionClose(wayId) {
- return function(graph) {
- return graph.replace(graph.entity(wayId).close());
- };
+ 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 start2(loc) {
- var startGraph = context.graph();
- var node = osmNode({ loc });
- var way = osmWay({ tags: defaultTags(loc) });
- context.perform(
- actionAddEntity(node),
- actionAddEntity(way),
- actionAddVertex(way.id, node.id),
- actionClose(way.id)
- );
- context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
+ function zoomOut(d3_event) {
+ if (d3_event.shiftKey)
+ return;
+ d3_event.preventDefault();
+ context.map().zoomOut();
}
- function startFromWay(loc, edge) {
- var startGraph = context.graph();
- var node = osmNode({ loc });
- var way = osmWay({ tags: defaultTags(loc) });
- 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 zoomInFurther(d3_event) {
+ if (d3_event.shiftKey)
+ return;
+ d3_event.preventDefault();
+ context.map().zoomInFurther();
}
- function startFromNode(node) {
- var startGraph = context.graph();
- var way = osmWay({ tags: defaultTags(node.loc) });
- context.perform(
- actionAddEntity(way),
- actionAddVertex(way.id, node.id),
- actionClose(way.id)
- );
- context.enter(modeDrawArea(context, way.id, startGraph, mode.button));
+ function zoomOutFurther(d3_event) {
+ if (d3_event.shiftKey)
+ return;
+ d3_event.preventDefault();
+ context.map().zoomOutFurther();
}
- mode.enter = function() {
- context.install(behavior);
- };
- mode.exit = function() {
- context.uninstall(behavior);
+ return function(selection2) {
+ var tooltipBehavior = uiTooltip().placement(_mainLocalizer.textDirection() === "rtl" ? "right" : "left").title(function(d2) {
+ if (d2.disabled()) {
+ return d2.disabledTitle;
+ }
+ return d2.title;
+ }).keys(function(d2) {
+ return [d2.key];
+ });
+ var lastPointerUpType;
+ var buttons = selection2.selectAll("button").data(zooms).enter().append("button").attr("class", function(d2) {
+ return d2.id;
+ }).on("pointerup.editor", function(d3_event) {
+ lastPointerUpType = d3_event.pointerType;
+ }).on("click.editor", function(d3_event, d2) {
+ if (!d2.disabled()) {
+ d2.action(d3_event);
+ } else if (lastPointerUpType === "touch" || lastPointerUpType === "pen") {
+ context.ui().flash.duration(2e3).iconName("#" + d2.icon).iconClass("disabled").label(d2.disabledTitle)();
+ }
+ lastPointerUpType = null;
+ }).call(tooltipBehavior);
+ buttons.each(function(d2) {
+ select_default2(this).call(svgIcon("#" + d2.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(d2) {
+ return d2.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);
};
- 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);
- function defaultTags(loc) {
- var defaultTags2 = {};
- if (mode.preset)
- defaultTags2 = mode.preset.setTags(defaultTags2, "line", false, loc);
- return defaultTags2;
- }
- function start2(loc) {
- var startGraph = context.graph();
- var node = osmNode({ loc });
- var way = osmWay({ tags: defaultTags(loc) });
- 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(loc) });
- context.perform(
- actionAddEntity(node),
- actionAddEntity(way),
- actionAddVertex(way.id, node.id),
- actionAddMidpoint({ loc, edge }, node)
- );
- context.enter(modeDrawLine(context, way.id, startGraph, mode.button));
+ // 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(d2) {
+ return d2;
+ }).length;
+ return _t.append("inspector.title_count", { title: _t("inspector.tags"), count });
+ }).expandedByDefault(false).disclosureContent(renderDisclosureContent);
+ var taginfo = services.taginfo;
+ var dispatch14 = dispatch_default("change");
+ var availableViews = [
+ { id: "list", icon: "#fas-th-list" },
+ { id: "text", icon: "#fas-i-cursor" }
+ ];
+ let _discardTags = {};
+ _mainFileFetcher.get("discarded").then((d2) => {
+ _discardTags = d2;
+ }).catch(() => {
+ });
+ 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 startFromNode(node) {
- var startGraph = context.graph();
- var way = osmWay({ tags: defaultTags(node.loc) });
- context.perform(
- actionAddEntity(way),
- actionAddVertex(way.id, node.id)
+ 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 i3 in missingKeys) {
+ _orderedKeys.push(missingKeys[i3]);
+ }
+ var rowData = _orderedKeys.map(function(key, i4) {
+ return { index: i4, 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(d2) {
+ return d2.id;
+ }).enter();
+ optionEnter.append("button").attr("class", function(d2) {
+ return "raw-tag-option raw-tag-option-" + d2.id + (_tagView === d2.id ? " selected" : "");
+ }).attr("aria-selected", function(d2) {
+ return _tagView === d2.id;
+ }).attr("role", "tab").attr("title", function(d2) {
+ return _t("icons." + d2.id);
+ }).on("click", function(d3_event, d2) {
+ _tagView = d2.id;
+ corePreferences("raw-tag-editor-view", d2.id);
+ wrap2.selectAll(".raw-tag-option").classed("selected", function(datum2) {
+ return datum2 === d2;
+ }).attr("aria-selected", function(datum2) {
+ return datum2 === d2;
+ });
+ wrap2.selectAll(".tag-text").classed("hide", d2.id !== "text").each(setTextareaHeight);
+ wrap2.selectAll(".tag-list, .add-row").classed("hide", d2.id !== "list");
+ }).each(function(d2) {
+ select_default2(this).call(svgIcon(d2.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 list2 = wrap2.selectAll(".tag-list").data([0]);
+ list2 = list2.enter().append("ul").attr("class", "tag-list" + (_tagView !== "list" ? " hide" : "")).merge(list2);
+ 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 = list2.selectAll(".tag-row").data(rowData, function(d2) {
+ return d2.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(a2, b2) {
+ return a2.index - b2.index;
+ });
+ items.each(function(d2) {
+ 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: d2.key };
+ if (typeof d2.value === "string") {
+ referenceOptions.value = d2.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(d2) {
+ return d2.key;
+ }).call(utilGetSetValue, function(d2) {
+ return d2.key;
+ }).attr("readonly", function(d2) {
+ return isReadOnly(d2) || null;
+ });
+ items.selectAll("input.value").attr("title", function(d2) {
+ return Array.isArray(d2.value) ? d2.value.filter(Boolean).join("\n") : d2.value;
+ }).classed("mixed", function(d2) {
+ return Array.isArray(d2.value);
+ }).attr("placeholder", function(d2) {
+ return typeof d2.value === "string" ? null : _t("inspector.multiple_values");
+ }).call(utilGetSetValue, function(d2) {
+ return typeof d2.value === "string" ? d2.value : "";
+ }).attr("readonly", function(d2) {
+ return isReadOnly(d2) || null;
+ });
+ items.selectAll("button.remove").on(
+ ("PointerEvent" in window ? "pointer" : "mouse") + "down",
+ // 'click' fires too late - #5878
+ (d3_event, d2) => {
+ if (d3_event.button !== 0)
+ return;
+ removeTag(d3_event, d2);
+ }
);
- 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);
- function defaultTags(loc) {
- var defaultTags2 = {};
- if (mode.preset)
- defaultTags2 = mode.preset.setTags(defaultTags2, "point", false, loc);
- return defaultTags2;
}
- function add(loc) {
- var node = osmNode({ loc, tags: defaultTags(loc) });
- context.perform(
- actionAddEntity(node),
- _t("operations.add.annotation.point")
- );
- enterSelectMode(node);
+ function isReadOnly(d2) {
+ for (var i3 = 0; i3 < _readOnlyTags.length; i3++) {
+ if (d2.key.match(_readOnlyTags[i3]) !== null) {
+ return true;
+ }
+ }
+ return false;
}
- function addWay(loc, edge) {
- var node = osmNode({ tags: defaultTags(loc) });
- context.perform(
- actionAddMidpoint({ loc, edge }, node),
- _t("operations.add.annotation.vertex")
- );
- enterSelectMode(node);
+ 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 enterSelectMode(node) {
- context.enter(
- modeSelect(context, [node.id]).newFeature(true)
- );
+ function stringify3(s2) {
+ return JSON.stringify(s2).slice(1, -1);
}
- function addNode(node) {
- const _defaultTags = defaultTags(node.loc);
- if (Object.keys(_defaultTags).length === 0) {
- enterSelectMode(node);
- return;
+ function unstringify(s2) {
+ var leading = "";
+ var trailing = "";
+ if (s2.length < 1 || s2.charAt(0) !== '"') {
+ leading = '"';
}
- var tags = Object.assign({}, node.tags);
- for (var key in _defaultTags) {
- tags[key] = _defaultTags[key];
+ if (s2.length < 2 || s2.charAt(s2.length - 1) !== '"' || s2.charAt(s2.length - 1) === '"' && s2.charAt(s2.length - 2) === "\\") {
+ trailing = '"';
}
- context.perform(
- actionChangeTags(node.id, tags),
- _t("operations.add.annotation.point")
- );
- enterSelectMode(node);
+ return JSON.parse(leading + s2 + trailing);
}
- function cancel() {
- context.enter(modeBrowse(context));
+ 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 ? stringify3(rawVal) : "";
+ return stringify3(row.key) + "=" + val;
+ }).join("\n");
+ if (_state !== "hover" && str.length) {
+ return str + "\n";
+ }
+ return str;
}
- 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)
+ function textChanged() {
+ var newText = this.value.trim();
+ var newTags = {};
+ newText.split("\n").forEach(function(row) {
+ var m2 = row.match(/^\s*([^=]+)=(.*)$/);
+ if (m2 !== null) {
+ var k2 = context.cleanTagKey(unstringify(m2[1].trim()));
+ var v2 = context.cleanTagValue(unstringify(m2[2].trim()));
+ newTags[k2] = v2;
+ }
+ });
+ var tagDiff = utilTagDiff(_tags, newTags);
+ if (!tagDiff.length)
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)
+ _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;
- var note = services.osm.getNote(selectedNoteID);
- if (!note) {
- context.enter(modeBrowse(context));
}
- return note;
+ scheduleChange();
}
- function selectNote(d3_event, drawn) {
- if (!checkSelectedID())
+ 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;
- 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));
+ 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].map(function(tagValue) {
+ if (!tagValue) {
+ return {
+ value: " ",
+ title: _t("inspector.empty"),
+ display: (selection2) => selection2.text("").classed("virtual-option", true).call(_t.append("inspector.empty"))
+ };
+ }
+ 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) {
+ const filtered = data.filter((d2) => _tags[d2.value] === void 0).filter((d2) => !(d2.value in _discardTags)).filter((d2) => !/_\d$/.test(d2)).filter((d2) => d2.value.toLowerCase().includes(value2.toLowerCase()));
+ 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) {
+ const filtered = data.filter((d2) => d2.value.toLowerCase().includes(value2.toLowerCase()));
+ callback(sort(value2, filtered));
+ }
+ });
+ }).caseSensitive(allowUpperCaseTagValues.test(utilGetSetValue(key))));
+ function sort(value2, data) {
+ var sameletter = [];
+ var other = [];
+ for (var i3 = 0; i3 < data.length; i3++) {
+ if (data[i3].value.substring(0, value2.length) === value2) {
+ sameletter.push(data[i3]);
+ } else {
+ other.push(data[i3]);
+ }
}
- } else {
- selection2.classed("selected", true);
- context.selectedNoteID(selectedNoteID);
+ return sameletter.concat(other);
}
}
- function esc() {
- if (context.container().select(".combobox").size())
- return;
- context.enter(modeBrowse(context));
+ function unbind() {
+ var row = select_default2(this);
+ row.selectAll("input.key").call(uiCombobox.off, context);
+ row.selectAll("input.value").call(uiCombobox.off, context);
}
- mode.zoomToSelected = function() {
- if (!services.osm)
+ function keyChange(d3_event, d2) {
+ if (select_default2(this).attr("readonly"))
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)
+ var kOld = d2.key;
+ if (_pendingChange && _pendingChange.hasOwnProperty(kOld) && _pendingChange[kOld] === void 0)
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)
+ var kNew = context.cleanTagKey(this.value.trim());
+ if (isReadOnly({ key: kNew })) {
+ this.value = kOld;
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(d2) {
- return d2.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(d2) {
- return d2.details || [];
- }).enter().append("li").attr("class", "error-detail-item").text(function(d2) {
- return d2;
- });
- items.exit().remove();
+ }
+ if (kNew && kNew !== kOld && _tags[kNew] !== void 0) {
+ this.value = kOld;
+ section.selection().selectAll(".tag-list input.value").each(function(d4) {
+ if (d4.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;
+ d2.key = kNew;
+ this.value = kNew;
+ scheduleChange();
}
- 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 valueChange(d3_event, d2) {
+ if (isReadOnly(d2))
+ return;
+ if (typeof d2.value !== "string" && !this.value)
+ return;
+ if (!this.value.trim())
+ return removeTag(d3_event, d2);
+ if (_pendingChange && _pendingChange.hasOwnProperty(d2.key) && _pendingChange[d2.key] === void 0)
+ return;
+ _pendingChange = _pendingChange || {};
+ _pendingChange[d2.key] = context.cleanTagValue(this.value);
+ scheduleChange();
}
- function keybindingOn() {
- select_default2(document).call(keybinding.on("\u238B", cancel, true));
+ function removeTag(d3_event, d2) {
+ if (isReadOnly(d2))
+ return;
+ if (d2.key === "") {
+ _showBlank = false;
+ section.reRender();
+ } else {
+ _orderedKeys = _orderedKeys.filter(function(key) {
+ return key !== d2.key;
+ });
+ _pendingChange = _pendingChange || {};
+ _pendingChange[d2.key] = void 0;
+ scheduleChange();
+ }
}
- function keybindingOff() {
- select_default2(document).call(keybinding.unbind);
+ function addTag() {
+ window.setTimeout(function() {
+ _showBlank = true;
+ section.reRender();
+ section.selection().selectAll(".tag-list li:last-child input.key").node().focus();
+ }, 20);
}
- 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)
+ function scheduleChange() {
+ var entityIDs = _entityIDs;
+ window.setTimeout(function() {
+ if (!_pendingChange)
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 }
- );
- });
+ dispatch14.call("change", this, entityIDs, _pendingChange);
+ _pendingChange = null;
+ }, 10);
}
- mode.selectedIDs = function() {
- return _conflictsUi ? _conflictsUi.shownEntityIds() : [];
- };
- mode.enter = function() {
- context.ui().sidebar.expand();
- function done() {
- context.ui().sidebar.show(commit);
+ section.state = function(val) {
+ if (!arguments.length)
+ return _state;
+ if (_state !== val) {
+ _orderedKeys = [];
+ _state = val;
}
- keybindingOn();
- context.container().selectAll(".main-content").classed("active", false).classed("inactive", true);
- var osm = context.connection();
- if (!osm) {
- cancel();
- return;
+ 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);
}
- if (osm.authenticated()) {
- done();
- } else {
- osm.authenticate(function(err) {
- if (err) {
- cancel();
- } else {
- done();
- }
- });
+ 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;
};
- mode.exit = function() {
- keybindingOff();
- context.container().selectAll(".main-content").classed("active", true).classed("inactive", false);
- context.ui().sidebar.hide();
+ section.readOnlyTags = function(val) {
+ if (!arguments.length)
+ return _readOnlyTags;
+ _readOnlyTags = val;
+ return section;
};
- return mode;
+ return utilRebind(section, dispatch14, "on");
}
- // 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) {
+ // 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));
- }
- return error;
+ }).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);
}
- mode.zoomToSelected = function() {
- if (!errorService)
- return;
- var error = errorService.getError(selectedErrorID);
- if (error) {
- context.map().centerZoomEase(error.loc, 20);
- }
+ dataEditor.datum = function(val) {
+ if (!arguments.length)
+ return _datum;
+ _datum = val;
+ return this;
};
- 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 dataEditor;
+ }
+
+ // 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;
- 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));
+ 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 x2 = containerLocGetter(d3_event)[0] - dragOffset;
+ sidebarWidth = isRTL ? containerWidth - x2 : x2;
+ 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 {
- selection2.classed("selected", true);
- context.selectedErrorID(selectedErrorID);
+ 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 esc() {
- if (context.container().select(".combobox").size())
+ function pointerup(d3_event) {
+ if (downPointerId !== (d3_event.pointerId || "mouse"))
return;
- context.enter(modeBrowse(context));
+ 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(d2) {
+ return d2.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 i3 = number_default(startMargin, endMargin);
+ return function(t2) {
+ var dx = lastMargin - Math.round(i3(t2));
+ 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() {
};
- 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([]);
+ sidebar.hover = function() {
};
- return mode;
+ 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/ui/tools/modes.js
@@ -72064,8 +74685,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
});
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)
+ context.history().on("change.undo_redo", function(difference2) {
+ if (difference2)
update();
});
context.on("enter.undo_redo", update);
@@ -72256,9 +74877,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
};
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"));
+ var heading2 = _paneSelection.append("div").attr("class", "pane-heading");
+ heading2.append("h2").text("").call(_label);
+ heading2.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);
@@ -72314,7 +74935,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
updateValue(d2, val);
});
sildersControlEnter.append("button").attr("title", function(d2) {
- return `${_t("background.reset")} ${_t("background." + d2)}`;
+ return "".concat(_t("background.reset"), " ").concat(_t("background." + d2));
}).attr("class", function(d2) {
return "display-option-reset display-option-reset-" + d2;
}).on("click", function(d3_event, d2) {
@@ -72355,28 +74976,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
var _currSettings = {
template: corePreferences("background-custom-template")
};
- var example = "https://{switch:a,b,c}.tile.openstreetmap.org/{zoom}/{x}/{y}.png";
+ var example = "https://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}\``;
+ 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(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");
@@ -72622,7 +75227,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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(d2) {
- return _t(`background.nudge.${d2[0]}`);
+ return _t("background.nudge.".concat(d2[0]));
}).attr("class", function(d2) {
return d2[0] + " nudge";
}).on("click", function(d3_event, d2) {
@@ -72960,7 +75565,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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 text = key[1].reduce(function(all, part) {
var subkey = helpkey + "." + part;
var depth = headings[subkey];
var hhh = depth ? Array(depth + 1).join("#") + " " : "";
@@ -72968,7 +75573,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}, "");
return {
title: _t.html(helpkey + ".title"),
- content: marked(text2.trim()).replace(//g, "").replace(/<\/code>/g, "")
+ content: marked(text.trim()).replace(//g, "").replace(/<\/code>/g, "")
};
});
var helpPane = uiPane("help", context).key(_t("help.key")).label(_t.append("help.title")).description(_t.append("help.title")).iconName("iD-icon-help");
@@ -73074,9 +75679,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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(d2) {
+ var list2 = selection2.selectAll(".issues-list").data([0]);
+ list2 = list2.enter().append("ul").attr("class", "layer-list issues-list " + severity + "s-list").merge(list2);
+ var items = list2.selectAll("li").data(issues, function(d2) {
return d2.key;
});
items.exit().remove();
@@ -73809,11 +76414,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
context.layers().on("change", () => updatePhotoList(photoList.select("ul")));
}
function updatePhotoList(container) {
+ var _a2;
function locationUnavailable(d2) {
return !(isArray_default(d2.loc) && isNumber_default(d2.loc[0]) && isNumber_default(d2.loc[1]));
}
container.selectAll("li.placeholder").remove();
- let selection2 = container.selectAll("li").data(photoLayer.getPhotos() ?? [], (d2) => d2.id);
+ let selection2 = container.selectAll("li").data((_a2 = photoLayer.getPhotos()) != null ? _a2 : [], (d2) => d2.id);
selection2.exit().remove();
const selectionEnter = selection2.enter().append("li");
selectionEnter.append("span").classed("filename", true);
@@ -73870,7 +76476,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return layerSupported(d2) && d2.layer.enabled();
}
function layerRendered(d2) {
- return d2.layer.rendered?.(context.map().zoom()) ?? true;
+ var _a2, _b, _c;
+ return (_c = (_b = (_a2 = d2.layer).rendered) == null ? void 0 : _b.call(_a2, context.map().zoom())) != null ? _c : true;
}
var ul = selection2.selectAll(".layer-list-photos").data([0]);
ul = ul.enter().append("ul").attr("class", "layer-list layer-list-photos").merge(ul);
@@ -74379,9 +76986,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (surfaceNode.focus) {
surfaceNode.focus();
}
- operations.forEach(function(operation) {
- if (operation.point)
- operation.point(anchorPoint);
+ operations.forEach(function(operation2) {
+ if (operation2.point)
+ operation2.point(anchorPoint);
});
_editMenu.anchorLoc(anchorPoint).triggerType(triggerType).operations(operations);
context.map().supersurface.call(_editMenu);
@@ -74527,6 +77134,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
_connection.loadEntityRelations(entityID, afterLoad(cid, callback));
}
};
+ context.loadNote = (entityID, callback) => {
+ if (_connection) {
+ const cid = _connection.getConnectionId();
+ _connection.loadEntityNote(entityID, afterLoad(cid, callback));
+ }
+ };
context.zoomToEntity = (entityID, zoomTo) => {
context.loadEntity(entityID, (err, result) => {
if (err)
@@ -74741,7 +77354,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
const filename = _assetPath + val;
return _assetMap[filename] || filename;
};
- context.imagePath = (val) => context.asset(`img/${val}`);
+ context.imagePath = (val) => context.asset("img/".concat(val));
context.reset = context.flush = () => {
context.debouncedSave.cancel();
Array.from(_deferred2).forEach((handle) => {
@@ -74924,11 +77537,11 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
// node_modules/name-suggestion-index/lib/simplify.js
var import_diacritics3 = __toESM(require_diacritics(), 1);
- function simplify2(str2) {
- if (typeof str2 !== "string")
+ function simplify2(str) {
+ if (typeof str !== "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\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()
+ str.replace(/&/g, "and").replace(/Ä°/ig, "i").replace(/[\s\-=_!"#%'*{},.\/:;?\(\)\[\]@\\$\^*+<>«»~`â\u00a1\u00a7\u00b6\u00b7\u00bf\u037e\u0387\u055a-\u055f\u0589\u05c0\u05c3\u05c6\u05f3\u05f4\u0609\u060a\u060c\u060d\u061b\u061e\u061f\u066a-\u066d\u06d4\u0700-\u070d\u07f7-\u07f9\u0830-\u083e\u085e\u0964\u0965\u0970\u0af0\u0df4\u0e4f\u0e5a\u0e5b\u0f04-\u0f12\u0f14\u0f85\u0fd0-\u0fd4\u0fd9\u0fda\u104a-\u104f\u10fb\u1360-\u1368\u166d\u166e\u16eb-\u16ed\u1735\u1736\u17d4-\u17d6\u17d8-\u17da\u1800-\u1805\u1807-\u180a\u1944\u1945\u1a1e\u1a1f\u1aa0-\u1aa6\u1aa8-\u1aad\u1b5a-\u1b60\u1bfc-\u1bff\u1c3b-\u1c3f\u1c7e\u1c7f\u1cc0-\u1cc7\u1cd3\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()
);
}
@@ -75313,7 +77926,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
const t2 = parts[0];
const k2 = parts[1];
const v2 = parts[2];
- const thiskv = `${k2}/${v2}`;
+ const thiskv = "".concat(k2, "/").concat(v2);
const tree = trees[t2];
let branch = that.matchIndex.get(thiskv);
if (!branch) {
@@ -75337,7 +77950,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
const alternateName = new RegExp(tree.nameTags.alternate, "i");
const notName = /:(colou?r|type|forward|backward|left|right|etymology|pronunciation|wikipedia)$/i;
const skipGenericKV = skipGenericKVMatches(t2, k2, v2);
- const genericKV = /* @__PURE__ */ new Set([`${k2}/yes`, `building/yes`]);
+ const genericKV = /* @__PURE__ */ new Set(["".concat(k2, "/yes"), "building/yes"]);
const matchGroupKV = /* @__PURE__ */ new Set();
Object.values(matchGroups).forEach((matchGroup) => {
const inGroup = matchGroup.some((otherkv) => otherkv === thiskv);
@@ -75348,7 +77961,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return;
matchGroupKV.add(otherkv);
const otherk = otherkv.split("/", 2)[0];
- genericKV.add(`${otherk}/yes`);
+ genericKV.add("".concat(otherk, "/yes"));
});
});
items.forEach((item) => {
@@ -75359,7 +77972,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (!item.matchTags.length)
delete item.matchTags;
}
- let kvTags = [`${thiskv}`].concat(item.matchTags || []);
+ let kvTags = ["".concat(thiskv)].concat(item.matchTags || []);
if (!skipGenericKV) {
kvTags = kvTags.concat(Array.from(genericKV));
}
@@ -75399,7 +78012,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
});
function insertName(which, t2, kv, nsimple, itemID) {
if (!nsimple) {
- that.warnings.push(`Warning: skipping empty ${which} name for item ${t2}/${kv}: ${itemID}`);
+ that.warnings.push("Warning: skipping empty ".concat(which, " name for item ").concat(t2, "/").concat(kv, ": ").concat(itemID));
return;
}
let branch = that.matchIndex.get(kv);
@@ -75419,11 +78032,11 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
leaf.add(itemID);
if (!/yes$/.test(kv)) {
- const kvnsimple = `${kv}/${nsimple}`;
+ const kvnsimple = "".concat(kv, "/").concat(nsimple);
const existing = seenTree.get(kvnsimple);
if (existing && existing !== t2) {
const items = Array.from(leaf);
- that.warnings.push(`Duplicate cache key "${kvnsimple}" in trees "${t2}" and "${existing}", check items: ${items}`);
+ that.warnings.push('Duplicate cache key "'.concat(kvnsimple, '" in trees "').concat(t2, '" and "').concat(existing, '", check items: ').concat(items));
return;
}
seenTree.set(kvnsimple, t2);
@@ -75464,7 +78077,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
try {
resolved = loco.resolveLocationSet(item.locationSet);
} catch (err) {
- console.warn(`buildLocationIndex: ${err.message}`);
+ console.warn("buildLocationIndex: ".concat(err.message));
}
if (!resolved || !resolved.id)
return;
@@ -75475,7 +78088,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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("buildLocationIndex: locationSet ".concat(resolved.id, " for ").concat(item.id, " resolves to an empty feature:"));
console.warn(JSON.stringify(feature3));
return;
}
@@ -75554,7 +78167,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
gatherResults("exclude");
return results.length ? results : null;
function gatherResults(which) {
- const kv = `${k2}/${v2}`;
+ const kv = "".concat(k2, "/").concat(v2);
let didMatch = tryMatch(which, kv);
if (didMatch)
return;
@@ -75663,7 +78276,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
function setNsiSources() {
const nsiVersion = package_default.dependencies["name-suggestion-index"] || package_default.devDependencies["name-suggestion-index"];
const v2 = (0, import_vparse2.default)(nsiVersion);
- const vMinor = `${v2.major}.${v2.minor}`;
+ const vMinor = "".concat(v2.major, ".").concat(v2.minor);
const cdn = nsiCdnUrl.replace("{version}", vMinor);
const sources = {
"nsi_data": cdn + "dist/nsi.min.json",
@@ -75686,6 +78299,17 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
_mainFileFetcher.get("nsi_features")
]).then((vals) => {
Object.values(vals[0].presets).forEach((preset) => preset.suggestion = true);
+ Object.values(vals[0].presets).forEach((preset) => {
+ if (preset.tags["brand:wikidata"]) {
+ preset.removeTags = { "brand:wikipedia": "*", ...preset.removeTags || preset.addTags || preset.tags };
+ }
+ if (preset.tags["operator:wikidata"]) {
+ preset.removeTags = { "operator:wikipedia": "*", ...preset.removeTags || preset.addTags || preset.tags };
+ }
+ if (preset.tags["network:wikidata"]) {
+ preset.removeTags = { "network:wikipedia": "*", ...preset.removeTags || preset.addTags || preset.tags };
+ }
+ });
_mainPresetIndex.merge({
presets: vals[0].presets,
featureCollection: vals[1]
@@ -75788,9 +78412,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (!vmap)
return;
if (vmap.get(osmvalue)) {
- primary.add(`${osmkey}/${osmvalue}`);
+ primary.add("".concat(osmkey, "/").concat(osmvalue));
} else if (osmvalue === "yes") {
- alternate.add(`${osmkey}/${osmvalue}`);
+ alternate.add("".concat(osmkey, "/").concat(osmvalue));
}
});
const preset = _mainPresetIndex.matchTags(tags, "area");
@@ -75942,7 +78566,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
if (replace && replace.wikipedia !== void 0) {
changed = true;
- const wpkey = `${prefix}wikipedia`;
+ const wpkey = "".concat(prefix, "wikipedia");
if (replace.wikipedia) {
newTags[wpkey] = replace.wikipedia;
} else {
@@ -75972,8 +78596,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (hits[0].match !== "primary" && hits[0].match !== "alternate")
break;
let itemID, item;
- for (let j3 = 0; j3 < hits.length; j3++) {
- const hit = hits[j3];
+ for (let j2 = 0; j2 < hits.length; j2++) {
+ const hit = hits[j2];
itemID = hit.itemID;
if (_nsi.dissolved[itemID])
continue;
@@ -75982,7 +78606,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
continue;
const mainTag = item.mainTag;
const itemQID = item.tags[mainTag];
- const notQID = newTags[`not:${mainTag}`];
+ const notQID = newTags["not:".concat(mainTag)];
if (
// Exceptions, skip this hit
!itemQID || itemQID === notQID || // No `*:wikidata` or matched a `not:*:wikidata`
@@ -76006,7 +78630,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
let preserveTags = item.preserveTags || properties.preserveTags || [];
["building", "emergency", "internet_access", "takeaway"].forEach((osmkey) => {
if (k2 !== osmkey)
- preserveTags.push(`^${osmkey}$`);
+ preserveTags.push("^".concat(osmkey, "$"));
});
const regexes = preserveTags.map((s2) => new RegExp(s2, "i"));
let keepTags = {};
@@ -76626,6 +79250,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
var _loadViewerPromise3;
var _vegbilderCache;
async function fetchAvailableLayers() {
+ var _a2, _b, _c;
const params = {
service: "WFS",
request: "GetCapabilities",
@@ -76645,12 +79270,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
let node;
const availableLayers = [];
while ((node = l2.iterateNext()) !== null) {
- const match = node.textContent?.match(regexMatcher);
+ const match = (_a2 = node.textContent) == null ? void 0 : _a2.match(regexMatcher);
if (match) {
availableLayers.push({
name: match[0],
- is_sphere: !!match.groups?.image_type,
- year: parseInt(match.groups?.year, 10)
+ is_sphere: !!((_b = match.groups) == null ? void 0 : _b.image_type),
+ year: parseInt((_c = match.groups) == null ? void 0 : _c.year, 10)
});
}
}
@@ -76837,14 +79462,14 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
} = properties;
let reference;
if (year >= 2020) {
- reference = `${road_class}${road_status}${road_number} S${section}D${subsection}`;
+ reference = "".concat(road_class).concat(road_status).concat(road_number, " S").concat(section, "D").concat(subsection);
if (junction_part) {
- reference = `${reference} M${anker_point} KD${junction_part}`;
+ reference = "".concat(reference, " M").concat(anker_point, " KD").concat(junction_part);
} else if (services_part) {
- reference = `${reference} M${anker_point} SD${services_part}`;
+ reference = "".concat(reference, " M").concat(anker_point, " SD").concat(services_part);
}
} else {
- reference = `${county_number}${road_class}${road_status}${road_number} HP${parcel}`;
+ reference = "".concat(county_number).concat(road_class).concat(road_status).concat(road_number, " HP").concat(parcel);
}
return reference;
}
@@ -76866,7 +79491,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return tiler9.getTiles(projection2).map((tile) => tile.extent);
}
function searchLimited3(limit, projection2, rtree) {
- limit ??= 5;
+ limit != null ? limit : limit = 5;
return partitionViewport3(projection2).reduce((result, extent) => {
const found = rtree.search(extent.bbox()).slice(0, limit).map((d2) => d2.data);
return result.concat(found);
@@ -76938,13 +79563,13 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
},
getSequenceForImage: function(image) {
- return _vegbilderCache?.image2sequence_map.get(image?.key);
+ return _vegbilderCache == null ? void 0 : _vegbilderCache.image2sequence_map.get(image == null ? void 0 : image.key);
},
loadImages: async function(context, margin) {
if (!_vegbilderCache) {
await this.reset();
}
- margin ??= 1;
+ margin != null ? margin : margin = 1;
const wfslayers = filterAvailableLayers(context.photos());
loadWFSLayers(context.projection, margin, wfslayers);
},
@@ -77000,7 +79625,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
attribution.append("span").attr("class", "captured_at").text(localeTimestamp(d2.captured_at));
}
attribution.append("a").attr("target", "_blank").attr("href", "https://vegvesen.no").call(_t.append("vegbilder.publisher"));
- attribution.append("a").attr("target", "_blank").attr("href", `https://vegbilder.atlas.vegvesen.no/?year=${d2.captured_at.getFullYear()}&lat=${d2.loc[1]}&lng=${d2.loc[0]}&view=image&imageId=${d2.key}`).call(_t.append("vegbilder.view_on"));
+ attribution.append("a").attr("target", "_blank").attr("href", "https://vegbilder.atlas.vegvesen.no/?year=".concat(d2.captured_at.getFullYear(), "&lat=").concat(d2.loc[1], "&lng=").concat(d2.loc[0], "&view=image&imageId=").concat(d2.key)).call(_t.append("vegbilder.view_on"));
_currentFrame = d2.is_sphere ? _pannellumFrame : _planeFrame;
_currentFrame.selectPhoto(d2, keepOrientation).showPhotoFrame(wrap2);
return this;
@@ -77027,20 +79652,21 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
// Reset is only necessary when interacting with the viewport because
// this implicitly changes the currently selected bubble/sequence
setStyles: function(context, hovered, reset) {
+ var _a2, _b;
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);
}
- const hoveredImageKey = hovered?.key;
+ const hoveredImageKey = hovered == null ? void 0 : hovered.key;
const hoveredSequence = this.getSequenceForImage(hovered);
- const hoveredSequenceKey = hoveredSequence?.key;
- const hoveredImageKeys = hoveredSequence?.images.map((d2) => d2.key) ?? [];
+ const hoveredSequenceKey = hoveredSequence == null ? void 0 : hoveredSequence.key;
+ const hoveredImageKeys = (_a2 = hoveredSequence == null ? void 0 : hoveredSequence.images.map((d2) => d2.key)) != null ? _a2 : [];
const viewer = context.container().select(".photoviewer");
const selected = viewer.empty() ? void 0 : viewer.datum();
- const selectedImageKey = selected?.key;
+ const selectedImageKey = selected == null ? void 0 : selected.key;
const selectedSequence = this.getSequenceForImage(selected);
- const selectedSequenceKey = selectedSequence?.key;
- const selectedImageKeys = selectedSequence?.images.map((d2) => d2.key) ?? [];
+ const selectedSequenceKey = selectedSequence == null ? void 0 : selectedSequence.key;
+ const selectedImageKeys = (_b = selectedSequence == null ? void 0 : selectedSequence.images.map((d2) => d2.key)) != null ? _b : [];
const highlightedImageKeys = utilArrayUnion(hoveredImageKeys, selectedImageKeys);
context.container().selectAll(".layer-vegbilder .viewfield-group").classed("highlighted", (d2) => highlightedImageKeys.indexOf(d2.key) !== -1).classed("hovered", (d2) => d2.key === hoveredImageKey).classed("currentView", (d2) => d2.key === selectedImageKey);
context.container().selectAll(".layer-vegbilder .sequence").classed("highlighted", (d2) => d2.key === hoveredSequenceKey).classed("currentView", (d2) => d2.key === selectedSequenceKey);
@@ -77076,9 +79702,28 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
};
// node_modules/osm-auth/src/osm-auth.mjs
- var import_store = __toESM(require_store_legacy(), 1);
function osmAuth(o2) {
var oauth2 = {};
+ var _store = null;
+ try {
+ _store = window.localStorage;
+ } catch (e3) {
+ var _mock = /* @__PURE__ */ new Map();
+ _store = {
+ isMocked: true,
+ hasItem: (k2) => _mock.has(k2),
+ getItem: (k2) => _mock.get(k2),
+ setItem: (k2, v2) => _mock.set(k2, v2),
+ removeItem: (k2) => _mock.delete(k2),
+ clear: () => _mock.clear()
+ };
+ }
+ function token(k2, v2) {
+ if (arguments.length === 1)
+ return _store.getItem(o2.url + k2);
+ else if (arguments.length === 2)
+ return _store.setItem(o2.url + k2, v2);
+ }
oauth2.authenticated = function() {
return !!token("oauth2_access_token");
};
@@ -77095,8 +79740,14 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return;
}
oauth2.logout();
- _generatePkceChallenge(function(pkce) {
- _authenticate(pkce, callback);
+ _preopenPopup(function(error, popup) {
+ if (error) {
+ callback(error);
+ } else {
+ _generatePkceChallenge(function(pkce) {
+ _authenticate(pkce, popup, callback);
+ });
+ }
});
};
oauth2.authenticateAsync = function() {
@@ -77107,15 +79758,45 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return new Promise((resolve, reject) => {
var errback = (err, result) => {
if (err) {
- reject(new Error(err));
+ reject(err);
} else {
resolve(result);
}
};
- _generatePkceChallenge((pkce) => _authenticate(pkce, errback));
+ _preopenPopup((error, popup) => {
+ if (error) {
+ errback(error);
+ } else {
+ _generatePkceChallenge((pkce) => _authenticate(pkce, popup, errback));
+ }
+ });
});
};
- function _authenticate(pkce, callback) {
+ function _preopenPopup(callback) {
+ if (o2.singlepage) {
+ callback(null, void 0);
+ return;
+ }
+ var w2 = 550;
+ var h2 = 610;
+ var settings = [
+ ["width", w2],
+ ["height", h2],
+ ["left", window.screen.width / 2 - w2 / 2],
+ ["top", window.screen.height / 2 - h2 / 2]
+ ].map(function(x2) {
+ return x2.join("=");
+ }).join(",");
+ var popup = window.open("about:blank", "oauth_window", settings);
+ if (popup) {
+ callback(null, popup);
+ } else {
+ var error = new Error("Popup was blocked");
+ error.status = "popup-blocked";
+ callback(error);
+ }
+ }
+ function _authenticate(pkce, popup, callback) {
var state = generateState();
var url = o2.url + "/oauth2/authorize?" + utilQsString2({
client_id: o2.client_id,
@@ -77127,8 +79808,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
code_challenge_method: pkce.code_challenge_method
});
if (o2.singlepage) {
- if (!import_store.default.enabled) {
- var error = new Error("local storage unavailable, but require in singlepage mode");
+ if (_store.isMocked) {
+ var error = new Error("localStorage unavailable, but required in singlepage mode");
error.status = "pkce-localstorage-unavailable";
callback(error);
return;
@@ -77142,24 +79823,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
window.location = url;
}
} else {
- var w2 = 600;
- var h2 = 550;
- var settings = [
- ["width", w2],
- ["height", h2],
- ["left", window.screen.width / 2 - w2 / 2],
- ["top", window.screen.height / 2 - h2 / 2]
- ].map(function(x2) {
- return x2.join("=");
- }).join(",");
- var popup = window.open("about:blank", "oauth_window", settings);
oauth2.popupWindow = popup;
popup.location = url;
- if (!popup) {
- error = new Error("Popup was blocked");
- error.status = "popup-blocked";
- callback(error);
- }
}
window.authComplete = function(url2) {
var params2 = utilStringQs2(url2.split("?")[1]);
@@ -77261,7 +79926,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
}
function _doXHR() {
- var url = options2.prefix !== false ? o2.url + options2.path : options2.path;
+ var url = options2.prefix !== false ? o2.apiUrl + options2.path : options2.path;
return oauth2.rawxhr(
options2.method,
url,
@@ -77315,6 +79980,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (!arguments.length)
return o2;
o2 = val;
+ o2.apiUrl = o2.apiUrl || "https://api.openstreetmap.org";
o2.url = o2.url || "https://www.openstreetmap.org";
o2.auto = o2.auto || false;
o2.singlepage = o2.singlepage || false;
@@ -77324,23 +79990,6 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
};
return oauth2.preauth(o2);
};
- var token;
- if (import_store.default.enabled) {
- token = function(x2, y2) {
- if (arguments.length === 1)
- return import_store.default.get(o2.url + x2);
- else if (arguments.length === 2)
- return import_store.default.set(o2.url + x2, y2);
- };
- } else {
- var storage = {};
- token = function(x2, y2) {
- if (arguments.length === 1)
- return storage[o2.url + x2];
- else if (arguments.length === 2)
- return storage[o2.url + x2] = y2;
- };
- }
oauth2.options(o2);
return oauth2;
}
@@ -77351,12 +80000,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return encodeURIComponent(key) + "=" + encodeURIComponent(obj[key]);
}).join("&");
}
- function utilStringQs2(str2) {
+ function utilStringQs2(str) {
var i3 = 0;
- while (i3 < str2.length && (str2[i3] === "?" || str2[i3] === "#"))
+ while (i3 < str.length && (str[i3] === "?" || str[i3] === "#"))
i3++;
- str2 = str2.slice(i3);
- return str2.split("&").reduce(function(obj, pair3) {
+ str = str.slice(i3);
+ return str.split("&").reduce(function(obj, pair3) {
var parts = pair3.split("=");
if (parts.length === 2) {
obj[parts[0]] = decodeURIComponent(parts[1]);
@@ -77419,9 +80068,11 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
var tiler6 = utilTiler();
var dispatch10 = dispatch_default("apiStatusChange", "authLoading", "authDone", "change", "loading", "loaded", "loadedNotes");
var urlroot = osmApiConnections[0].url;
+ var apiUrlroot = osmApiConnections[0].apiUrl || urlroot;
var redirectPath = window.location.origin + window.location.pathname;
var oauth = osmAuth({
url: urlroot,
+ apiUrl: apiUrlroot,
client_id: osmApiConnections[0].client_id,
client_secret: osmApiConnections[0].client_secret,
scope: "read_prefs write_prefs write_api read_gpx write_notes",
@@ -77537,8 +80188,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (comment.nodeName === "comment") {
var childNodes = comment.childNodes;
var parsedComment = {};
- for (var j3 = 0; j3 < childNodes.length; j3++) {
- var node = childNodes[j3];
+ for (var j2 = 0; j2 < childNodes.length; j2++) {
+ var node = childNodes[j2];
var nodeName = node.nodeName;
if (nodeName === "#text")
continue;
@@ -77738,15 +80389,19 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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);
+ if (!_noteCache.note[uid]) {
+ let coincident = false;
+ const epsilon3 = 1e-5;
+ do {
+ if (coincident) {
+ props.loc = geoVecAdd(props.loc, [epsilon3, epsilon3]);
+ }
+ const bbox2 = geoExtent(props.loc).bbox();
+ coincident = _noteCache.rtree.search(bbox2).length;
+ } while (coincident);
+ } else {
+ props.loc = _noteCache.note[uid].loc;
+ }
for (var i3 = 0; i3 < childNodes.length; i3++) {
var node = childNodes[i3];
var nodeName = node.nodeName;
@@ -77761,7 +80416,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
var note = new osmNote(props);
var item = encodeNoteRtree(note);
_noteCache.note[note.id] = note;
- _noteCache.rtree.insert(item);
+ updateRtree4(item, true);
return note;
},
user: function parseUser2(obj, uid) {
@@ -77889,12 +80544,15 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
getUrlRoot: function() {
return urlroot;
},
+ getApiUrlRoot: function() {
+ return apiUrlroot;
+ },
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);
+ var precision3 = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2));
+ return urlroot + "/history#map=" + Math.floor(zoom) + "/" + center[1].toFixed(precision3) + "/" + center[0].toFixed(precision3);
},
entityURL: function(entity) {
return urlroot + "/" + entity.type + "/" + entity.osmId();
@@ -77949,9 +80607,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
}
if (this.authenticated()) {
- return oauth.xhr({ method: "GET", path }, done);
+ return oauth.xhr({
+ method: "GET",
+ path
+ }, done);
} else {
- var url = urlroot + path;
+ var url = apiUrlroot + path;
var controller = new AbortController();
var fn;
if (path.indexOf(".json") !== -1) {
@@ -77991,6 +80652,19 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
options2
);
},
+ // Load a single note by id , XML format
+ // GET /api/0.6/notes/#id
+ loadEntityNote: function(id2, callback) {
+ var options2 = { skipSeen: false };
+ this.loadFromAPI(
+ "/api/0.6/notes/" + id2,
+ 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) {
@@ -78124,12 +80798,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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));
+ 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);
@@ -78148,10 +80822,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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)
- );
+ oauth.xhr({
+ method: "GET",
+ path: "/api/0.6/user/" + uid + ".json"
+ }, wrapcb(this, done, _connectionID));
function done(err, payload) {
if (err)
return callback(err);
@@ -78169,10 +80843,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (_userDetails) {
return callback(void 0, _userDetails);
}
- oauth.xhr(
- { method: "GET", path: "/api/0.6/user/details.json" },
- wrapcb(this, done, _connectionID)
- );
+ oauth.xhr({
+ method: "GET",
+ path: "/api/0.6/user/details.json"
+ }, wrapcb(this, done, _connectionID));
function done(err, payload) {
if (err)
return callback(err);
@@ -78198,10 +80872,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (err) {
return callback(err);
}
- oauth.xhr(
- { method: "GET", path: "/api/0.6/changesets?user=" + user.id },
- wrapcb(this, done, _connectionID)
- );
+ oauth.xhr({
+ method: "GET",
+ path: "/api/0.6/changesets?user=" + user.id
+ }, wrapcb(this, done, _connectionID));
}
function done(err, xml) {
if (err) {
@@ -78222,7 +80896,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
// Fetch the status of the OSM API
// GET /api/capabilities
status: function(callback) {
- var url = urlroot + "/api/capabilities";
+ var url = apiUrlroot + "/api/capabilities";
var errback = wrapcb(this, done, _connectionID);
xml_default(url).then(function(data) {
errback(null, data);
@@ -78400,10 +81074,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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)
- );
+ _noteCache.inflightPost[note.id] = oauth.xhr({
+ method: "POST",
+ path
+ }, wrapcb(this, done, _connectionID));
function done(err, xml) {
delete _noteCache.inflightPost[note.id];
if (err) {
@@ -78445,10 +81119,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (note.newComment) {
path += "?" + utilQsString({ text: note.newComment });
}
- _noteCache.inflightPost[note.id] = oauth.xhr(
- { method: "POST", path },
- wrapcb(this, done, _connectionID)
- );
+ _noteCache.inflightPost[note.id] = oauth.xhr({
+ method: "POST",
+ path
+ }, wrapcb(this, done, _connectionID));
function done(err, xml) {
delete _noteCache.inflightPost[note.id];
if (err) {
@@ -78479,8 +81153,15 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
},
switch: function(newOptions) {
urlroot = newOptions.url;
- var oldOptions = utilObjectOmit(oauth.options(), "access_token");
- oauth.options(Object.assign(oldOptions, newOptions));
+ apiUrlroot = newOptions.apiUrl || urlroot;
+ if (newOptions.url && !newOptions.apiUrl) {
+ newOptions = {
+ ...newOptions,
+ apiUrl: newOptions.url
+ };
+ }
+ const oldOptions = utilObjectOmit(oauth.options(), "access_token");
+ oauth.options({ ...oldOptions, ...newOptions });
this.reset();
this.userChangesets(function() {
});
@@ -78872,8 +81553,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
var wikis = [rtypeWiki, tagWiki, keyWiki];
for (i3 in wikis) {
var wiki = wikis[i3];
- for (var j3 in langCodes) {
- var code = langCodes[j3];
+ 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) {
@@ -78909,64 +81590,11 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
// modules/services/streetside.js
var import_rbush11 = __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 t2 = window.setTimeout(function() {
- callback(window.JSONP_FIX);
- }, window.JSONP_DELAY || 0);
- request3.abort = function() {
- window.clearTimeout(t2);
- };
- }
- return request3;
- }
- function rand() {
- var chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz";
- var c2 = "";
- var i3 = -1;
- while (++i3 < 15)
- c2 += chars.charAt(Math.floor(Math.random() * 52));
- return c2;
- }
- function create2(url2) {
- var e3 = url2.match(/callback=(\w+)/);
- var c2 = e3 ? e3[1] : rand();
- jsonpCache[c2] = function(data) {
- if (jsonpCache[c2]) {
- callback(data);
- }
- finalize();
- };
- function finalize() {
- delete jsonpCache[c2];
- script.remove();
- }
- request3.abort = finalize;
- return "jsonpCache." + c2;
- }
- 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 streetsideApi = "https://dev.virtualearth.net/REST/v1/Imagery/MetaData/Streetside?mapArea={bbox}&key={key}&count={count}";
+ var maxResults2 = 500;
+ var bubbleAppKey = utilAesDecrypt("5c875730b09c6b422433e807e1ff060b6536c791dbfffcffc4c6b18a1bdba1f14593d151adb50e19e1be1ab19aef813bf135d0f103475e5c724dec94389e45d0");
var pannellumViewerCSS2 = "pannellum/pannellum.css";
var pannellumViewerJS2 = "pannellum/pannellum.js";
- var maxResults2 = 2e3;
var tileZoom3 = 16.5;
var tiler7 = utilTiler().zoomExtent([tileZoom3, tileZoom3]).skipNullIsland(true);
var dispatch11 = dispatch_default("loadedImages", "viewerChanged");
@@ -79020,37 +81648,37 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
const id2 = tile.id + "," + String(nextPage);
if (cache.loaded[id2] || cache.inflight[id2])
return;
- cache.inflight[id2] = getBubbles(url, tile, (bubbles) => {
+ cache.inflight[id2] = getBubbles(url, tile, (response) => {
cache.loaded[id2] = true;
delete cache.inflight[id2];
- if (!bubbles)
+ if (!response)
return;
- bubbles.shift();
- const features = bubbles.map((bubble) => {
- if (cache.points[bubble.id])
+ if (response.resourceSets[0].resources.length === maxResults2) {
+ const split = tile.extent.split();
+ loadNextTilePage2(which, url, { id: tile.id + ",a", extent: split[0] });
+ loadNextTilePage2(which, url, { id: tile.id + ",b", extent: split[1] });
+ loadNextTilePage2(which, url, { id: tile.id + ",c", extent: split[2] });
+ loadNextTilePage2(which, url, { id: tile.id + ",d", extent: split[3] });
+ }
+ const features = response.resourceSets[0].resources.map((bubble) => {
+ const bubbleId = bubble.imageUrl;
+ if (cache.points[bubbleId])
return null;
- const loc = [bubble.lo, bubble.la];
+ const loc = [bubble.lon, bubble.lat];
const d2 = {
loc,
- key: bubble.id,
+ key: bubbleId,
+ imageUrl: bubble.imageUrl.replace(
+ "{subdomain}",
+ bubble.imageUrlSubdomains[0]
+ ),
ca: bubble.he,
- captured_at: bubble.cd,
+ captured_at: bubble.vintageEnd,
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] = d2;
- if (bubble.pr === void 0) {
- cache.leaders.push(bubble.id);
- }
+ cache.points[bubbleId] = d2;
return {
minX: loc[0],
minY: loc[1],
@@ -79060,67 +81688,32 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
};
}).filter(Boolean);
cache.rtree.load(features);
- connectSequences();
if (which === "bubbles") {
dispatch11.call("loadedImages");
}
});
}
- function connectSequences() {
- let cache = _ssCache.bubbles;
- let keepLeaders = [];
- for (let i3 = 0; i3 < cache.leaders.length; i3++) {
- let bubble = cache.points[cache.leaders[i3]];
- 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 j3 = 0; j3 < sequence.bubbles.length; j3++) {
- sequence.bubbles[j3].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((d2) => d2.loc)
- };
- } else {
- keepLeaders.push(cache.leaders[i3]);
- }
- }
- 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) {
+ let urlForRequest = url.replace("{key}", bubbleAppKey).replace("{bbox}", [rect[1], rect[0], rect[3], rect[2]].join(",")).replace("{count}", maxResults2);
+ const controller = new AbortController();
+ fetch(urlForRequest, { signal: controller.signal }).then(function(response) {
+ if (!response.ok) {
+ throw new Error(response.status + " " + response.statusText);
+ }
+ return response.json();
+ }).then(function(result) {
+ if (!result) {
callback(null);
+ }
+ return callback(result || []);
+ }).catch(function(err) {
+ if (err.name === "AbortError") {
} else {
- callback(data);
+ throw new Error(err);
}
});
+ return controller;
}
function partitionViewport4(projection2) {
let z2 = geoScaleToZoom(projection2.scale());
@@ -79559,7 +82152,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
Object.values(_ssCache.bubbles.inflight).forEach(abortRequest6);
}
_ssCache = {
- bubbles: { inflight: {}, loaded: {}, nextPage: {}, rtree: new import_rbush11.default(), points: {}, leaders: [] },
+ bubbles: { inflight: {}, loaded: {}, nextPage: {}, rtree: new import_rbush11.default(), points: {} },
sequences: {}
};
},
@@ -79595,7 +82188,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
loadBubbles: function(projection2, margin) {
if (margin === void 0)
margin = 2;
- loadTiles3("bubbles", bubbleApi, projection2, margin);
+ loadTiles3("bubbles", streetsideApi, projection2, margin);
},
viewer: function() {
return _pannellumViewer2;
@@ -79795,13 +82388,6 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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=" + d2.loc[1] + "~" + d2.loc[0] + "&lvl=17&dir=" + d2.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(d2.key) + "&focus=photo&lat=" + d2.loc[1] + "&lng=" + d2.loc[0] + "&z=17").call(_t.append("streetside.report"));
- let bubbleIdQuadKey = d2.key.toString(4);
- const paddingNeeded = 16 - bubbleIdQuadKey.length;
- for (let i3 = 0; i3 < paddingNeeded; i3++) {
- bubbleIdQuadKey = "0" + bubbleIdQuadKey;
- }
- const imgUrlPrefix = streetsideImagesApi + "hs" + bubbleIdQuadKey;
- const imgUrlSuffix = ".jpg?g=13515&n=z";
const faceKeys = ["01", "02", "03", "10", "11", "12"];
let quadKeys = getQuadKeys();
let faces = faceKeys.map((faceKey) => {
@@ -79809,7 +82395,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
const xy = qkToXY(quadKey);
return {
face: faceKey,
- url: imgUrlPrefix + faceKey + quadKey + imgUrlSuffix,
+ url: d2.imageUrl.replace("{faceId}", faceKey).replace("{tileId}", quadKey),
x: xy[0],
y: xy[1]
};
@@ -80080,8 +82666,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (err) {
callback(err);
} else {
- var f3 = filterKeys(params.filter);
- var result = d2.data.filter(f3).sort(sortKeys).map(valKey);
+ var f2 = filterKeys(params.filter);
+ var result = d2.data.filter(f2).sort(sortKeys).map(valKey);
_taginfoCache[url] = result;
callback(null, result);
}
@@ -80103,8 +82689,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (err) {
callback(err);
} else {
- var f3 = filterMultikeys(prefix);
- var result = d2.data.filter(f3).map(valKey);
+ var f2 = filterMultikeys(prefix);
+ var result = d2.data.filter(f2).map(valKey);
_taginfoCache[url] = result;
callback(null, result);
}
@@ -80131,8 +82717,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
callback(err);
} else {
var allowUpperCase = allowUpperCaseTagValues.test(params.key);
- var f3 = filterValues(allowUpperCase);
- var result = d2.data.filter(f3).map(valKeyDescription);
+ var f2 = filterValues(allowUpperCase);
+ var result = d2.data.filter(f2).map(valKeyDescription);
_taginfoCache[url] = result;
callback(null, result);
}
@@ -80154,8 +82740,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (err) {
callback(err);
} else {
- var f3 = filterRoles(geometry);
- var result = d2.data.filter(f3).map(roleKey);
+ var f2 = filterRoles(geometry);
+ var result = d2.data.filter(f2).map(roleKey);
_taginfoCache[url] = result;
callback(null, result);
}
@@ -80191,7 +82777,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
// 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_polygon_clipping = __toESM(require_polygon_clipping_umd());
var import_pbf2 = __toESM(require_pbf());
var import_vector_tile2 = __toESM(require_vector_tile());
var tiler8 = utilTiler().tileSize(512).margin(1);
@@ -80238,7 +82824,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
var merged = mergeCache[propertyhash];
if (merged && merged.length) {
var other = merged[0];
- var coords = import_polygon_clipping2.default.union(
+ var coords = import_polygon_clipping.default.union(
feature3.geometry.coordinates,
other.geometry.coordinates
);
@@ -80246,9 +82832,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
continue;
}
merged.push(feature3);
- for (var j3 = 0; j3 < merged.length; j3++) {
- merged[j3].geometry.coordinates = coords;
- merged[j3].__featurehash__ = featurehash;
+ for (var j2 = 0; j2 < merged.length; j2++) {
+ merged[j2].geometry.coordinates = coords;
+ merged[j2].__featurehash__ = featurehash;
}
} else {
mergeCache[propertyhash] = [feature3];
@@ -80321,8 +82907,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
var features = source.loaded[tiles[i3].id];
if (!features || !features.length)
continue;
- for (var j3 = 0; j3 < features.length; j3++) {
- var feature3 = features[j3];
+ for (var j2 = 0; j2 < features.length; j2++) {
+ var feature3 = features[j2];
var hash = feature3.__featurehash__;
if (seen[hash])
continue;
@@ -80643,10 +83229,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
throw new Error("No Results");
}
if (callback) {
- var list = result.query.pages[Object.keys(result.query.pages)[0]];
+ var list2 = result.query.pages[Object.keys(result.query.pages)[0]];
var translations = {};
- if (list && list.langlinks) {
- list.langlinks.forEach(function(d2) {
+ if (list2 && list2.langlinks) {
+ list2.langlinks.forEach(function(d2) {
translations[d2.lang] = d2["*"];
});
}
@@ -80675,11 +83261,11 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
var pannellumViewerCSS3 = "pannellum/pannellum.css";
var pannellumViewerJS3 = "pannellum/pannellum.js";
var resolution = 1080;
- var _mlyActiveImage2;
- var _mlyCache2;
+ var _activeImage;
+ var _cache4;
var _loadViewerPromise5;
var _pannellumViewer3;
- var _mlySceneOptions = {
+ var _sceneOptions2 = {
showFullscreenCtrl: false,
autoLoad: true,
yaw: 0,
@@ -80713,8 +83299,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
});
}
function loadTile4(which, url, tile) {
- const cache = _mlyCache2.requests;
- const tileId = `${tile.id}-${which}`;
+ const cache = _cache4.requests;
+ const tileId = "".concat(tile.id, "-").concat(which);
if (cache.loaded[tileId] || cache.inflight[tileId])
return;
const controller = new AbortController();
@@ -80750,7 +83336,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
let features, cache, layer, i3, feature3, loc, d2;
if (vectorTile.layers.hasOwnProperty(pointLayer)) {
features = [];
- cache = _mlyCache2.images;
+ cache = _cache4.images;
layer = vectorTile.layers[pointLayer];
for (i3 = 0; i3 < layer.length; i3++) {
feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
@@ -80782,7 +83368,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
}
if (vectorTile.layers.hasOwnProperty(lineLayer)) {
- cache = _mlyCache2.sequences;
+ cache = _cache4.sequences;
layer = vectorTile.layers[lineLayer];
for (i3 = 0; i3 < layer.length; i3++) {
feature3 = layer.feature(i3).toGeoJSON(tile.xyz[0], tile.xyz[1], tile.xyz[2]);
@@ -80795,7 +83381,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
}
function getImageData(imageId, sequenceId) {
- return fetch(apiUrl2 + `/api/sequence-detail?sequence_uuid=${sequenceId}`, { method: "GET" }).then(function(response) {
+ return fetch(apiUrl2 + "/api/sequence-detail?sequence_uuid=".concat(sequenceId), { method: "GET" }).then(function(response) {
if (!response.ok) {
throw new Error(response.status + " " + response.statusText);
}
@@ -80803,38 +83389,38 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}).then(function(data) {
let index = data.data.findIndex((feature3) => feature3.id === imageId);
const { filename, uploaded_hash } = data.data[index];
- _mlySceneOptions.panorama = imageBaseUrl + "/" + uploaded_hash + "/" + filename + "/" + resolution;
+ _sceneOptions2.panorama = imageBaseUrl + "/" + uploaded_hash + "/" + filename + "/" + resolution;
});
}
var mapilio_default = {
// Initialize Mapilio
init: function() {
- if (!_mlyCache2) {
+ if (!_cache4) {
this.reset();
}
this.event = utilRebind(this, dispatch13, "on");
},
// Reset cache and state
reset: function() {
- if (_mlyCache2) {
- Object.values(_mlyCache2.requests.inflight).forEach(function(request3) {
+ if (_cache4) {
+ Object.values(_cache4.requests.inflight).forEach(function(request3) {
request3.abort();
});
}
- _mlyCache2 = {
+ _cache4 = {
images: { rtree: new import_rbush12.default(), forImageId: {} },
sequences: { rtree: new import_rbush12.default(), lineString: {} },
requests: { loaded: {}, inflight: {} }
};
- _mlyActiveImage2 = null;
+ _activeImage = null;
},
// Get visible images
images: function(projection2) {
const limit = 5;
- return searchLimited5(limit, projection2, _mlyCache2.images.rtree);
+ return searchLimited5(limit, projection2, _cache4.images.rtree);
},
cachedImage: function(imageKey) {
- return _mlyCache2.images.forImageId[imageKey];
+ return _cache4.images.forImageId[imageKey];
},
// Load images in the visible area
loadImages: function(projection2) {
@@ -80854,14 +83440,14 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
const bbox2 = geoExtent(projection2.invert(min3), projection2.invert(max3)).bbox();
const sequenceIds = {};
let lineStrings = [];
- _mlyCache2.images.rtree.search(bbox2).forEach(function(d2) {
+ _cache4.images.rtree.search(bbox2).forEach(function(d2) {
if (d2.data.sequence_id) {
sequenceIds[d2.data.sequence_id] = true;
}
});
Object.keys(sequenceIds).forEach(function(sequenceId) {
- if (_mlyCache2.sequences.lineString[sequenceId]) {
- lineStrings = lineStrings.concat(_mlyCache2.sequences.lineString[sequenceId]);
+ if (_cache4.sequences.lineString[sequenceId]) {
+ lineStrings = lineStrings.concat(_cache4.sequences.lineString[sequenceId]);
}
});
return lineStrings;
@@ -80869,20 +83455,20 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
// Set the currently visible image
setActiveImage: function(image) {
if (image) {
- _mlyActiveImage2 = {
+ _activeImage = {
id: image.id,
sequence_id: image.sequence_id
};
} else {
- _mlyActiveImage2 = null;
+ _activeImage = 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 = _mlyActiveImage2 && _mlyActiveImage2.sequence_id;
- const selectedImageId = _mlyActiveImage2 && _mlyActiveImage2.id;
+ const selectedSequenceId = _activeImage && _activeImage.sequence_id;
+ const selectedImageId = _activeImage && _activeImage.id;
const markers = context.container().selectAll(".layer-mapilio .viewfield-group");
const sequences = context.container().selectAll(".layer-mapilio .sequence");
markers.classed("highlighted", function(d2) {
@@ -80921,7 +83507,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
"default": { firstScene: sceneID },
scenes: {}
};
- options2.scenes[sceneID] = _mlySceneOptions;
+ options2.scenes[sceneID] = _sceneOptions2;
_pannellumViewer3 = window.pannellum.viewer("ideditor-viewer-mapilio-pnlm", options2);
},
selectImage: function(context, id2) {
@@ -80941,7 +83527,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
attribution.append("span").attr("class", "captured_at").text(localeDateString2(d2.capture_time));
attribution.append("span").text("|");
}
- attribution.append("a").attr("class", "image-link").attr("target", "_blank").attr("href", `https://mapilio.com/app?lat=${d2.loc[1]}&lng=${d2.loc[0]}&zoom=17&pId=${d2.id}`).text("mapilio.com");
+ attribution.append("a").attr("class", "image-link").attr("target", "_blank").attr("href", "https://mapilio.com/app?lat=".concat(d2.loc[1], "&lng=").concat(d2.loc[0], "&zoom=17&pId=").concat(d2.id)).text("mapilio.com");
wrap2.transition().duration(100).call(imgZoom3.transform, identity2);
wrap2.selectAll("img").remove();
getImageData(d2.id, d2.sequence_id).then(function() {
@@ -80951,7 +83537,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
} else {
_currScene2 += 1;
let sceneID = _currScene2.toString();
- _pannellumViewer3.addScene(sceneID, _mlySceneOptions).loadScene(sceneID);
+ _pannellumViewer3.addScene(sceneID, _sceneOptions2).loadScene(sceneID);
if (_currScene2 > 2) {
sceneID = (_currScene2 - 1).toString();
_pannellumViewer3.removeScene(sceneID);
@@ -80980,9 +83566,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
let wrap2 = context.container().select("#ideditor-viewer-mapilio-simple");
let imgWrap = wrap2.select("img");
if (!imgWrap.empty()) {
- imgWrap.attr("src", _mlySceneOptions.panorama);
+ imgWrap.attr("src", _sceneOptions2.panorama);
} else {
- wrap2.append("img").attr("src", _mlySceneOptions.panorama);
+ wrap2.append("img").attr("src", _sceneOptions2.panorama);
}
},
ensureViewerLoaded: function(context) {
@@ -81025,13 +83611,13 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
});
function step(stepBy) {
return function() {
- if (!_mlyActiveImage2)
+ if (!_activeImage)
return;
- const imageId = _mlyActiveImage2.id;
+ const imageId = _activeImage.id;
const nextIndex = imageId + stepBy;
if (!nextIndex)
return;
- const nextImage = _mlyCache2.images.forImageId[nextIndex];
+ const nextImage = _cache4.images.forImageId[nextIndex];
context.map().centerEase(nextImage.loc);
that.selectImage(context, nextImage.id);
};
@@ -81066,7 +83652,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
},
// Return the current cache
cache: function() {
- return _mlyCache2;
+ return _cache4;
}
};
@@ -81545,19 +84131,19 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}) : [];
}
var _candidates = candidateWays();
- var operation = function() {
+ var operation2 = function() {
var candidate = _candidates[0];
context.enter(
modeDrawLine(context, candidate.id, context.graph(), "line", candidate.affix(_vertex.id), true)
);
};
- operation.relatedEntityIds = function() {
+ operation2.relatedEntityIds = function() {
return _candidates.length ? [_candidates[0].id] : [];
};
- operation.available = function() {
+ operation2.available = function() {
return _geometries.vertex.length === 1 && _geometries.line.length <= 1 && !context.features().hasHiddenConnections(_vertex, context.graph());
};
- operation.disabled = function() {
+ operation2.disabled = function() {
if (_candidates.length === 0) {
return "not_eligible";
} else if (_candidates.length > 1) {
@@ -81565,18 +84151,18 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return false;
};
- operation.tooltip = function() {
- var disable = operation.disabled();
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
return disable ? _t.append("operations.continue." + disable) : _t.append("operations.continue.description");
};
- operation.annotation = function() {
+ operation2.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;
+ operation2.id = "continue";
+ operation2.keys = [_t("operations.continue.key")];
+ operation2.title = _t.append("operations.continue.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/operations/copy.js
@@ -81587,7 +84173,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return entity.hasInterestingTags() || entity.geometry(context.graph()) !== "vertex";
});
}
- var operation = function() {
+ var operation2 = function() {
var graph = context.graph();
var selected = groupEntities(getFilteredIdsToCopy(), graph);
var canCopy = [];
@@ -81651,37 +84237,37 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return descendants;
}
- operation.available = function() {
+ operation2.available = function() {
return getFilteredIdsToCopy().length > 0;
};
- operation.disabled = function() {
+ operation2.disabled = function() {
var extent = utilTotalExtent(getFilteredIdsToCopy(), context.graph());
if (extent.percentContainedIn(context.map().extent()) < 0.8) {
return "too_large";
}
return false;
};
- operation.availableForKeypress = function() {
+ operation2.availableForKeypress = function() {
var selection2 = window.getSelection && window.getSelection();
return !selection2 || !selection2.toString();
};
- operation.tooltip = function() {
- var disable = operation.disabled();
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
return disable ? _t.append("operations.copy." + disable, { n: selectedIDs.length }) : _t.append("operations.copy.description", { n: selectedIDs.length });
};
- operation.annotation = function() {
+ operation2.annotation = function() {
return _t("operations.copy.annotation", { n: selectedIDs.length });
};
var _point;
- operation.point = function(val) {
+ operation2.point = function(val) {
_point = val;
- return operation;
+ return operation2;
};
- operation.id = "copy";
- operation.keys = [uiCmd("\u2318C")];
- operation.title = _t.append("operations.copy.title");
- operation.behavior = behaviorOperation(context).which(operation);
- return operation;
+ operation2.id = "copy";
+ operation2.keys = [uiCmd("\u2318C")];
+ operation2.title = _t.append("operations.copy.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/operations/disconnect.js
@@ -81777,21 +84363,21 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
}
var _extent = utilTotalExtent(_disconnectingVertexIds, context.graph());
- var operation = function() {
+ var operation2 = function() {
context.perform(function(graph) {
return _actions.reduce(function(graph2, action) {
return action(graph2);
}, graph);
- }, operation.annotation());
+ }, operation2.annotation());
context.validator().validate();
};
- operation.relatedEntityIds = function() {
+ operation2.relatedEntityIds = function() {
if (_vertexIDs.length) {
return _disconnectingWayIds;
}
return _disconnectingVertexIds;
};
- operation.available = function() {
+ operation2.available = function() {
if (_actions.length === 0)
return false;
if (_otherIDs.length !== 0)
@@ -81805,7 +84391,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return false;
return true;
};
- operation.disabled = function() {
+ operation2.disabled = function() {
var reason;
for (var actionIndex in _actions) {
reason = _actions[actionIndex].disabled(context.graph());
@@ -81838,18 +84424,18 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return false;
}
};
- operation.tooltip = function() {
- var disable = operation.disabled();
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
return disable ? _t.append("operations.disconnect." + disable) : _t.append("operations.disconnect.description." + _descriptionID);
};
- operation.annotation = function() {
+ operation2.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;
+ operation2.id = "disconnect";
+ operation2.keys = [_t("operations.disconnect.key")];
+ operation2.title = _t.append("operations.disconnect.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/operations/downgrade.js
@@ -81898,9 +84484,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return null;
}
- var buildingKeysToKeep = ["architect", "building", "height", "layer", "source", "type", "wheelchair"];
+ var buildingKeysToKeep = ["architect", "building", "height", "layer", "nycdoitt:bin", "source", "type", "wheelchair"];
var addressKeysToKeep = ["source"];
- var operation = function() {
+ var operation2 = function() {
context.perform(function(graph) {
for (var i3 in selectedIDs) {
var entityID = selectedIDs[i3];
@@ -81924,14 +84510,14 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
graph = actionChangeTags(entityID, tags)(graph);
}
return graph;
- }, operation.annotation());
+ }, operation2.annotation());
context.validator().validate();
context.enter(modeSelect(context, selectedIDs));
};
- operation.available = function() {
+ operation2.available = function() {
return _downgradeType;
};
- operation.disabled = function() {
+ operation2.disabled = function() {
if (selectedIDs.some(hasWikidataTag)) {
return "has_wikidata_tag";
}
@@ -81941,11 +84527,11 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return entity.tags.wikidata && entity.tags.wikidata.trim().length > 0;
}
};
- operation.tooltip = function() {
- var disable = operation.disabled();
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
return disable ? _t.append("operations.downgrade." + disable + "." + _multi) : _t.append("operations.downgrade.description." + _downgradeType);
};
- operation.annotation = function() {
+ operation2.annotation = function() {
var suffix;
if (_downgradeType === "building_address") {
suffix = "generic";
@@ -81954,11 +84540,11 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
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;
+ operation2.id = "downgrade";
+ operation2.keys = [uiCmd("\u232B")];
+ operation2.title = _t.append("operations.downgrade.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/operations/extract.js
@@ -81984,23 +84570,23 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
_extent = _extent ? _extent.extend(entity.extent(graph)) : entity.extent(graph);
return actionExtract(entityID, context.projection);
}).filter(Boolean);
- var operation = function() {
+ var operation2 = function() {
var combinedAction = function(graph) {
_actions.forEach(function(action) {
graph = action(graph);
});
return graph;
};
- context.perform(combinedAction, operation.annotation());
+ context.perform(combinedAction, operation2.annotation());
var extractedNodeIDs = _actions.map(function(action) {
return action.getExtractedNodeID();
});
context.enter(modeSelect(context, extractedNodeIDs));
};
- operation.available = function() {
+ operation2.available = function() {
return _actions.length && selectedIDs.length === _actions.length;
};
- operation.disabled = function() {
+ operation2.disabled = function() {
if (_extent && _extent.percentContainedIn(context.map().extent()) < 0.8) {
return "too_large";
} else if (selectedIDs.some(function(entityID) {
@@ -82010,22 +84596,22 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return false;
};
- operation.tooltip = function() {
- var disableReason = operation.disabled();
+ operation2.tooltip = function() {
+ var disableReason = operation2.disabled();
if (disableReason) {
return _t.append("operations.extract." + disableReason + "." + _amount);
} else {
return _t.append("operations.extract.description." + _geometryID + "." + _amount);
}
};
- operation.annotation = function() {
+ operation2.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;
+ operation2.id = "extract";
+ operation2.keys = [_t("operations.extract.key")];
+ operation2.title = _t.append("operations.extract.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/operations/merge.js
@@ -82052,10 +84638,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return mergePolygon;
return mergeNodes;
}
- var operation = function() {
- if (operation.disabled())
+ var operation2 = function() {
+ if (operation2.disabled())
return;
- context.perform(_action, operation.annotation());
+ context.perform(_action, operation2.annotation());
context.validator().validate();
var resultIDs = selectedIDs.filter(context.hasEntity);
if (resultIDs.length > 1) {
@@ -82067,10 +84653,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
context.enter(modeSelect(context, resultIDs));
};
- operation.available = function() {
+ operation2.available = function() {
return selectedIDs.length >= 2;
};
- operation.disabled = function() {
+ operation2.disabled = function() {
var actionDisabled = _action.disabled(context.graph());
if (actionDisabled)
return actionDisabled;
@@ -82080,8 +84666,8 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return false;
};
- operation.tooltip = function() {
- var disabled = operation.disabled();
+ operation2.tooltip = function() {
+ var disabled = operation2.disabled();
if (disabled) {
if (disabled === "conflicting_relations") {
return _t.append("operations.merge.conflicting_relations");
@@ -82096,20 +84682,20 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return _t.append("operations.merge.description");
};
- operation.annotation = function() {
+ operation2.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;
+ operation2.id = "merge";
+ operation2.keys = [_t("operations.merge.key")];
+ operation2.title = _t.append("operations.merge.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/operations/paste.js
function operationPaste(context) {
var _pastePoint;
- var operation = function() {
+ var operation2 = function() {
if (!_pastePoint)
return;
var oldIDs = context.copyIDs();
@@ -82140,20 +84726,20 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
var copyPoint = context.copyLonLat() && projection2(context.copyLonLat()) || projection2(extent.center());
var delta = geoVecSubtract(_pastePoint, copyPoint);
- context.replace(actionMove(newIDs, delta, projection2), operation.annotation());
+ context.replace(actionMove(newIDs, delta, projection2), operation2.annotation());
context.enter(modeSelect(context, newIDs));
};
- operation.point = function(val) {
+ operation2.point = function(val) {
_pastePoint = val;
- return operation;
+ return operation2;
};
- operation.available = function() {
+ operation2.available = function() {
return context.mode().id === "browse";
};
- operation.disabled = function() {
+ operation2.disabled = function() {
return !context.copyIDs().length;
};
- operation.tooltip = function() {
+ operation2.tooltip = function() {
var oldGraph = context.copyGraph();
var ids = context.copyIDs();
if (!ids.length) {
@@ -82161,25 +84747,25 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return _t.append("operations.paste.description", { feature: utilDisplayLabel(oldGraph.entity(ids[0]), oldGraph), n: ids.length });
};
- operation.annotation = function() {
+ operation2.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;
+ operation2.id = "paste";
+ operation2.keys = [uiCmd("\u2318V")];
+ operation2.title = _t.append("operations.paste.title");
+ return operation2;
}
// modules/operations/reverse.js
function operationReverse(context, selectedIDs) {
- var operation = function() {
+ var operation2 = function() {
context.perform(function combinedReverseAction(graph) {
actions().forEach(function(action) {
graph = action(graph);
});
return graph;
- }, operation.annotation());
+ }, operation2.annotation());
context.validator().validate();
};
function actions(situation) {
@@ -82212,24 +84798,24 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return "point";
return "feature";
}
- operation.available = function(situation) {
+ operation2.available = function(situation) {
return actions(situation).length > 0;
};
- operation.disabled = function() {
+ operation2.disabled = function() {
return false;
};
- operation.tooltip = function() {
+ operation2.tooltip = function() {
return _t.append("operations.reverse.description." + reverseTypeID());
};
- operation.annotation = function() {
+ operation2.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;
+ operation2.id = "reverse";
+ operation2.keys = [_t("operations.reverse.key")];
+ operation2.title = _t.append("operations.reverse.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/operations/split.js
@@ -82260,20 +84846,20 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
_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) {
+ var operation2 = function() {
+ var difference2 = context.perform(_action, operation2.annotation());
+ var idsToSelect = _vertexIds.concat(difference2.extantIDs().filter(function(id2) {
return context.entity(id2).type === "way";
}));
context.enter(modeSelect(context, idsToSelect));
};
- operation.relatedEntityIds = function() {
+ operation2.relatedEntityIds = function() {
return _selectedWayIds.length ? [] : _ways.map((way) => way.id);
};
- operation.available = function() {
+ operation2.available = function() {
return _isAvailable;
};
- operation.disabled = function() {
+ operation2.disabled = function() {
var reason = _action.disabled(context.graph());
if (reason) {
return reason;
@@ -82282,25 +84868,25 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return false;
};
- operation.tooltip = function() {
- var disable = operation.disabled();
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
return disable ? _t.append("operations.split." + disable) : _t.append("operations.split.description." + _geometry + "." + _waysAmount + "." + _nodesAmount + "_node");
};
- operation.annotation = function() {
+ operation2.annotation = function() {
return _t("operations.split.annotation." + _geometry, { n: _ways.length });
};
- operation.icon = function() {
+ operation2.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;
+ operation2.id = "split";
+ operation2.keys = [_t("operations.split.key")];
+ operation2.title = _t.append("operations.split.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/operations/straighten.js
@@ -82359,18 +84945,18 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
}
return null;
}
- function operation() {
+ function operation2() {
if (!_action)
return;
- context.perform(_action, operation.annotation());
+ context.perform(_action, operation2.annotation());
window.setTimeout(function() {
context.validator().validate();
}, 300);
}
- operation.available = function() {
+ operation2.available = function() {
return Boolean(_action);
};
- operation.disabled = function() {
+ operation2.disabled = function() {
var reason = _action.disabled(context.graph());
if (reason) {
return reason;
@@ -82400,18 +84986,18 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return false;
}
};
- operation.tooltip = function() {
- var disable = operation.disabled();
+ operation2.tooltip = function() {
+ var disable = operation2.disabled();
return disable ? _t.append("operations.straighten." + disable + "." + _amount) : _t.append("operations.straighten.description." + _geometry + (_wayIDs.length === 1 ? "" : "s"));
};
- operation.annotation = function() {
+ operation2.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;
+ operation2.id = "straighten";
+ operation2.keys = [_t("operations.straighten.key")];
+ operation2.title = _t.append("operations.straighten.title");
+ operation2.behavior = behaviorOperation(context).which(operation2);
+ return operation2;
}
// modules/modes/select.js
@@ -82543,9 +85129,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
return mode;
};
function loadOperations() {
- _operations.forEach(function(operation) {
- if (operation.behavior) {
- context.uninstall(operation.behavior);
+ _operations.forEach(function(operation2) {
+ if (operation2.behavior) {
+ context.uninstall(operation2.behavior);
}
});
_operations = Object.values(operations_exports).map(function(o2) {
@@ -82557,12 +85143,12 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
operationCopy(context, selectedIDs),
operationDowngrade(context, selectedIDs),
operationDelete(context, selectedIDs)
- ]).filter(function(operation) {
- return operation.available();
+ ]).filter(function(operation2) {
+ return operation2.available();
});
- _operations.forEach(function(operation) {
- if (operation.behavior) {
- context.install(operation.behavior);
+ _operations.forEach(function(operation2) {
+ if (operation2.behavior) {
+ context.install(operation2.behavior);
}
});
context.ui().closeEditMenu();
@@ -82780,13 +85366,13 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (!parentId)
return;
var way = context.entity(parentId);
- var length = way.nodes.length;
+ var length2 = 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;
+ index = length2 - 2;
}
if (index !== -1) {
context.enter(
@@ -82801,10 +85387,10 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (!parentId)
return;
var way = context.entity(parentId);
- var length = way.nodes.length;
+ var length2 = way.nodes.length;
var curr = way.nodes.indexOf(selectedIDs[0]);
var index = -1;
- if (curr < length - 1) {
+ if (curr < length2 - 1) {
index = curr + 1;
} else if (way.isClosed()) {
index = 0;
@@ -82859,9 +85445,9 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
mode.exit = function() {
_newFeature = false;
_focusedVertexIds = null;
- _operations.forEach(function(operation) {
- if (operation.behavior) {
- context.uninstall(operation.behavior);
+ _operations.forEach(function(operation2) {
+ if (operation2.behavior) {
+ context.uninstall(operation2.behavior);
}
});
_operations = [];
@@ -83070,7 +85656,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
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 precision3 = Math.max(0, Math.ceil(Math.log(zoom) / Math.LN2));
var oldParams = utilObjectOmit(
utilStringQs(window.location.hash),
["comment", "source", "hashtags", "walkthrough"]
@@ -83083,7 +85669,7 @@ this.ifd0Offset: ${this.ifd0Offset}, file.byteLength: ${e3.byteLength}`), e3.tif
if (selected.length) {
newParams.id = selected.join(",");
}
- newParams.map = zoom.toFixed(2) + "/" + center[1].toFixed(precision2) + "/" + center[0].toFixed(precision2);
+ newParams.map = zoom.toFixed(2) + "/" + center[1].toFixed(precision3) + "/" + center[0].toFixed(precision3);
return Object.assign(oldParams, newParams);
}
function computedHash() {