Branch/co-authors in commit (#24768)

- **branch selector in commit box**
- **TEMP**
- **Add co-authors toggle button**

Closes #ISSUE

Release Notes:

- N/A *or* Added/Fixed/Improved ...

---------

Co-authored-by: Mikayla <mikayla@zed.dev>
This commit is contained in:
Conrad Irwin 2025-02-12 20:53:52 -07:00 committed by GitHub
parent 71867096c8
commit 21a1541a70
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 148 additions and 123 deletions

2
Cargo.lock generated
View file

@ -5358,6 +5358,7 @@ dependencies = [
"fuzzy", "fuzzy",
"git", "git",
"gpui", "gpui",
"itertools 0.14.0",
"language", "language",
"menu", "menu",
"multi_buffer", "multi_buffer",
@ -5372,7 +5373,6 @@ dependencies = [
"settings", "settings",
"theme", "theme",
"time", "time",
"time_format",
"ui", "ui",
"util", "util",
"windows 0.58.0", "windows 0.58.0",

View file

@ -14,15 +14,16 @@ path = "src/git_ui.rs"
[dependencies] [dependencies]
anyhow.workspace = true anyhow.workspace = true
buffer_diff.workspace = true
collections.workspace = true collections.workspace = true
db.workspace = true db.workspace = true
buffer_diff.workspace = true
editor.workspace = true editor.workspace = true
feature_flags.workspace = true feature_flags.workspace = true
futures.workspace = true futures.workspace = true
fuzzy.workspace = true fuzzy.workspace = true
git.workspace = true git.workspace = true
gpui.workspace = true gpui.workspace = true
itertools.workspace = true
language.workspace = true language.workspace = true
menu.workspace = true menu.workspace = true
multi_buffer.workspace = true multi_buffer.workspace = true
@ -37,7 +38,6 @@ serde_json.workspace = true
settings.workspace = true settings.workspace = true
theme.workspace = true theme.workspace = true
time.workspace = true time.workspace = true
time_format.workspace = true
ui.workspace = true ui.workspace = true
util.workspace = true util.workspace = true
workspace.workspace = true workspace.workspace = true

View file

@ -8,12 +8,13 @@ use collections::HashMap;
use db::kvp::KEY_VALUE_STORE; use db::kvp::KEY_VALUE_STORE;
use editor::commit_tooltip::CommitTooltip; use editor::commit_tooltip::CommitTooltip;
use editor::{ use editor::{
actions::MoveToEnd, scroll::ScrollbarAutoHide, Editor, EditorElement, EditorMode, scroll::ScrollbarAutoHide, Editor, EditorElement, EditorMode, EditorSettings, MultiBuffer,
EditorSettings, MultiBuffer, ShowScrollbar, ShowScrollbar,
}; };
use git::repository::{CommitDetails, ResetMode}; use git::repository::{CommitDetails, ResetMode};
use git::{repository::RepoPath, status::FileStatus, Commit, ToggleStaged}; use git::{repository::RepoPath, status::FileStatus, Commit, ToggleStaged};
use gpui::*; use gpui::*;
use itertools::Itertools;
use language::{markdown, Buffer, File, ParsedMarkdown}; use language::{markdown, Buffer, File, ParsedMarkdown};
use menu::{SelectFirst, SelectLast, SelectNext, SelectPrev}; use menu::{SelectFirst, SelectLast, SelectNext, SelectPrev};
use multi_buffer::ExcerptInfo; use multi_buffer::ExcerptInfo;
@ -27,8 +28,8 @@ use settings::Settings as _;
use std::{collections::HashSet, path::PathBuf, sync::Arc, time::Duration, usize}; use std::{collections::HashSet, path::PathBuf, sync::Arc, time::Duration, usize};
use time::OffsetDateTime; use time::OffsetDateTime;
use ui::{ use ui::{
prelude::*, ButtonLike, Checkbox, CheckboxWithLabel, Divider, DividerColor, ElevationIndex, prelude::*, ButtonLike, Checkbox, Divider, DividerColor, ElevationIndex, IndentGuideColors,
IndentGuideColors, ListItem, ListItemSpacing, Scrollbar, ScrollbarState, Tooltip, ListItem, ListItemSpacing, Scrollbar, ScrollbarState, Tooltip,
}; };
use util::{maybe, ResultExt, TryFutureExt}; use util::{maybe, ResultExt, TryFutureExt};
use workspace::{ use workspace::{
@ -45,7 +46,7 @@ actions!(
OpenMenu, OpenMenu,
FocusEditor, FocusEditor,
FocusChanges, FocusChanges,
FillCoAuthors, ToggleFillCoAuthors,
] ]
); );
@ -154,7 +155,7 @@ pub struct GitPanel {
conflicted_count: usize, conflicted_count: usize,
conflicted_staged_count: usize, conflicted_staged_count: usize,
current_modifiers: Modifiers, current_modifiers: Modifiers,
enable_auto_coauthors: bool, add_coauthors: bool,
entries: Vec<GitListEntry>, entries: Vec<GitListEntry>,
entries_by_path: collections::HashMap<RepoPath, usize>, entries_by_path: collections::HashMap<RepoPath, usize>,
focus_handle: FocusHandle, focus_handle: FocusHandle,
@ -260,7 +261,7 @@ impl GitPanel {
conflicted_count: 0, conflicted_count: 0,
conflicted_staged_count: 0, conflicted_staged_count: 0,
current_modifiers: window.modifiers(), current_modifiers: window.modifiers(),
enable_auto_coauthors: true, add_coauthors: true,
entries: Vec::new(), entries: Vec::new(),
entries_by_path: HashMap::default(), entries_by_path: HashMap::default(),
focus_handle: cx.focus_handle(), focus_handle: cx.focus_handle(),
@ -696,11 +697,14 @@ impl GitPanel {
return; return;
} }
let message = self.commit_editor.read(cx).text(cx); let mut message = self.commit_editor.read(cx).text(cx);
if message.trim().is_empty() { if message.trim().is_empty() {
self.commit_editor.read(cx).focus_handle(cx).focus(window); self.commit_editor.read(cx).focus_handle(cx).focus(window);
return; return;
} }
if self.add_coauthors {
self.fill_co_authors(&mut message, cx);
}
let task = if self.has_staged_changes() { let task = if self.has_staged_changes() {
// Repository serializes all git operations, so we can just send a commit immediately // Repository serializes all git operations, so we can just send a commit immediately
@ -781,19 +785,70 @@ impl GitPanel {
self.pending_commit = Some(task); self.pending_commit = Some(task);
} }
fn fill_co_authors(&mut self, _: &FillCoAuthors, window: &mut Window, cx: &mut Context<Self>) { fn potential_co_authors(&self, cx: &App) -> Vec<(String, String)> {
const CO_AUTHOR_PREFIX: &str = "Co-authored-by: "; let mut new_co_authors = Vec::new();
let project = self.project.read(cx);
let Some(room) = self let Some(room) = self
.workspace .workspace
.upgrade() .upgrade()
.and_then(|workspace| workspace.read(cx).active_call()?.read(cx).room().cloned()) .and_then(|workspace| workspace.read(cx).active_call()?.read(cx).room().cloned())
else { else {
return; return Vec::default();
}; };
let mut existing_text = self.commit_editor.read(cx).text(cx); let room = room.read(cx);
existing_text.make_ascii_lowercase();
for (peer_id, collaborator) in project.collaborators() {
if collaborator.is_host {
continue;
}
let Some(participant) = room.remote_participant_for_peer_id(*peer_id) else {
continue;
};
if participant.can_write() && participant.user.email.is_some() {
let email = participant.user.email.clone().unwrap();
new_co_authors.push((
participant
.user
.name
.clone()
.unwrap_or_else(|| participant.user.github_login.clone()),
email,
))
}
}
if !project.is_local() && !project.is_read_only(cx) {
if let Some(user) = room.local_participant_user(cx) {
if let Some(email) = user.email.clone() {
new_co_authors.push((
user.name
.clone()
.unwrap_or_else(|| user.github_login.clone()),
email.clone(),
))
}
}
}
new_co_authors
}
fn toggle_fill_co_authors(
&mut self,
_: &ToggleFillCoAuthors,
_: &mut Window,
cx: &mut Context<Self>,
) {
self.add_coauthors = !self.add_coauthors;
cx.notify();
}
fn fill_co_authors(&mut self, message: &mut String, cx: &mut Context<Self>) {
const CO_AUTHOR_PREFIX: &str = "Co-authored-by: ";
let existing_text = message.to_ascii_lowercase();
let lowercase_co_author_prefix = CO_AUTHOR_PREFIX.to_lowercase(); let lowercase_co_author_prefix = CO_AUTHOR_PREFIX.to_lowercase();
let mut ends_with_co_authors = false; let mut ends_with_co_authors = false;
let existing_co_authors = existing_text let existing_co_authors = existing_text
@ -810,70 +865,32 @@ impl GitPanel {
}) })
.collect::<HashSet<_>>(); .collect::<HashSet<_>>();
let project = self.project.read(cx); let new_co_authors = self
let room = room.read(cx); .potential_co_authors(cx)
let mut new_co_authors = Vec::new(); .into_iter()
.filter(|(_, email)| {
!existing_co_authors
.iter()
.any(|existing| existing.contains(email.as_str()))
})
.collect::<Vec<_>>();
for (peer_id, collaborator) in project.collaborators() {
if collaborator.is_host {
continue;
}
let Some(participant) = room.remote_participant_for_peer_id(*peer_id) else {
continue;
};
if participant.can_write() && participant.user.email.is_some() {
let email = participant.user.email.clone().unwrap();
if !existing_co_authors.contains(&email.as_ref()) {
new_co_authors.push((
participant
.user
.name
.clone()
.unwrap_or_else(|| participant.user.github_login.clone()),
email,
))
}
}
}
if !project.is_local() && !project.is_read_only(cx) {
if let Some(user) = room.local_participant_user(cx) {
if let Some(email) = user.email.clone() {
if !existing_co_authors.contains(&email.as_ref()) {
new_co_authors.push((
user.name
.clone()
.unwrap_or_else(|| user.github_login.clone()),
email.clone(),
))
}
}
}
}
if new_co_authors.is_empty() { if new_co_authors.is_empty() {
return; return;
} }
self.commit_editor.update(cx, |editor, cx| {
let editor_end = editor.buffer().read(cx).read(cx).len();
let mut edit = String::new();
if !ends_with_co_authors { if !ends_with_co_authors {
edit.push('\n'); message.push('\n');
} }
for (name, email) in new_co_authors { for (name, email) in new_co_authors {
edit.push('\n'); message.push('\n');
edit.push_str(CO_AUTHOR_PREFIX); message.push_str(CO_AUTHOR_PREFIX);
edit.push_str(&name); message.push_str(&name);
edit.push_str(" <"); message.push_str(" <");
edit.push_str(&email); message.push_str(&email);
edit.push('>'); message.push('>');
} }
message.push('\n');
editor.edit(Some((editor_end..editor_end, edit)), cx);
editor.move_to_end(&MoveToEnd, window, cx);
editor.focus_handle(cx).focus(window);
});
} }
fn schedule_update( fn schedule_update(
@ -1046,11 +1063,6 @@ impl GitPanel {
cx.notify(); cx.notify();
} }
fn toggle_auto_coauthors(&mut self, cx: &mut Context<Self>) {
self.enable_auto_coauthors = !self.enable_auto_coauthors;
cx.notify();
}
fn header_state(&self, header_type: Section) -> ToggleState { fn header_state(&self, header_type: Section) -> ToggleState {
let (staged_count, count) = match header_type { let (staged_count, count) = match header_type {
Section::New => (self.new_staged_count, self.new_count), Section::New => (self.new_staged_count, self.new_count),
@ -1241,14 +1253,59 @@ impl GitPanel {
cx.listener(move |this, _: &ClickEvent, window, cx| this.commit_changes(window, cx)) cx.listener(move |this, _: &ClickEvent, window, cx| this.commit_changes(window, cx))
}); });
let enable_coauthors = CheckboxWithLabel::new( let potential_co_authors = self.potential_co_authors(cx);
"enable-coauthors", let enable_coauthors = if potential_co_authors.is_empty() {
Label::new("Add Co-authors") None
.color(Color::Disabled) } else {
.size(LabelSize::XSmall), Some(
self.enable_auto_coauthors.into(), IconButton::new("co-authors", IconName::Person)
cx.listener(move |this, _, _, cx| this.toggle_auto_coauthors(cx)), .icon_color(Color::Disabled)
.selected_icon_color(Color::Selected)
.toggle_state(self.add_coauthors)
.tooltip(move |_, cx| {
let title = format!(
"Add co-authored-by:{}{}",
if potential_co_authors.len() == 1 {
""
} else {
"\n"
},
potential_co_authors
.iter()
.map(|(name, email)| format!(" {} <{}>", name, email))
.join("\n")
); );
Tooltip::simple(title, cx)
})
.on_click(cx.listener(|this, _, _, cx| {
this.add_coauthors = !this.add_coauthors;
cx.notify();
})),
)
};
let branch = self
.active_repository
.as_ref()
.and_then(|repo| repo.read(cx).branch().map(|b| b.name.clone()))
.unwrap_or_else(|| "<no branch>".into());
let branch_selector = Button::new("branch-selector", branch)
.color(Color::Muted)
.style(ButtonStyle::Subtle)
.icon(IconName::GitBranch)
.icon_size(IconSize::Small)
.icon_color(Color::Muted)
.size(ButtonSize::Compact)
.icon_position(IconPosition::Start)
.tooltip(Tooltip::for_action_title(
"Switch Branch",
&zed_actions::git::Branch,
))
.on_click(cx.listener(|_, _, window, cx| {
window.dispatch_action(zed_actions::git::Branch.boxed_clone(), cx);
}))
.style(ButtonStyle::Transparent);
let footer_size = px(32.); let footer_size = px(32.);
let gap = px(16.0); let gap = px(16.0);
@ -1274,7 +1331,7 @@ impl GitPanel {
.left_2() .left_2()
.h(footer_size) .h(footer_size)
.flex_none() .flex_none()
.child(enable_coauthors), .child(branch_selector),
) )
.child( .child(
h_flex() h_flex()
@ -1283,6 +1340,7 @@ impl GitPanel {
.right_2() .right_2()
.h(footer_size) .h(footer_size)
.flex_none() .flex_none()
.children(enable_coauthors)
.child(commit_button), .child(commit_button),
) )
} }
@ -1301,32 +1359,6 @@ impl GitPanel {
}) { }) {
return None; return None;
} }
let _branch_selector = Button::new("branch-selector", branch.name.clone())
.color(Color::Muted)
.style(ButtonStyle::Subtle)
.icon(IconName::GitBranch)
.icon_size(IconSize::Small)
.icon_color(Color::Muted)
.size(ButtonSize::Compact)
.icon_position(IconPosition::Start)
.tooltip(Tooltip::for_action_title(
"Switch Branch",
&zed_actions::git::Branch,
))
.on_click(cx.listener(|_, _, window, cx| {
window.dispatch_action(zed_actions::git::Branch.boxed_clone(), cx);
}))
.style(ButtonStyle::Transparent);
let _timestamp = Label::new(time_format::format_local_timestamp(
OffsetDateTime::from_unix_timestamp(commit.commit_timestamp).log_err()?,
OffsetDateTime::now_utc(),
time_format::TimestampFormat::Relative,
))
.size(LabelSize::Small)
.color(Color::Muted);
let tooltip = if self.has_staged_changes() { let tooltip = if self.has_staged_changes() {
"git reset HEAD^ --soft" "git reset HEAD^ --soft"
} else { } else {
@ -1374,13 +1406,6 @@ impl GitPanel {
.icon_position(IconPosition::Start) .icon_position(IconPosition::Start)
.tooltip(Tooltip::for_action_title(tooltip, &git::Uncommit)) .tooltip(Tooltip::for_action_title(tooltip, &git::Uncommit))
.on_click(cx.listener(|this, _, window, cx| this.uncommit(window, cx))), .on_click(cx.listener(|this, _, window, cx| this.uncommit(window, cx))),
// .child(
// panel_filled_button("Push")
// .icon(IconName::ArrowUp)
// .icon_size(IconSize::Small)
// .icon_color(Color::Muted)
// .icon_position(IconPosition::Start), // .disabled(true),
// ),
), ),
) )
} }
@ -1857,7 +1882,7 @@ impl Render for GitPanel {
.on_action(cx.listener(Self::focus_editor)) .on_action(cx.listener(Self::focus_editor))
.on_action(cx.listener(Self::toggle_staged_for_selected)) .on_action(cx.listener(Self::toggle_staged_for_selected))
.when(has_write_access && has_co_authors, |git_panel| { .when(has_write_access && has_co_authors, |git_panel| {
git_panel.on_action(cx.listener(Self::fill_co_authors)) git_panel.on_action(cx.listener(Self::toggle_fill_co_authors))
}) })
// .on_action(cx.listener(|this, &OpenSelected, cx| this.open_selected(&OpenSelected, cx))) // .on_action(cx.listener(|this, &OpenSelected, cx| this.open_selected(&OpenSelected, cx)))
.on_hover(cx.listener(|this, hovered, window, cx| { .on_hover(cx.listener(|this, hovered, window, cx| {