mirror of
https://github.com/zed-industries/zed.git
synced 2024-09-18 18:08:07 +03:00
WIP
This commit is contained in:
parent
dcee8439b6
commit
f403d87eff
2
Cargo.lock
generated
2
Cargo.lock
generated
@ -979,6 +979,7 @@ name = "context_menu"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"gpui",
|
||||
"settings",
|
||||
"theme",
|
||||
]
|
||||
|
||||
@ -3457,6 +3458,7 @@ dependencies = [
|
||||
name = "project_panel"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"context_menu",
|
||||
"editor",
|
||||
"futures",
|
||||
"gpui",
|
||||
|
@ -149,6 +149,7 @@ pub fn init_tracing(config: &Config) -> Option<()> {
|
||||
use tracing_subscriber::layer::SubscriberExt;
|
||||
let rust_log = config.rust_log.clone()?;
|
||||
|
||||
println!("HEY!");
|
||||
LogTracer::init().log_err()?;
|
||||
|
||||
let open_telemetry_layer = config
|
||||
|
@ -9,4 +9,5 @@ doctest = false
|
||||
|
||||
[dependencies]
|
||||
gpui = { path = "../gpui" }
|
||||
settings = { path = "../settings" }
|
||||
theme = { path = "../theme" }
|
||||
|
@ -1,6 +1,10 @@
|
||||
use gpui::{Entity, View};
|
||||
use gpui::{
|
||||
elements::*, geometry::vector::Vector2F, Action, Entity, RenderContext, View, ViewContext,
|
||||
};
|
||||
use settings::Settings;
|
||||
use std::{marker::PhantomData, sync::Arc};
|
||||
|
||||
enum ContextMenuItem {
|
||||
pub enum ContextMenuItem {
|
||||
Item {
|
||||
label: String,
|
||||
action: Box<dyn Action>,
|
||||
@ -8,21 +12,116 @@ enum ContextMenuItem {
|
||||
Separator,
|
||||
}
|
||||
|
||||
pub struct ContextMenu {
|
||||
pub struct ContextMenu<T> {
|
||||
position: Vector2F,
|
||||
items: Vec<ContextMenuItem>,
|
||||
items: Arc<[ContextMenuItem]>,
|
||||
state: UniformListState,
|
||||
selected_index: Option<usize>,
|
||||
widest_item_index: Option<usize>,
|
||||
visible: bool,
|
||||
_phantom: PhantomData<T>,
|
||||
}
|
||||
|
||||
impl Entity for ContextMenu {
|
||||
impl<T: 'static> Entity for ContextMenu<T> {
|
||||
type Event = ();
|
||||
}
|
||||
|
||||
impl View for ContextMenu {
|
||||
impl<T: 'static> View for ContextMenu<T> {
|
||||
fn ui_name() -> &'static str {
|
||||
"ContextMenu"
|
||||
}
|
||||
|
||||
fn render(&mut self, cx: &mut gpui::RenderContext<'_, Self>) -> gpui::ElementBox {
|
||||
Overlay::new().with_abs_position(self.position).boxed()
|
||||
fn render(&mut self, cx: &mut RenderContext<Self>) -> ElementBox {
|
||||
if !self.visible {
|
||||
return Empty::new().boxed();
|
||||
}
|
||||
|
||||
let theme = &cx.global::<Settings>().theme;
|
||||
let menu_style = &theme.project_panel.context_menu;
|
||||
let separator_style = menu_style.separator;
|
||||
let item_style = menu_style.item.clone();
|
||||
let items = self.items.clone();
|
||||
let selected_ix = self.selected_index;
|
||||
Overlay::new(
|
||||
UniformList::new(
|
||||
self.state.clone(),
|
||||
self.items.len(),
|
||||
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()
|
||||
})
|
||||
.on_click(move |_, _, cx| {
|
||||
cx.dispatch_any_action(action.boxed_clone())
|
||||
})
|
||||
.boxed()
|
||||
}
|
||||
ContextMenuItem::Separator => {
|
||||
Empty::new().contained().with_style(separator_style).boxed()
|
||||
}
|
||||
}
|
||||
}))
|
||||
},
|
||||
)
|
||||
.with_width_from_item(self.widest_item_index)
|
||||
.boxed(),
|
||||
)
|
||||
.with_abs_position(self.position)
|
||||
.contained()
|
||||
.with_style(menu_style.container)
|
||||
.boxed()
|
||||
}
|
||||
|
||||
fn on_blur(&mut self, cx: &mut ViewContext<Self>) {
|
||||
self.visible = false;
|
||||
cx.notify();
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: 'static> ContextMenu<T> {
|
||||
pub fn new() -> Self {
|
||||
Self {
|
||||
position: Default::default(),
|
||||
items: Arc::from([]),
|
||||
state: Default::default(),
|
||||
selected_index: Default::default(),
|
||||
widest_item_index: Default::default(),
|
||||
visible: false,
|
||||
_phantom: PhantomData,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn show(
|
||||
&mut self,
|
||||
position: Vector2F,
|
||||
items: impl IntoIterator<Item = ContextMenuItem>,
|
||||
cx: &mut ViewContext<Self>,
|
||||
) {
|
||||
self.items = items.into_iter().collect();
|
||||
self.widest_item_index = self
|
||||
.items
|
||||
.iter()
|
||||
.enumerate()
|
||||
.max_by_key(|(_, item)| match item {
|
||||
ContextMenuItem::Item { label, .. } => label.chars().count(),
|
||||
ContextMenuItem::Separator => 0,
|
||||
})
|
||||
.map(|(ix, _)| ix);
|
||||
self.position = position;
|
||||
self.visible = true;
|
||||
cx.focus_self();
|
||||
cx.notify();
|
||||
}
|
||||
}
|
||||
|
@ -8,6 +8,7 @@ path = "src/project_panel.rs"
|
||||
doctest = false
|
||||
|
||||
[dependencies]
|
||||
context_menu = { path = "../context_menu" }
|
||||
editor = { path = "../editor" }
|
||||
gpui = { path = "../gpui" }
|
||||
project = { path = "../project" }
|
||||
|
@ -1,17 +1,18 @@
|
||||
use context_menu::{ContextMenu, ContextMenuItem};
|
||||
use editor::{Cancel, Editor};
|
||||
use futures::stream::StreamExt;
|
||||
use gpui::{
|
||||
actions,
|
||||
anyhow::{anyhow, Result},
|
||||
elements::{
|
||||
ChildView, ConstrainedBox, Empty, Flex, Label, MouseEventHandler, Overlay, ParentElement,
|
||||
ChildView, ConstrainedBox, Empty, Flex, Label, MouseEventHandler, ParentElement,
|
||||
ScrollTarget, Stack, Svg, UniformList, UniformListState,
|
||||
},
|
||||
geometry::vector::Vector2F,
|
||||
impl_internal_actions, keymap,
|
||||
platform::CursorStyle,
|
||||
AppContext, Element, ElementBox, Entity, ModelHandle, MutableAppContext, PromptLevel,
|
||||
RenderContext, Task, View, ViewContext, ViewHandle, WeakViewHandle,
|
||||
AppContext, Element, ElementBox, Entity, ModelHandle, MutableAppContext, PromptLevel, Task,
|
||||
View, ViewContext, ViewHandle, WeakViewHandle,
|
||||
};
|
||||
use project::{Entry, EntryKind, Project, ProjectEntryId, ProjectPath, Worktree, WorktreeId};
|
||||
use settings::Settings;
|
||||
@ -37,7 +38,7 @@ pub struct ProjectPanel {
|
||||
selection: Option<Selection>,
|
||||
edit_state: Option<EditState>,
|
||||
filename_editor: ViewHandle<Editor>,
|
||||
context_menu: Option<ContextMenu>,
|
||||
context_menu: ViewHandle<ContextMenu<Self>>,
|
||||
handle: WeakViewHandle<Self>,
|
||||
}
|
||||
|
||||
@ -83,11 +84,6 @@ pub struct DeployContextMenu {
|
||||
pub entry_id: Option<ProjectEntryId>,
|
||||
}
|
||||
|
||||
pub struct ContextMenu {
|
||||
pub position: Vector2F,
|
||||
pub entry_id: Option<ProjectEntryId>,
|
||||
}
|
||||
|
||||
actions!(
|
||||
project_panel,
|
||||
[
|
||||
@ -170,7 +166,7 @@ impl ProjectPanel {
|
||||
selection: None,
|
||||
edit_state: None,
|
||||
filename_editor,
|
||||
context_menu: None,
|
||||
context_menu: cx.add_view(|_| ContextMenu::new()),
|
||||
handle: cx.weak_handle(),
|
||||
};
|
||||
this.update_visible_entries(None, cx);
|
||||
@ -211,9 +207,22 @@ impl ProjectPanel {
|
||||
}
|
||||
|
||||
fn deploy_context_menu(&mut self, action: &DeployContextMenu, cx: &mut ViewContext<Self>) {
|
||||
self.context_menu = Some(ContextMenu {
|
||||
position: action.position,
|
||||
entry_id: action.entry_id,
|
||||
self.context_menu.update(cx, |menu, cx| {
|
||||
menu.show(
|
||||
action.position,
|
||||
[
|
||||
ContextMenuItem::Item {
|
||||
label: "New File".to_string(),
|
||||
action: Box::new(AddFile),
|
||||
},
|
||||
ContextMenuItem::Item {
|
||||
label: "New Directory".to_string(),
|
||||
action: Box::new(AddDirectory),
|
||||
},
|
||||
ContextMenuItem::Separator,
|
||||
],
|
||||
cx,
|
||||
);
|
||||
});
|
||||
cx.notify();
|
||||
}
|
||||
@ -883,24 +892,6 @@ impl ProjectPanel {
|
||||
.with_cursor_style(CursorStyle::PointingHand)
|
||||
.boxed()
|
||||
}
|
||||
|
||||
fn render_context_menu(&self, cx: &mut RenderContext<Self>) -> Option<ElementBox> {
|
||||
self.context_menu.as_ref().map(|menu| {
|
||||
let style = &cx.global::<Settings>().theme.project_panel.context_menu;
|
||||
|
||||
Overlay::new(
|
||||
Flex::column()
|
||||
.with_child(
|
||||
Label::new("Add File".to_string(), style.item.label.clone()).boxed(),
|
||||
)
|
||||
.contained()
|
||||
.with_style(style.container)
|
||||
.boxed(),
|
||||
)
|
||||
.with_abs_position(menu.position)
|
||||
.named("Project Panel Context Menu")
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl View for ProjectPanel {
|
||||
@ -943,7 +934,7 @@ impl View for ProjectPanel {
|
||||
.with_style(container_style)
|
||||
.boxed(),
|
||||
)
|
||||
.with_children(self.render_context_menu(cx))
|
||||
.with_child(ChildView::new(&self.context_menu).boxed())
|
||||
.boxed()
|
||||
}
|
||||
|
||||
|
@ -244,7 +244,8 @@ pub struct ProjectPanelEntry {
|
||||
pub struct ContextMenu {
|
||||
#[serde(flatten)]
|
||||
pub container: ContainerStyle,
|
||||
pub item: ContextMenuItem,
|
||||
pub item: Interactive<ContextMenuItem>,
|
||||
pub separator: ContainerStyle,
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug, Deserialize, Default)]
|
||||
|
@ -43,7 +43,9 @@ export default function projectPanel(theme: Theme) {
|
||||
right: 6,
|
||||
top: 2,
|
||||
},
|
||||
label: text(theme, "sans", "secondary", { size: "sm" }),
|
||||
item: {
|
||||
label: text(theme, "sans", "secondary", { size: "sm" }),
|
||||
},
|
||||
shadow: shadow(theme),
|
||||
}
|
||||
};
|
||||
|
Loading…
Reference in New Issue
Block a user