2022-08-30 22:37:54 +00:00
|
|
|
use std::any::Any;
|
|
|
|
|
|
|
|
use gpui::{
|
|
|
|
AnyViewHandle, AnyWeakViewHandle, AppContext, MutableAppContext, Subscription, Task,
|
|
|
|
ViewContext, ViewHandle, WeakViewHandle,
|
|
|
|
};
|
|
|
|
use project::search::SearchQuery;
|
|
|
|
|
2022-11-15 01:31:12 +00:00
|
|
|
use crate::{item::WeakItemHandle, Item, ItemHandle};
|
2022-08-30 22:37:54 +00:00
|
|
|
|
2022-08-31 04:53:49 +00:00
|
|
|
#[derive(Debug)]
|
2022-08-30 22:37:54 +00:00
|
|
|
pub enum SearchEvent {
|
2022-09-01 00:41:53 +00:00
|
|
|
MatchesInvalidated,
|
|
|
|
ActiveMatchChanged,
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
|
|
|
|
2022-09-01 20:45:46 +00:00
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
|
2022-08-30 22:37:54 +00:00
|
|
|
pub enum Direction {
|
|
|
|
Prev,
|
|
|
|
Next,
|
|
|
|
}
|
|
|
|
|
2022-09-01 20:45:46 +00:00
|
|
|
#[derive(Clone, Copy, Debug, Default)]
|
|
|
|
pub struct SearchOptions {
|
|
|
|
pub case: bool,
|
|
|
|
pub word: bool,
|
|
|
|
pub regex: bool,
|
|
|
|
}
|
|
|
|
|
2022-08-30 22:37:54 +00:00
|
|
|
pub trait SearchableItem: Item {
|
2022-08-31 04:53:49 +00:00
|
|
|
type Match: Any + Sync + Send + Clone;
|
|
|
|
|
2022-09-01 20:45:46 +00:00
|
|
|
fn supported_options() -> SearchOptions {
|
|
|
|
SearchOptions {
|
|
|
|
case: true,
|
|
|
|
word: true,
|
|
|
|
regex: true,
|
|
|
|
}
|
|
|
|
}
|
2022-08-30 22:37:54 +00:00
|
|
|
fn to_search_event(event: &Self::Event) -> Option<SearchEvent>;
|
2022-09-01 00:41:53 +00:00
|
|
|
fn clear_matches(&mut self, cx: &mut ViewContext<Self>);
|
|
|
|
fn update_matches(&mut self, matches: Vec<Self::Match>, cx: &mut ViewContext<Self>);
|
2022-08-30 22:37:54 +00:00
|
|
|
fn query_suggestion(&mut self, cx: &mut ViewContext<Self>) -> String;
|
2022-09-01 20:45:46 +00:00
|
|
|
fn activate_match(
|
2022-08-30 22:37:54 +00:00
|
|
|
&mut self,
|
|
|
|
index: usize,
|
2022-08-31 04:53:49 +00:00
|
|
|
matches: Vec<Self::Match>,
|
2022-08-30 22:37:54 +00:00
|
|
|
cx: &mut ViewContext<Self>,
|
|
|
|
);
|
2022-09-01 20:45:46 +00:00
|
|
|
fn match_index_for_direction(
|
2022-08-30 22:37:54 +00:00
|
|
|
&mut self,
|
2022-09-01 20:45:46 +00:00
|
|
|
matches: &Vec<Self::Match>,
|
|
|
|
mut current_index: usize,
|
|
|
|
direction: Direction,
|
|
|
|
_: &mut ViewContext<Self>,
|
|
|
|
) -> usize {
|
|
|
|
match direction {
|
|
|
|
Direction::Prev => {
|
|
|
|
if current_index == 0 {
|
|
|
|
matches.len() - 1
|
|
|
|
} else {
|
|
|
|
current_index - 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Direction::Next => {
|
|
|
|
current_index += 1;
|
|
|
|
if current_index == matches.len() {
|
|
|
|
0
|
|
|
|
} else {
|
|
|
|
current_index
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-09-01 00:41:53 +00:00
|
|
|
fn find_matches(
|
|
|
|
&mut self,
|
|
|
|
query: SearchQuery,
|
|
|
|
cx: &mut ViewContext<Self>,
|
|
|
|
) -> Task<Vec<Self::Match>>;
|
2022-08-30 22:37:54 +00:00
|
|
|
fn active_match_index(
|
|
|
|
&mut self,
|
2022-08-31 04:53:49 +00:00
|
|
|
matches: Vec<Self::Match>,
|
2022-08-30 22:37:54 +00:00
|
|
|
cx: &mut ViewContext<Self>,
|
|
|
|
) -> Option<usize>;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait SearchableItemHandle: ItemHandle {
|
|
|
|
fn downgrade(&self) -> Box<dyn WeakSearchableItemHandle>;
|
|
|
|
fn boxed_clone(&self) -> Box<dyn SearchableItemHandle>;
|
2022-09-01 20:45:46 +00:00
|
|
|
fn supported_options(&self) -> SearchOptions;
|
2022-09-06 21:39:58 +00:00
|
|
|
fn subscribe_to_search_events(
|
2022-08-30 22:37:54 +00:00
|
|
|
&self,
|
|
|
|
cx: &mut MutableAppContext,
|
|
|
|
handler: Box<dyn Fn(SearchEvent, &mut MutableAppContext)>,
|
|
|
|
) -> Subscription;
|
2022-09-01 20:45:46 +00:00
|
|
|
fn clear_matches(&self, cx: &mut MutableAppContext);
|
|
|
|
fn update_matches(&self, matches: &Vec<Box<dyn Any + Send>>, cx: &mut MutableAppContext);
|
2022-08-30 22:37:54 +00:00
|
|
|
fn query_suggestion(&self, cx: &mut MutableAppContext) -> String;
|
2022-09-01 20:45:46 +00:00
|
|
|
fn activate_match(
|
2022-08-30 22:37:54 +00:00
|
|
|
&self,
|
|
|
|
index: usize,
|
|
|
|
matches: &Vec<Box<dyn Any + Send>>,
|
|
|
|
cx: &mut MutableAppContext,
|
|
|
|
);
|
2022-09-01 20:45:46 +00:00
|
|
|
fn match_index_for_direction(
|
2022-08-30 22:37:54 +00:00
|
|
|
&self,
|
|
|
|
matches: &Vec<Box<dyn Any + Send>>,
|
2022-09-01 20:45:46 +00:00
|
|
|
current_index: usize,
|
|
|
|
direction: Direction,
|
2022-08-30 22:37:54 +00:00
|
|
|
cx: &mut MutableAppContext,
|
2022-09-01 20:45:46 +00:00
|
|
|
) -> usize;
|
|
|
|
fn find_matches(
|
2022-08-30 22:37:54 +00:00
|
|
|
&self,
|
|
|
|
query: SearchQuery,
|
|
|
|
cx: &mut MutableAppContext,
|
|
|
|
) -> Task<Vec<Box<dyn Any + Send>>>;
|
|
|
|
fn active_match_index(
|
|
|
|
&self,
|
|
|
|
matches: &Vec<Box<dyn Any + Send>>,
|
|
|
|
cx: &mut MutableAppContext,
|
|
|
|
) -> Option<usize>;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: SearchableItem> SearchableItemHandle for ViewHandle<T> {
|
|
|
|
fn downgrade(&self) -> Box<dyn WeakSearchableItemHandle> {
|
|
|
|
Box::new(self.downgrade())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn boxed_clone(&self) -> Box<dyn SearchableItemHandle> {
|
|
|
|
Box::new(self.clone())
|
|
|
|
}
|
|
|
|
|
2022-09-01 20:45:46 +00:00
|
|
|
fn supported_options(&self) -> SearchOptions {
|
|
|
|
T::supported_options()
|
|
|
|
}
|
|
|
|
|
2022-09-06 21:39:58 +00:00
|
|
|
fn subscribe_to_search_events(
|
2022-08-30 22:37:54 +00:00
|
|
|
&self,
|
|
|
|
cx: &mut MutableAppContext,
|
|
|
|
handler: Box<dyn Fn(SearchEvent, &mut MutableAppContext)>,
|
|
|
|
) -> Subscription {
|
|
|
|
cx.subscribe(self, move |_, event, cx| {
|
|
|
|
if let Some(search_event) = T::to_search_event(event) {
|
|
|
|
handler(search_event, cx)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-09-01 20:45:46 +00:00
|
|
|
fn clear_matches(&self, cx: &mut MutableAppContext) {
|
2022-09-01 00:41:53 +00:00
|
|
|
self.update(cx, |this, cx| this.clear_matches(cx));
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
2022-09-01 20:45:46 +00:00
|
|
|
fn update_matches(&self, matches: &Vec<Box<dyn Any + Send>>, cx: &mut MutableAppContext) {
|
2022-08-31 04:53:49 +00:00
|
|
|
let matches = downcast_matches(matches);
|
2022-09-01 00:41:53 +00:00
|
|
|
self.update(cx, |this, cx| this.update_matches(matches, cx));
|
2022-08-31 04:53:49 +00:00
|
|
|
}
|
2022-08-30 22:37:54 +00:00
|
|
|
fn query_suggestion(&self, cx: &mut MutableAppContext) -> String {
|
|
|
|
self.update(cx, |this, cx| this.query_suggestion(cx))
|
|
|
|
}
|
2022-09-01 20:45:46 +00:00
|
|
|
fn activate_match(
|
2022-08-30 22:37:54 +00:00
|
|
|
&self,
|
|
|
|
index: usize,
|
|
|
|
matches: &Vec<Box<dyn Any + Send>>,
|
|
|
|
cx: &mut MutableAppContext,
|
|
|
|
) {
|
2022-08-31 04:53:49 +00:00
|
|
|
let matches = downcast_matches(matches);
|
2022-09-01 20:45:46 +00:00
|
|
|
self.update(cx, |this, cx| this.activate_match(index, matches, cx));
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
2022-09-01 20:45:46 +00:00
|
|
|
fn match_index_for_direction(
|
2022-08-30 22:37:54 +00:00
|
|
|
&self,
|
|
|
|
matches: &Vec<Box<dyn Any + Send>>,
|
2022-09-01 20:45:46 +00:00
|
|
|
current_index: usize,
|
|
|
|
direction: Direction,
|
2022-08-30 22:37:54 +00:00
|
|
|
cx: &mut MutableAppContext,
|
2022-09-01 20:45:46 +00:00
|
|
|
) -> usize {
|
2022-08-31 04:53:49 +00:00
|
|
|
let matches = downcast_matches(matches);
|
2022-08-30 22:37:54 +00:00
|
|
|
self.update(cx, |this, cx| {
|
2022-09-01 20:45:46 +00:00
|
|
|
this.match_index_for_direction(&matches, current_index, direction, cx)
|
|
|
|
})
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
2022-09-01 20:45:46 +00:00
|
|
|
fn find_matches(
|
2022-08-30 22:37:54 +00:00
|
|
|
&self,
|
|
|
|
query: SearchQuery,
|
|
|
|
cx: &mut MutableAppContext,
|
|
|
|
) -> Task<Vec<Box<dyn Any + Send>>> {
|
2022-09-01 00:41:53 +00:00
|
|
|
let matches = self.update(cx, |this, cx| this.find_matches(query, cx));
|
2022-08-31 04:53:49 +00:00
|
|
|
cx.foreground().spawn(async {
|
|
|
|
let matches = matches.await;
|
|
|
|
matches
|
|
|
|
.into_iter()
|
|
|
|
.map::<Box<dyn Any + Send>, _>(|range| Box::new(range))
|
|
|
|
.collect()
|
|
|
|
})
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
|
|
|
fn active_match_index(
|
|
|
|
&self,
|
|
|
|
matches: &Vec<Box<dyn Any + Send>>,
|
|
|
|
cx: &mut MutableAppContext,
|
|
|
|
) -> Option<usize> {
|
2022-08-31 04:53:49 +00:00
|
|
|
let matches = downcast_matches(matches);
|
2022-08-30 22:37:54 +00:00
|
|
|
self.update(cx, |this, cx| this.active_match_index(matches, cx))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-31 04:53:49 +00:00
|
|
|
fn downcast_matches<T: Any + Clone>(matches: &Vec<Box<dyn Any + Send>>) -> Vec<T> {
|
|
|
|
matches
|
|
|
|
.iter()
|
|
|
|
.map(|range| range.downcast_ref::<T>().cloned())
|
|
|
|
.collect::<Option<Vec<_>>>()
|
|
|
|
.expect(
|
|
|
|
"SearchableItemHandle function called with vec of matches of a different type than expected",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-08-30 22:37:54 +00:00
|
|
|
impl From<Box<dyn SearchableItemHandle>> for AnyViewHandle {
|
|
|
|
fn from(this: Box<dyn SearchableItemHandle>) -> Self {
|
2023-04-02 20:57:06 +00:00
|
|
|
this.as_any().clone()
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<&Box<dyn SearchableItemHandle>> for AnyViewHandle {
|
|
|
|
fn from(this: &Box<dyn SearchableItemHandle>) -> Self {
|
2023-04-02 20:57:06 +00:00
|
|
|
this.as_any().clone()
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq for Box<dyn SearchableItemHandle> {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.id() == other.id() && self.window_id() == other.window_id()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for Box<dyn SearchableItemHandle> {}
|
|
|
|
|
|
|
|
pub trait WeakSearchableItemHandle: WeakItemHandle {
|
|
|
|
fn upgrade(&self, cx: &AppContext) -> Option<Box<dyn SearchableItemHandle>>;
|
|
|
|
|
2023-04-02 21:42:03 +00:00
|
|
|
fn into_any(self) -> AnyWeakViewHandle;
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: SearchableItem> WeakSearchableItemHandle for WeakViewHandle<T> {
|
|
|
|
fn upgrade(&self, cx: &AppContext) -> Option<Box<dyn SearchableItemHandle>> {
|
|
|
|
Some(Box::new(self.upgrade(cx)?))
|
|
|
|
}
|
|
|
|
|
2023-04-02 21:42:03 +00:00
|
|
|
fn into_any(self) -> AnyWeakViewHandle {
|
|
|
|
self.into_any()
|
2022-08-30 22:37:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq for Box<dyn WeakSearchableItemHandle> {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.id() == other.id() && self.window_id() == other.window_id()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for Box<dyn WeakSearchableItemHandle> {}
|
|
|
|
|
|
|
|
impl std::hash::Hash for Box<dyn WeakSearchableItemHandle> {
|
|
|
|
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
|
|
|
|
(self.id(), self.window_id()).hash(state)
|
|
|
|
}
|
|
|
|
}
|