
Similar issue as in https://github.com/zed-industries/zed/pull/26721. Release Notes: - N/A
250 lines
7.2 KiB
Rust
250 lines
7.2 KiB
Rust
use std::sync::Arc;
|
|
|
|
use fuzzy::StringMatchCandidate;
|
|
use gpui::{App, DismissEvent, Entity, FocusHandle, Focusable, Task, WeakEntity};
|
|
use picker::{Picker, PickerDelegate};
|
|
use ui::{prelude::*, ListItem};
|
|
|
|
use crate::context_picker::{ConfirmBehavior, ContextPicker};
|
|
use crate::context_store::{self, ContextStore};
|
|
use crate::thread::ThreadId;
|
|
use crate::thread_store::ThreadStore;
|
|
|
|
pub struct ThreadContextPicker {
|
|
picker: Entity<Picker<ThreadContextPickerDelegate>>,
|
|
}
|
|
|
|
impl ThreadContextPicker {
|
|
pub fn new(
|
|
thread_store: WeakEntity<ThreadStore>,
|
|
context_picker: WeakEntity<ContextPicker>,
|
|
context_store: WeakEntity<context_store::ContextStore>,
|
|
confirm_behavior: ConfirmBehavior,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) -> Self {
|
|
let delegate = ThreadContextPickerDelegate::new(
|
|
thread_store,
|
|
context_picker,
|
|
context_store,
|
|
confirm_behavior,
|
|
);
|
|
let picker = cx.new(|cx| Picker::uniform_list(delegate, window, cx));
|
|
|
|
ThreadContextPicker { picker }
|
|
}
|
|
}
|
|
|
|
impl Focusable for ThreadContextPicker {
|
|
fn focus_handle(&self, cx: &App) -> FocusHandle {
|
|
self.picker.focus_handle(cx)
|
|
}
|
|
}
|
|
|
|
impl Render for ThreadContextPicker {
|
|
fn render(&mut self, _window: &mut Window, _cx: &mut Context<Self>) -> impl IntoElement {
|
|
self.picker.clone()
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct ThreadContextEntry {
|
|
pub id: ThreadId,
|
|
pub summary: SharedString,
|
|
}
|
|
|
|
pub struct ThreadContextPickerDelegate {
|
|
thread_store: WeakEntity<ThreadStore>,
|
|
context_picker: WeakEntity<ContextPicker>,
|
|
context_store: WeakEntity<context_store::ContextStore>,
|
|
confirm_behavior: ConfirmBehavior,
|
|
matches: Vec<ThreadContextEntry>,
|
|
selected_index: usize,
|
|
}
|
|
|
|
impl ThreadContextPickerDelegate {
|
|
pub fn new(
|
|
thread_store: WeakEntity<ThreadStore>,
|
|
context_picker: WeakEntity<ContextPicker>,
|
|
context_store: WeakEntity<context_store::ContextStore>,
|
|
confirm_behavior: ConfirmBehavior,
|
|
) -> Self {
|
|
ThreadContextPickerDelegate {
|
|
thread_store,
|
|
context_picker,
|
|
context_store,
|
|
confirm_behavior,
|
|
matches: Vec::new(),
|
|
selected_index: 0,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl PickerDelegate for ThreadContextPickerDelegate {
|
|
type ListItem = ListItem;
|
|
|
|
fn match_count(&self) -> usize {
|
|
self.matches.len()
|
|
}
|
|
|
|
fn selected_index(&self) -> usize {
|
|
self.selected_index
|
|
}
|
|
|
|
fn set_selected_index(
|
|
&mut self,
|
|
ix: usize,
|
|
_window: &mut Window,
|
|
_cx: &mut Context<Picker<Self>>,
|
|
) {
|
|
self.selected_index = ix;
|
|
}
|
|
|
|
fn placeholder_text(&self, _window: &mut Window, _cx: &mut App) -> Arc<str> {
|
|
"Search threads…".into()
|
|
}
|
|
|
|
fn update_matches(
|
|
&mut self,
|
|
query: String,
|
|
window: &mut Window,
|
|
cx: &mut Context<Picker<Self>>,
|
|
) -> Task<()> {
|
|
let Ok(threads) = self.thread_store.update(cx, |this, _cx| {
|
|
this.threads()
|
|
.into_iter()
|
|
.map(|thread| ThreadContextEntry {
|
|
id: thread.id,
|
|
summary: thread.summary,
|
|
})
|
|
.collect::<Vec<_>>()
|
|
}) else {
|
|
return Task::ready(());
|
|
};
|
|
|
|
let executor = cx.background_executor().clone();
|
|
let search_task = cx.background_spawn(async move {
|
|
if query.is_empty() {
|
|
threads
|
|
} else {
|
|
let candidates = threads
|
|
.iter()
|
|
.enumerate()
|
|
.map(|(id, thread)| StringMatchCandidate::new(id, &thread.summary))
|
|
.collect::<Vec<_>>();
|
|
let matches = fuzzy::match_strings(
|
|
&candidates,
|
|
&query,
|
|
false,
|
|
100,
|
|
&Default::default(),
|
|
executor,
|
|
)
|
|
.await;
|
|
|
|
matches
|
|
.into_iter()
|
|
.map(|mat| threads[mat.candidate_id].clone())
|
|
.collect()
|
|
}
|
|
});
|
|
|
|
cx.spawn_in(window, |this, mut cx| async move {
|
|
let matches = search_task.await;
|
|
this.update(&mut cx, |this, cx| {
|
|
this.delegate.matches = matches;
|
|
this.delegate.selected_index = 0;
|
|
cx.notify();
|
|
})
|
|
.ok();
|
|
})
|
|
}
|
|
|
|
fn confirm(&mut self, _secondary: bool, window: &mut Window, cx: &mut Context<Picker<Self>>) {
|
|
let Some(entry) = self.matches.get(self.selected_index) else {
|
|
return;
|
|
};
|
|
|
|
let Some(thread_store) = self.thread_store.upgrade() else {
|
|
return;
|
|
};
|
|
|
|
let open_thread_task = thread_store.update(cx, |this, cx| this.open_thread(&entry.id, cx));
|
|
|
|
cx.spawn_in(window, |this, mut cx| async move {
|
|
let thread = open_thread_task.await?;
|
|
this.update_in(&mut cx, |this, window, cx| {
|
|
this.delegate
|
|
.context_store
|
|
.update(cx, |context_store, cx| context_store.add_thread(thread, cx))
|
|
.ok();
|
|
|
|
match this.delegate.confirm_behavior {
|
|
ConfirmBehavior::KeepOpen => {}
|
|
ConfirmBehavior::Close => this.delegate.dismissed(window, cx),
|
|
}
|
|
})
|
|
})
|
|
.detach_and_log_err(cx);
|
|
}
|
|
|
|
fn dismissed(&mut self, _window: &mut Window, cx: &mut Context<Picker<Self>>) {
|
|
self.context_picker
|
|
.update(cx, |_, cx| {
|
|
cx.emit(DismissEvent);
|
|
})
|
|
.ok();
|
|
}
|
|
|
|
fn render_match(
|
|
&self,
|
|
ix: usize,
|
|
selected: bool,
|
|
_window: &mut Window,
|
|
cx: &mut Context<Picker<Self>>,
|
|
) -> Option<Self::ListItem> {
|
|
let thread = &self.matches[ix];
|
|
|
|
Some(ListItem::new(ix).inset(true).toggle_state(selected).child(
|
|
render_thread_context_entry(thread, self.context_store.clone(), cx),
|
|
))
|
|
}
|
|
}
|
|
|
|
pub fn render_thread_context_entry(
|
|
thread: &ThreadContextEntry,
|
|
context_store: WeakEntity<ContextStore>,
|
|
cx: &mut App,
|
|
) -> Div {
|
|
let added = context_store.upgrade().map_or(false, |ctx_store| {
|
|
ctx_store.read(cx).includes_thread(&thread.id).is_some()
|
|
});
|
|
|
|
h_flex()
|
|
.gap_1p5()
|
|
.w_full()
|
|
.justify_between()
|
|
.child(
|
|
h_flex()
|
|
.gap_1p5()
|
|
.max_w_72()
|
|
.child(
|
|
Icon::new(IconName::MessageCircle)
|
|
.size(IconSize::XSmall)
|
|
.color(Color::Muted),
|
|
)
|
|
.child(Label::new(thread.summary.clone()).truncate()),
|
|
)
|
|
.when(added, |el| {
|
|
el.child(
|
|
h_flex()
|
|
.gap_1()
|
|
.child(
|
|
Icon::new(IconName::Check)
|
|
.size(IconSize::Small)
|
|
.color(Color::Success),
|
|
)
|
|
.child(Label::new("Added").size(LabelSize::Small)),
|
|
)
|
|
})
|
|
}
|