Ignore history items' paths when matching search queries (#3107)

Follow-up of https://github.com/zed-industries/zed/pull/3059 

Before: 

![image](https://github.com/zed-industries/zed/assets/2690773/4eb2d2d1-1aa3-40b8-b782-bf2bc5f17b43)

After:

![image](https://github.com/zed-industries/zed/assets/2690773/5587d46b-9198-45fe-9372-114a95d4b7d6)

Release Notes:

- N/A
This commit is contained in:
Kirill Bulatov 2023-10-09 22:35:11 +02:00 committed by GitHub
commit 0823a18cff
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 126 additions and 9 deletions

View file

@ -107,13 +107,23 @@ fn matching_history_item_paths(
) -> HashMap<Arc<Path>, PathMatch> { ) -> HashMap<Arc<Path>, PathMatch> {
let history_items_by_worktrees = history_items let history_items_by_worktrees = history_items
.iter() .iter()
.map(|found_path| { .filter_map(|found_path| {
let path = &found_path.project.path;
let candidate = PathMatchCandidate { let candidate = PathMatchCandidate {
path, path: &found_path.project.path,
char_bag: CharBag::from_iter(path.to_string_lossy().to_lowercase().chars()), // Only match history items names, otherwise their paths may match too many queries, producing false positives.
// E.g. `foo` would match both `something/foo/bar.rs` and `something/foo/foo.rs` and if the former is a history item,
// it would be shown first always, despite the latter being a better match.
char_bag: CharBag::from_iter(
found_path
.project
.path
.file_name()?
.to_string_lossy()
.to_lowercase()
.chars(),
),
}; };
(found_path.project.worktree_id, candidate) Some((found_path.project.worktree_id, candidate))
}) })
.fold( .fold(
HashMap::default(), HashMap::default(),
@ -1803,6 +1813,113 @@ mod tests {
}); });
} }
#[gpui::test]
async fn test_history_items_vs_very_good_external_match(
deterministic: Arc<gpui::executor::Deterministic>,
cx: &mut gpui::TestAppContext,
) {
let app_state = init_test(cx);
app_state
.fs
.as_fake()
.insert_tree(
"/src",
json!({
"collab_ui": {
"first.rs": "// First Rust file",
"second.rs": "// Second Rust file",
"third.rs": "// Third Rust file",
"collab_ui.rs": "// Fourth Rust file",
}
}),
)
.await;
let project = Project::test(app_state.fs.clone(), ["/src".as_ref()], cx).await;
let window = cx.add_window(|cx| Workspace::test_new(project, cx));
let workspace = window.root(cx);
// generate some history to select from
open_close_queried_buffer(
"fir",
1,
"first.rs",
window.into(),
&workspace,
&deterministic,
cx,
)
.await;
open_close_queried_buffer(
"sec",
1,
"second.rs",
window.into(),
&workspace,
&deterministic,
cx,
)
.await;
open_close_queried_buffer(
"thi",
1,
"third.rs",
window.into(),
&workspace,
&deterministic,
cx,
)
.await;
open_close_queried_buffer(
"sec",
1,
"second.rs",
window.into(),
&workspace,
&deterministic,
cx,
)
.await;
cx.dispatch_action(window.into(), Toggle);
let query = "collab_ui";
let finder = cx.read(|cx| workspace.read(cx).modal::<FileFinder>().unwrap());
finder
.update(cx, |finder, cx| {
finder.delegate_mut().update_matches(query.to_string(), cx)
})
.await;
finder.read_with(cx, |finder, _| {
let delegate = finder.delegate();
assert!(
delegate.matches.history.is_empty(),
"History items should not math query {query}, they should be matched by name only"
);
let search_entries = delegate
.matches
.search
.iter()
.map(|e| e.path.to_path_buf())
.collect::<Vec<_>>();
assert_eq!(
search_entries.len(),
4,
"All history and the new file should be found after query {query} as search results"
);
assert_eq!(
search_entries,
vec![
PathBuf::from("collab_ui/collab_ui.rs"),
PathBuf::from("collab_ui/third.rs"),
PathBuf::from("collab_ui/first.rs"),
PathBuf::from("collab_ui/second.rs"),
],
"Despite all search results having the same directory name, the most matching one should be on top"
);
});
}
async fn open_close_queried_buffer( async fn open_close_queried_buffer(
input: &str, input: &str,
expected_matches: usize, expected_matches: usize,

View file

@ -441,7 +441,7 @@ mod tests {
score, score,
worktree_id: 0, worktree_id: 0,
positions: Vec::new(), positions: Vec::new(),
path: candidate.path.clone(), path: Arc::from(candidate.path),
path_prefix: "".into(), path_prefix: "".into(),
distance_to_relative_ancestor: usize::MAX, distance_to_relative_ancestor: usize::MAX,
}, },

View file

@ -14,7 +14,7 @@ use crate::{
#[derive(Clone, Debug)] #[derive(Clone, Debug)]
pub struct PathMatchCandidate<'a> { pub struct PathMatchCandidate<'a> {
pub path: &'a Arc<Path>, pub path: &'a Path,
pub char_bag: CharBag, pub char_bag: CharBag,
} }
@ -120,7 +120,7 @@ pub fn match_fixed_path_set(
score, score,
worktree_id, worktree_id,
positions: Vec::new(), positions: Vec::new(),
path: candidate.path.clone(), path: Arc::from(candidate.path),
path_prefix: Arc::from(""), path_prefix: Arc::from(""),
distance_to_relative_ancestor: usize::MAX, distance_to_relative_ancestor: usize::MAX,
}, },
@ -195,7 +195,7 @@ pub async fn match_path_sets<'a, Set: PathMatchCandidateSet<'a>>(
score, score,
worktree_id, worktree_id,
positions: Vec::new(), positions: Vec::new(),
path: candidate.path.clone(), path: Arc::from(candidate.path),
path_prefix: candidate_set.prefix(), path_prefix: candidate_set.prefix(),
distance_to_relative_ancestor: relative_to.as_ref().map_or( distance_to_relative_ancestor: relative_to.as_ref().map_or(
usize::MAX, usize::MAX,