use std::ops::{Deref, Range};
use crate::prelude::*;
#[derive(Lens)]
pub struct VirtualList {
scroll_to_cursor: bool,
on_change: Option<Box<dyn Fn(&mut EventContext, Range<usize>)>>,
}
pub enum VirtualListEvent {
SetScrollY(f32),
}
#[derive(Lens)]
struct VirtualListData {
num_items: usize,
item_height: f32,
visible_range: Range<usize>,
scroll_y: f32,
}
impl VirtualListData {
fn evaluate_index(index: usize, start: usize, end: usize) -> usize {
match end - start {
0 => 0,
len => start + (len - (start % len) + index) % len,
}
}
fn visible_item_index(index: usize) -> impl Lens<Target = usize> {
Self::visible_range.map(move |range| Self::evaluate_index(index, range.start, range.end))
}
fn recalc(&mut self, cx: &mut EventContext) {
if self.num_items == 0 {
self.visible_range = 0..0;
return;
}
let current = cx.current();
let current_height = cx.cache.get_height(current);
if current_height == f32::MAX {
return;
}
let item_height = self.item_height;
let total_height = item_height * (self.num_items as f32);
let visible_height = current_height / cx.scale_factor();
let mut num_visible_items = (visible_height / item_height).ceil();
num_visible_items += 1.0; let visible_items_height = item_height * num_visible_items;
let empty_height = (total_height - visible_items_height).max(0.0);
let visible_start = empty_height * self.scroll_y;
let visible_end = visible_start + visible_items_height;
let start_index = (visible_start / item_height).trunc() as usize;
let end_index = 1 + (visible_end / item_height).trunc() as usize;
self.visible_range = start_index..end_index.min(self.num_items);
}
}
impl Model for VirtualListData {
fn event(&mut self, cx: &mut EventContext, event: &mut Event) {
event.map(|virtual_list_event, _| match virtual_list_event {
VirtualListEvent::SetScrollY(scroll_y) => {
self.scroll_y = *scroll_y;
self.recalc(cx);
}
});
event.map(|window_event, _| match window_event {
WindowEvent::GeometryChanged(geo) => {
if geo.intersects(GeoChanged::WIDTH_CHANGED | GeoChanged::HEIGHT_CHANGED) {
self.recalc(cx);
}
}
_ => {}
});
}
}
impl VirtualList {
pub fn new<V: View, L: Lens, T: 'static>(
cx: &mut Context,
list: L,
item_height: f32,
item_content: impl 'static + Copy + Fn(&mut Context, usize, MapRef<L, T>) -> Handle<V>,
) -> Handle<Self>
where
L::Target: Deref<Target = [T]>,
{
Self::new_generic(
cx,
list,
|list| list.len(),
|list, index| &list[index],
item_height,
item_content,
)
}
pub fn new_generic<V: View, L: Lens, T: 'static>(
cx: &mut Context,
list: L,
list_len: impl 'static + Fn(&L::Target) -> usize,
list_index: impl 'static + Copy + Fn(&L::Target, usize) -> &T,
item_height: f32,
item_content: impl 'static + Copy + Fn(&mut Context, usize, MapRef<L, T>) -> Handle<V>,
) -> Handle<Self> {
let vl = cx.current;
let num_items = list.map(list_len);
Self { scroll_to_cursor: true, on_change: None }.build(cx, |cx| {
Binding::new(cx, num_items, move |cx, lens| {
let num_items = lens.get(cx);
let mut data =
VirtualListData { num_items, item_height, visible_range: 0..0, scroll_y: 0.0 };
data.recalc(&mut EventContext::new_with_current(cx, vl));
data.build(cx);
});
ScrollView::new(cx, 0.0, 0.0, false, true, move |cx| {
Binding::new(cx, num_items, move |cx, lens| {
let num_items = lens.get(cx);
cx.emit(ScrollEvent::SetY(0.0));
VStack::new(cx, |cx| {
let num_visible_items = VirtualListData::visible_range.map(Range::len);
Binding::new(cx, num_visible_items, move |cx, lens| {
for i in 0..lens.get(cx).min(num_items) {
let item_index = VirtualListData::visible_item_index(i);
Binding::new(cx, item_index, move |cx, lens| {
let index = lens.get(cx);
HStack::new(cx, move |cx| {
let item =
list.map_ref(move |list| list_index(list, index));
item_content(cx, index, item).height(Percentage(100.0));
})
.height(Pixels(item_height))
.position_type(PositionType::SelfDirected)
.bind(
item_index,
move |handle, lens| {
let index = lens.get(&handle);
handle.top(Pixels(index as f32 * item_height));
},
);
});
}
})
})
.height(Pixels(num_items as f32 * item_height));
})
})
.scroll_to_cursor(true)
.on_scroll(|cx, _, y| {
if y.is_finite() {
cx.emit(VirtualListEvent::SetScrollY(y));
}
});
})
}
}
impl View for VirtualList {
fn element(&self) -> Option<&'static str> {
Some("virtual-list")
}
}
impl Handle<'_, VirtualList> {
pub fn scroll_to_cursor(self, flag: bool) -> Self {
self.modify(|virtual_list: &mut VirtualList| {
virtual_list.scroll_to_cursor = flag;
})
}
}
#[cfg(test)]
mod tests {
use super::*;
fn evaluate_indices(range: Range<usize>) -> Vec<usize> {
(0..range.len())
.map(|index| VirtualListData::evaluate_index(index, range.start, range.end))
.collect()
}
#[test]
fn test_evaluate_index() {
assert_eq!(evaluate_indices(0..4), [0, 1, 2, 3]);
assert_eq!(evaluate_indices(1..5), [4, 1, 2, 3]);
assert_eq!(evaluate_indices(2..6), [4, 5, 2, 3]);
assert_eq!(evaluate_indices(3..7), [4, 5, 6, 3]);
assert_eq!(evaluate_indices(4..8), [4, 5, 6, 7]);
assert_eq!(evaluate_indices(5..9), [8, 5, 6, 7]);
assert_eq!(evaluate_indices(6..10), [8, 9, 6, 7]);
assert_eq!(evaluate_indices(7..11), [8, 9, 10, 7]);
assert_eq!(evaluate_indices(8..12), [8, 9, 10, 11]);
assert_eq!(evaluate_indices(9..13), [12, 9, 10, 11]);
}
}