Git panel refinements 2 (#21947)
Add entry list, scrollbar Release Notes: - N/A
This commit is contained in:
parent
b38e9e44d6
commit
9f0f63f92b
5 changed files with 472 additions and 23 deletions
2
Cargo.lock
generated
2
Cargo.lock
generated
|
@ -5177,7 +5177,9 @@ name = "git_ui"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
|
"collections",
|
||||||
"db",
|
"db",
|
||||||
|
"git",
|
||||||
"gpui",
|
"gpui",
|
||||||
"project",
|
"project",
|
||||||
"schemars",
|
"schemars",
|
||||||
|
|
|
@ -25,6 +25,8 @@ settings.workspace = true
|
||||||
ui.workspace = true
|
ui.workspace = true
|
||||||
util.workspace = true
|
util.workspace = true
|
||||||
workspace.workspace = true
|
workspace.workspace = true
|
||||||
|
git.workspace = true
|
||||||
|
collections.workspace = true
|
||||||
|
|
||||||
[target.'cfg(windows)'.dependencies]
|
[target.'cfg(windows)'.dependencies]
|
||||||
windows.workspace = true
|
windows.workspace = true
|
||||||
|
|
|
@ -4,14 +4,17 @@
|
||||||
|
|
||||||
### List
|
### List
|
||||||
|
|
||||||
- [ ] Git status item
|
- [x] Add uniform list
|
||||||
|
- [x] Git status item
|
||||||
- [ ] Directory item
|
- [ ] Directory item
|
||||||
- [ ] Scrollbar
|
- [x] Scrollbar
|
||||||
- [ ] Add indent size setting
|
- [ ] Add indent size setting
|
||||||
- [ ] Add tree settings
|
- [ ] Add tree settings
|
||||||
|
|
||||||
### List Items
|
### List Items
|
||||||
|
|
||||||
|
- [x] Checkbox for staging
|
||||||
|
- [x] Git status icon
|
||||||
- [ ] Context menu
|
- [ ] Context menu
|
||||||
- [ ] Discard Changes
|
- [ ] Discard Changes
|
||||||
- ---
|
- ---
|
||||||
|
@ -35,7 +38,7 @@
|
||||||
|
|
||||||
- [ ] ChangedLineCount (new)
|
- [ ] ChangedLineCount (new)
|
||||||
- takes `lines_added: usize, lines_removed: usize`, returns a added/removed badge
|
- takes `lines_added: usize, lines_removed: usize`, returns a added/removed badge
|
||||||
- [ ] GitStatusIcon (new)
|
- [x] GitStatusIcon (new)
|
||||||
- [ ] Checkbox
|
- [ ] Checkbox
|
||||||
- update checkbox design
|
- update checkbox design
|
||||||
- [ ] ScrollIndicator
|
- [ ] ScrollIndicator
|
||||||
|
|
|
@ -1,16 +1,30 @@
|
||||||
use std::sync::Arc;
|
use collections::HashMap;
|
||||||
use util::TryFutureExt;
|
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 db::kvp::KEY_VALUE_STORE;
|
||||||
use gpui::*;
|
use gpui::*;
|
||||||
use project::{Fs, Project};
|
use project::{Entry, EntryKind, Fs, Project, ProjectEntryId, WorktreeId};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use settings::Settings as _;
|
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::dock::{DockPosition, Panel, PanelEvent};
|
||||||
use workspace::Workspace;
|
use workspace::Workspace;
|
||||||
|
|
||||||
use crate::settings::GitPanelSettings;
|
use crate::{git_status_icon, settings::GitPanelSettings};
|
||||||
use crate::{CommitAllChanges, CommitStagedChanges, DiscardAll, StageAll, UnstageAll};
|
use crate::{CommitAllChanges, CommitStagedChanges, DiscardAll, StageAll, UnstageAll};
|
||||||
|
|
||||||
actions!(git_panel, [ToggleFocus]);
|
actions!(git_panel, [ToggleFocus]);
|
||||||
|
@ -28,6 +42,30 @@ pub fn init(cx: &mut AppContext) {
|
||||||
.detach();
|
.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)]
|
#[derive(Serialize, Deserialize)]
|
||||||
struct SerializedGitPanel {
|
struct SerializedGitPanel {
|
||||||
width: Option<Pixels>,
|
width: Option<Pixels>,
|
||||||
|
@ -35,13 +73,22 @@ struct SerializedGitPanel {
|
||||||
|
|
||||||
pub struct GitPanel {
|
pub struct GitPanel {
|
||||||
_workspace: WeakView<Workspace>,
|
_workspace: WeakView<Workspace>,
|
||||||
|
current_modifiers: Modifiers,
|
||||||
focus_handle: FocusHandle,
|
focus_handle: FocusHandle,
|
||||||
fs: Arc<dyn Fs>,
|
fs: Arc<dyn Fs>,
|
||||||
|
hide_scrollbar_task: Option<Task<()>>,
|
||||||
pending_serialization: Task<Option<()>>,
|
pending_serialization: Task<Option<()>>,
|
||||||
project: Model<Project>,
|
project: Model<Project>,
|
||||||
width: Option<Pixels>,
|
scroll_handle: UniformListScrollHandle,
|
||||||
|
scrollbar_state: ScrollbarState,
|
||||||
|
selected_item: Option<usize>,
|
||||||
|
show_scrollbar: bool,
|
||||||
|
expanded_dir_ids: HashMap<WorktreeId, Vec<ProjectEntryId>>,
|
||||||
|
|
||||||
current_modifiers: Modifiers,
|
// 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>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GitPanel {
|
impl GitPanel {
|
||||||
|
@ -57,21 +104,57 @@ impl GitPanel {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new(workspace: &mut Workspace, cx: &mut ViewContext<Workspace>) -> View<Self> {
|
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 fs = workspace.app_state().fs.clone();
|
||||||
let weak_workspace = workspace.weak_handle();
|
let weak_workspace = workspace.weak_handle();
|
||||||
|
let project = workspace.project().clone();
|
||||||
|
|
||||||
cx.new_view(|cx| Self {
|
let git_panel = cx.new_view(|cx: &mut ViewContext<Self>| {
|
||||||
_workspace: weak_workspace,
|
let focus_handle = cx.focus_handle();
|
||||||
focus_handle: cx.focus_handle(),
|
cx.on_focus(&focus_handle, Self::focus_in).detach();
|
||||||
fs,
|
cx.on_focus_out(&focus_handle, |this, _, cx| {
|
||||||
pending_serialization: Task::ready(None),
|
this.hide_scrollbar(cx);
|
||||||
project,
|
})
|
||||||
|
.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();
|
||||||
|
|
||||||
current_modifiers: cx.modifiers(),
|
let scroll_handle = UniformListScrollHandle::new();
|
||||||
|
|
||||||
width: Some(px(360.)),
|
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>) {
|
fn serialize(&mut self, cx: &mut ViewContext<Self>) {
|
||||||
|
@ -98,6 +181,40 @@ impl GitPanel {
|
||||||
dispatch_context
|
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(
|
fn handle_modifiers_changed(
|
||||||
&mut self,
|
&mut self,
|
||||||
event: &ModifiersChangedEvent,
|
event: &ModifiersChangedEvent,
|
||||||
|
@ -106,6 +223,34 @@ impl GitPanel {
|
||||||
self.current_modifiers = event.modifiers;
|
self.current_modifiers = event.modifiers;
|
||||||
cx.notify();
|
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 {
|
impl GitPanel {
|
||||||
|
@ -140,6 +285,147 @@ impl GitPanel {
|
||||||
// todo!(): Implement all_staged
|
// todo!(): Implement all_staged
|
||||||
true
|
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 {
|
impl GitPanel {
|
||||||
|
@ -168,6 +454,8 @@ impl GitPanel {
|
||||||
pub fn render_panel_header(&self, cx: &mut ViewContext<Self>) -> impl IntoElement {
|
pub fn render_panel_header(&self, cx: &mut ViewContext<Self>) -> impl IntoElement {
|
||||||
let focus_handle = self.focus_handle(cx).clone();
|
let focus_handle = self.focus_handle(cx).clone();
|
||||||
|
|
||||||
|
let changes_string = format!("{} changes", self.entry_count());
|
||||||
|
|
||||||
h_flex()
|
h_flex()
|
||||||
.h(px(32.))
|
.h(px(32.))
|
||||||
.items_center()
|
.items_center()
|
||||||
|
@ -177,7 +465,7 @@ impl GitPanel {
|
||||||
h_flex()
|
h_flex()
|
||||||
.gap_2()
|
.gap_2()
|
||||||
.child(Checkbox::new("all-changes", true.into()).disabled(true))
|
.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(div().flex_grow())
|
||||||
.child(
|
.child(
|
||||||
|
@ -283,6 +571,113 @@ impl GitPanel {
|
||||||
.text_color(Color::Placeholder.color(cx)),
|
.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 {
|
impl Render for GitPanel {
|
||||||
|
@ -309,6 +704,15 @@ impl Render for GitPanel {
|
||||||
this.commit_all_changes(&CommitAllChanges, cx)
|
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()
|
.size_full()
|
||||||
.overflow_hidden()
|
.overflow_hidden()
|
||||||
.font_buffer(cx)
|
.font_buffer(cx)
|
||||||
|
@ -316,7 +720,11 @@ impl Render for GitPanel {
|
||||||
.bg(ElevationIndex::Surface.bg(cx))
|
.bg(ElevationIndex::Surface.bg(cx))
|
||||||
.child(self.render_panel_header(cx))
|
.child(self.render_panel_header(cx))
|
||||||
.child(self.render_divider(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_divider(cx))
|
||||||
.child(self.render_commit_editor(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 EventEmitter<PanelEvent> for GitPanel {}
|
||||||
|
|
||||||
impl Panel for GitPanel {
|
impl Panel for GitPanel {
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
use ::settings::Settings;
|
use ::settings::Settings;
|
||||||
use gpui::{actions, AppContext};
|
use git::repository::GitFileStatus;
|
||||||
|
use gpui::{actions, AppContext, Hsla};
|
||||||
use settings::GitPanelSettings;
|
use settings::GitPanelSettings;
|
||||||
|
use ui::{Color, Icon, IconName, IntoElement};
|
||||||
|
|
||||||
pub mod git_panel;
|
pub mod git_panel;
|
||||||
mod settings;
|
mod settings;
|
||||||
|
@ -19,3 +21,33 @@ actions!(
|
||||||
pub fn init(cx: &mut AppContext) {
|
pub fn init(cx: &mut AppContext) {
|
||||||
GitPanelSettings::register(cx);
|
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)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue