Constrain context menu to the width of the widest item
Co-Authored-By: Nathan Sobo <nathan@zed.dev>
This commit is contained in:
parent
f403d87eff
commit
3b2f1644fb
8 changed files with 200 additions and 93 deletions
|
@ -1,8 +1,8 @@
|
||||||
use gpui::{
|
use gpui::{
|
||||||
elements::*, geometry::vector::Vector2F, Action, Entity, RenderContext, View, ViewContext,
|
elements::*, geometry::vector::Vector2F, Action, Axis, Entity, RenderContext, SizeConstraint,
|
||||||
|
View, ViewContext,
|
||||||
};
|
};
|
||||||
use settings::Settings;
|
use settings::Settings;
|
||||||
use std::{marker::PhantomData, sync::Arc};
|
|
||||||
|
|
||||||
pub enum ContextMenuItem {
|
pub enum ContextMenuItem {
|
||||||
Item {
|
Item {
|
||||||
|
@ -12,75 +12,51 @@ pub enum ContextMenuItem {
|
||||||
Separator,
|
Separator,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct ContextMenu<T> {
|
pub struct ContextMenu {
|
||||||
position: Vector2F,
|
position: Vector2F,
|
||||||
items: Arc<[ContextMenuItem]>,
|
items: Vec<ContextMenuItem>,
|
||||||
state: UniformListState,
|
widest_item_index: usize,
|
||||||
selected_index: Option<usize>,
|
selected_index: Option<usize>,
|
||||||
widest_item_index: Option<usize>,
|
|
||||||
visible: bool,
|
visible: bool,
|
||||||
_phantom: PhantomData<T>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: 'static> Entity for ContextMenu<T> {
|
impl Entity for ContextMenu {
|
||||||
type Event = ();
|
type Event = ();
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: 'static> View for ContextMenu<T> {
|
impl View for ContextMenu {
|
||||||
fn ui_name() -> &'static str {
|
fn ui_name() -> &'static str {
|
||||||
"ContextMenu"
|
"ContextMenu"
|
||||||
}
|
}
|
||||||
|
|
||||||
fn render(&mut self, cx: &mut RenderContext<Self>) -> ElementBox {
|
fn render(&mut self, cx: &mut RenderContext<Self>) -> ElementBox {
|
||||||
|
enum Tag {}
|
||||||
|
|
||||||
if !self.visible {
|
if !self.visible {
|
||||||
return Empty::new().boxed();
|
return Empty::new().boxed();
|
||||||
}
|
}
|
||||||
|
|
||||||
let theme = &cx.global::<Settings>().theme;
|
let style = cx.global::<Settings>().theme.context_menu.clone();
|
||||||
let menu_style = &theme.project_panel.context_menu;
|
|
||||||
let separator_style = menu_style.separator;
|
let mut widest_item = self.render_menu_item::<()>(self.widest_item_index, cx, &style);
|
||||||
let item_style = menu_style.item.clone();
|
|
||||||
let items = self.items.clone();
|
|
||||||
let selected_ix = self.selected_index;
|
|
||||||
Overlay::new(
|
Overlay::new(
|
||||||
UniformList::new(
|
Flex::column()
|
||||||
self.state.clone(),
|
.with_children(
|
||||||
self.items.len(),
|
(0..self.items.len()).map(|ix| self.render_menu_item::<Tag>(ix, cx, &style)),
|
||||||
move |range, elements, cx| {
|
|
||||||
let start = range.start;
|
|
||||||
elements.extend(items[range].iter().enumerate().map(|(ix, item)| {
|
|
||||||
let item_ix = start + ix;
|
|
||||||
match item {
|
|
||||||
ContextMenuItem::Item { label, action } => {
|
|
||||||
let action = action.boxed_clone();
|
|
||||||
MouseEventHandler::new::<T, _, _>(item_ix, cx, |state, _| {
|
|
||||||
let style =
|
|
||||||
item_style.style_for(state, Some(item_ix) == selected_ix);
|
|
||||||
Flex::row()
|
|
||||||
.with_child(
|
|
||||||
Label::new(label.to_string(), style.label.clone())
|
|
||||||
.boxed(),
|
|
||||||
)
|
)
|
||||||
.boxed()
|
.constrained()
|
||||||
})
|
.dynamically(move |constraint, cx| {
|
||||||
.on_click(move |_, _, cx| {
|
SizeConstraint::strict_along(
|
||||||
cx.dispatch_any_action(action.boxed_clone())
|
Axis::Horizontal,
|
||||||
})
|
widest_item.layout(constraint, cx).x(),
|
||||||
.boxed()
|
|
||||||
}
|
|
||||||
ContextMenuItem::Separator => {
|
|
||||||
Empty::new().contained().with_style(separator_style).boxed()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
},
|
|
||||||
)
|
)
|
||||||
.with_width_from_item(self.widest_item_index)
|
})
|
||||||
|
.contained()
|
||||||
|
.with_style(style.container)
|
||||||
.boxed(),
|
.boxed(),
|
||||||
)
|
)
|
||||||
.with_abs_position(self.position)
|
.with_abs_position(self.position)
|
||||||
.contained()
|
|
||||||
.with_style(menu_style.container)
|
|
||||||
.boxed()
|
.boxed()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -90,16 +66,14 @@ impl<T: 'static> View for ContextMenu<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: 'static> ContextMenu<T> {
|
impl ContextMenu {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
Self {
|
Self {
|
||||||
position: Default::default(),
|
position: Default::default(),
|
||||||
items: Arc::from([]),
|
items: Default::default(),
|
||||||
state: Default::default(),
|
|
||||||
selected_index: Default::default(),
|
selected_index: Default::default(),
|
||||||
widest_item_index: Default::default(),
|
widest_item_index: Default::default(),
|
||||||
visible: false,
|
visible: false,
|
||||||
_phantom: PhantomData,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -109,7 +83,9 @@ impl<T: 'static> ContextMenu<T> {
|
||||||
items: impl IntoIterator<Item = ContextMenuItem>,
|
items: impl IntoIterator<Item = ContextMenuItem>,
|
||||||
cx: &mut ViewContext<Self>,
|
cx: &mut ViewContext<Self>,
|
||||||
) {
|
) {
|
||||||
self.items = items.into_iter().collect();
|
let mut items = items.into_iter().peekable();
|
||||||
|
assert!(items.peek().is_some(), "must have at least one item");
|
||||||
|
self.items = items.collect();
|
||||||
self.widest_item_index = self
|
self.widest_item_index = self
|
||||||
.items
|
.items
|
||||||
.iter()
|
.iter()
|
||||||
|
@ -118,10 +94,39 @@ impl<T: 'static> ContextMenu<T> {
|
||||||
ContextMenuItem::Item { label, .. } => label.chars().count(),
|
ContextMenuItem::Item { label, .. } => label.chars().count(),
|
||||||
ContextMenuItem::Separator => 0,
|
ContextMenuItem::Separator => 0,
|
||||||
})
|
})
|
||||||
.map(|(ix, _)| ix);
|
.unwrap()
|
||||||
|
.0;
|
||||||
self.position = position;
|
self.position = position;
|
||||||
self.visible = true;
|
self.visible = true;
|
||||||
cx.focus_self();
|
cx.focus_self();
|
||||||
cx.notify();
|
cx.notify();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn render_menu_item<T: 'static>(
|
||||||
|
&self,
|
||||||
|
ix: usize,
|
||||||
|
cx: &mut RenderContext<ContextMenu>,
|
||||||
|
style: &theme::ContextMenu,
|
||||||
|
) -> ElementBox {
|
||||||
|
match &self.items[ix] {
|
||||||
|
ContextMenuItem::Item { label, action } => {
|
||||||
|
let action = action.boxed_clone();
|
||||||
|
MouseEventHandler::new::<T, _, _>(ix, cx, |state, _| {
|
||||||
|
let style = style.item.style_for(state, Some(ix) == self.selected_index);
|
||||||
|
Flex::row()
|
||||||
|
.with_child(Label::new(label.to_string(), style.label.clone()).boxed())
|
||||||
|
.boxed()
|
||||||
|
})
|
||||||
|
.on_click(move |_, _, cx| cx.dispatch_any_action(action.boxed_clone()))
|
||||||
|
.boxed()
|
||||||
|
}
|
||||||
|
ContextMenuItem::Separator => Empty::new()
|
||||||
|
.contained()
|
||||||
|
.with_style(style.separator)
|
||||||
|
.constrained()
|
||||||
|
.with_height(1.)
|
||||||
|
.flex(1., false)
|
||||||
|
.boxed(),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,46 +9,121 @@ use crate::{
|
||||||
|
|
||||||
pub struct ConstrainedBox {
|
pub struct ConstrainedBox {
|
||||||
child: ElementBox,
|
child: ElementBox,
|
||||||
constraint: SizeConstraint,
|
constraint: Constraint,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum Constraint {
|
||||||
|
Static(SizeConstraint),
|
||||||
|
Dynamic(Box<dyn FnMut(SizeConstraint, &mut LayoutContext) -> SizeConstraint>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToJson for Constraint {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
match self {
|
||||||
|
Constraint::Static(constraint) => constraint.to_json(),
|
||||||
|
Constraint::Dynamic(_) => "dynamic".into(),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ConstrainedBox {
|
impl ConstrainedBox {
|
||||||
pub fn new(child: ElementBox) -> Self {
|
pub fn new(child: ElementBox) -> Self {
|
||||||
Self {
|
Self {
|
||||||
child,
|
child,
|
||||||
constraint: SizeConstraint {
|
constraint: Constraint::Static(Default::default()),
|
||||||
min: Vector2F::zero(),
|
|
||||||
max: Vector2F::splat(f32::INFINITY),
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn dynamically(
|
||||||
|
mut self,
|
||||||
|
constraint: impl 'static + FnMut(SizeConstraint, &mut LayoutContext) -> SizeConstraint,
|
||||||
|
) -> Self {
|
||||||
|
self.constraint = Constraint::Dynamic(Box::new(constraint));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
pub fn with_min_width(mut self, min_width: f32) -> Self {
|
pub fn with_min_width(mut self, min_width: f32) -> Self {
|
||||||
self.constraint.min.set_x(min_width);
|
if let Constraint::Dynamic(_) = self.constraint {
|
||||||
|
self.constraint = Constraint::Static(Default::default());
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Constraint::Static(constraint) = &mut self.constraint {
|
||||||
|
constraint.min.set_x(min_width);
|
||||||
|
} else {
|
||||||
|
unreachable!()
|
||||||
|
}
|
||||||
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_max_width(mut self, max_width: f32) -> Self {
|
pub fn with_max_width(mut self, max_width: f32) -> Self {
|
||||||
self.constraint.max.set_x(max_width);
|
if let Constraint::Dynamic(_) = self.constraint {
|
||||||
|
self.constraint = Constraint::Static(Default::default());
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Constraint::Static(constraint) = &mut self.constraint {
|
||||||
|
constraint.max.set_x(max_width);
|
||||||
|
} else {
|
||||||
|
unreachable!()
|
||||||
|
}
|
||||||
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_max_height(mut self, max_height: f32) -> Self {
|
pub fn with_max_height(mut self, max_height: f32) -> Self {
|
||||||
self.constraint.max.set_y(max_height);
|
if let Constraint::Dynamic(_) = self.constraint {
|
||||||
|
self.constraint = Constraint::Static(Default::default());
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Constraint::Static(constraint) = &mut self.constraint {
|
||||||
|
constraint.max.set_y(max_height);
|
||||||
|
} else {
|
||||||
|
unreachable!()
|
||||||
|
}
|
||||||
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_width(mut self, width: f32) -> Self {
|
pub fn with_width(mut self, width: f32) -> Self {
|
||||||
self.constraint.min.set_x(width);
|
if let Constraint::Dynamic(_) = self.constraint {
|
||||||
self.constraint.max.set_x(width);
|
self.constraint = Constraint::Static(Default::default());
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Constraint::Static(constraint) = &mut self.constraint {
|
||||||
|
constraint.min.set_x(width);
|
||||||
|
constraint.max.set_x(width);
|
||||||
|
} else {
|
||||||
|
unreachable!()
|
||||||
|
}
|
||||||
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_height(mut self, height: f32) -> Self {
|
pub fn with_height(mut self, height: f32) -> Self {
|
||||||
self.constraint.min.set_y(height);
|
if let Constraint::Dynamic(_) = self.constraint {
|
||||||
self.constraint.max.set_y(height);
|
self.constraint = Constraint::Static(Default::default());
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Constraint::Static(constraint) = &mut self.constraint {
|
||||||
|
constraint.min.set_y(height);
|
||||||
|
constraint.max.set_y(height);
|
||||||
|
} else {
|
||||||
|
unreachable!()
|
||||||
|
}
|
||||||
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn constraint(
|
||||||
|
&mut self,
|
||||||
|
input_constraint: SizeConstraint,
|
||||||
|
cx: &mut LayoutContext,
|
||||||
|
) -> SizeConstraint {
|
||||||
|
match &mut self.constraint {
|
||||||
|
Constraint::Static(constraint) => *constraint,
|
||||||
|
Constraint::Dynamic(compute_constraint) => compute_constraint(input_constraint, cx),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Element for ConstrainedBox {
|
impl Element for ConstrainedBox {
|
||||||
|
@ -57,13 +132,14 @@ impl Element for ConstrainedBox {
|
||||||
|
|
||||||
fn layout(
|
fn layout(
|
||||||
&mut self,
|
&mut self,
|
||||||
mut constraint: SizeConstraint,
|
mut parent_constraint: SizeConstraint,
|
||||||
cx: &mut LayoutContext,
|
cx: &mut LayoutContext,
|
||||||
) -> (Vector2F, Self::LayoutState) {
|
) -> (Vector2F, Self::LayoutState) {
|
||||||
constraint.min = constraint.min.max(self.constraint.min);
|
let constraint = self.constraint(parent_constraint, cx);
|
||||||
constraint.max = constraint.max.min(self.constraint.max);
|
parent_constraint.min = parent_constraint.min.max(constraint.min);
|
||||||
constraint.max = constraint.max.max(constraint.min);
|
parent_constraint.max = parent_constraint.max.min(constraint.max);
|
||||||
let size = self.child.layout(constraint, cx);
|
parent_constraint.max = parent_constraint.max.max(parent_constraint.min);
|
||||||
|
let size = self.child.layout(parent_constraint, cx);
|
||||||
(size, ())
|
(size, ())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,6 +172,6 @@ impl Element for ConstrainedBox {
|
||||||
_: &Self::PaintState,
|
_: &Self::PaintState,
|
||||||
cx: &DebugContext,
|
cx: &DebugContext,
|
||||||
) -> json::Value {
|
) -> json::Value {
|
||||||
json!({"type": "ConstrainedBox", "set_constraint": self.constraint.to_json(), "child": self.child.debug(cx)})
|
json!({"type": "ConstrainedBox", "assigned_constraint": self.constraint.to_json(), "child": self.child.debug(cx)})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -524,6 +524,15 @@ impl SizeConstraint {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Default for SizeConstraint {
|
||||||
|
fn default() -> Self {
|
||||||
|
SizeConstraint {
|
||||||
|
min: Vector2F::zero(),
|
||||||
|
max: Vector2F::splat(f32::INFINITY),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl ToJson for SizeConstraint {
|
impl ToJson for SizeConstraint {
|
||||||
fn to_json(&self) -> serde_json::Value {
|
fn to_json(&self) -> serde_json::Value {
|
||||||
json!({
|
json!({
|
||||||
|
|
|
@ -38,7 +38,7 @@ pub struct ProjectPanel {
|
||||||
selection: Option<Selection>,
|
selection: Option<Selection>,
|
||||||
edit_state: Option<EditState>,
|
edit_state: Option<EditState>,
|
||||||
filename_editor: ViewHandle<Editor>,
|
filename_editor: ViewHandle<Editor>,
|
||||||
context_menu: ViewHandle<ContextMenu<Self>>,
|
context_menu: ViewHandle<ContextMenu>,
|
||||||
handle: WeakViewHandle<Self>,
|
handle: WeakViewHandle<Self>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -220,6 +220,14 @@ impl ProjectPanel {
|
||||||
action: Box::new(AddDirectory),
|
action: Box::new(AddDirectory),
|
||||||
},
|
},
|
||||||
ContextMenuItem::Separator,
|
ContextMenuItem::Separator,
|
||||||
|
ContextMenuItem::Item {
|
||||||
|
label: "Rename".to_string(),
|
||||||
|
action: Box::new(Rename),
|
||||||
|
},
|
||||||
|
ContextMenuItem::Item {
|
||||||
|
label: "Delete".to_string(),
|
||||||
|
action: Box::new(Delete),
|
||||||
|
},
|
||||||
],
|
],
|
||||||
cx,
|
cx,
|
||||||
);
|
);
|
||||||
|
|
|
@ -19,6 +19,7 @@ pub struct Theme {
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub name: String,
|
pub name: String,
|
||||||
pub workspace: Workspace,
|
pub workspace: Workspace,
|
||||||
|
pub context_menu: ContextMenu,
|
||||||
pub chat_panel: ChatPanel,
|
pub chat_panel: ChatPanel,
|
||||||
pub contacts_panel: ContactsPanel,
|
pub contacts_panel: ContactsPanel,
|
||||||
pub contact_finder: ContactFinder,
|
pub contact_finder: ContactFinder,
|
||||||
|
@ -226,7 +227,6 @@ pub struct ProjectPanel {
|
||||||
pub ignored_entry_fade: f32,
|
pub ignored_entry_fade: f32,
|
||||||
pub filename_editor: FieldEditor,
|
pub filename_editor: FieldEditor,
|
||||||
pub indent_width: f32,
|
pub indent_width: f32,
|
||||||
pub context_menu: ContextMenu,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, Deserialize, Default)]
|
#[derive(Clone, Debug, Deserialize, Default)]
|
||||||
|
|
|
@ -9,6 +9,7 @@ import projectPanel from "./projectPanel";
|
||||||
import search from "./search";
|
import search from "./search";
|
||||||
import picker from "./picker";
|
import picker from "./picker";
|
||||||
import workspace from "./workspace";
|
import workspace from "./workspace";
|
||||||
|
import contextMenu from "./contextMenu";
|
||||||
import projectDiagnostics from "./projectDiagnostics";
|
import projectDiagnostics from "./projectDiagnostics";
|
||||||
import contactNotification from "./contactNotification";
|
import contactNotification from "./contactNotification";
|
||||||
|
|
||||||
|
@ -20,6 +21,7 @@ export default function app(theme: Theme): Object {
|
||||||
return {
|
return {
|
||||||
picker: picker(theme),
|
picker: picker(theme),
|
||||||
workspace: workspace(theme),
|
workspace: workspace(theme),
|
||||||
|
contextMenu: contextMenu(theme),
|
||||||
editor: editor(theme),
|
editor: editor(theme),
|
||||||
projectDiagnostics: projectDiagnostics(theme),
|
projectDiagnostics: projectDiagnostics(theme),
|
||||||
commandPalette: commandPalette(theme),
|
commandPalette: commandPalette(theme),
|
||||||
|
|
23
styles/src/styleTree/contextMenu.ts
Normal file
23
styles/src/styleTree/contextMenu.ts
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
import Theme from "../themes/common/theme";
|
||||||
|
import { shadow, text } from "./components";
|
||||||
|
|
||||||
|
export default function contextMenu(theme: Theme) {
|
||||||
|
return {
|
||||||
|
background: "#ff0000",
|
||||||
|
// background: backgroundColor(theme, 300, "base"),
|
||||||
|
cornerRadius: 6,
|
||||||
|
padding: {
|
||||||
|
bottom: 2,
|
||||||
|
left: 6,
|
||||||
|
right: 6,
|
||||||
|
top: 2,
|
||||||
|
},
|
||||||
|
shadow: shadow(theme),
|
||||||
|
item: {
|
||||||
|
label: text(theme, "sans", "secondary", { size: "sm" }),
|
||||||
|
},
|
||||||
|
separator: {
|
||||||
|
background: "#00ff00"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -32,21 +32,5 @@ export default function projectPanel(theme: Theme) {
|
||||||
text: text(theme, "mono", "primary", { size: "sm" }),
|
text: text(theme, "mono", "primary", { size: "sm" }),
|
||||||
selection: player(theme, 1).selection,
|
selection: player(theme, 1).selection,
|
||||||
},
|
},
|
||||||
contextMenu: {
|
|
||||||
width: 100,
|
|
||||||
// background: "#ff0000",
|
|
||||||
background: backgroundColor(theme, 300, "base"),
|
|
||||||
cornerRadius: 6,
|
|
||||||
padding: {
|
|
||||||
bottom: 2,
|
|
||||||
left: 6,
|
|
||||||
right: 6,
|
|
||||||
top: 2,
|
|
||||||
},
|
|
||||||
item: {
|
|
||||||
label: text(theme, "sans", "secondary", { size: "sm" }),
|
|
||||||
},
|
|
||||||
shadow: shadow(theme),
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue