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:
parent
71867096c8
commit
21a1541a70
3 changed files with 148 additions and 123 deletions
2
Cargo.lock
generated
2
Cargo.lock
generated
|
@ -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",
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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| {
|
if !ends_with_co_authors {
|
||||||
let editor_end = editor.buffer().read(cx).read(cx).len();
|
message.push('\n');
|
||||||
let mut edit = String::new();
|
}
|
||||||
if !ends_with_co_authors {
|
for (name, email) in new_co_authors {
|
||||||
edit.push('\n');
|
message.push('\n');
|
||||||
}
|
message.push_str(CO_AUTHOR_PREFIX);
|
||||||
for (name, email) in new_co_authors {
|
message.push_str(&name);
|
||||||
edit.push('\n');
|
message.push_str(" <");
|
||||||
edit.push_str(CO_AUTHOR_PREFIX);
|
message.push_str(&email);
|
||||||
edit.push_str(&name);
|
message.push('>');
|
||||||
edit.push_str(" <");
|
}
|
||||||
edit.push_str(&email);
|
message.push('\n');
|
||||||
edit.push('>');
|
|
||||||
}
|
|
||||||
|
|
||||||
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| {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue