Compare commits

...

1 Commits

Author SHA1 Message Date
Thorsten Ball
fe78378631 WIP: Show last error always 2024-08-13 14:59:57 +02:00
3 changed files with 61 additions and 36 deletions

View File

@@ -13,9 +13,9 @@ use crate::{
terminal_inline_assistant::TerminalInlineAssistant,
Assist, ConfirmCommand, Context, ContextEvent, ContextId, ContextStore, CycleMessageRole,
DebugWorkflowSteps, DeployHistory, DeployPromptLibrary, InlineAssist, InlineAssistId,
InlineAssistant, InsertIntoEditor, MessageStatus, ModelSelector, PendingSlashCommand,
PendingSlashCommandStatus, QuoteSelection, RemoteContextMetadata, ResolvedWorkflowStep,
SavedContextMetadata, Split, ToggleFocus, ToggleModelSelector,
InlineAssistant, InsertIntoEditor, MessageId, MessageStatus, ModelSelector,
PendingSlashCommand, PendingSlashCommandStatus, QuoteSelection, RemoteContextMetadata,
ResolvedWorkflowStep, SavedContextMetadata, Split, ToggleFocus, ToggleModelSelector,
};
use crate::{ContextStoreEvent, ShowConfiguration};
use anyhow::{anyhow, Result};
@@ -1711,7 +1711,7 @@ pub struct ContextEditor {
workflow_steps: HashMap<Range<language::Anchor>, WorkflowStep>,
active_workflow_step: Option<ActiveWorkflowStep>,
assistant_panel: WeakView<AssistantPanel>,
error_message: Option<SharedString>,
error_message_handle: Option<(MessageId, PopoverMenuHandle<ErrorPopover>)>,
debug_inspector: Option<ContextInspector>,
show_accept_terms: bool,
}
@@ -1772,7 +1772,7 @@ impl ContextEditor {
workflow_steps: HashMap::default(),
active_workflow_step: None,
assistant_panel,
error_message: None,
error_message_handle: None,
debug_inspector: None,
show_accept_terms: false,
};
@@ -1818,7 +1818,9 @@ impl ContextEditor {
}
if !self.apply_active_workflow_step(cx) {
self.error_message = None;
if let Some((_, handle)) = self.error_message_handle.take() {
handle.hide(cx);
}
self.send_to_model(cx);
cx.notify();
}
@@ -2338,8 +2340,13 @@ impl ContextEditor {
}
}
ContextEvent::Operation(_) => {}
ContextEvent::AssistError(error_message) => {
self.error_message = Some(SharedString::from(error_message.clone()));
ContextEvent::AssistError(message_id) => {
self.error_message_handle = Some((*message_id, PopoverMenuHandle::default()));
cx.defer(|this, cx| {
if let Some((_, handle)) = &mut this.error_message_handle {
handle.show(cx);
}
});
}
}
}
@@ -2951,6 +2958,7 @@ impl ContextEditor {
style: BlockStyle::Sticky,
render: Box::new({
let context = self.context.clone();
let handle = self.error_message_handle.clone();
move |cx| {
let message_id = message.id;
let show_spinner = message.role == Role::Assistant
@@ -3021,6 +3029,17 @@ impl ContextEditor {
cx,
)
});
let handle = if let Some((id, handle)) = handle.clone() {
if id == message.id {
Some(handle)
} else {
None
}
} else {
None
};
h_flex()
.id(("message_header", message_id.as_u64()))
.pl(cx.gutter_dimensions.full_width())
@@ -3039,7 +3058,10 @@ impl ContextEditor {
focus_handle: cx.focus_handle(),
}))
})
.trigger(trigger),
.trigger(trigger)
.when_some(handle, |el, handle| {
el.with_handle(handle)
}),
)
} else {
None

View File

@@ -281,7 +281,7 @@ impl ContextOperation {
#[derive(Debug, Clone)]
pub enum ContextEvent {
AssistError(String),
AssistError(MessageId),
MessagesEdited,
SummaryChanged,
WorkflowStepsRemoved(Vec<Range<language::Anchor>>),
@@ -1704,8 +1704,8 @@ impl Context {
.err()
.map(|error| error.to_string().trim().to_string());
if let Some(error_message) = error_message.as_ref() {
cx.emit(ContextEvent::AssistError(error_message.to_string()));
if error_message.is_some() {
cx.emit(ContextEvent::AssistError(assistant_message_id));
}
this.update_metadata(assistant_message_id, cx, |metadata| {

View File

@@ -1,4 +1,4 @@
use std::{cell::RefCell, rc::Rc};
use std::sync::{Arc, Mutex};
use gpui::{
anchored, deferred, div, point, prelude::FluentBuilder, px, size, AnchorCorner, AnyElement,
@@ -13,7 +13,7 @@ pub trait PopoverTrigger: IntoElement + Clickable + Selectable + 'static {}
impl<T: IntoElement + Clickable + Selectable + 'static> PopoverTrigger for T {}
pub struct PopoverMenuHandle<M>(Rc<RefCell<Option<PopoverMenuHandleState<M>>>>);
pub struct PopoverMenuHandle<M>(Arc<Mutex<Option<PopoverMenuHandleState<M>>>>);
impl<M> Clone for PopoverMenuHandle<M> {
fn clone(&self) -> Self {
@@ -23,33 +23,36 @@ impl<M> Clone for PopoverMenuHandle<M> {
impl<M> Default for PopoverMenuHandle<M> {
fn default() -> Self {
Self(Rc::default())
Self(Arc::default())
}
}
unsafe impl<M: ManagedView> Send for PopoverMenuHandle<M> {}
unsafe impl<M: ManagedView> Sync for PopoverMenuHandle<M> {}
struct PopoverMenuHandleState<M> {
menu_builder: Rc<dyn Fn(&mut WindowContext) -> Option<View<M>>>,
menu: Rc<RefCell<Option<View<M>>>>,
menu_builder: Arc<dyn Fn(&mut WindowContext) -> Option<View<M>>>,
menu: Arc<Mutex<Option<View<M>>>>,
}
impl<M: ManagedView> PopoverMenuHandle<M> {
pub fn show(&self, cx: &mut WindowContext) {
if let Some(state) = self.0.borrow().as_ref() {
if let Some(state) = self.0.lock().unwrap().as_ref() {
show_menu(&state.menu_builder, &state.menu, cx);
}
}
pub fn hide(&self, cx: &mut WindowContext) {
if let Some(state) = self.0.borrow().as_ref() {
if let Some(menu) = state.menu.borrow().as_ref() {
if let Some(state) = self.0.lock().unwrap().as_ref() {
if let Some(menu) = state.menu.lock().unwrap().as_ref() {
menu.update(cx, |_, cx| cx.emit(DismissEvent));
}
}
}
pub fn toggle(&self, cx: &mut WindowContext) {
if let Some(state) = self.0.borrow().as_ref() {
if state.menu.borrow().is_some() {
if let Some(state) = self.0.lock().unwrap().as_ref() {
if state.menu.lock().unwrap().is_some() {
self.hide(cx);
} else {
self.show(cx);
@@ -63,13 +66,13 @@ pub struct PopoverMenu<M: ManagedView> {
child_builder: Option<
Box<
dyn FnOnce(
Rc<RefCell<Option<View<M>>>>,
Option<Rc<dyn Fn(&mut WindowContext) -> Option<View<M>> + 'static>>,
Arc<Mutex<Option<View<M>>>>,
Option<Arc<dyn Fn(&mut WindowContext) -> Option<View<M>> + 'static>>,
) -> AnyElement
+ 'static,
>,
>,
menu_builder: Option<Rc<dyn Fn(&mut WindowContext) -> Option<View<M>> + 'static>>,
menu_builder: Option<Arc<dyn Fn(&mut WindowContext) -> Option<View<M>> + 'static>>,
anchor: AnchorCorner,
attach: Option<AnchorCorner>,
offset: Option<Point<Pixels>>,
@@ -98,7 +101,7 @@ impl<M: ManagedView> PopoverMenu<M> {
}
pub fn menu(mut self, f: impl Fn(&mut WindowContext) -> Option<View<M>> + 'static) -> Self {
self.menu_builder = Some(Rc::new(f));
self.menu_builder = Some(Arc::new(f));
self
}
@@ -109,7 +112,7 @@ impl<M: ManagedView> PopoverMenu<M> {
pub fn trigger<T: PopoverTrigger>(mut self, t: T) -> Self {
self.child_builder = Some(Box::new(|menu, builder| {
let open = menu.borrow().is_some();
let open = menu.lock().unwrap().is_some();
t.selected(open)
.when_some(builder, |el, builder| {
el.on_click(move |_, cx| show_menu(&builder, &menu, cx))
@@ -160,8 +163,8 @@ impl<M: ManagedView> PopoverMenu<M> {
}
fn show_menu<M: ManagedView>(
builder: &Rc<dyn Fn(&mut WindowContext) -> Option<View<M>>>,
menu: &Rc<RefCell<Option<View<M>>>>,
builder: &Arc<dyn Fn(&mut WindowContext) -> Option<View<M>>>,
menu: &Arc<Mutex<Option<View<M>>>>,
cx: &mut WindowContext,
) {
let Some(new_menu) = (builder)(cx) else {
@@ -176,24 +179,24 @@ fn show_menu<M: ManagedView>(
cx.focus(previous_focus_handle);
}
}
*menu2.borrow_mut() = None;
*menu2.lock().unwrap() = None;
cx.refresh();
})
.detach();
cx.focus_view(&new_menu);
*menu.borrow_mut() = Some(new_menu);
*menu.lock().unwrap() = Some(new_menu);
cx.refresh();
}
pub struct PopoverMenuElementState<M> {
menu: Rc<RefCell<Option<View<M>>>>,
menu: Arc<Mutex<Option<View<M>>>>,
child_bounds: Option<Bounds<Pixels>>,
}
impl<M> Clone for PopoverMenuElementState<M> {
fn clone(&self) -> Self {
Self {
menu: Rc::clone(&self.menu),
menu: Arc::clone(&self.menu),
child_bounds: self.child_bounds,
}
}
@@ -202,7 +205,7 @@ impl<M> Clone for PopoverMenuElementState<M> {
impl<M> Default for PopoverMenuElementState<M> {
fn default() -> Self {
Self {
menu: Rc::default(),
menu: Arc::default(),
child_bounds: None,
}
}
@@ -233,7 +236,7 @@ impl<M: ManagedView> Element for PopoverMenu<M> {
let element_state = element_state.unwrap_or_default();
let mut menu_layout_id = None;
let menu_element = element_state.menu.borrow_mut().as_mut().map(|menu| {
let menu_element = element_state.menu.lock().unwrap().as_mut().map(|menu| {
let mut anchored = anchored().snap_to_window().anchor(self.anchor);
if let Some(child_bounds) = element_state.child_bounds {
anchored = anchored.position(
@@ -254,7 +257,7 @@ impl<M: ManagedView> Element for PopoverMenu<M> {
if let Some(trigger_handle) = self.trigger_handle.take() {
if let Some(menu_builder) = self.menu_builder.clone() {
*trigger_handle.0.borrow_mut() = Some(PopoverMenuHandleState {
*trigger_handle.0.lock().unwrap() = Some(PopoverMenuHandleState {
menu_builder,
menu: element_state.menu.clone(),
});