use std::rc::Rc; use editor::Editor; use gpui::{AppContext, FocusHandle, Model, View, WeakModel, WeakView}; use language::Buffer; use ui::{prelude::*, KeyBinding, PopoverMenu, PopoverMenuHandle, Tooltip}; use workspace::Workspace; use crate::context_picker::{ConfirmBehavior, ContextPicker}; use crate::context_store::ContextStore; use crate::thread::Thread; use crate::thread_store::ThreadStore; use crate::ui::ContextPill; use crate::{AssistantPanel, ToggleContextPicker}; pub struct ContextStrip { context_store: Model, context_picker: View, context_picker_menu_handle: PopoverMenuHandle, focus_handle: FocusHandle, suggest_context_kind: SuggestContextKind, workspace: WeakView, } impl ContextStrip { pub fn new( context_store: Model, workspace: WeakView, thread_store: Option>, focus_handle: FocusHandle, context_picker_menu_handle: PopoverMenuHandle, suggest_context_kind: SuggestContextKind, cx: &mut ViewContext, ) -> Self { Self { context_store: context_store.clone(), context_picker: cx.new_view(|cx| { ContextPicker::new( workspace.clone(), thread_store.clone(), context_store.downgrade(), ConfirmBehavior::KeepOpen, cx, ) }), context_picker_menu_handle, focus_handle, suggest_context_kind, workspace, } } fn suggested_context(&self, cx: &ViewContext) -> Option { match self.suggest_context_kind { SuggestContextKind::File => self.suggested_file(cx), SuggestContextKind::Thread => self.suggested_thread(cx), } } fn suggested_file(&self, cx: &ViewContext) -> Option { let workspace = self.workspace.upgrade()?; let active_item = workspace.read(cx).active_item(cx)?; let editor = active_item.to_any().downcast::().ok()?.read(cx); let active_buffer = editor.buffer().read(cx).as_singleton()?; let path = active_buffer.read(cx).file()?.path(); if self.context_store.read(cx).included_file(path).is_some() { return None; } let title = path.to_string_lossy().into_owned().into(); Some(SuggestedContext::File { title, buffer: active_buffer.downgrade(), }) } fn suggested_thread(&self, cx: &ViewContext) -> Option { let workspace = self.workspace.upgrade()?; let active_thread = workspace .read(cx) .panel::(cx)? .read(cx) .active_thread(cx); let weak_active_thread = active_thread.downgrade(); let active_thread = active_thread.read(cx); if self .context_store .read(cx) .included_thread(active_thread.id()) .is_some() { return None; } Some(SuggestedContext::Thread { title: active_thread.summary().unwrap_or("Active Thread".into()), thread: weak_active_thread, }) } } impl Render for ContextStrip { fn render(&mut self, cx: &mut ViewContext) -> impl IntoElement { let context_store = self.context_store.read(cx); let context = context_store.context().clone(); let context_picker = self.context_picker.clone(); let focus_handle = self.focus_handle.clone(); let suggested_context = self.suggested_context(cx); h_flex() .flex_wrap() .gap_1() .child( PopoverMenu::new("context-picker") .menu(move |_cx| Some(context_picker.clone())) .trigger( IconButton::new("add-context", IconName::Plus) .icon_size(IconSize::Small) .style(ui::ButtonStyle::Filled) .tooltip({ let focus_handle = focus_handle.clone(); move |cx| { Tooltip::for_action_in( "Add Context", &ToggleContextPicker, &focus_handle, cx, ) } }), ) .attach(gpui::Corner::TopLeft) .anchor(gpui::Corner::BottomLeft) .offset(gpui::Point { x: px(0.0), y: px(-16.0), }) .with_handle(self.context_picker_menu_handle.clone()), ) .when(context.is_empty() && suggested_context.is_none(), { |parent| { parent.child( h_flex() .ml_1p5() .gap_2() .child( Label::new("Add Context") .size(LabelSize::Small) .color(Color::Muted), ) .opacity(0.5) .children( KeyBinding::for_action_in(&ToggleContextPicker, &focus_handle, cx) .map(|binding| binding.into_any_element()), ), ) } }) .children(context.iter().map(|context| { ContextPill::new(context.clone()).on_remove({ let context = context.clone(); let context_store = self.context_store.clone(); Rc::new(cx.listener(move |_this, _event, cx| { context_store.update(cx, |this, _cx| { this.remove_context(&context.id); }); cx.notify(); })) }) })) .when_some(suggested_context, |el, suggested| { el.child( Button::new("add-suggested-context", suggested.title().clone()) .on_click({ let context_store = self.context_store.clone(); cx.listener(move |_this, _event, cx| { context_store.update(cx, |context_store, cx| { suggested.accept(context_store, cx); }); cx.notify(); }) }) .icon(IconName::Plus) .icon_position(IconPosition::Start) .icon_size(IconSize::XSmall) .icon_color(Color::Muted) .label_size(LabelSize::Small) .style(ButtonStyle::Filled) .tooltip(|cx| { Tooltip::with_meta("Suggested Context", None, "Click to add it", cx) }), ) }) .when(!context.is_empty(), { move |parent| { parent.child( IconButton::new("remove-all-context", IconName::Eraser) .icon_size(IconSize::Small) .tooltip(move |cx| Tooltip::text("Remove All Context", cx)) .on_click({ let context_store = self.context_store.clone(); cx.listener(move |_this, _event, cx| { context_store.update(cx, |this, _cx| this.clear()); cx.notify(); }) }), ) } }) } } pub enum SuggestContextKind { File, Thread, } #[derive(Clone)] pub enum SuggestedContext { File { title: SharedString, buffer: WeakModel, }, Thread { title: SharedString, thread: WeakModel, }, } impl SuggestedContext { pub fn title(&self) -> &SharedString { match self { Self::File { title, .. } => title, Self::Thread { title, .. } => title, } } pub fn accept(&self, context_store: &mut ContextStore, cx: &mut AppContext) { match self { Self::File { buffer, title: _ } => { if let Some(buffer) = buffer.upgrade() { context_store.insert_file(buffer.read(cx)); }; } Self::Thread { thread, title: _ } => { if let Some(thread) = thread.upgrade() { context_store.insert_thread(thread.read(cx)); }; } } } }