diff --git a/crates/collab_ui/src/collaborator_list_popover.rs b/crates/collab_ui/src/collaborator_list_popover.rs index cdc053d83f..e6bebf861b 100644 --- a/crates/collab_ui/src/collaborator_list_popover.rs +++ b/crates/collab_ui/src/collaborator_list_popover.rs @@ -1,6 +1,9 @@ use call::ActiveCall; use client::UserStore; -use gpui::{elements::*, Entity, ModelHandle, MouseButton, RenderContext, View, ViewContext}; +use gpui::Action; +use gpui::{ + actions, elements::*, Entity, ModelHandle, MouseButton, RenderContext, View, ViewContext, +}; use settings::Settings; use crate::collab_titlebar_item::ToggleCollaboratorList; @@ -9,6 +12,13 @@ pub(crate) enum Event { Dismissed, } +enum Collaborator { + SelfUser { username: String }, + RemoteUser { username: String }, +} + +actions!(collaborator_list_popover, [NoOp]); + pub(crate) struct CollaboratorListPopover { list_state: ListState, } @@ -52,14 +62,18 @@ impl CollaboratorListPopover { let mut collaborators = user_store .read(cx) .current_user() - .map(|u| u.github_login.clone()) + .map(|u| Collaborator::SelfUser { + username: u.github_login.clone(), + }) .into_iter() .collect::>(); //TODO: What should the canonical sort here look like, consult contacts list implementation if let Some(room) = active_call.read(cx).room() { for participant in room.read(cx).remote_participants() { - collaborators.push(participant.1.user.github_login.clone()); + collaborators.push(Collaborator::RemoteUser { + username: participant.1.user.github_login.clone(), + }); } } @@ -69,15 +83,83 @@ impl CollaboratorListPopover { Orientation::Top, 0., cx, - move |_, index, cx| { - let theme = &cx.global::().theme; - Label::new( - collaborators[index].clone(), - theme.contact_list.contact_username.text.clone(), - ) - .boxed() + move |_, index, cx| match &collaborators[index] { + Collaborator::SelfUser { username } => render_collaborator_list_entry( + index, + username, + None::, + None, + Svg::new("icons/chevron_right_12.svg"), + NoOp, + "Leave call".to_owned(), + cx, + ), + + Collaborator::RemoteUser { username } => render_collaborator_list_entry( + index, + username, + Some(NoOp), + Some(format!("Follow {username}")), + Svg::new("icons/x_mark_12.svg"), + NoOp, + format!("Remove {username} from call"), + cx, + ), }, ), } } } + +fn render_collaborator_list_entry( + index: usize, + username: &str, + username_action: Option, + username_tooltip: Option, + icon: Svg, + icon_action: IA, + icon_tooltip: String, + cx: &mut RenderContext, +) -> ElementBox { + enum Username {} + enum UsernameTooltip {} + enum Icon {} + enum IconTooltip {} + + let theme = &cx.global::().theme; + let username_theme = theme.contact_list.contact_username.text.clone(); + let tooltip_theme = theme.tooltip.clone(); + + let username = MouseEventHandler::::new(index, cx, |_, _| { + Label::new(username.to_owned(), username_theme.clone()).boxed() + }) + .on_click(MouseButton::Left, move |_, cx| { + if let Some(username_action) = username_action.clone() { + cx.dispatch_action(username_action); + } + }); + + Flex::row() + .with_child(if let Some(username_tooltip) = username_tooltip { + username + .with_tooltip::( + index, + username_tooltip, + None, + tooltip_theme.clone(), + cx, + ) + .boxed() + } else { + username.boxed() + }) + .with_child( + MouseEventHandler::::new(index, cx, |_, _| icon.boxed()) + .on_click(MouseButton::Left, move |_, cx| { + cx.dispatch_action(icon_action.clone()) + }) + .with_tooltip::(index, icon_tooltip, None, tooltip_theme, cx) + .boxed(), + ) + .boxed() +}