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

New triangle collapse arrow for large collapsible header #2920

Merged
merged 7 commits into from
Aug 11, 2023
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
49 changes: 47 additions & 2 deletions crates/re_ui/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ use std::{ops::RangeInclusive, sync::Arc};
use parking_lot::Mutex;

use crate::list_item::ListItem;
use egui::emath::Rot2;
use egui::{pos2, Align2, Color32, Mesh, NumExt, Rect, Shape, Vec2};

#[derive(Clone)]
Expand Down Expand Up @@ -576,12 +577,16 @@ impl ReUi {
egui::Vec2::splat(icon_width),
);
let icon_response = header_response.clone().with_new_rect(icon_rect);
egui::collapsing_header::paint_default_icon(ui, openness, &icon_response);
Self::paint_collapsing_triangle(ui, openness, &icon_response);

let visuals = ui.style().interact(&header_response);

let optical_vertical_alignment = 0.5; // improves perceived vertical alignment
let text_pos = icon_response.rect.right_center()
+ egui::vec2(space_after_icon, -0.5 * galley.size().y);
+ egui::vec2(
space_after_icon,
-0.5 * galley.size().y + optical_vertical_alignment,
);
ui.painter()
.galley_with_color(text_pos, galley, visuals.text_color());

Expand All @@ -607,6 +612,46 @@ impl ReUi {
});
}

/// Paint a collapsing triangle with rounded corners.
///
/// Alternative to [`egui::collapsing_header::paint_default_icon`].
pub fn paint_collapsing_triangle(ui: &mut egui::Ui, openness: f32, response: &egui::Response) {
let visuals = ui.style().interact(response);

let rect = response.rect;
let extent = response.rect.width().min(response.rect.height());

// Normalised in [0, 1]^2 space.
// Note on how these coords have been computed: https://github.com/rerun-io/rerun/pull/2920
// Discussion on the future of icons: https://github.com/rerun-io/rerun/issues/2960
let mut points = vec![
pos2(0.80387, 0.470537),
pos2(0.816074, 0.5),
pos2(0.80387, 0.529463),
pos2(0.316248, 1.017085),
pos2(0.286141, 1.029362),
pos2(0.257726, 1.017592),
pos2(0.245118, 0.987622),
pos2(0.245118, 0.012378),
pos2(0.257726, -0.017592),
pos2(0.286141, -0.029362),
pos2(0.316248, -0.017085),
pos2(0.80387, 0.470537),
];
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Perhaps link to #2960 here so we remember to update this code once that issue is resolved


use std::f32::consts::TAU;
let rotation = Rot2::from_angle(egui::remap(openness, 0.0..=1.0, 0.0..=TAU / 4.0));
for p in &mut points {
*p = rect.center() + rotation * (*p - pos2(0.5, 0.5)) * extent;
}

ui.painter().add(Shape::convex_polygon(
points,
visuals.fg_stroke.color,
egui::Stroke::NONE,
));
}

/// Workaround for putting a label into a grid at the top left of its row.
#[allow(clippy::unused_self)]
pub fn grid_left_hand_label(&self, ui: &mut egui::Ui, label: &str) -> egui::Response {
Expand Down