1 let abortController = null;
2 const languagesToRequest = [...new Set(OSM.preferred_languages.map(l => l.toLowerCase()))];
3 const wikisToRequest = [...new Set([...OSM.preferred_languages, "en"].map(l => l.split("-")[0] + "wiki"))];
4 const isOfExpectedLanguage = ({ language }) => languagesToRequest[0].startsWith(language) || language === "mul";
6 export function element(type) {
10 page.pushstate = page.popstate = function (path, id, version) {
11 OSM.loadSidebarContent(path, function () {
12 page._addObject(type, id, version);
13 $(".numbered_pagination").trigger("numbered_pagination:enable");
14 abortController = new AbortController();
18 page.load = function (path, id, version) {
19 page._addObject(type, id, version, true);
20 $(".numbered_pagination").trigger("numbered_pagination:enable");
21 abortController = new AbortController();
24 page.unload = function () {
26 $(".numbered_pagination").trigger("numbered_pagination:disable");
27 abortController?.abort();
30 page._addObject = function () {};
31 page._removeObject = function () {};
37 export function mappedElement(type) {
38 return function (map) {
39 const page = element(type)(map);
41 page._addObject = function (type, id, version, center) {
42 const hashParams = OSM.parseHash();
43 map.addObject({ type: type, id: parseInt(id, 10), version: version && parseInt(version, 10) }, function (bounds) {
44 if (!hashParams.center && bounds.isValid() &&
45 (center || !map.getBounds().contains(bounds))) {
46 OSM.router.withoutMoveListener(function () {
47 map.fitBounds(bounds);
53 page._removeObject = function () {
61 $(document).on("click", "button.wdt-preview", e => previewWikidataValue($(e.currentTarget)));
63 function previewWikidataValue($btn) {
64 if (!OSM.WIKIDATA_API_URL) return;
65 const items = $btn.data("qids");
66 if (!items?.length) return;
67 $btn.prop("disabled", true);
68 fetch(OSM.WIKIDATA_API_URL + "?" + new URLSearchParams({
69 action: "wbgetentities",
73 props: "labels|sitelinks/urls|claims|descriptions",
74 languages: languagesToRequest.join("|"),
76 sitefilter: wikisToRequest.join("|")
78 headers: { "Api-User-Agent": "OSM-TagPreview (https://github.com/openstreetmap/openstreetmap-website)" },
79 signal: abortController?.signal
81 .then(response => response.ok ? response.json() : Promise.reject(response))
82 .then(({ entities }) => {
83 if (!entities) return Promise.reject(entities);
88 .filter(qid => entities[qid])
89 .map(qid => getLocalizedResponse(entities[qid]))
90 .filter(data => data.label || data.icon || data.description || data.article)
91 .map(data => renderWikidataResponse(data, $btn.siblings(`a[href*="wikidata.org/entity/${data.qid}"]`)))
94 .catch(() => $btn.prop("disabled", false));
97 function getLocalizedResponse(entity) {
98 const siteScheme = OSM.isDark("bs") ? "Q6545942" : "Q101608434";
99 const scheme = ({ qualifiers }) => qualifiers?.P8798?.some(q => q?.datavalue?.value?.id === siteScheme) ?? 0;
100 const rank = ({ rank }) => ({ preferred: 2, normal: 0, deprecated: -2 })[rank] ?? 0;
101 const toBestClaim = (out, claim) => (rank(claim) + scheme(claim) > rank(out) + scheme(out)) ? claim : out;
102 const toFirstOf = (property) => (out, localization) => out ?? property[localization];
105 label: languagesToRequest.reduce(toFirstOf(entity.labels), null),
107 "P8972", // small logo or icon
108 "P154", // logo image
109 "P14" // traffic sign
110 ].reduce((out, prop) => out ?? entity.claims[prop]?.reduce(toBestClaim)?.mainsnak?.datavalue?.value, null),
111 description: languagesToRequest.reduce(toFirstOf(entity.descriptions), null),
112 article: wikisToRequest.reduce(toFirstOf(entity.sitelinks), null)
114 if (data.article) data.article.language = data.article.site.replace("wiki", "");
118 function renderWikidataResponse({ icon, label, article, description }, $link) {
119 const localeName = new Intl.DisplayNames(OSM.preferred_languages, { type: "language" });
120 const cell = $("<td>")
122 .addClass("bg-body-tertiary");
124 if (icon && OSM.WIKIMEDIA_COMMONS_URL) {
125 let src = OSM.WIKIMEDIA_COMMONS_URL + "Special:Redirect/file/" + encodeURIComponent(icon) + "?mobileaction=toggle_view_desktop";
126 if (!icon.endsWith(".svg")) src += "&width=128";
128 .attr("href", OSM.WIKIMEDIA_COMMONS_URL + "File:" + encodeURIComponent(icon) + `?uselang=${OSM.i18n.locale}`)
129 .append($("<img>").attr({ src, height: "32" }))
130 .addClass("float-end mb-1 ms-2")
134 const link = $link.clone()
138 if (!isOfExpectedLanguage(label)) {
139 link.attr("lang", label.language);
140 link.after($("<sup>").text(" " + localeName.of(label.language)));
144 const link = $("<a>")
145 .attr("href", article.url + `?uselang=${OSM.i18n.locale}`)
146 .text(label ? OSM.i18n.t("javascripts.element.wikipedia") : article.title)
153 if (!isOfExpectedLanguage(article)) {
154 link.attr("lang", article.language);
155 link.after($("<sup>").text(" " + localeName.of(article.language)));
159 const text = $("<div>")
160 .text(description.value)
164 if (!isOfExpectedLanguage(description)) {
165 text.attr("lang", description.language);
168 return $("<tr>").append(cell);