use egui::{Rect, Ui, UiBuilder, Vec2, Vec2b, pos2, vec2};
#[derive(Clone, Copy, Debug)]
pub struct SplitScroll {
pub scroll_enabled: Vec2b,
pub fixed_size: Vec2,
pub scroll_outer_size: Vec2,
pub scroll_content_size: Vec2,
pub stick_to_bottom: bool,
}
pub trait SplitScrollDelegate {
fn left_top_ui(&mut self, ui: &mut Ui);
fn right_top_ui(&mut self, ui: &mut Ui);
fn left_bottom_ui(&mut self, ui: &mut Ui);
fn right_bottom_ui(&mut self, ui: &mut Ui);
fn finish(&mut self, _ui: &mut Ui) {}
}
impl SplitScroll {
pub fn show(self, ui: &mut Ui, delegate: &mut dyn SplitScrollDelegate) {
let Self {
scroll_enabled,
fixed_size,
scroll_outer_size,
scroll_content_size,
stick_to_bottom,
} = self;
ui.scope(|ui| {
ui.visuals_mut().clip_rect_margin = 0.0;
let mut rect = ui.cursor();
rect.max = rect.min + fixed_size + scroll_outer_size;
ui.shrink_clip_rect(rect);
let rect = rect;
let bottom_right_rect = Rect::from_min_max(rect.min + fixed_size, rect.max);
let scroll_offset = {
let mut scroll_ui = ui.new_child(UiBuilder::new().max_rect(rect));
egui::ScrollArea::new(scroll_enabled)
.auto_shrink(false)
.scroll_bar_rect(bottom_right_rect)
.stick_to_bottom(stick_to_bottom)
.show_viewport(&mut scroll_ui, |ui, scroll_offset| {
ui.set_min_size(fixed_size + scroll_content_size);
let mut shrunk_rect = ui.max_rect();
shrunk_rect.min += fixed_size;
let mut shrunk_ui = ui.new_child(UiBuilder::new().max_rect(shrunk_rect));
shrunk_ui.shrink_clip_rect(bottom_right_rect);
delegate.right_bottom_ui(&mut shrunk_ui);
scroll_offset.min
})
.inner
};
{
let left_top_rect = rect
.with_max_x(rect.left() + fixed_size.x)
.with_max_y(rect.top() + fixed_size.y);
let mut left_top_ui = ui.new_child(UiBuilder::new().max_rect(left_top_rect));
left_top_ui.shrink_clip_rect(left_top_rect);
delegate.left_top_ui(&mut left_top_ui);
}
{
let right_top_outer_rect = rect
.with_min_x(rect.left() + fixed_size.x)
.with_max_y(rect.top() + fixed_size.y);
let right_top_content_rect = Rect::from_min_size(
pos2(right_top_outer_rect.min.x - scroll_offset.x, rect.min.y),
vec2(scroll_content_size.x, fixed_size.y),
);
let mut right_top_ui =
ui.new_child(UiBuilder::new().max_rect(right_top_content_rect));
right_top_ui.shrink_clip_rect(right_top_outer_rect);
delegate.right_top_ui(&mut right_top_ui);
}
{
let left_bottom_outer_rect = rect
.with_max_x(rect.left() + fixed_size.x)
.with_min_y(rect.top() + fixed_size.y);
let left_bottom_content_rect = Rect::from_min_size(
pos2(rect.min.x, left_bottom_outer_rect.min.y - scroll_offset.y),
vec2(fixed_size.x, scroll_content_size.y),
);
let mut left_bottom_ui =
ui.new_child(UiBuilder::new().max_rect(left_bottom_content_rect));
left_bottom_ui.shrink_clip_rect(left_bottom_outer_rect);
delegate.left_bottom_ui(&mut left_bottom_ui);
}
delegate.finish(ui);
ui.advance_cursor_after_rect(rect);
});
}
}