Cargo.lock 🔗
@@ -5177,7 +5177,9 @@ name = "git_ui"
version = "0.1.0"
dependencies = [
"anyhow",
+ "collections",
"db",
+ "git",
"gpui",
"project",
"schemars",
Nate Butler created
Add entry list, scrollbar
Release Notes:
- N/A
Cargo.lock | 2
crates/git_ui/Cargo.toml | 2
crates/git_ui/TODO.md | 9
crates/git_ui/src/git_panel.rs | 452 ++++++++++++++++++++++++++++++++++-
crates/git_ui/src/git_ui.rs | 34 ++
5 files changed, 474 insertions(+), 25 deletions(-)
@@ -5177,7 +5177,9 @@ name = "git_ui"
version = "0.1.0"
dependencies = [
"anyhow",
+ "collections",
"db",
+ "git",
"gpui",
"project",
"schemars",
@@ -25,6 +25,8 @@ settings.workspace = true
ui.workspace = true
util.workspace = true
workspace.workspace = true
+git.workspace = true
+collections.workspace = true
[target.'cfg(windows)'.dependencies]
windows.workspace = true
@@ -4,14 +4,17 @@
### List
-- [ ] Git status item
+- [x] Add uniform list
+- [x] Git status item
- [ ] Directory item
-- [ ] Scrollbar
+- [x] Scrollbar
- [ ] Add indent size setting
- [ ] Add tree settings
### List Items
+- [x] Checkbox for staging
+- [x] Git status icon
- [ ] Context menu
- [ ] Discard Changes
- ---
@@ -35,7 +38,7 @@
- [ ] ChangedLineCount (new)
- takes `lines_added: usize, lines_removed: usize`, returns a added/removed badge
-- [ ] GitStatusIcon (new)
+- [x] GitStatusIcon (new)
- [ ] Checkbox
- update checkbox design
- [ ] ScrollIndicator
@@ -1,16 +1,30 @@
-use std::sync::Arc;
-use util::TryFutureExt;
+use collections::HashMap;
+use std::{
+ cell::OnceCell,
+ collections::HashSet,
+ ffi::OsStr,
+ ops::Range,
+ path::{Path, PathBuf},
+ sync::Arc,
+ time::Duration,
+};
+
+use git::repository::GitFileStatus;
+
+use util::{ResultExt, TryFutureExt};
use db::kvp::KEY_VALUE_STORE;
use gpui::*;
-use project::{Fs, Project};
+use project::{Entry, EntryKind, Fs, Project, ProjectEntryId, WorktreeId};
use serde::{Deserialize, Serialize};
use settings::Settings as _;
-use ui::{prelude::*, Checkbox, Divider, DividerColor, ElevationIndex, Tooltip};
+use ui::{
+ prelude::*, Checkbox, Divider, DividerColor, ElevationIndex, Scrollbar, ScrollbarState, Tooltip,
+};
use workspace::dock::{DockPosition, Panel, PanelEvent};
use workspace::Workspace;
-use crate::settings::GitPanelSettings;
+use crate::{git_status_icon, settings::GitPanelSettings};
use crate::{CommitAllChanges, CommitStagedChanges, DiscardAll, StageAll, UnstageAll};
actions!(git_panel, [ToggleFocus]);
@@ -28,6 +42,30 @@ pub fn init(cx: &mut AppContext) {
.detach();
}
+#[derive(Debug)]
+pub enum Event {
+ Focus,
+}
+
+pub struct GitStatusEntry {}
+
+#[derive(Debug, PartialEq, Eq, Clone)]
+struct EntryDetails {
+ filename: String,
+ display_name: String,
+ path: Arc<Path>,
+ kind: EntryKind,
+ depth: usize,
+ is_expanded: bool,
+ status: Option<GitFileStatus>,
+}
+
+impl EntryDetails {
+ pub fn is_dir(&self) -> bool {
+ self.kind.is_dir()
+ }
+}
+
#[derive(Serialize, Deserialize)]
struct SerializedGitPanel {
width: Option<Pixels>,
@@ -35,13 +73,22 @@ struct SerializedGitPanel {
pub struct GitPanel {
_workspace: WeakView<Workspace>,
+ current_modifiers: Modifiers,
focus_handle: FocusHandle,
fs: Arc<dyn Fs>,
+ hide_scrollbar_task: Option<Task<()>>,
pending_serialization: Task<Option<()>>,
project: Model<Project>,
+ scroll_handle: UniformListScrollHandle,
+ scrollbar_state: ScrollbarState,
+ selected_item: Option<usize>,
+ show_scrollbar: bool,
+ expanded_dir_ids: HashMap<WorktreeId, Vec<ProjectEntryId>>,
+
+ // The entries that are currently shown in the panel, aka
+ // not hidden by folding or such
+ visible_entries: Vec<(WorktreeId, Vec<Entry>, OnceCell<HashSet<Arc<Path>>>)>,
width: Option<Pixels>,
-
- current_modifiers: Modifiers,
}
impl GitPanel {
@@ -57,21 +104,57 @@ impl GitPanel {
}
pub fn new(workspace: &mut Workspace, cx: &mut ViewContext<Workspace>) -> View<Self> {
- let project = workspace.project().clone();
let fs = workspace.app_state().fs.clone();
let weak_workspace = workspace.weak_handle();
+ let project = workspace.project().clone();
- cx.new_view(|cx| Self {
- _workspace: weak_workspace,
- focus_handle: cx.focus_handle(),
- fs,
- pending_serialization: Task::ready(None),
- project,
-
- current_modifiers: cx.modifiers(),
-
- width: Some(px(360.)),
- })
+ let git_panel = cx.new_view(|cx: &mut ViewContext<Self>| {
+ let focus_handle = cx.focus_handle();
+ cx.on_focus(&focus_handle, Self::focus_in).detach();
+ cx.on_focus_out(&focus_handle, |this, _, cx| {
+ this.hide_scrollbar(cx);
+ })
+ .detach();
+ cx.subscribe(&project, |this, _project, event, cx| match event {
+ project::Event::WorktreeRemoved(id) => {
+ this.expanded_dir_ids.remove(id);
+ this.update_visible_entries(None, cx);
+ cx.notify();
+ }
+ project::Event::WorktreeUpdatedEntries(_, _)
+ | project::Event::WorktreeAdded(_)
+ | project::Event::WorktreeOrderChanged => {
+ this.update_visible_entries(None, cx);
+ cx.notify();
+ }
+ _ => {}
+ })
+ .detach();
+
+ let scroll_handle = UniformListScrollHandle::new();
+
+ let mut this = Self {
+ _workspace: weak_workspace,
+ focus_handle: cx.focus_handle(),
+ fs,
+ pending_serialization: Task::ready(None),
+ project,
+ visible_entries: Vec::new(),
+ current_modifiers: cx.modifiers(),
+ expanded_dir_ids: Default::default(),
+
+ width: Some(px(360.)),
+ scrollbar_state: ScrollbarState::new(scroll_handle.clone()).parent_view(cx.view()),
+ scroll_handle,
+ selected_item: None,
+ show_scrollbar: !Self::should_autohide_scrollbar(cx),
+ hide_scrollbar_task: None,
+ };
+ this.update_visible_entries(None, cx);
+ this
+ });
+
+ git_panel
}
fn serialize(&mut self, cx: &mut ViewContext<Self>) {
@@ -98,6 +181,40 @@ impl GitPanel {
dispatch_context
}
+ fn focus_in(&mut self, cx: &mut ViewContext<Self>) {
+ if !self.focus_handle.contains_focused(cx) {
+ cx.emit(Event::Focus);
+ }
+ }
+
+ fn should_show_scrollbar(_cx: &AppContext) -> bool {
+ // todo!(): plug into settings
+ true
+ }
+
+ fn should_autohide_scrollbar(_cx: &AppContext) -> bool {
+ // todo!(): plug into settings
+ true
+ }
+
+ fn hide_scrollbar(&mut self, cx: &mut ViewContext<Self>) {
+ const SCROLLBAR_SHOW_INTERVAL: Duration = Duration::from_secs(1);
+ if !Self::should_autohide_scrollbar(cx) {
+ return;
+ }
+ self.hide_scrollbar_task = Some(cx.spawn(|panel, mut cx| async move {
+ cx.background_executor()
+ .timer(SCROLLBAR_SHOW_INTERVAL)
+ .await;
+ panel
+ .update(&mut cx, |panel, cx| {
+ panel.show_scrollbar = false;
+ cx.notify();
+ })
+ .log_err();
+ }))
+ }
+
fn handle_modifiers_changed(
&mut self,
event: &ModifiersChangedEvent,
@@ -106,6 +223,34 @@ impl GitPanel {
self.current_modifiers = event.modifiers;
cx.notify();
}
+
+ fn calculate_depth_and_difference(
+ entry: &Entry,
+ visible_worktree_entries: &HashSet<Arc<Path>>,
+ ) -> (usize, usize) {
+ let (depth, difference) = entry
+ .path
+ .ancestors()
+ .skip(1) // Skip the entry itself
+ .find_map(|ancestor| {
+ if let Some(parent_entry) = visible_worktree_entries.get(ancestor) {
+ let entry_path_components_count = entry.path.components().count();
+ let parent_path_components_count = parent_entry.components().count();
+ let difference = entry_path_components_count - parent_path_components_count;
+ let depth = parent_entry
+ .ancestors()
+ .skip(1)
+ .filter(|ancestor| visible_worktree_entries.contains(*ancestor))
+ .count();
+ Some((depth + 1, difference))
+ } else {
+ None
+ }
+ })
+ .unwrap_or((0, 0));
+
+ (depth, difference)
+ }
}
impl GitPanel {
@@ -140,6 +285,147 @@ impl GitPanel {
// todo!(): Implement all_staged
true
}
+
+ fn no_entries(&self) -> bool {
+ self.visible_entries.is_empty()
+ }
+
+ fn entry_count(&self) -> usize {
+ self.visible_entries
+ .iter()
+ .map(|(_, entries, _)| {
+ entries
+ .iter()
+ .filter(|entry| entry.git_status.is_some())
+ .count()
+ })
+ .sum()
+ }
+
+ fn for_each_visible_entry(
+ &self,
+ range: Range<usize>,
+ cx: &mut ViewContext<Self>,
+ mut callback: impl FnMut(ProjectEntryId, EntryDetails, &mut ViewContext<Self>),
+ ) {
+ let mut ix = 0;
+ for (worktree_id, visible_worktree_entries, entries_paths) in &self.visible_entries {
+ if ix >= range.end {
+ return;
+ }
+
+ if ix + visible_worktree_entries.len() <= range.start {
+ ix += visible_worktree_entries.len();
+ continue;
+ }
+
+ let end_ix = range.end.min(ix + visible_worktree_entries.len());
+ // let entry_range = range.start.saturating_sub(ix)..end_ix - ix;
+ if let Some(worktree) = self.project.read(cx).worktree_for_id(*worktree_id, cx) {
+ let snapshot = worktree.read(cx).snapshot();
+ let root_name = OsStr::new(snapshot.root_name());
+ let expanded_entry_ids = self
+ .expanded_dir_ids
+ .get(&snapshot.id())
+ .map(Vec::as_slice)
+ .unwrap_or(&[]);
+
+ let entry_range = range.start.saturating_sub(ix)..end_ix - ix;
+ let entries = entries_paths.get_or_init(|| {
+ visible_worktree_entries
+ .iter()
+ .map(|e| (e.path.clone()))
+ .collect()
+ });
+
+ for entry in visible_worktree_entries[entry_range].iter() {
+ let status = entry.git_status;
+ let is_expanded = expanded_entry_ids.binary_search(&entry.id).is_ok();
+
+ let (depth, difference) = Self::calculate_depth_and_difference(entry, entries);
+
+ let filename = match difference {
+ diff if diff > 1 => entry
+ .path
+ .iter()
+ .skip(entry.path.components().count() - diff)
+ .collect::<PathBuf>()
+ .to_str()
+ .unwrap_or_default()
+ .to_string(),
+ _ => entry
+ .path
+ .file_name()
+ .map(|name| name.to_string_lossy().into_owned())
+ .unwrap_or_else(|| root_name.to_string_lossy().to_string()),
+ };
+
+ let display_name = entry.path.to_string_lossy().into_owned();
+
+ let details = EntryDetails {
+ filename,
+ display_name,
+ kind: entry.kind,
+ is_expanded,
+ path: entry.path.clone(),
+ status,
+ depth,
+ };
+ callback(entry.id, details, cx);
+ }
+ }
+ ix = end_ix;
+ }
+ }
+
+ // todo!(): Update expanded directory state
+ fn update_visible_entries(
+ &mut self,
+ new_selected_entry: Option<(WorktreeId, ProjectEntryId)>,
+ cx: &mut ViewContext<Self>,
+ ) {
+ let project = self.project.read(cx);
+ self.visible_entries.clear();
+ for worktree in project.visible_worktrees(cx) {
+ let snapshot = worktree.read(cx).snapshot();
+ let worktree_id = snapshot.id();
+
+ let mut visible_worktree_entries = Vec::new();
+ let mut entry_iter = snapshot.entries(true, 0);
+ while let Some(entry) = entry_iter.entry() {
+ // Only include entries with a git status
+ if entry.git_status.is_some() {
+ visible_worktree_entries.push(entry.clone());
+ }
+ entry_iter.advance();
+ }
+
+ snapshot.propagate_git_statuses(&mut visible_worktree_entries);
+ project::sort_worktree_entries(&mut visible_worktree_entries);
+
+ if !visible_worktree_entries.is_empty() {
+ self.visible_entries
+ .push((worktree_id, visible_worktree_entries, OnceCell::new()));
+ }
+ }
+
+ if let Some((worktree_id, entry_id)) = new_selected_entry {
+ self.selected_item = self.visible_entries.iter().enumerate().find_map(
+ |(worktree_index, (id, entries, _))| {
+ if *id == worktree_id {
+ entries
+ .iter()
+ .position(|entry| entry.id == entry_id)
+ .map(|entry_index| worktree_index * entries.len() + entry_index)
+ } else {
+ None
+ }
+ },
+ );
+ }
+
+ cx.notify();
+ }
}
impl GitPanel {
@@ -168,6 +454,8 @@ impl GitPanel {
pub fn render_panel_header(&self, cx: &mut ViewContext<Self>) -> impl IntoElement {
let focus_handle = self.focus_handle(cx).clone();
+ let changes_string = format!("{} changes", self.entry_count());
+
h_flex()
.h(px(32.))
.items_center()
@@ -177,7 +465,7 @@ impl GitPanel {
h_flex()
.gap_2()
.child(Checkbox::new("all-changes", true.into()).disabled(true))
- .child(div().text_buffer(cx).text_ui_sm(cx).child("0 changes")),
+ .child(div().text_buffer(cx).text_ui_sm(cx).child(changes_string)),
)
.child(div().flex_grow())
.child(
@@ -283,6 +571,113 @@ impl GitPanel {
.text_color(Color::Placeholder.color(cx)),
)
}
+
+ fn render_scrollbar(&self, cx: &mut ViewContext<Self>) -> Option<Stateful<Div>> {
+ if !Self::should_show_scrollbar(cx)
+ || !(self.show_scrollbar || self.scrollbar_state.is_dragging())
+ {
+ return None;
+ }
+ Some(
+ div()
+ .occlude()
+ .id("project-panel-vertical-scroll")
+ .on_mouse_move(cx.listener(|_, _, cx| {
+ cx.notify();
+ cx.stop_propagation()
+ }))
+ .on_hover(|_, cx| {
+ cx.stop_propagation();
+ })
+ .on_any_mouse_down(|_, cx| {
+ cx.stop_propagation();
+ })
+ .on_mouse_up(
+ MouseButton::Left,
+ cx.listener(|this, _, cx| {
+ if !this.scrollbar_state.is_dragging()
+ && !this.focus_handle.contains_focused(cx)
+ {
+ this.hide_scrollbar(cx);
+ cx.notify();
+ }
+
+ cx.stop_propagation();
+ }),
+ )
+ .on_scroll_wheel(cx.listener(|_, _, cx| {
+ cx.notify();
+ }))
+ .h_full()
+ .absolute()
+ .right_1()
+ .top_1()
+ .bottom_1()
+ .w(px(12.))
+ .cursor_default()
+ .children(Scrollbar::vertical(
+ // percentage as f32..end_offset as f32,
+ self.scrollbar_state.clone(),
+ )),
+ )
+ }
+
+ fn render_entries(&self, cx: &mut ViewContext<Self>) -> impl IntoElement {
+ let item_count = self
+ .visible_entries
+ .iter()
+ .map(|(_, worktree_entries, _)| worktree_entries.len())
+ .sum();
+ h_flex()
+ .size_full()
+ .overflow_hidden()
+ .child(
+ uniform_list(cx.view().clone(), "entries", item_count, {
+ |this, range, cx| {
+ let mut items = Vec::with_capacity(range.end - range.start);
+ this.for_each_visible_entry(range, cx, |id, details, cx| {
+ items.push(this.render_entry(id, details, cx));
+ });
+ items
+ }
+ })
+ .size_full()
+ .with_sizing_behavior(ListSizingBehavior::Infer)
+ .with_horizontal_sizing_behavior(ListHorizontalSizingBehavior::Unconstrained)
+ // .with_width_from_item(self.max_width_item_index)
+ .track_scroll(self.scroll_handle.clone()),
+ )
+ .children(self.render_scrollbar(cx))
+ }
+
+ fn render_entry(
+ &self,
+ id: ProjectEntryId,
+ details: EntryDetails,
+ cx: &ViewContext<Self>,
+ ) -> impl IntoElement {
+ let id = id.to_proto() as usize;
+ let checkbox_id = ElementId::Name(format!("checkbox_{}", id).into());
+ let is_staged = Selection::Selected;
+
+ h_flex()
+ .id(id)
+ .h(px(28.))
+ .w_full()
+ .pl(px(12. + 12. * details.depth as f32))
+ .pr(px(4.))
+ .items_center()
+ .gap_2()
+ .font_buffer(cx)
+ .text_ui_sm(cx)
+ .when(!details.is_dir(), |this| {
+ this.child(Checkbox::new(checkbox_id, is_staged))
+ })
+ .when_some(details.status, |this, status| {
+ this.child(git_status_icon(status))
+ })
+ .child(h_flex().gap_1p5().child(details.display_name.clone()))
+ }
}
impl Render for GitPanel {
@@ -309,6 +704,15 @@ impl Render for GitPanel {
this.commit_all_changes(&CommitAllChanges, cx)
}))
})
+ .on_hover(cx.listener(|this, hovered, cx| {
+ if *hovered {
+ this.show_scrollbar = true;
+ this.hide_scrollbar_task.take();
+ cx.notify();
+ } else if !this.focus_handle.contains_focused(cx) {
+ this.hide_scrollbar(cx);
+ }
+ }))
.size_full()
.overflow_hidden()
.font_buffer(cx)
@@ -316,7 +720,11 @@ impl Render for GitPanel {
.bg(ElevationIndex::Surface.bg(cx))
.child(self.render_panel_header(cx))
.child(self.render_divider(cx))
- .child(self.render_empty_state(cx))
+ .child(if !self.no_entries() {
+ self.render_entries(cx).into_any_element()
+ } else {
+ self.render_empty_state(cx).into_any_element()
+ })
.child(self.render_divider(cx))
.child(self.render_commit_editor(cx))
}
@@ -328,6 +736,8 @@ impl FocusableView for GitPanel {
}
}
+impl EventEmitter<Event> for GitPanel {}
+
impl EventEmitter<PanelEvent> for GitPanel {}
impl Panel for GitPanel {
@@ -1,6 +1,8 @@
use ::settings::Settings;
-use gpui::{actions, AppContext};
+use git::repository::GitFileStatus;
+use gpui::{actions, AppContext, Hsla};
use settings::GitPanelSettings;
+use ui::{Color, Icon, IconName, IntoElement};
pub mod git_panel;
mod settings;
@@ -19,3 +21,33 @@ actions!(
pub fn init(cx: &mut AppContext) {
GitPanelSettings::register(cx);
}
+
+const ADDED_COLOR: Hsla = Hsla {
+ h: 142. / 360.,
+ s: 0.68,
+ l: 0.45,
+ a: 1.0,
+};
+const MODIFIED_COLOR: Hsla = Hsla {
+ h: 48. / 360.,
+ s: 0.76,
+ l: 0.47,
+ a: 1.0,
+};
+const REMOVED_COLOR: Hsla = Hsla {
+ h: 355. / 360.,
+ s: 0.65,
+ l: 0.65,
+ a: 1.0,
+};
+
+// todo!(): Add updated status colors to theme
+pub fn git_status_icon(status: GitFileStatus) -> impl IntoElement {
+ match status {
+ GitFileStatus::Added => Icon::new(IconName::SquarePlus).color(Color::Custom(ADDED_COLOR)),
+ GitFileStatus::Modified => {
+ Icon::new(IconName::SquareDot).color(Color::Custom(MODIFIED_COLOR))
+ }
+ GitFileStatus::Conflict => Icon::new(IconName::Warning).color(Color::Custom(REMOVED_COLOR)),
+ }
+}