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

InfiniteList widget #160

Open
sumibi-yakitori opened this issue Jan 15, 2020 · 11 comments
Open

InfiniteList widget #160

sumibi-yakitori opened this issue Jan 15, 2020 · 11 comments
Labels
feature New feature or request help wanted Extra attention is needed question Further information is requested rendering widget
Milestone

Comments

@sumibi-yakitori
Copy link
Contributor

sumibi-yakitori commented Jan 15, 2020

I am writing an app that loads more than 400 PNGs into a scroll view.

But, it seems that VRAM is out of memory because too many images are being loaded.

I want to unload images outside the rendering range of the scroll view.
Is there a way to know the display status of children elements in scroll view?

@hecrj hecrj added feature New feature or request help wanted Extra attention is needed question Further information is requested labels Jan 15, 2020
@hecrj
Copy link
Member

hecrj commented Jan 15, 2020

We are currently not performing any culling of rendering primitives. We haven't focused much on performance or memory footprint yet. However, it may be worth it to add a layer boundary check here:

https://github.com/hecrj/iced/blob/e6aa25a1032f583ced7f6f02806991ffb4cde140/wgpu/src/renderer.rs#L246-L252

It should be a matter of checking rectangle intersection. Once images are culled, the renderer cache will automatically deallocate memory of images that fall out of bounds. I will see if I can add that in a bit!

However, the deallocation strategy is very simplistic and I have the feeling this will not be enough for a smooth scrolling experience, as the renderer currently loads images using blocking I/O. We should fix that eventually.

There is also the fact that we are issuing a draw call per image, instead of using a texture atlas to batch the work (there are some initial efforts in #154).

@hecrj hecrj added this to the 0.1.0 milestone Jan 15, 2020
@sumibi-yakitori
Copy link
Contributor Author

sumibi-yakitori commented Jan 15, 2020

It is wonderful!

Thanks for supporting this particular use case in this early project.

@Maldela
Copy link
Contributor

Maldela commented Jan 17, 2020

Maybe that logic could be handled in the scroll view itself. Make the view somehow instantiate only those widgets that are actually being shown. I know Qt does it like that.

@hecrj
Copy link
Member

hecrj commented Jan 17, 2020

@Maldela Yes, there are different levels where we can perform optimizations.

The renderer one I mentioned is widget-agnostic and culls primitives directly. Therefore, it's more general. It should benefit custom widgets too.

Checking boundaries of the children in the Scrollable implementation to avoid generating unnecessary primitives altogether would be the next step. However, we cannot rely only on that. A simple, efficient implementation will only be able to check the direct children of the Scrollable and, given widgets can be nested infinitely, there could be many out-of-bounds primitives after culling (think of a Scrollable with a huge Column as a single child).

As an alternative, we could provide Widget::draw with a clip region. Scrollable would create a new region and feed it to its children. This way, culling could be performed efficiently at any nesting level.

In any case, the renderer optimization should take care of the extreme cases for now.

@hecrj hecrj modified the milestones: 0.1.0, 0.2.0 Apr 2, 2020
@tarkah
Copy link
Member

tarkah commented Apr 18, 2020

I have a Scrollable with >1k Button / Text elements. CPU usage is very high when cursor is moving inside the app, I assume from from having to push each element on every redraw? Worth noting I have subscriptions enabled to capture keyboard events. If I filter down to a reasonable number of elements, everything is snappy and CPU usage is very low.

I'd think I wouldn't want to push 1k records in the first place, only however many elements are "in view". Any chance of exposing helper methods to calculate how many elements are "in view" and what the current offset is? That way I could just push what's needed on each redraw.

Screenshot from 2020-04-18 12-02-20
Screenshot from 2020-04-18 12-01-28

@hecrj
Copy link
Member

hecrj commented Apr 18, 2020

@tarkah For this kind of use case, there are plans to create an alternative to Scrollable with a retained/lazy API (i.e. owning its contents/producing only visible elements). We have an ongoing discussion in the Zulip server about building this InfiniteList widget.

EDIT: That application looks cool! 😄

@tarkah
Copy link
Member

tarkah commented Apr 18, 2020

Thanks! Perfect, I'll follow progress there.

@travistrue2008
Copy link

@hecrj Would that InfiniteList widget be similar to table widgets in libraries like Qt or UIKit for iOS where it puts the responsibility of dequeuing renderable rows onto the implementor? It's a pretty common pattern that I've seen in UI frameworks. Basically, there are a set of closures that one could provide to the table view to keep track of everything. Whenever the set of items is changed, the user calculates the total height of the table, and sets the scrollable content height and width if need-be, then a item_height closure is called once per item which should return the current height for that item at that index. This gives the table view an idea of where each child row's y-position starts and ends while allowing each cell to have different heights if need be. This also allows the table to infer which row bounds are in view at any given time since it'll always know its own scroll offset and dimensions. Then whenever the items change, the view is resized, or scrolled a dequeuing closure is called once per item that the table view expects to be in-bounds. This closure's job is to match which type of cell to render, and assign data to it, so that it's updated. Then the view draws all of those pre-instantiated elements at that starting offset which is the first element that's just off-screen.

From a 3D graphics perspective, you won't ever need to resize your vertex buffers except for when the table view grows. Eventually, it'll never grow anymore because that element will typically be within the window's view except for crazy situations when tables need to be nested inside of tables... That's probably on them to figure out though lol. I'm not familiar with Vulkan/wgpu, but I'm guessing GLSL's used under-the-hood all the same, so you should be able to pass all viewable elements' data as a UBO, and have a vertex attribute to associate each quad to the index of the table cell struct in the UBO.

Sorry for the crazy wordiness. This looks like an incredibly exciting project. The Rust community needs more UI crates like this :)

@hecrj hecrj modified the milestones: 0.2.0, 0.3.0 Nov 26, 2020
@hecrj hecrj modified the milestones: 0.3.0, 0.4.0 Mar 31, 2021
@hecrj hecrj changed the title Element rendering state support InfiniteList widget Jan 18, 2022
@hecrj hecrj modified the milestones: 0.4.0, 0.5.0 Apr 12, 2022
@hecrj hecrj modified the milestones: 0.5.0, 0.6.0 Nov 9, 2022
@hecrj hecrj modified the milestones: 0.6.0, 0.7.0 Dec 7, 2022
@hecrj hecrj modified the milestones: 0.7.0, 0.8.0 Jan 14, 2023
@hecrj hecrj modified the milestones: 0.8.0, 0.9.0 Feb 18, 2023
@hecrj hecrj modified the milestones: 0.9.0, 0.10.0 Apr 13, 2023
@hecrj hecrj modified the milestones: 0.10.0, 0.11.0 Jul 28, 2023
@hecrj hecrj modified the milestones: 0.12, 0.13 Feb 11, 2024
@hecrj hecrj modified the milestones: 0.13, 0.14 Sep 18, 2024
@CelestialCrafter
Copy link

any updates on this?

@nednoodlehead
Copy link

@CelestialCrafter there is a branch (feature/list-widget-reloaded) that has some work. I use it in my current application punge. It is working pretty well. I have no complaints (besides it was left behind for 0.13 :( )

hecrj mentioned that he wanted to fix it / make it better (in the discord), so hopefully we'll see some work on it in the coming months.

@CelestialCrafter
Copy link

ohh nice, my app is gonna be rendering ~100k images in a list so.. kinda need infinite lists! sucks that it's ~500 commits behind though but whatever!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
feature New feature or request help wanted Extra attention is needed question Further information is requested rendering widget
Projects
None yet
Development

Successfully merging a pull request may close this issue.

7 participants