Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Use baselayerchange/overlaylayerchange instead of layeradd/layerremove #5474

Open
wants to merge 1 commit into
base: master
Choose a base branch
from

Conversation

deevroman
Copy link

@deevroman deevroman commented Jan 6, 2025

Description

This is fix #5466 After merging openstreetmap/leaflet-osm#43 flame graph looks like this:

https://share.firefox.dev/4a2Wc9Z

Most of the time is spent on two handlers that update information about the current location and elements in the right sidebar.

Unfortunately, replacing layeradd layerremove with baselayerchange overlaylayerchange (#5466 (comment)) is difficult, since in addition to the Map Data layer, other elements may be displayed on the map. For example, the currently selected object on the map. It is not clear how to separate changes to Map Data objects and changes to the selected element. It is necessary to track changes in the active element, otherwise at least the Edit button will break.

Therefore, I implemented a little ugly, but safer in terms of breaking business logic. When adding the Map Data layer, I set the pause flag in the slowdown handlers. While the Map Data is rendering, the user cannot interact with the interface, so it seems safe to pause the event handlers.

Total 15s vs 0.5s when rendering 5000 elements.

https://share.firefox.dev/4h4IK7C

@deevroman deevroman marked this pull request as ready for review January 6, 2025 14:00
@AntonKhorev
Copy link
Collaborator

For example, the currently selected object on the map. It is not clear how to separate changes to Map Data objects and changes to the selected element. It is necessary to track changes in the active element, otherwise at least the Edit button will break.

addObject/removeObject in leaflet.map.js could fire some event for active object change. The listener that calls updateLinks could listen to that event.

@deevroman
Copy link
Author

I also saw a problem in PR: when you click the Map Data checkbox, the cookie with the current location is no longer updated

@deevroman deevroman marked this pull request as draft January 6, 2025 19:57
@deevroman deevroman force-pushed the speedup-datalayer-render branch 5 times, most recently from a359de9 to a32249e Compare January 10, 2025 23:27
@deevroman deevroman changed the title Suspend the layer event handlers when rendering Map Data Use baselayerchange/overlaylayerchange instead of layeradd/layerremove Jan 10, 2025
@deevroman deevroman marked this pull request as ready for review January 10, 2025 23:51
@deevroman
Copy link
Author

deevroman commented Jan 10, 2025

Okay, firing events from addObject / removeObject turned out to be the best idea. And I changed this PR to use baselayerchange/overlaylayerchange

I also replaced map.on("layeradd",...), which tracks the layer's on/off state with dataLayer.on("add", ...). Using overlaylayerchange in this case doesn't work, because this event isn't thrown when opening a new page with ...&layers=D Plus add / remove looks cleaner in my opinion.

@deevroman deevroman marked this pull request as draft January 10, 2025 23:59
@deevroman deevroman force-pushed the speedup-datalayer-render branch from a32249e to 5b37b54 Compare January 11, 2025 00:11
@deevroman deevroman marked this pull request as ready for review January 11, 2025 00:15
@mmd-osm
Copy link
Contributor

mmd-osm commented Jan 11, 2025

With the latest changes I was able to render Lake Huron locally in a reasonable amount of time (<20s, including 4.5s API response time; 455k nodes). Also map data display is extremely fast now (e.g. 100ms instead of 3s). That is definitely a significant speed-up.

image

Ideas for follow up PRs:

Furthermore, switching from XML to JSON has improved the response time from 20s to 8s (both values include 4.5s API time). For reference, I've uploaded the quick hack here: https://gist.github.com/mmd-osm/d45ff3b4f41908b1c075d70d1b6fdebe (edit: added fetch based + error handling alternative).

image

We should think about adding some error handling:

image
image

@HolgerJeromin
Copy link
Contributor

HolgerJeromin commented Jan 12, 2025

Jquery uses JSON.parse for parsing of ajax() result.

In a next step switching to fetch would allow using
https://developer.mozilla.org/en-US/docs/Web/API/Response/json
Where the parsing is done without blocking the main thread.

@deevroman deevroman force-pushed the speedup-datalayer-render branch from 5b37b54 to 30403cb Compare January 13, 2025 14:50
@AntonKhorev
Copy link
Collaborator

AntonKhorev commented Jan 14, 2025

We probably need to fire baselayerchange either from map.setState or from map.updateLayers. Otherwise if you edit &layers=X in the location bar, share urls are updated to its previous value.

If we follow the same principles as leaflet.layers.js, baselayerchange should be fired by the same function that does map.addLayer, so I guess map.updateLayers should fire it.

@deevroman
Copy link
Author

deevroman commented Jan 14, 2025

Otherwise if you edit &layers=X in the location bar, share urls are updated to its previous value.

Do I understand correctly that you are talking about the current behavior of the website, and not after PR? At least in Firefox, changing &layers=X does not change the layers on the map, only if you open in a new tab

@mmd-osm
Copy link
Contributor

mmd-osm commented Jan 14, 2025

@AntonKhorev 's comment wasn't exactly clear to me either. I think it's the following. Note that once you change the layer in the URL (in the example below from "C" to "H"), the share link "layer" attribute is updated accordingly:

Peek 2025-01-14 22-58

This doesn't seem to work anymore with the changes in this PR.

Besides, there may be some timing issue in the current code, as the behavior is not 100% reproducible for me, even today. Sometimes the map layer changes along with the share link, in other cases, the old layer is still shown.

@deevroman
Copy link
Author

deevroman commented Jan 14, 2025

Ah, I understand. I only tested changing the letters that represent the overlay (D/N/G). Changing the letters that respond to tile layers works now. PR broke it. I'll try to fix it

@AntonKhorev
Copy link
Collaborator

AntonKhorev commented Jan 14, 2025

Leaflet ships with a layers control which lets you switch between layers. That thing is responsible for firing baselayerchange events. We don't use this control because we have our own layers menu. Therefore we're responsible for firing that event, which we do:

map.fire("baselayerchange", { layer: layer });

The problem is that our layers menu is not the only thing that can switch layers. All other places where layer change happens need also to fire baselayerchange. I think that map.updateLayers is the only such place. Lack of baselayerchange there becomes more noticeable with this PR because it uses this event.

Similar story with overlaylayerchange, although it's not a standard Leaflet event. Leaflet layers controluses overlayadd and overlayremove.

UPD: we're doing this wrong, see the following comments...

@deevroman deevroman force-pushed the speedup-datalayer-render branch from 30403cb to ad9dcfe Compare January 14, 2025 23:50
@deevroman
Copy link
Author

Okay, I added the baselayerchange firing. So, when changing D/N/G, the update does not work now, so firing overlaylayerchange is better done in a separate PR.

@@ -116,6 +116,7 @@ L.OSM.Map = L.Map.extend({
} else {
this.removeLayer(this.baseLayers[i]);
}
this.fire("baselayerchange", { layer: this.baseLayers[i] });
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This fires the event for every possible base layer. It's supposed to be fired only for the layer that became visible, that's how it works in leaflet.layers.js.

Copy link
Author

@deevroman deevroman Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Implemented a check that there is already a layer on the map. But the code doesn’t look so attractive anymore, so it might be better to add add/remove handlers for all base layers. For some reason the naive addition in initialize() does not work yet

layer.on("add remove", function () {
  this.fire("baselayerchange", { layer: layer });
})

Copy link
Collaborator

@AntonKhorev AntonKhorev Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You could put this.fire ... next to this.addLayer ..., that's one extra line of code. (with a caveat that if you're switching to the same layer, baselayerchange still fires; if that's wrong, our layer menu is also wrong)

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, looks like our layers menu is wrong. Leaflet's layer selector doesn't fire baselayerchange if you clicking an already active layer, and if you add an already added layer using map.addLayer, layer events also don't happen.

@deevroman deevroman force-pushed the speedup-datalayer-render branch from ad9dcfe to f10083a Compare January 16, 2025 01:41
@deevroman deevroman force-pushed the speedup-datalayer-render branch from f10083a to 12fee32 Compare January 16, 2025 01:42
this.addLayer(this.baseLayers[i]);
} else {
baselayerChanged = this.hasLayer(this.baseLayers[i]);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

baselayerchange is supposed to be fired once for the layer that became visible. It's not supposed to be fired for removed layers. See that null in Leafet code?
https://github.com/Leaflet/Leaflet/blob/c511e66c20dce06ac05c43cf287b3ecdba00d924/src/control/Control.Layers.js#L317-L319

"This event would be triggered after the other layer events, and it would include a layer property set to the resulting base layer." Leaflet/Leaflet#1064

Copy link
Author

@deevroman deevroman Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So the next handler also violates this expectation?

input.on("click", function () {
layers.forEach(function (other) {
if (other === layer) {
map.addLayer(other);
} else {
map.removeLayer(other);
}
});
map.fire("baselayerchange", { layer: layer });
});

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The answer to the above question was yes but it probably didn't affect anything. However if we start using baselayerchange in more places, it might. For example, this code, if rewritten for baselayerchange wouldn't work correctly, if it expects this event to happen like Leaflet would fire it:

map.on("layeradd", function (e) {
if (e.layer.options) {
var goal = OSM.MATOMO.goals[e.layer.options.layerId];
if (goal) {
$("body").trigger("matomogoal", goal);
}
}
});

Copy link
Author

@deevroman deevroman Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It seems that "baselayerchange overlaylayerchange" will work in this place (provided that baselayerchange overlaylayerchange is firing in all places that the layer is added)

@AntonKhorev
Copy link
Collaborator

Do we fix baselayerchange everywhere before doing this PR that changes listeners for this event?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

"layeradd layerremove" event handlers dramatically slow down Map data layer rendering
4 participants