mirror of
https://github.com/zed-industries/zed.git
synced 2025-01-12 05:15:00 +00:00
Add file_finder module
Still need to wire up key bindings to make it toggle. Co-Authored-By: Antonio Scandurra <me@as-cii.com>
This commit is contained in:
parent
0c59937a65
commit
f5df3681f8
10 changed files with 508 additions and 25 deletions
|
@ -3,8 +3,9 @@ use crate::{
|
|||
executor::{self, ForegroundTask},
|
||||
keymap::{self, Keystroke},
|
||||
platform::{self, App as _, WindowOptions},
|
||||
presenter::Presenter,
|
||||
util::post_inc,
|
||||
AssetCache, AssetSource, FontCache, Presenter,
|
||||
AssetCache, AssetSource, FontCache,
|
||||
};
|
||||
use anyhow::{anyhow, Result};
|
||||
use keymap::MatchResult;
|
||||
|
@ -1440,8 +1441,8 @@ impl<'a, T: Entity> ModelContext<'a, T> {
|
|||
self.app
|
||||
}
|
||||
|
||||
pub fn background_executor(&self) -> Arc<executor::Background> {
|
||||
self.app.background.clone()
|
||||
pub fn background_executor(&self) -> &Arc<executor::Background> {
|
||||
&self.app.background
|
||||
}
|
||||
|
||||
pub fn halt_stream(&mut self) {
|
||||
|
@ -1633,6 +1634,10 @@ impl<'a, T: View> ViewContext<'a, T> {
|
|||
self.app
|
||||
}
|
||||
|
||||
pub fn background_executor(&self) -> &Arc<executor::Background> {
|
||||
&self.app.background
|
||||
}
|
||||
|
||||
pub fn focus<S>(&mut self, handle: S)
|
||||
where
|
||||
S: Into<AnyViewHandle>,
|
||||
|
|
|
@ -10,6 +10,7 @@ mod stack;
|
|||
mod svg;
|
||||
mod uniform_list;
|
||||
|
||||
pub use crate::presenter::ChildView;
|
||||
pub use align::*;
|
||||
pub use constrained_box::*;
|
||||
pub use container::*;
|
||||
|
|
|
@ -30,30 +30,28 @@ impl UniformListState {
|
|||
}
|
||||
}
|
||||
|
||||
pub struct UniformList<F, G>
|
||||
pub struct UniformList<F>
|
||||
where
|
||||
F: Fn(Range<usize>, &AppContext) -> G,
|
||||
G: Iterator<Item = Box<dyn Element>>,
|
||||
F: Fn(Range<usize>, &mut Vec<Box<dyn Element>>, &AppContext),
|
||||
{
|
||||
state: UniformListState,
|
||||
item_count: usize,
|
||||
build_items: F,
|
||||
append_items: F,
|
||||
scroll_max: Option<f32>,
|
||||
items: Vec<Box<dyn Element>>,
|
||||
origin: Option<Vector2F>,
|
||||
size: Option<Vector2F>,
|
||||
}
|
||||
|
||||
impl<F, G> UniformList<F, G>
|
||||
impl<F> UniformList<F>
|
||||
where
|
||||
F: Fn(Range<usize>, &AppContext) -> G,
|
||||
G: Iterator<Item = Box<dyn Element>>,
|
||||
F: Fn(Range<usize>, &mut Vec<Box<dyn Element>>, &AppContext),
|
||||
{
|
||||
pub fn new(state: UniformListState, item_count: usize, build_items: F) -> Self {
|
||||
Self {
|
||||
state,
|
||||
item_count,
|
||||
build_items,
|
||||
append_items: build_items,
|
||||
scroll_max: None,
|
||||
items: Default::default(),
|
||||
origin: None,
|
||||
|
@ -115,10 +113,9 @@ where
|
|||
}
|
||||
}
|
||||
|
||||
impl<F, G> Element for UniformList<F, G>
|
||||
impl<F> Element for UniformList<F>
|
||||
where
|
||||
F: Fn(Range<usize>, &AppContext) -> G,
|
||||
G: Iterator<Item = Box<dyn Element>>,
|
||||
F: Fn(Range<usize>, &mut Vec<Box<dyn Element>>, &AppContext),
|
||||
{
|
||||
fn layout(
|
||||
&mut self,
|
||||
|
@ -135,8 +132,9 @@ where
|
|||
let mut item_constraint =
|
||||
SizeConstraint::new(vec2f(size.x(), 0.0), vec2f(size.x(), f32::INFINITY));
|
||||
|
||||
let first_item = (self.build_items)(0..1, app).next();
|
||||
if let Some(mut first_item) = first_item {
|
||||
self.items.clear();
|
||||
(self.append_items)(0..1, &mut self.items, app);
|
||||
if let Some(first_item) = self.items.first_mut() {
|
||||
let mut item_size = first_item.layout(item_constraint, ctx, app);
|
||||
item_size.set_x(size.x());
|
||||
item_constraint.min = item_size;
|
||||
|
@ -158,7 +156,7 @@ where
|
|||
start + (size.y() / item_size.y()).ceil() as usize + 1,
|
||||
);
|
||||
self.items.clear();
|
||||
self.items.extend((self.build_items)(start..end, app));
|
||||
(self.append_items)(start..end, &mut self.items, app);
|
||||
|
||||
self.scroll_max = Some(item_size.y() * self.item_count as f32 - size.y());
|
||||
|
||||
|
|
|
@ -18,4 +18,7 @@ pub mod platform;
|
|||
pub use pathfinder_color as color;
|
||||
pub use pathfinder_geometry as geometry;
|
||||
pub use platform::Event;
|
||||
pub use presenter::*;
|
||||
pub use presenter::{
|
||||
AfterLayoutContext, Axis, EventContext, LayoutContext, PaintContext, SizeConstraint,
|
||||
Vector2FExt,
|
||||
};
|
||||
|
|
476
zed/src/file_finder.rs
Normal file
476
zed/src/file_finder.rs
Normal file
|
@ -0,0 +1,476 @@
|
|||
use crate::{
|
||||
editor::{buffer_view, BufferView},
|
||||
settings::Settings,
|
||||
util, watch,
|
||||
workspace::{Workspace, WorkspaceView},
|
||||
worktree::{match_paths, PathMatch, Worktree},
|
||||
};
|
||||
use gpui::{
|
||||
color::{ColorF, ColorU},
|
||||
elements::*,
|
||||
fonts::{Properties, Weight},
|
||||
geometry::vector::vec2f,
|
||||
keymap::{self, Binding},
|
||||
App, AppContext, Axis, Border, Entity, ModelHandle, View, ViewContext, ViewHandle,
|
||||
WeakViewHandle,
|
||||
};
|
||||
use std::cmp;
|
||||
|
||||
pub struct FileFinder {
|
||||
handle: WeakViewHandle<Self>,
|
||||
settings: watch::Receiver<Settings>,
|
||||
workspace: ModelHandle<Workspace>,
|
||||
query_buffer: ViewHandle<BufferView>,
|
||||
search_count: usize,
|
||||
latest_search_id: usize,
|
||||
matches: Vec<PathMatch>,
|
||||
selected: usize,
|
||||
list_state: UniformListState,
|
||||
}
|
||||
|
||||
pub fn init(app: &mut App) {
|
||||
app.add_action("file_finder:toggle", FileFinder::toggle);
|
||||
app.add_action("file_finder:confirm", FileFinder::confirm);
|
||||
app.add_action("file_finder:select", FileFinder::select);
|
||||
app.add_action("buffer:move_up", FileFinder::select_prev);
|
||||
app.add_action("buffer:move_down", FileFinder::select_next);
|
||||
app.add_action("uniform_list:scroll", FileFinder::scroll);
|
||||
|
||||
app.add_bindings(vec![
|
||||
Binding::new("cmd-p", "file_finder:toggle", None),
|
||||
Binding::new("escape", "file_finder:toggle", Some("FileFinder")),
|
||||
Binding::new("enter", "file_finder:confirm", Some("FileFinder")),
|
||||
]);
|
||||
}
|
||||
|
||||
pub enum Event {
|
||||
Selected(usize, usize),
|
||||
Dismissed,
|
||||
}
|
||||
|
||||
impl Entity for FileFinder {
|
||||
type Event = Event;
|
||||
}
|
||||
|
||||
impl View for FileFinder {
|
||||
fn ui_name() -> &'static str {
|
||||
"FileFinder"
|
||||
}
|
||||
|
||||
fn render(&self, _: &AppContext) -> Box<dyn Element> {
|
||||
Align::new(
|
||||
ConstrainedBox::new(
|
||||
Container::new(
|
||||
Flex::new(Axis::Vertical)
|
||||
.with_child(ChildView::new(self.query_buffer.id()).boxed())
|
||||
.with_child(Expanded::new(1.0, self.render_matches()).boxed())
|
||||
.boxed(),
|
||||
)
|
||||
.with_margin_top(12.0)
|
||||
.with_uniform_padding(6.0)
|
||||
.with_corner_radius(6.0)
|
||||
.with_background_color(ColorU::new(0xff, 0xf2, 0xf2, 0xff))
|
||||
// .with_background_color(ColorU::new(0xf2, 0xf2, 0xf2, 0xff))
|
||||
.with_shadow(
|
||||
vec2f(0.0, 4.0),
|
||||
12.0,
|
||||
ColorF::new(0.0, 0.0, 0.0, 0.25).to_u8(),
|
||||
)
|
||||
.boxed(),
|
||||
)
|
||||
.with_max_width(600.0)
|
||||
.with_max_height(400.0)
|
||||
.boxed(),
|
||||
)
|
||||
.top_center()
|
||||
.boxed()
|
||||
}
|
||||
|
||||
fn on_focus(&mut self, ctx: &mut ViewContext<Self>) {
|
||||
ctx.focus(&self.query_buffer);
|
||||
}
|
||||
|
||||
fn keymap_context(&self, _: &AppContext) -> keymap::Context {
|
||||
let mut ctx = Self::default_keymap_context();
|
||||
ctx.set.insert("menu".into());
|
||||
ctx
|
||||
}
|
||||
}
|
||||
|
||||
impl FileFinder {
|
||||
fn render_matches(&self) -> Box<dyn Element> {
|
||||
if self.matches.is_empty() {
|
||||
let settings = smol::block_on(self.settings.read());
|
||||
return Container::new(
|
||||
Label::new(
|
||||
"No matches".into(),
|
||||
settings.ui_font_family,
|
||||
settings.ui_font_size,
|
||||
)
|
||||
.boxed(),
|
||||
)
|
||||
.with_margin_top(6.0)
|
||||
.boxed();
|
||||
}
|
||||
|
||||
let handle = self.handle.clone();
|
||||
let list = UniformList::new(
|
||||
self.list_state.clone(),
|
||||
self.matches.len(),
|
||||
move |mut range, items, app| {
|
||||
let finder = handle.upgrade(app).unwrap();
|
||||
let finder = finder.as_ref(app);
|
||||
let start = range.start;
|
||||
range.end = cmp::min(range.end, finder.matches.len());
|
||||
items.extend(finder.matches[range].iter().enumerate().filter_map(
|
||||
move |(i, path_match)| finder.render_match(path_match, start + i, app),
|
||||
));
|
||||
},
|
||||
);
|
||||
|
||||
Container::new(list.boxed())
|
||||
.with_background_color(ColorU::new(0xf7, 0xf7, 0xf7, 0xff))
|
||||
.with_border(Border::all(1.0, ColorU::new(0xdb, 0xdb, 0xdc, 0xff)))
|
||||
.with_margin_top(6.0)
|
||||
.boxed()
|
||||
}
|
||||
|
||||
fn render_match(
|
||||
&self,
|
||||
path_match: &PathMatch,
|
||||
index: usize,
|
||||
app: &AppContext,
|
||||
) -> Option<Box<dyn Element>> {
|
||||
let tree_id = path_match.tree_id;
|
||||
let entry_id = path_match.entry_id;
|
||||
|
||||
self.worktree(tree_id, app).map(|tree| {
|
||||
let path = tree.entry_path(entry_id).unwrap();
|
||||
let file_name = path
|
||||
.file_name()
|
||||
.unwrap_or_default()
|
||||
.to_string_lossy()
|
||||
.to_string();
|
||||
|
||||
let mut path = path.to_string_lossy().to_string();
|
||||
if path_match.skipped_prefix_len > 0 {
|
||||
let mut i = 0;
|
||||
path.retain(|_| util::post_inc(&mut i) >= path_match.skipped_prefix_len)
|
||||
}
|
||||
|
||||
let path_positions = path_match.positions.clone();
|
||||
let file_name_start = path.chars().count() - file_name.chars().count();
|
||||
let mut file_name_positions = Vec::new();
|
||||
file_name_positions.extend(path_positions.iter().filter_map(|pos| {
|
||||
if pos >= &file_name_start {
|
||||
Some(pos - file_name_start)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}));
|
||||
|
||||
let settings = smol::block_on(self.settings.read());
|
||||
let highlight_color = ColorU::new(0x30, 0x4e, 0xe2, 0xff);
|
||||
let bold = *Properties::new().weight(Weight::BOLD);
|
||||
|
||||
let mut container = Container::new(
|
||||
Flex::row()
|
||||
.with_child(
|
||||
Container::new(
|
||||
LineBox::new(
|
||||
settings.ui_font_family,
|
||||
settings.ui_font_size,
|
||||
Svg::new("icons/file-16.svg".into()).boxed(),
|
||||
)
|
||||
.boxed(),
|
||||
)
|
||||
.with_padding_right(6.0)
|
||||
.boxed(),
|
||||
)
|
||||
.with_child(
|
||||
Expanded::new(
|
||||
1.0,
|
||||
Flex::column()
|
||||
.with_child(
|
||||
Label::new(
|
||||
file_name,
|
||||
settings.ui_font_family,
|
||||
settings.ui_font_size,
|
||||
)
|
||||
.with_highlights(highlight_color, bold, file_name_positions)
|
||||
.boxed(),
|
||||
)
|
||||
.with_child(
|
||||
Label::new(
|
||||
path.into(),
|
||||
settings.ui_font_family,
|
||||
settings.ui_font_size,
|
||||
)
|
||||
.with_highlights(highlight_color, bold, path_positions)
|
||||
.boxed(),
|
||||
)
|
||||
.boxed(),
|
||||
)
|
||||
.boxed(),
|
||||
)
|
||||
.boxed(),
|
||||
)
|
||||
.with_uniform_padding(6.0);
|
||||
|
||||
if index == self.selected || index < self.matches.len() - 1 {
|
||||
container =
|
||||
container.with_border(Border::bottom(1.0, ColorU::new(0xdb, 0xdb, 0xdc, 0xff)));
|
||||
}
|
||||
|
||||
if index == self.selected {
|
||||
container = container.with_background_color(ColorU::new(0xdb, 0xdb, 0xdc, 0xff));
|
||||
}
|
||||
|
||||
EventHandler::new(container.boxed())
|
||||
.on_mouse_down(move |ctx, _| {
|
||||
ctx.dispatch_action("file_finder:select", (tree_id, entry_id));
|
||||
true
|
||||
})
|
||||
.boxed()
|
||||
})
|
||||
}
|
||||
|
||||
fn toggle(workspace_view: &mut WorkspaceView, _: &(), ctx: &mut ViewContext<WorkspaceView>) {
|
||||
workspace_view.toggle_modal(ctx, |ctx, workspace_view| {
|
||||
let handle = ctx.add_view(|ctx| {
|
||||
Self::new(
|
||||
workspace_view.settings.clone(),
|
||||
workspace_view.workspace.clone(),
|
||||
ctx,
|
||||
)
|
||||
});
|
||||
ctx.subscribe_to_view(&handle, Self::on_event);
|
||||
handle
|
||||
});
|
||||
}
|
||||
|
||||
fn on_event(
|
||||
workspace_view: &mut WorkspaceView,
|
||||
_: ViewHandle<FileFinder>,
|
||||
event: &Event,
|
||||
ctx: &mut ViewContext<WorkspaceView>,
|
||||
) {
|
||||
match event {
|
||||
Event::Selected(tree_id, entry_id) => {
|
||||
workspace_view.open_entry((*tree_id, *entry_id), ctx);
|
||||
workspace_view.dismiss_modal(ctx);
|
||||
}
|
||||
Event::Dismissed => {
|
||||
workspace_view.dismiss_modal(ctx);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub fn new(
|
||||
settings: watch::Receiver<Settings>,
|
||||
workspace: ModelHandle<Workspace>,
|
||||
ctx: &mut ViewContext<Self>,
|
||||
) -> Self {
|
||||
ctx.observe(&workspace, Self::workspace_updated);
|
||||
|
||||
let query_buffer = ctx.add_view(|ctx| BufferView::single_line(settings.clone(), ctx));
|
||||
ctx.subscribe_to_view(&query_buffer, Self::on_query_buffer_event);
|
||||
|
||||
settings.notify_view_on_change(ctx);
|
||||
|
||||
Self {
|
||||
handle: ctx.handle(),
|
||||
settings,
|
||||
workspace,
|
||||
query_buffer,
|
||||
search_count: 0,
|
||||
latest_search_id: 0,
|
||||
matches: Vec::new(),
|
||||
selected: 0,
|
||||
list_state: UniformListState::new(),
|
||||
}
|
||||
}
|
||||
|
||||
fn workspace_updated(&mut self, _: ModelHandle<Workspace>, ctx: &mut ViewContext<Self>) {
|
||||
self.spawn_search(self.query_buffer.as_ref(ctx).text(ctx.app()), ctx);
|
||||
}
|
||||
|
||||
fn on_query_buffer_event(
|
||||
&mut self,
|
||||
_: ViewHandle<BufferView>,
|
||||
event: &buffer_view::Event,
|
||||
ctx: &mut ViewContext<Self>,
|
||||
) {
|
||||
use buffer_view::Event::*;
|
||||
match event {
|
||||
Edited => {
|
||||
let query = self.query_buffer.as_ref(ctx).text(ctx.app());
|
||||
if query.is_empty() {
|
||||
self.latest_search_id = util::post_inc(&mut self.search_count);
|
||||
self.matches.clear();
|
||||
ctx.notify();
|
||||
} else {
|
||||
self.spawn_search(query, ctx);
|
||||
}
|
||||
}
|
||||
Blurred => ctx.emit(Event::Dismissed),
|
||||
Activate => {}
|
||||
}
|
||||
}
|
||||
|
||||
fn select_prev(&mut self, _: &(), ctx: &mut ViewContext<Self>) {
|
||||
if self.selected > 0 {
|
||||
self.selected -= 1;
|
||||
}
|
||||
self.list_state.scroll_to(self.selected);
|
||||
ctx.notify();
|
||||
}
|
||||
|
||||
fn select_next(&mut self, _: &(), ctx: &mut ViewContext<Self>) {
|
||||
if self.selected + 1 < self.matches.len() {
|
||||
self.selected += 1;
|
||||
}
|
||||
self.list_state.scroll_to(self.selected);
|
||||
ctx.notify();
|
||||
}
|
||||
|
||||
fn scroll(&mut self, _: &f32, ctx: &mut ViewContext<Self>) {
|
||||
ctx.notify();
|
||||
}
|
||||
|
||||
fn confirm(&mut self, _: &(), ctx: &mut ViewContext<Self>) {
|
||||
if let Some(m) = self.matches.get(self.selected) {
|
||||
ctx.emit(Event::Selected(m.tree_id, m.entry_id));
|
||||
}
|
||||
}
|
||||
|
||||
fn select(&mut self, entry: &(usize, usize), ctx: &mut ViewContext<Self>) {
|
||||
let (tree_id, entry_id) = *entry;
|
||||
log::info!("selected item! {} {}", tree_id, entry_id);
|
||||
ctx.emit(Event::Selected(tree_id, entry_id));
|
||||
}
|
||||
|
||||
fn spawn_search(&mut self, query: String, ctx: &mut ViewContext<Self>) {
|
||||
log::info!("spawn search!");
|
||||
|
||||
let worktrees = self.worktrees(ctx.app());
|
||||
let search_id = util::post_inc(&mut self.search_count);
|
||||
let task = ctx.background_executor().spawn(async move {
|
||||
let matches = match_paths(worktrees.as_slice(), &query, false, false, 100);
|
||||
(search_id, matches)
|
||||
});
|
||||
|
||||
ctx.spawn(task, Self::update_matches).detach();
|
||||
}
|
||||
|
||||
fn update_matches(
|
||||
&mut self,
|
||||
(search_id, matches): (usize, Vec<PathMatch>),
|
||||
ctx: &mut ViewContext<Self>,
|
||||
) {
|
||||
if search_id >= self.latest_search_id {
|
||||
self.latest_search_id = search_id;
|
||||
self.matches = matches;
|
||||
self.selected = 0;
|
||||
self.list_state.scroll_to(0);
|
||||
ctx.notify();
|
||||
}
|
||||
}
|
||||
|
||||
fn worktree<'a>(&'a self, tree_id: usize, app: &'a AppContext) -> Option<&'a Worktree> {
|
||||
self.workspace
|
||||
.as_ref(app)
|
||||
.worktrees()
|
||||
.get(&tree_id)
|
||||
.map(|worktree| worktree.as_ref(app))
|
||||
}
|
||||
|
||||
fn worktrees(&self, app: &AppContext) -> Vec<Worktree> {
|
||||
self.workspace
|
||||
.as_ref(app)
|
||||
.worktrees()
|
||||
.iter()
|
||||
.map(|worktree| worktree.as_ref(app).clone())
|
||||
.collect()
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
use crate::{
|
||||
editor, settings,
|
||||
workspace::{Workspace, WorkspaceView},
|
||||
};
|
||||
use anyhow::Result;
|
||||
use gpui::App;
|
||||
use smol::fs;
|
||||
use tempdir::TempDir;
|
||||
|
||||
#[test]
|
||||
fn test_matching_paths() -> Result<()> {
|
||||
App::test((), |mut app| async move {
|
||||
let tmp_dir = TempDir::new("example")?;
|
||||
fs::create_dir(tmp_dir.path().join("a")).await?;
|
||||
fs::write(tmp_dir.path().join("a/banana"), "banana").await?;
|
||||
fs::write(tmp_dir.path().join("a/bandana"), "bandana").await?;
|
||||
super::init(&mut app);
|
||||
editor::init(&mut app);
|
||||
|
||||
let settings = settings::channel(&app.fonts()).unwrap().1;
|
||||
let workspace = app.add_model(|ctx| Workspace::new(vec![tmp_dir.path().into()], ctx));
|
||||
let (window_id, workspace_view) =
|
||||
app.add_window(|ctx| WorkspaceView::new(workspace.clone(), settings, ctx));
|
||||
app.finish_pending_tasks().await; // Open and populate worktree.
|
||||
app.dispatch_action(
|
||||
window_id,
|
||||
vec![workspace_view.id()],
|
||||
"file_finder:toggle".into(),
|
||||
(),
|
||||
);
|
||||
let (finder, query_buffer) = workspace_view.read(&app, |view, ctx| {
|
||||
let finder = view
|
||||
.modal()
|
||||
.cloned()
|
||||
.unwrap()
|
||||
.downcast::<FileFinder>()
|
||||
.unwrap();
|
||||
let query_buffer = finder.as_ref(ctx).query_buffer.clone();
|
||||
(finder, query_buffer)
|
||||
});
|
||||
|
||||
let chain = vec![finder.id(), query_buffer.id()];
|
||||
app.dispatch_action(window_id, chain.clone(), "buffer:insert", "b".to_string());
|
||||
app.dispatch_action(window_id, chain.clone(), "buffer:insert", "n".to_string());
|
||||
app.dispatch_action(window_id, chain.clone(), "buffer:insert", "a".to_string());
|
||||
app.finish_pending_tasks().await; // Complete path search.
|
||||
|
||||
// let view_state = finder.state(&app);
|
||||
// assert!(view_state.matches.len() > 1);
|
||||
// app.dispatch_action(
|
||||
// window_id,
|
||||
// vec![workspace_view.id(), finder.id()],
|
||||
// "menu:select_next",
|
||||
// (),
|
||||
// );
|
||||
// app.dispatch_action(
|
||||
// window_id,
|
||||
// vec![workspace_view.id(), finder.id()],
|
||||
// "file_finder:confirm",
|
||||
// (),
|
||||
// );
|
||||
// app.finish_pending_tasks().await; // Load Buffer and open BufferView.
|
||||
// let active_pane = workspace_view.read(&app, |view, _| view.active_pane().clone());
|
||||
// assert_eq!(
|
||||
// active_pane.state(&app),
|
||||
// pane::State {
|
||||
// tabs: vec![pane::TabState {
|
||||
// title: "bandana".into(),
|
||||
// active: true,
|
||||
// }]
|
||||
// }
|
||||
// );
|
||||
Ok(())
|
||||
})
|
||||
}
|
||||
}
|
|
@ -1,5 +1,6 @@
|
|||
pub mod assets;
|
||||
pub mod editor;
|
||||
pub mod file_finder;
|
||||
mod operation_queue;
|
||||
pub mod settings;
|
||||
mod sum_tree;
|
||||
|
|
|
@ -4,7 +4,7 @@ use log::LevelFilter;
|
|||
use simplelog::SimpleLogger;
|
||||
use std::{fs, path::PathBuf};
|
||||
use zed::{
|
||||
assets, editor, settings,
|
||||
assets, editor, file_finder, settings,
|
||||
workspace::{self, OpenParams},
|
||||
};
|
||||
|
||||
|
@ -20,6 +20,7 @@ fn main() {
|
|||
.on_finish_launching(move || {
|
||||
workspace::init(&mut app);
|
||||
editor::init(&mut app);
|
||||
file_finder::init(&mut app);
|
||||
|
||||
if stdout_is_a_pty() {
|
||||
app.platform().activate(true);
|
||||
|
|
|
@ -1,8 +1,7 @@
|
|||
use super::{ItemViewHandle, SplitDirection};
|
||||
use crate::{settings::Settings, watch};
|
||||
use gpui::{
|
||||
color::ColorU, elements::*, keymap::Binding, App, AppContext, Border, ChildView, Entity, View,
|
||||
ViewContext,
|
||||
color::ColorU, elements::*, keymap::Binding, App, AppContext, Border, Entity, View, ViewContext,
|
||||
};
|
||||
use std::cmp;
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@ use anyhow::{anyhow, Result};
|
|||
use gpui::{
|
||||
color::{rgbu, ColorU},
|
||||
elements::*,
|
||||
Axis, Border, ChildView,
|
||||
Axis, Border,
|
||||
};
|
||||
|
||||
#[derive(Clone, Debug, Eq, PartialEq)]
|
||||
|
|
|
@ -1,9 +1,8 @@
|
|||
use super::{pane, Pane, PaneGroup, SplitDirection, Workspace};
|
||||
use crate::{settings::Settings, watch};
|
||||
use gpui::{color::rgbu, ChildView};
|
||||
use gpui::{
|
||||
elements::*, AnyViewHandle, AppContext, Entity, ModelHandle, MutableAppContext, View,
|
||||
ViewContext, ViewHandle,
|
||||
color::rgbu, elements::*, AnyViewHandle, AppContext, Entity, ModelHandle, MutableAppContext,
|
||||
View, ViewContext, ViewHandle,
|
||||
};
|
||||
use log::{error, info};
|
||||
use std::{collections::HashSet, path::PathBuf};
|
||||
|
|
Loading…
Reference in a new issue