1 OSM.initializeDataLayer = function (map) {
 
   2   let dataLoader, loadedBounds;
 
   3   const dataLayer = map.dataLayer;
 
  20   dataLayer.isWayArea = function () {
 
  24   dataLayer.on("click", function (e) {
 
  28   dataLayer.on("add", function () {
 
  29     map.fire("overlayadd", { layer: this });
 
  30     map.on("moveend", updateData);
 
  34   dataLayer.on("remove", function () {
 
  35     if (dataLoader) dataLoader.abort();
 
  37     map.off("moveend", updateData);
 
  38     $("#browse_status").empty();
 
  39     map.fire("overlayremove", { layer: this });
 
  42   function updateData() {
 
  43     const bounds = map.getBounds();
 
  44     if (!loadedBounds || !loadedBounds.contains(bounds)) {
 
  49   function displayFeatureWarning(num_features, add, cancel) {
 
  50     $("#browse_status").html(
 
  51       $("<div class='p-3'>").append(
 
  52         $("<div class='d-flex'>").append(
 
  53           $("<h2 class='flex-grow-1 text-break'>")
 
  54             .text(OSM.i18n.t("browse.start_rjs.load_data")),
 
  56             $("<button type='button' class='btn-close'>")
 
  57               .attr("aria-label", OSM.i18n.t("javascripts.close"))
 
  59         $("<p class='alert alert-warning'>")
 
  60           .text(OSM.i18n.t("browse.start_rjs.feature_warning", { num_features })),
 
  61         $("<input type='submit' class='btn btn-primary d-block mx-auto'>")
 
  62           .val(OSM.i18n.t("browse.start_rjs.load_data"))
 
  66   function displayLoadError(message, close) {
 
  67     $("#browse_status").html(
 
  68       $("<div class='p-3'>").append(
 
  69         $("<div class='d-flex'>").append(
 
  70           $("<h2 class='flex-grow-1 text-break'>")
 
  71             .text(OSM.i18n.t("browse.start_rjs.load_data")),
 
  73             $("<button type='button' class='btn-close'>")
 
  74               .attr("aria-label", OSM.i18n.t("javascripts.close"))
 
  76         $("<p class='alert alert-warning'>")
 
  77           .text(OSM.i18n.t("browse.start_rjs.feature_error", { message: message }))));
 
  81     const bounds = map.getBounds();
 
  82     const url = "/api/" + OSM.API_VERSION + "/map.json?bbox=" + bounds.toBBoxString();
 
  85      * Modern browsers are quite happy showing far more than 100 features in
 
  86      * the data browser, so increase the limit to 4000.
 
  88     const maxFeatures = 4000;
 
  90     if (dataLoader) dataLoader.abort();
 
  92     $("#layers-data-loading").remove();
 
  94     const spanLoading = $("<span>")
 
  95       .attr("id", "layers-data-loading")
 
  96       .attr("class", "spinner-border spinner-border-sm ms-1")
 
  97       .attr("role", "status")
 
  98       .html("<span class='visually-hidden'>" + OSM.i18n.t("browse.start_rjs.loading") + "</span>")
 
  99       .appendTo($("#label-layers-data"));
 
 101     dataLoader = new AbortController();
 
 102     fetch(url, { signal: dataLoader.signal })
 
 104         if (response.ok) return response.json();
 
 105         const status = response.statusText || response.status;
 
 106         if (response.status !== 400) throw new Error(status);
 
 107         return response.text().then(text => {
 
 108           throw new Error(text || status);
 
 111       .then(function (data) {
 
 112         dataLayer.clearLayers();
 
 114         const features = dataLayer.buildFeatures(data);
 
 116         function addFeatures() {
 
 117           $("#browse_status").empty();
 
 118           dataLayer.addData(features);
 
 119           loadedBounds = bounds;
 
 122         function cancelAddFeatures() {
 
 123           $("#browse_status").empty();
 
 126         if (features.length < maxFeatures) {
 
 129           displayFeatureWarning(features.length, addFeatures, cancelAddFeatures);
 
 132         if (map._objectLayer) {
 
 133           map._objectLayer.bringToFront();
 
 136       .catch(function (error) {
 
 137         if (error.name === "AbortError") return;
 
 139         displayLoadError(error?.message, () => {
 
 140           $("#browse_status").empty();
 
 145         spanLoading.remove();
 
 149   function onSelect(layer) {
 
 150     OSM.router.route("/" + layer.feature.type + "/" + layer.feature.id);