assistant2: Add the ability to collapse chat messages (#11194)

This PR adds the ability to collapse/uncollapse chat messages.

I think the spacing might be a little off with the collapsed
calculations, so we'll need to look into that.


https://github.com/zed-industries/zed/assets/1486634/4009c831-b44e-4b30-85ed-0266cb5b8a26

Release Notes:

- N/A
This commit is contained in:
Marshall Bowers 2024-04-29 19:25:58 -04:00 committed by GitHub
parent ce643e6bef
commit 04cd8dd0f2
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 34 additions and 10 deletions

1
Cargo.lock generated
View file

@ -379,6 +379,7 @@ dependencies = [
"assets", "assets",
"assistant_tooling", "assistant_tooling",
"client", "client",
"collections",
"editor", "editor",
"env_logger", "env_logger",
"feature_flags", "feature_flags",

View file

@ -12,6 +12,7 @@ path = "src/assistant2.rs"
anyhow.workspace = true anyhow.workspace = true
assistant_tooling.workspace = true assistant_tooling.workspace = true
client.workspace = true client.workspace = true
collections.workspace = true
editor.workspace = true editor.workspace = true
feature_flags.workspace = true feature_flags.workspace = true
fs.workspace = true fs.workspace = true

View file

@ -7,6 +7,7 @@ use ::ui::{div, prelude::*, Color, ViewContext};
use anyhow::{Context, Result}; use anyhow::{Context, Result};
use assistant_tooling::{ToolFunctionCall, ToolRegistry}; use assistant_tooling::{ToolFunctionCall, ToolRegistry};
use client::{proto, Client, UserStore}; use client::{proto, Client, UserStore};
use collections::HashMap;
use completion_provider::*; use completion_provider::*;
use editor::Editor; use editor::Editor;
use feature_flags::FeatureFlagAppExt as _; use feature_flags::FeatureFlagAppExt as _;
@ -214,6 +215,7 @@ struct AssistantChat {
composer_editor: View<Editor>, composer_editor: View<Editor>,
user_store: Model<UserStore>, user_store: Model<UserStore>,
next_message_id: MessageId, next_message_id: MessageId,
collapsed_messages: HashMap<MessageId, bool>,
pending_completion: Option<Task<()>>, pending_completion: Option<Task<()>>,
tool_registry: Arc<ToolRegistry>, tool_registry: Arc<ToolRegistry>,
} }
@ -250,6 +252,7 @@ impl AssistantChat {
user_store, user_store,
language_registry, language_registry,
next_message_id: MessageId(0), next_message_id: MessageId(0),
collapsed_messages: HashMap::default(),
pending_completion: None, pending_completion: None,
tool_registry, tool_registry,
} }
@ -496,6 +499,15 @@ impl AssistantChat {
} }
} }
fn is_message_collapsed(&self, id: &MessageId) -> bool {
self.collapsed_messages.get(id).copied().unwrap_or_default()
}
fn toggle_message_collapsed(&mut self, id: MessageId) {
let entry = self.collapsed_messages.entry(id).or_insert(false);
*entry = !*entry;
}
fn render_error( fn render_error(
&self, &self,
error: Option<SharedString>, error: Option<SharedString>,
@ -531,8 +543,13 @@ impl AssistantChat {
*id, *id,
UserOrAssistant::User(self.user_store.read(cx).current_user()), UserOrAssistant::User(self.user_store.read(cx).current_user()),
body.clone().into_any_element(), body.clone().into_any_element(),
false, self.is_message_collapsed(id),
Box::new(|_, _| {}), Box::new(cx.listener({
let id = *id;
move |assistant_chat, _event, _cx| {
assistant_chat.toggle_message_collapsed(id)
}
})),
)) ))
.into_any(), .into_any(),
ChatMessage::Assistant(AssistantMessage { ChatMessage::Assistant(AssistantMessage {
@ -554,8 +571,13 @@ impl AssistantChat {
*id, *id,
UserOrAssistant::Assistant, UserOrAssistant::Assistant,
assistant_body.into_any_element(), assistant_body.into_any_element(),
false, self.is_message_collapsed(id),
Box::new(|_, _| {}), Box::new(cx.listener({
let id = *id;
move |assistant_chat, _event, _cx| {
assistant_chat.toggle_message_collapsed(id)
}
})),
)) ))
// TODO: Should the errors and tool calls get passed into `ChatMessage`? // TODO: Should the errors and tool calls get passed into `ChatMessage`?
.child(self.render_error(error.clone(), ix, cx)) .child(self.render_error(error.clone(), ix, cx))
@ -665,7 +687,7 @@ impl Render for AssistantChat {
} }
} }
#[derive(Debug, Copy, Clone, Eq, PartialEq)] #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Hash, Clone, Copy)]
struct MessageId(usize); struct MessageId(usize);
impl MessageId { impl MessageId {

View file

@ -1,7 +1,7 @@
use std::sync::Arc; use std::sync::Arc;
use client::User; use client::User;
use gpui::AnyElement; use gpui::{AnyElement, ClickEvent};
use ui::{prelude::*, Avatar}; use ui::{prelude::*, Avatar};
use crate::MessageId; use crate::MessageId;
@ -17,7 +17,7 @@ pub struct ChatMessage {
player: UserOrAssistant, player: UserOrAssistant,
message: AnyElement, message: AnyElement,
collapsed: bool, collapsed: bool,
on_collapse: Box<dyn Fn(bool, &mut WindowContext) + 'static>, on_collapse_handle_click: Box<dyn Fn(&ClickEvent, &mut WindowContext) + 'static>,
} }
impl ChatMessage { impl ChatMessage {
@ -26,14 +26,14 @@ impl ChatMessage {
player: UserOrAssistant, player: UserOrAssistant,
message: AnyElement, message: AnyElement,
collapsed: bool, collapsed: bool,
on_collapse: Box<dyn Fn(bool, &mut WindowContext) + 'static>, on_collapse_handle_click: Box<dyn Fn(&ClickEvent, &mut WindowContext) + 'static>,
) -> Self { ) -> Self {
Self { Self {
id, id,
player, player,
message, message,
collapsed, collapsed,
on_collapse, on_collapse_handle_click,
} }
} }
} }
@ -53,7 +53,7 @@ impl RenderOnce for ChatMessage {
.w_1() .w_1()
.mx_2() .mx_2()
.h_full() .h_full()
.on_click(move |_event, cx| (self.on_collapse)(!self.collapsed, cx)) .on_click(self.on_collapse_handle_click)
.child( .child(
div() div()
.w_px() .w_px()