Skip to content

Commit

Permalink
Define min/max bounds for bbox updates to avoid constant updates whil…
Browse files Browse the repository at this point in the history
…e locate control is active
  • Loading branch information
cdauth committed Apr 26, 2024
1 parent 1ae85c2 commit 4152c66
Showing 1 changed file with 35 additions and 10 deletions.
45 changes: 35 additions & 10 deletions leaflet/src/bbox-handler.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import type Client from "facilmap-client";
import type { ClientEvents } from "facilmap-client";
import type { EventHandler } from "facilmap-types";
import { Handler, LatLng, LatLngBounds, Map } from "leaflet";
import { Bounds, Handler, LatLng, LatLngBounds, Map } from "leaflet";
import { leafletToFmBbox } from "./utils/leaflet";

const flyToBkp = Map.prototype.flyTo;
Expand All @@ -11,9 +11,23 @@ Map.prototype.flyTo = function(...args) {
return flyToBkp.apply(this, args);
};

function extendBounds(bounds: Bounds, pixels: number): Bounds {
return new Bounds(
[bounds.getTopLeft().x - pixels, bounds.getTopLeft().y - pixels],
[bounds.getBottomRight().x + pixels, bounds.getBottomRight().y + pixels]
);
}

export default class BboxHandler extends Handler {

margin = 50;
/**
* The min and max values how much wider the map pixel bounds should be used than what they actually are.
* By default, the max value is used. If the map is panned and the last bounds are still within the range
* of the min value, no update is triggered. This avoids an update every second when the locate control
* adjusts the map position very slightly.
*/
margin = [50, 400];
lastBounds: { bounds: Bounds; zoom: number } | undefined = undefined;

client: Client;

Expand All @@ -26,17 +40,23 @@ export default class BboxHandler extends Handler {
if (!this._map._loaded && (center == null || zoom == null))
return;

const pixelBounds = this._map.getPixelBounds(center, zoom);
pixelBounds.min!.x -= this.margin;
pixelBounds.min!.y -= this.margin;
pixelBounds.max!.x += this.margin;
pixelBounds.max!.y += this.margin;
const resolvedZoom = zoom ?? this._map.getZoom();
const pixelBounds = this._map.getPixelBounds(center, resolvedZoom);

if (this.lastBounds && this.lastBounds.zoom === resolvedZoom) {
const minBounds = extendBounds(pixelBounds, this.margin[0]);
if (this.lastBounds.bounds.contains(minBounds)) {
return;
}
}

const maxBounds = extendBounds(pixelBounds, this.margin[1]);
const bounds = new LatLngBounds(
this._map.unproject(pixelBounds.getBottomLeft(), zoom),
this._map.unproject(pixelBounds.getTopRight(), zoom)
this._map.unproject(maxBounds.getBottomLeft(), resolvedZoom),
this._map.unproject(maxBounds.getTopRight(), resolvedZoom)
);
void this.client.updateBbox(leafletToFmBbox(bounds ?? this._map.getBounds(), zoom ?? this._map.getZoom()));
void this.client.updateBbox(leafletToFmBbox(bounds, resolvedZoom));
this.lastBounds = { bounds: maxBounds, zoom: resolvedZoom };
}

shouldUpdateBbox(): boolean {
Expand Down Expand Up @@ -72,12 +92,17 @@ export default class BboxHandler extends Handler {
this._map.on("viewreset", this.handleViewReset);
this._map.on("fmFlyTo", this.handleFlyTo);
this.client.on("emitResolve", this.handleEmitResolve);

if (this.shouldUpdateBbox()) {
this.updateBbox();
}
}

removeHooks(): void {
this._map.off("moveend", this.handleMoveEnd);
this._map.off("viewreset", this.handleViewReset);
this._map.off("fmFlyTo", this.handleFlyTo);
this.client.removeListener("emitResolve", this.handleEmitResolve);
this.lastBounds = undefined;
}
}

0 comments on commit 4152c66

Please sign in to comment.