feat: pr file tree item switching

This commit is contained in:
2026-05-28 22:28:59 +01:00
parent 75c3b73665
commit 101b144f53
6 changed files with 201 additions and 109 deletions

View File

@@ -28,6 +28,7 @@ pub(crate) struct FileTree {
pub(crate) struct Item { pub(crate) struct Item {
item: util::file::FileTreeItem, item: util::file::FileTreeItem,
is_expanded: bool, is_expanded: bool,
is_highlighed: bool,
on_click: Box<dyn Fn(&mut gpui::Window, &mut gpui::App) + 'static>, on_click: Box<dyn Fn(&mut gpui::Window, &mut gpui::App) + 'static>,
} }
@@ -38,6 +39,7 @@ struct FileTreeStateInner {
list_state: gpui::ListState, list_state: gpui::ListState,
collapsed_dirs: HashSet<Arc<str>>, collapsed_dirs: HashSet<Arc<str>>,
visible_items: Vec<usize>, visible_items: Vec<usize>,
highlighted_items: HashSet<usize>,
} }
pub(crate) fn file_tree( pub(crate) fn file_tree(
@@ -57,6 +59,7 @@ impl FileTreeState {
list_state: gpui::ListState::new(0, gpui::ListAlignment::Top, px(50.)), list_state: gpui::ListState::new(0, gpui::ListAlignment::Top, px(50.)),
collapsed_dirs: HashSet::new(), collapsed_dirs: HashSet::new(),
visible_items: Vec::new(), visible_items: Vec::new(),
highlighted_items: HashSet::new(),
}))) })))
} }
@@ -66,6 +69,12 @@ impl FileTreeState {
state.visible_items = (0..items.len()).collect::<Vec<_>>(); state.visible_items = (0..items.len()).collect::<Vec<_>>();
} }
pub(crate) fn highlight_item(&mut self, index: usize) {
let mut state = self.0.borrow_mut();
state.highlighted_items.clear();
state.highlighted_items.insert(index);
}
pub(crate) fn toggle_directory( pub(crate) fn toggle_directory(
&mut self, &mut self,
dir_path: &Arc<str>, dir_path: &Arc<str>,
@@ -125,6 +134,7 @@ impl gpui::RenderOnce for FileTree {
let item = (self.item)(item_index, window, cx); let item = (self.item)(item_index, window, cx);
let item = Item { let item = Item {
is_expanded: !state.collapsed_dirs.contains(&item.full_path), is_expanded: !state.collapsed_dirs.contains(&item.full_path),
is_highlighed: state.highlighted_items.contains(&item_index),
item, item,
on_click: Box::new(move |window, cx| { on_click: Box::new(move |window, cx| {
if let Some(f) = &on_item_click { if let Some(f) = &on_item_click {
@@ -153,7 +163,13 @@ impl gpui::RenderOnce for Item {
.py_0p5() .py_0p5()
.pl(px(8. + (8 * self.item.level) as f32)) .pl(px(8. + (8 * self.item.level) as f32))
.rounded_sm() .rounded_sm()
.hover(|it| it.bg(theme.colors.surface_hover)) .hover(|it| {
if self.is_highlighed {
it
} else {
it.bg(theme.colors.surface_hover)
}
})
.on_click(move |_, window, cx| (self.on_click)(window, cx)) .on_click(move |_, window, cx| (self.on_click)(window, cx))
.when( .when(
matches!(self.item.kind, FileTreeItemKind::Directory), matches!(self.item.kind, FileTreeItemKind::Directory),
@@ -171,6 +187,10 @@ impl gpui::RenderOnce for Item {
.when(matches!(self.item.kind, FileTreeItemKind::File), |it| { .when(matches!(self.item.kind, FileTreeItemKind::File), |it| {
it.child(font_icon(FontIcon::FileBracesCorner).size_3()) it.child(font_icon(FontIcon::FileBracesCorner).size_3())
}) })
.when(self.is_highlighed, |it| {
it.bg(theme.colors.accent_muted)
.text_color(theme.colors.accent_fg)
})
.child(text(gpui::SharedString::new(self.item.name)).text_sm()) .child(text(gpui::SharedString::new(self.item.name)).text_sm())
} }
} }

View File

@@ -1,4 +1,5 @@
mod issue_list; mod issue_list;
mod pull_request_change_view;
mod pull_request_diff_view; mod pull_request_diff_view;
mod pull_request_file_tree; mod pull_request_file_tree;
mod pull_request_view; mod pull_request_view;

View File

@@ -0,0 +1,119 @@
use std::sync::Arc;
use crate::{
api, app,
component::file_tree::{FileTreeState, file_tree},
query::{self, QueryStatus, read_query, use_query, watch_query},
screen::dashboard::pull_request_diff_view::PullRequestDiffView,
util::{
self,
file::{FileTreeItem, FileTreeItemKind},
},
};
use gpui::{AppContext, ParentElement, Styled, div};
pub(crate) struct PullRequestChangeView {
selected_file_path: Option<Arc<str>>,
diff_view: gpui::Entity<PullRequestDiffView>,
file_tree_query: query::Entity<api::issues::FetchPullRequestFileTree>,
file_tree_state: FileTreeState,
file_tree_items: Vec<FileTreeItem>,
}
pub(crate) fn new(
pr_id: api::issues::Id,
cx: &mut gpui::Context<PullRequestChangeView>,
) -> PullRequestChangeView {
let mut view = PullRequestChangeView {
selected_file_path: None,
diff_view: cx.new(|cx| PullRequestDiffView::new(pr_id.clone(), cx)),
file_tree_query: use_query(
api::issues::FetchPullRequestFileTree {
id: pr_id,
first: 100,
},
cx,
),
file_tree_state: FileTreeState::new(),
file_tree_items: Vec::new(),
};
view.on_create(cx);
view
}
impl PullRequestChangeView {
fn on_create(&mut self, cx: &mut gpui::Context<Self>) {
watch_query(
&self.file_tree_query,
|this, query, cx| {
if let QueryStatus::Loaded(changed_files) = read_query(query, cx) {
let tree = util::file::build_file_tree(changed_files, |f| &f.path);
this.file_tree_state.reset(&tree);
this.file_tree_items = tree;
cx.notify();
};
},
cx,
)
.detach();
}
fn handle_file_tree_item_click(&mut self, i: usize, cx: &mut gpui::Context<Self>) {
let item = &self.file_tree_items[i];
match item.kind {
| FileTreeItemKind::Directory => {
self.file_tree_state
.toggle_directory(&item.full_path, &self.file_tree_items);
cx.notify();
}
| FileTreeItemKind::File => {
self.selected_file_path = Some(Arc::clone(&item.full_path));
self.file_tree_state.highlight_item(i);
self.diff_view.update(cx, |diff_view, cx| {
diff_view.show_diff_for_file(&item.full_path, cx);
});
cx.notify();
}
}
}
}
impl gpui::Render for PullRequestChangeView {
fn render(
&mut self,
_window: &mut gpui::Window,
cx: &mut gpui::Context<Self>,
) -> impl gpui::IntoElement {
let theme = app::current_theme(cx);
let weak = cx.entity();
div()
.size_full()
.flex()
.flex_row()
.child(
div()
.flex()
.w_80()
.h_full()
.bg(theme.colors.surface_chrome)
.border_r_1()
.border_color(theme.colors.border_muted)
.p_1()
.child(
file_tree(self.file_tree_state.clone(), move |i, _, cx| {
weak.read(cx).file_tree_items[i].clone()
})
.on_item_click(cx.listener(|this, i, _, cx| {
this.handle_file_tree_item_click(*i, cx);
})),
),
)
.child(
gpui::AnyView::from(self.diff_view.clone())
.cached(gpui::StyleRefinement::default().size_full()),
)
}
}

View File

@@ -1,69 +1,42 @@
use std::sync::Arc; use std::sync::Arc;
use gpui::{AppContext, IntoElement, div};
use crate::{ use crate::{
api, app, api::{self},
component::{ app,
diff_view::{DiffViewContent, DiffViewState, diff_view}, component::diff_view::{DiffViewContent, DiffViewState, diff_view},
text::text,
},
query::{self, QueryStatus, read_query, use_query, watch_query}, query::{self, QueryStatus, read_query, use_query, watch_query},
screen::dashboard::pull_request_file_tree::{self, PullRequestFileTree}, util,
util::{self},
}; };
use gpui::{AppContext, ParentElement, Styled, div};
pub(crate) struct PullRequestDiffView { pub(crate) struct PullRequestDiffView {
selected_file_path: Option<Arc<str>>,
pr_query: query::Entity<api::issues::FetchPullRequest>, pr_query: query::Entity<api::issues::FetchPullRequest>,
file_tree_query: query::Entity<api::issues::FetchPullRequestFileTree>,
content_diff_query: Option<query::Entity<api::repo::FetchFileDiff>>, content_diff_query: Option<query::Entity<api::repo::FetchFileDiff>>,
diff_view_state: DiffViewState, diff_view_state: DiffViewState,
diff_view_content: Option<DiffViewContent>, diff_view_content: Option<DiffViewContent>,
current_file_path: Option<Arc<str>>,
file_tree: gpui::Entity<PullRequestFileTree>,
}
pub(crate) fn new(
pr_id: api::issues::Id,
cx: &mut gpui::Context<PullRequestDiffView>,
) -> PullRequestDiffView {
let mut view = PullRequestDiffView {
selected_file_path: None,
pr_query: use_query(api::issues::FetchPullRequest { id: pr_id.clone() }, cx),
file_tree_query: use_query(
api::issues::FetchPullRequestFileTree {
id: pr_id.clone(),
first: 100,
},
cx,
),
content_diff_query: None,
diff_view_state: DiffViewState::new(),
diff_view_content: None,
file_tree: cx.new(|cx| pull_request_file_tree::new(pr_id, cx)),
};
view.on_create(cx);
view
} }
impl PullRequestDiffView { impl PullRequestDiffView {
fn on_create(&mut self, cx: &mut gpui::Context<Self>) { pub(crate) fn new(pr_id: api::issues::Id, cx: &mut gpui::Context<Self>) -> Self {
_ = cx let mut s = Self {
.observe(&self.pr_query, |this, _, cx| { pr_query: use_query(api::issues::FetchPullRequest { id: pr_id }, cx),
this.start_content_queries(cx); content_diff_query: None,
}) diff_view_state: DiffViewState::new(),
.detach(); diff_view_content: None,
current_file_path: None,
};
s.on_create(cx);
s
}
_ = cx pub(crate) fn show_diff_for_file(
.observe(&self.file_tree_query, |this, _, cx| { &mut self,
this.start_content_queries(cx); file_path: &Arc<str>,
}) cx: &mut gpui::Context<Self>,
.detach(); ) {
self.current_file_path = Some(Arc::clone(file_path));
// if pr is already loaded, start content queries
self.start_content_queries(cx); self.start_content_queries(cx);
} }
@@ -71,14 +44,7 @@ impl PullRequestDiffView {
if self.content_diff_query.is_some() { if self.content_diff_query.is_some() {
return; return;
} }
let Some(selected_file_path) = self.current_file_path.as_deref() else {
if self.selected_file_path.is_none()
&& let QueryStatus::Loaded(files) = read_query(&self.file_tree_query, cx)
{
self.selected_file_path = files.first().map(|file| Arc::clone(&file.path));
}
let Some(selected_file_path) = self.selected_file_path.as_deref() else {
return; return;
}; };
@@ -115,6 +81,14 @@ impl PullRequestDiffView {
self.content_diff_query = Some(content_diff_query); self.content_diff_query = Some(content_diff_query);
} }
fn on_create(&mut self, cx: &mut gpui::Context<Self>) {
_ = cx
.observe(&self.pr_query, |this, _, cx| {
this.start_content_queries(cx);
})
.detach();
}
fn sync_content_diff_query( fn sync_content_diff_query(
&mut self, &mut self,
query: &query::Entity<api::repo::FetchFileDiff>, query: &query::Entity<api::repo::FetchFileDiff>,
@@ -122,8 +96,8 @@ impl PullRequestDiffView {
) { ) {
if let Some(diff) = { if let Some(diff) = {
match read_query(query, cx) { match read_query(query, cx) {
| QueryStatus::Loaded(diff) => Some(Arc::clone(diff)), | QueryStatus::Loaded(diff) => Some(Arc::clone(diff)),
| _ => None, | _ => None,
} }
} { } {
self.load_diff_view(diff, cx); self.load_diff_view(diff, cx);
@@ -145,7 +119,7 @@ impl PullRequestDiffView {
let theme_syntax = theme.syntax; let theme_syntax = theme.syntax;
if let Some(path) = &self.selected_file_path { if let Some(path) = &self.current_file_path {
let path = Arc::clone(&path); let path = Arc::clone(&path);
let file_type = util::file::file_type_from_path(&path); let file_type = util::file::file_type_from_path(&path);
@@ -158,16 +132,16 @@ impl PullRequestDiffView {
_ = cx _ = cx
.spawn(async move |weak, cx| match tokio::join!(t1, t2) { .spawn(async move |weak, cx| match tokio::join!(t1, t2) {
| (Some(old_side_highlights), Some(new_side_highlights)) => { | (Some(old_side_highlights), Some(new_side_highlights)) => {
_ = weak.update(cx, |this, cx| { _ = weak.update(cx, |this, cx| {
this.diff_view_state this.diff_view_state
.set_old_side_highlights(old_side_highlights); .set_old_side_highlights(old_side_highlights);
this.diff_view_state this.diff_view_state
.set_new_side_highlights(new_side_highlights); .set_new_side_highlights(new_side_highlights);
cx.notify(); cx.notify();
}); });
} }
| _ => {} | _ => {}
}) })
.detach(); .detach();
} }
@@ -178,42 +152,20 @@ impl gpui::Render for PullRequestDiffView {
fn render( fn render(
&mut self, &mut self,
_window: &mut gpui::Window, _window: &mut gpui::Window,
cx: &mut gpui::Context<Self>, cx: &mut gpui::prelude::Context<Self>,
) -> impl gpui::IntoElement { ) -> impl gpui::IntoElement {
let theme = app::current_theme(cx);
let content_diff = self let content_diff = self
.content_diff_query .content_diff_query
.as_ref() .as_ref()
.map(|q| read_query(q, cx)) .map(|q| read_query(q, cx))
.unwrap_or(QueryStatus::Loading); .unwrap_or(QueryStatus::Loading);
match content_diff { match (content_diff, &self.diff_view_content) {
| QueryStatus::Err(_) | QueryStatus::Loading => div() | (QueryStatus::Loaded(_), Some(content)) => {
.size_full() diff_view(self.diff_view_state.clone(), content.clone()).into_any_element()
.bg(theme.colors.surface) }
.p_4()
.child(text("asd")),
| QueryStatus::Loaded(_) => match &self.diff_view_content { | (_, _) => div().into_any_element(),
| Some(content) => div()
.size_full()
.flex()
.flex_row()
.child(
div()
.flex()
.w_80()
.h_full()
.bg(theme.colors.surface_chrome)
.border_r_1()
.border_color(theme.colors.border_muted)
.p_1()
.child(self.file_tree.clone()),
)
.child(diff_view(self.diff_view_state.clone(), content.clone())),
| None => div(),
},
} }
} }
} }

View File

@@ -16,14 +16,14 @@ use crate::{
text::text, text::text,
}, },
query::{self, QueryStatus, read_query, use_query, watch_query}, query::{self, QueryStatus, read_query, use_query, watch_query},
screen::dashboard::pull_request_diff_view::{self, PullRequestDiffView}, screen::dashboard::pull_request_change_view::{self, PullRequestChangeView},
}; };
pub(crate) struct PullRequestView { pub(crate) struct PullRequestView {
current_tab: Tab, current_tab: Tab,
markdown_viewer: Option<gpui::Entity<MarkdownText>>, markdown_viewer: Option<gpui::Entity<MarkdownText>>,
diff_view: Option<gpui::Entity<PullRequestDiffView>>, diff_view: Option<gpui::Entity<PullRequestChangeView>>,
pull_request_query: Option<query::Entity<api::issues::FetchPullRequest>>, pull_request_query: Option<query::Entity<api::issues::FetchPullRequest>>,
} }
@@ -101,7 +101,7 @@ impl PullRequestView {
} }
}; };
self.diff_view = pr_id.map(|id| cx.new(|cx| pull_request_diff_view::new(id, cx))); self.diff_view = pr_id.map(|id| cx.new(|cx| pull_request_change_view::new(id, cx)));
cx.notify(); cx.notify();
} }

View File

@@ -4,7 +4,7 @@ use crate::{
api, app, api, app,
screen::dashboard::{ screen::dashboard::{
issue_list::{self, IssueList}, issue_list::{self, IssueList},
pull_request_diff_view::{self, PullRequestDiffView}, pull_request_change_view::{self, PullRequestChangeView},
pull_request_view::{self, PullRequestView}, pull_request_view::{self, PullRequestView},
titlebar::{self, TitleBar}, titlebar::{self, TitleBar},
}, },
@@ -14,7 +14,7 @@ pub(crate) struct Screen {
titlebar: gpui::Entity<TitleBar>, titlebar: gpui::Entity<TitleBar>,
issue_list: gpui::Entity<IssueList>, issue_list: gpui::Entity<IssueList>,
pull_request_view: gpui::Entity<PullRequestView>, pull_request_view: gpui::Entity<PullRequestView>,
pull_request_diff_view: Option<gpui::Entity<PullRequestDiffView>>, pull_request_change_view: Option<gpui::Entity<PullRequestChangeView>>,
issue_filter: Option<&'static str>, issue_filter: Option<&'static str>,
} }
@@ -24,7 +24,7 @@ pub(crate) fn new(cx: &mut gpui::Context<Screen>) -> Screen {
titlebar: cx.new(titlebar::new), titlebar: cx.new(titlebar::new),
issue_list: cx.new(issue_list::new), issue_list: cx.new(issue_list::new),
pull_request_view: cx.new(pull_request_view::new), pull_request_view: cx.new(pull_request_view::new),
pull_request_diff_view: None, pull_request_change_view: None,
issue_filter: None, issue_filter: None,
}; };
@@ -54,8 +54,8 @@ impl Screen {
println!("change displayed pull request: {:?}", id); println!("change displayed pull request: {:?}", id);
cx.notify(); cx.notify();
}); });
self.pull_request_diff_view = self.pull_request_change_view =
Some(cx.new(|cx| pull_request_diff_view::new(id.clone(), cx))); Some(cx.new(|cx| pull_request_change_view::new(id.clone(), cx)));
} }
} }