mirror of https://github.com/emilk/egui.git
Emil Ernerfeldt
2 months ago
committed by
GitHub
8 changed files with 165 additions and 16 deletions
@ -0,0 +1,120 @@ |
|||||
|
use emath::Align; |
||||
|
|
||||
|
use crate::{Layout, Ui, UiBuilder}; |
||||
|
|
||||
|
/// Put some widgets on the left and right sides of a ui.
|
||||
|
///
|
||||
|
/// The result will look like this:
|
||||
|
/// ```text
|
||||
|
/// parent Ui
|
||||
|
/// ______________________________________________________
|
||||
|
/// | | | | ^
|
||||
|
/// | -> left widgets -> | gap | <- right widgets <- | | height
|
||||
|
/// |____________________| |_____________________| v
|
||||
|
/// | |
|
||||
|
/// | |
|
||||
|
/// ```
|
||||
|
///
|
||||
|
/// The width of the gap is dynamic, based on the max width of the parent [`Ui`].
|
||||
|
/// When the parent is being auto-sized ([`Ui::is_sizing_pass`]) the gap will be as small as possible.
|
||||
|
///
|
||||
|
/// If the parent is not wide enough to fit all widgets, the parent will be expanded to the right.
|
||||
|
///
|
||||
|
/// The left widgets are first added to the ui, left-to-right.
|
||||
|
/// Then the right widgets are added, right-to-left.
|
||||
|
///
|
||||
|
/// ```
|
||||
|
/// # egui::__run_test_ui(|ui| {
|
||||
|
/// egui::containers::Sides::new().show(ui,
|
||||
|
/// |ui| {
|
||||
|
/// ui.label("Left");
|
||||
|
/// },
|
||||
|
/// |ui| {
|
||||
|
/// ui.label("Right");
|
||||
|
/// }
|
||||
|
/// );
|
||||
|
/// # });
|
||||
|
/// ```
|
||||
|
#[must_use = "You should call sides.show()"] |
||||
|
#[derive(Clone, Copy, Debug, Default)] |
||||
|
pub struct Sides { |
||||
|
height: Option<f32>, |
||||
|
spacing: Option<f32>, |
||||
|
} |
||||
|
|
||||
|
impl Sides { |
||||
|
#[inline] |
||||
|
pub fn new() -> Self { |
||||
|
Default::default() |
||||
|
} |
||||
|
|
||||
|
/// The minimum height of the sides.
|
||||
|
///
|
||||
|
/// The content will be centered vertically within this height.
|
||||
|
/// The default height is [`crate::Spacing::interact_size`]`.y`.
|
||||
|
#[inline] |
||||
|
pub fn height(mut self, height: f32) -> Self { |
||||
|
self.height = Some(height); |
||||
|
self |
||||
|
} |
||||
|
|
||||
|
/// The horizontal spacing between the left and right UIs.
|
||||
|
///
|
||||
|
/// This is the minimum gap.
|
||||
|
/// The default is [`crate::Spacing::item_spacing`]`.x`.
|
||||
|
#[inline] |
||||
|
pub fn spacing(mut self, spacing: f32) -> Self { |
||||
|
self.spacing = Some(spacing); |
||||
|
self |
||||
|
} |
||||
|
|
||||
|
pub fn show( |
||||
|
self, |
||||
|
ui: &mut Ui, |
||||
|
add_left: impl FnOnce(&mut Ui), |
||||
|
add_right: impl FnOnce(&mut Ui), |
||||
|
) { |
||||
|
let Self { height, spacing } = self; |
||||
|
let height = height.unwrap_or_else(|| ui.spacing().interact_size.y); |
||||
|
let spacing = spacing.unwrap_or_else(|| ui.spacing().item_spacing.x); |
||||
|
|
||||
|
let mut top_rect = ui.max_rect(); |
||||
|
top_rect.max.y = top_rect.min.y + height; |
||||
|
|
||||
|
let left_rect = { |
||||
|
let left_max_rect = top_rect; |
||||
|
let mut left_ui = ui.new_child( |
||||
|
UiBuilder::new() |
||||
|
.max_rect(left_max_rect) |
||||
|
.layout(Layout::left_to_right(Align::Center)), |
||||
|
); |
||||
|
add_left(&mut left_ui); |
||||
|
left_ui.min_rect() |
||||
|
}; |
||||
|
|
||||
|
let right_rect = { |
||||
|
let right_max_rect = top_rect.with_min_x(left_rect.max.x); |
||||
|
let mut right_ui = ui.new_child( |
||||
|
UiBuilder::new() |
||||
|
.max_rect(right_max_rect) |
||||
|
.layout(Layout::right_to_left(Align::Center)), |
||||
|
); |
||||
|
add_right(&mut right_ui); |
||||
|
right_ui.min_rect() |
||||
|
}; |
||||
|
|
||||
|
let mut final_rect = left_rect.union(right_rect); |
||||
|
let min_width = left_rect.width() + spacing + right_rect.width(); |
||||
|
|
||||
|
if ui.is_sizing_pass() { |
||||
|
// Make as small as possible:
|
||||
|
final_rect.max.x = left_rect.min.x + min_width; |
||||
|
} else { |
||||
|
// If the rects overlap, make sure we expand the allocated rect so that the parent
|
||||
|
// ui knows we overflowed, and resizes:
|
||||
|
final_rect.max.x = final_rect.max.x.max(left_rect.min.x + min_width); |
||||
|
} |
||||
|
|
||||
|
ui.advance_cursor_after_rect(final_rect); |
||||
|
} |
||||
|
} |
Loading…
Reference in new issue