-
Notifications
You must be signed in to change notification settings - Fork 1.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add
egui::Sides
for adding UI on left and right sides (#5036)
* Closes #5015
- Loading branch information
Showing
8 changed files
with
165 additions
and
16 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -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); | ||
} | ||
} |
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters