Merge pull request #2186 from zed-industries/better-vim-matching-motion

Better vim matching motion
This commit is contained in:
Kay Simmons 2023-02-17 22:10:28 -08:00 committed by GitHub
commit ac3e8f61ef
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
14 changed files with 303 additions and 191 deletions

View file

@ -77,14 +77,14 @@ use std::{
cmp::{self, Ordering, Reverse}, cmp::{self, Ordering, Reverse},
mem, mem,
num::NonZeroU32, num::NonZeroU32,
ops::{Deref, DerefMut, Range, RangeInclusive}, ops::{Deref, DerefMut, Range},
path::Path, path::Path,
sync::Arc, sync::Arc,
time::{Duration, Instant}, time::{Duration, Instant},
}; };
pub use sum_tree::Bias; pub use sum_tree::Bias;
use theme::{DiagnosticStyle, Theme}; use theme::{DiagnosticStyle, Theme};
use util::{post_inc, ResultExt, TryFutureExt}; use util::{post_inc, ResultExt, TryFutureExt, RangeExt};
use workspace::{ItemNavHistory, ViewId, Workspace, WorkspaceId}; use workspace::{ItemNavHistory, ViewId, Workspace, WorkspaceId};
use crate::git::diff_hunk_to_display; use crate::git::diff_hunk_to_display;
@ -6993,21 +6993,6 @@ pub fn split_words<'a>(text: &'a str) -> impl std::iter::Iterator<Item = &'a str
.flat_map(|word| word.split_inclusive('_')) .flat_map(|word| word.split_inclusive('_'))
} }
trait RangeExt<T> {
fn sorted(&self) -> Range<T>;
fn to_inclusive(&self) -> RangeInclusive<T>;
}
impl<T: Ord + Clone> RangeExt<T> for Range<T> {
fn sorted(&self) -> Self {
cmp::min(&self.start, &self.end).clone()..cmp::max(&self.start, &self.end).clone()
}
fn to_inclusive(&self) -> RangeInclusive<T> {
self.start.clone()..=self.end.clone()
}
}
trait RangeToAnchorExt { trait RangeToAnchorExt {
fn to_anchors(self, snapshot: &MultiBufferSnapshot) -> Range<Anchor>; fn to_anchors(self, snapshot: &MultiBufferSnapshot) -> Range<Anchor>;
} }

View file

@ -385,9 +385,13 @@ impl MultiBuffer {
_ => Default::default(), _ => Default::default(),
}; };
#[allow(clippy::type_complexity)] struct BufferEdit {
let mut buffer_edits: HashMap<usize, Vec<(Range<usize>, Arc<str>, bool, u32)>> = range: Range<usize>,
Default::default(); new_text: Arc<str>,
is_insertion: bool,
original_indent_column: u32,
}
let mut buffer_edits: HashMap<usize, Vec<BufferEdit>> = Default::default();
let mut cursor = snapshot.excerpts.cursor::<usize>(); let mut cursor = snapshot.excerpts.cursor::<usize>();
for (ix, (range, new_text)) in edits.enumerate() { for (ix, (range, new_text)) in edits.enumerate() {
let new_text: Arc<str> = new_text.into(); let new_text: Arc<str> = new_text.into();
@ -422,12 +426,12 @@ impl MultiBuffer {
buffer_edits buffer_edits
.entry(start_excerpt.buffer_id) .entry(start_excerpt.buffer_id)
.or_insert(Vec::new()) .or_insert(Vec::new())
.push(( .push(BufferEdit {
buffer_start..buffer_end, range: buffer_start..buffer_end,
new_text, new_text,
true, is_insertion: true,
original_indent_column, original_indent_column,
)); });
} else { } else {
let start_excerpt_range = buffer_start let start_excerpt_range = buffer_start
..start_excerpt ..start_excerpt
@ -444,21 +448,21 @@ impl MultiBuffer {
buffer_edits buffer_edits
.entry(start_excerpt.buffer_id) .entry(start_excerpt.buffer_id)
.or_insert(Vec::new()) .or_insert(Vec::new())
.push(( .push(BufferEdit {
start_excerpt_range, range: start_excerpt_range,
new_text.clone(), new_text: new_text.clone(),
true, is_insertion: true,
original_indent_column, original_indent_column,
)); });
buffer_edits buffer_edits
.entry(end_excerpt.buffer_id) .entry(end_excerpt.buffer_id)
.or_insert(Vec::new()) .or_insert(Vec::new())
.push(( .push(BufferEdit {
end_excerpt_range, range: end_excerpt_range,
new_text.clone(), new_text: new_text.clone(),
false, is_insertion: false,
original_indent_column, original_indent_column,
)); });
cursor.seek(&range.start, Bias::Right, &()); cursor.seek(&range.start, Bias::Right, &());
cursor.next(&()); cursor.next(&());
@ -469,19 +473,19 @@ impl MultiBuffer {
buffer_edits buffer_edits
.entry(excerpt.buffer_id) .entry(excerpt.buffer_id)
.or_insert(Vec::new()) .or_insert(Vec::new())
.push(( .push(BufferEdit {
excerpt.range.context.to_offset(&excerpt.buffer), range: excerpt.range.context.to_offset(&excerpt.buffer),
new_text.clone(), new_text: new_text.clone(),
false, is_insertion: false,
original_indent_column, original_indent_column,
)); });
cursor.next(&()); cursor.next(&());
} }
} }
} }
for (buffer_id, mut edits) in buffer_edits { for (buffer_id, mut edits) in buffer_edits {
edits.sort_unstable_by_key(|(range, _, _, _)| range.start); edits.sort_unstable_by_key(|edit| edit.range.start);
self.buffers.borrow()[&buffer_id] self.buffers.borrow()[&buffer_id]
.buffer .buffer
.update(cx, |buffer, cx| { .update(cx, |buffer, cx| {
@ -490,14 +494,19 @@ impl MultiBuffer {
let mut original_indent_columns = Vec::new(); let mut original_indent_columns = Vec::new();
let mut deletions = Vec::new(); let mut deletions = Vec::new();
let empty_str: Arc<str> = "".into(); let empty_str: Arc<str> = "".into();
while let Some(( while let Some(BufferEdit {
mut range, mut range,
new_text, new_text,
mut is_insertion, mut is_insertion,
original_indent_column, original_indent_column,
)) = edits.next() }) = edits.next()
{
while let Some(BufferEdit {
range: next_range,
is_insertion: next_is_insertion,
..
}) = edits.peek()
{ {
while let Some((next_range, _, next_is_insertion, _)) = edits.peek() {
if range.end >= next_range.start { if range.end >= next_range.start {
range.end = cmp::max(next_range.end, range.end); range.end = cmp::max(next_range.end, range.end);
is_insertion |= *next_is_insertion; is_insertion |= *next_is_insertion;
@ -2621,6 +2630,9 @@ impl MultiBufferSnapshot {
self.parse_count self.parse_count
} }
/// Returns the smallest enclosing bracket ranges containing the given range or
/// None if no brackets contain range or the range is not contained in a single
/// excerpt
pub fn innermost_enclosing_bracket_ranges<T: ToOffset>( pub fn innermost_enclosing_bracket_ranges<T: ToOffset>(
&self, &self,
range: Range<T>, range: Range<T>,
@ -2648,25 +2660,38 @@ impl MultiBufferSnapshot {
result result
} }
/// Returns enclosinng bracket ranges containing the given range or returns None if the range is /// Returns enclosing bracket ranges containing the given range or returns None if the range is
/// not contained in a single excerpt /// not contained in a single excerpt
pub fn enclosing_bracket_ranges<'a, T: ToOffset>( pub fn enclosing_bracket_ranges<'a, T: ToOffset>(
&'a self, &'a self,
range: Range<T>, range: Range<T>,
) -> Option<impl Iterator<Item = (Range<usize>, Range<usize>)> + 'a> { ) -> Option<impl Iterator<Item = (Range<usize>, Range<usize>)> + 'a> {
let range = range.start.to_offset(self)..range.end.to_offset(self); let range = range.start.to_offset(self)..range.end.to_offset(self);
self.excerpt_containing(range.clone())
.map(|(excerpt, excerpt_offset)| { self.bracket_ranges(range.clone()).map(|range_pairs| {
range_pairs
.filter(move |(open, close)| open.start <= range.start && close.end >= range.end)
})
}
/// Returns bracket range pairs overlapping the given `range` or returns None if the `range` is
/// not contained in a single excerpt
pub fn bracket_ranges<'a, T: ToOffset>(
&'a self,
range: Range<T>,
) -> Option<impl Iterator<Item = (Range<usize>, Range<usize>)> + 'a> {
let range = range.start.to_offset(self)..range.end.to_offset(self);
let excerpt = self.excerpt_containing(range.clone());
excerpt.map(|(excerpt, excerpt_offset)| {
let excerpt_buffer_start = excerpt.range.context.start.to_offset(&excerpt.buffer); let excerpt_buffer_start = excerpt.range.context.start.to_offset(&excerpt.buffer);
let excerpt_buffer_end = excerpt_buffer_start + excerpt.text_summary.len; let excerpt_buffer_end = excerpt_buffer_start + excerpt.text_summary.len;
let start_in_buffer = let start_in_buffer = excerpt_buffer_start + range.start.saturating_sub(excerpt_offset);
excerpt_buffer_start + range.start.saturating_sub(excerpt_offset);
let end_in_buffer = excerpt_buffer_start + range.end.saturating_sub(excerpt_offset); let end_in_buffer = excerpt_buffer_start + range.end.saturating_sub(excerpt_offset);
excerpt excerpt
.buffer .buffer
.enclosing_bracket_ranges(start_in_buffer..end_in_buffer) .bracket_ranges(start_in_buffer..end_in_buffer)
.filter_map(move |(start_bracket_range, end_bracket_range)| { .filter_map(move |(start_bracket_range, end_bracket_range)| {
if start_bracket_range.start < excerpt_buffer_start if start_bracket_range.start < excerpt_buffer_start
|| end_bracket_range.end > excerpt_buffer_end || end_bracket_range.end > excerpt_buffer_end
@ -2939,6 +2964,10 @@ impl MultiBufferSnapshot {
cursor.seek(&range.start, Bias::Right, &()); cursor.seek(&range.start, Bias::Right, &());
let start_excerpt = cursor.item(); let start_excerpt = cursor.item();
if range.start == range.end {
return start_excerpt.map(|excerpt| (excerpt, *cursor.start()));
}
cursor.seek(&range.end, Bias::Right, &()); cursor.seek(&range.end, Bias::Right, &());
let end_excerpt = cursor.item(); let end_excerpt = cursor.item();

View file

@ -62,7 +62,7 @@ impl<'a> EditorLspTestContext<'a> {
params params
.fs .fs
.as_fake() .as_fake()
.insert_tree("/root", json!({ "dir": { file_name: "" }})) .insert_tree("/root", json!({ "dir": { file_name.clone(): "" }}))
.await; .await;
let (window_id, workspace) = cx.add_window(|cx| { let (window_id, workspace) = cx.add_window(|cx| {
@ -107,7 +107,7 @@ impl<'a> EditorLspTestContext<'a> {
}, },
lsp, lsp,
workspace, workspace,
buffer_lsp_url: lsp::Url::from_file_path("/root/dir/file.rs").unwrap(), buffer_lsp_url: lsp::Url::from_file_path(format!("/root/dir/{file_name}")).unwrap(),
} }
} }
@ -122,7 +122,33 @@ impl<'a> EditorLspTestContext<'a> {
..Default::default() ..Default::default()
}, },
Some(tree_sitter_rust::language()), Some(tree_sitter_rust::language()),
); )
.with_queries(LanguageQueries {
indents: Some(Cow::from(indoc! {r#"
[
((where_clause) _ @end)
(field_expression)
(call_expression)
(assignment_expression)
(let_declaration)
(let_chain)
(await_expression)
] @indent
(_ "[" "]" @end) @indent
(_ "<" ">" @end) @indent
(_ "{" "}" @end) @indent
(_ "(" ")" @end) @indent"#})),
brackets: Some(Cow::from(indoc! {r#"
("(" @open ")" @close)
("[" @open "]" @close)
("{" @open "}" @close)
("<" @open ">" @close)
("\"" @open "\"" @close)
(closure_parameters "|" @open "|" @close)"#})),
..Default::default()
})
.expect("Could not parse queries");
Self::new(language, capabilities, cx).await Self::new(language, capabilities, cx).await
} }
@ -148,7 +174,7 @@ impl<'a> EditorLspTestContext<'a> {
("\"" @open "\"" @close)"#})), ("\"" @open "\"" @close)"#})),
..Default::default() ..Default::default()
}) })
.expect("Could not parse brackets"); .expect("Could not parse queries");
Self::new(language, capabilities, cx).await Self::new(language, capabilities, cx).await
} }

View file

@ -41,7 +41,7 @@ pub use text::{Buffer as TextBuffer, BufferSnapshot as TextBufferSnapshot, Opera
use theme::SyntaxTheme; use theme::SyntaxTheme;
#[cfg(any(test, feature = "test-support"))] #[cfg(any(test, feature = "test-support"))]
use util::RandomCharIter; use util::RandomCharIter;
use util::TryFutureExt as _; use util::{RangeExt, TryFutureExt as _};
#[cfg(any(test, feature = "test-support"))] #[cfg(any(test, feature = "test-support"))]
pub use {tree_sitter_rust, tree_sitter_typescript}; pub use {tree_sitter_rust, tree_sitter_typescript};
@ -1389,12 +1389,12 @@ impl Buffer {
.enumerate() .enumerate()
.zip(&edit_operation.as_edit().unwrap().new_text) .zip(&edit_operation.as_edit().unwrap().new_text)
.map(|((ix, (range, _)), new_text)| { .map(|((ix, (range, _)), new_text)| {
let new_text_len = new_text.len(); let new_text_length = new_text.len();
let old_start = range.start.to_point(&before_edit); let old_start = range.start.to_point(&before_edit);
let new_start = (delta + range.start as isize) as usize; let new_start = (delta + range.start as isize) as usize;
delta += new_text_len as isize - (range.end as isize - range.start as isize); delta += new_text_length as isize - (range.end as isize - range.start as isize);
let mut range_of_insertion_to_indent = 0..new_text_len; let mut range_of_insertion_to_indent = 0..new_text_length;
let mut first_line_is_new = false; let mut first_line_is_new = false;
let mut original_indent_column = None; let mut original_indent_column = None;
@ -2358,18 +2358,18 @@ impl BufferSnapshot {
Some(items) Some(items)
} }
pub fn enclosing_bracket_ranges<'a, T: ToOffset>( /// Returns bracket range pairs overlapping or adjacent to `range`
pub fn bracket_ranges<'a, T: ToOffset>(
&'a self, &'a self,
range: Range<T>, range: Range<T>,
) -> impl Iterator<Item = (Range<usize>, Range<usize>)> + 'a { ) -> impl Iterator<Item = (Range<usize>, Range<usize>)> + 'a {
// Find bracket pairs that *inclusively* contain the given range. // Find bracket pairs that *inclusively* contain the given range.
let range = range.start.to_offset(self)..range.end.to_offset(self); let range = range.start.to_offset(self).saturating_sub(1)
..self.len().min(range.end.to_offset(self) + 1);
let mut matches = self.syntax.matches( let mut matches = self.syntax.matches(range.clone(), &self.text, |grammar| {
range.start.saturating_sub(1)..self.len().min(range.end + 1), grammar.brackets_config.as_ref().map(|c| &c.query)
&self.text, });
|grammar| grammar.brackets_config.as_ref().map(|c| &c.query),
);
let configs = matches let configs = matches
.grammars() .grammars()
.iter() .iter()
@ -2393,7 +2393,8 @@ impl BufferSnapshot {
let Some((open, close)) = open.zip(close) else { continue }; let Some((open, close)) = open.zip(close) else { continue };
if open.start > range.start || close.end < range.end { let bracket_range = open.start..=close.end;
if !bracket_range.overlaps(&range) {
continue; continue;
} }

View file

@ -578,7 +578,7 @@ async fn test_symbols_containing(cx: &mut gpui::TestAppContext) {
#[gpui::test] #[gpui::test]
fn test_enclosing_bracket_ranges(cx: &mut MutableAppContext) { fn test_enclosing_bracket_ranges(cx: &mut MutableAppContext) {
let mut assert = |selection_text, range_markers| { let mut assert = |selection_text, range_markers| {
assert_enclosing_bracket_pairs(selection_text, range_markers, rust_lang(), cx) assert_bracket_pairs(selection_text, range_markers, rust_lang(), cx)
}; };
assert( assert(
@ -696,7 +696,7 @@ fn test_enclosing_bracket_ranges_where_brackets_are_not_outermost_children(
cx: &mut MutableAppContext, cx: &mut MutableAppContext,
) { ) {
let mut assert = |selection_text, bracket_pair_texts| { let mut assert = |selection_text, bracket_pair_texts| {
assert_enclosing_bracket_pairs(selection_text, bracket_pair_texts, javascript_lang(), cx) assert_bracket_pairs(selection_text, bracket_pair_texts, javascript_lang(), cx)
}; };
assert( assert(
@ -710,6 +710,7 @@ fn test_enclosing_bracket_ranges_where_brackets_are_not_outermost_children(
}"}], }"}],
); );
eprintln!("-----------------------");
// Regression test: even though the parent node of the parentheses (the for loop) does // Regression test: even though the parent node of the parentheses (the for loop) does
// intersect the given range, the parentheses themselves do not contain the range, so // intersect the given range, the parentheses themselves do not contain the range, so
// they should not be returned. Only the curly braces contain the range. // they should not be returned. Only the curly braces contain the range.
@ -2047,7 +2048,7 @@ fn get_tree_sexp(buffer: &ModelHandle<Buffer>, cx: &gpui::TestAppContext) -> Str
} }
// Assert that the enclosing bracket ranges around the selection match the pairs indicated by the marked text in `range_markers` // Assert that the enclosing bracket ranges around the selection match the pairs indicated by the marked text in `range_markers`
fn assert_enclosing_bracket_pairs( fn assert_bracket_pairs(
selection_text: &'static str, selection_text: &'static str,
bracket_pair_texts: Vec<&'static str>, bracket_pair_texts: Vec<&'static str>,
language: Language, language: Language,
@ -2072,9 +2073,7 @@ fn assert_enclosing_bracket_pairs(
.collect::<Vec<_>>(); .collect::<Vec<_>>();
assert_set_eq!( assert_set_eq!(
buffer buffer.bracket_ranges(selection_range).collect::<Vec<_>>(),
.enclosing_bracket_ranges(selection_range)
.collect::<Vec<_>>(),
bracket_pairs bracket_pairs
); );
} }

View file

@ -5,6 +5,7 @@ edition = "2021"
publish = false publish = false
[lib] [lib]
path = "src/util.rs"
doctest = false doctest = false
[features] [features]
@ -22,7 +23,6 @@ serde_json = { version = "1.0", features = ["preserve_order"], optional = true }
git2 = { version = "0.15", default-features = false, optional = true } git2 = { version = "0.15", default-features = false, optional = true }
dirs = "3.0" dirs = "3.0"
[dev-dependencies] [dev-dependencies]
tempdir = { version = "0.3.7" } tempdir = { version = "0.3.7" }
serde_json = { version = "1.0", features = ["preserve_order"] } serde_json = { version = "1.0", features = ["preserve_order"] }

View file

@ -3,16 +3,17 @@ pub mod paths;
#[cfg(any(test, feature = "test-support"))] #[cfg(any(test, feature = "test-support"))]
pub mod test; pub mod test;
pub use backtrace::Backtrace;
use futures::Future;
use rand::{seq::SliceRandom, Rng};
use std::{ use std::{
cmp::Ordering, cmp::{self, Ordering},
ops::AddAssign, ops::{AddAssign, Range, RangeInclusive},
pin::Pin, pin::Pin,
task::{Context, Poll}, task::{Context, Poll},
}; };
pub use backtrace::Backtrace;
use futures::Future;
use rand::{seq::SliceRandom, Rng};
#[derive(Debug, Default)] #[derive(Debug, Default)]
pub struct StaffMode(pub bool); pub struct StaffMode(pub bool);
@ -245,6 +246,46 @@ macro_rules! async_iife {
}; };
} }
pub trait RangeExt<T> {
fn sorted(&self) -> Self;
fn to_inclusive(&self) -> RangeInclusive<T>;
fn overlaps(&self, other: &Range<T>) -> bool;
}
impl<T: Ord + Clone> RangeExt<T> for Range<T> {
fn sorted(&self) -> Self {
cmp::min(&self.start, &self.end).clone()..cmp::max(&self.start, &self.end).clone()
}
fn to_inclusive(&self) -> RangeInclusive<T> {
self.start.clone()..=self.end.clone()
}
fn overlaps(&self, other: &Range<T>) -> bool {
self.contains(&other.start)
|| self.contains(&other.end)
|| other.contains(&self.start)
|| other.contains(&self.end)
}
}
impl<T: Ord + Clone> RangeExt<T> for RangeInclusive<T> {
fn sorted(&self) -> Self {
cmp::min(self.start(), self.end()).clone()..=cmp::max(self.start(), self.end()).clone()
}
fn to_inclusive(&self) -> RangeInclusive<T> {
self.clone()
}
fn overlaps(&self, other: &Range<T>) -> bool {
self.contains(&other.start)
|| self.contains(&other.end)
|| other.contains(&self.start())
|| other.contains(&self.end())
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;

View file

@ -3,7 +3,7 @@ use std::sync::Arc;
use editor::{ use editor::{
char_kind, char_kind,
display_map::{DisplaySnapshot, ToDisplayPoint}, display_map::{DisplaySnapshot, ToDisplayPoint},
movement, Bias, CharKind, DisplayPoint, movement, Bias, CharKind, DisplayPoint, ToOffset,
}; };
use gpui::{actions, impl_actions, MutableAppContext}; use gpui::{actions, impl_actions, MutableAppContext};
use language::{Point, Selection, SelectionGoal}; use language::{Point, Selection, SelectionGoal};
@ -450,19 +450,53 @@ fn end_of_document(map: &DisplaySnapshot, point: DisplayPoint, line: usize) -> D
map.clip_point(new_point, Bias::Left) map.clip_point(new_point, Bias::Left)
} }
fn matching(map: &DisplaySnapshot, point: DisplayPoint) -> DisplayPoint { fn matching(map: &DisplaySnapshot, display_point: DisplayPoint) -> DisplayPoint {
let offset = point.to_offset(map, Bias::Left); // https://github.com/vim/vim/blob/1d87e11a1ef201b26ed87585fba70182ad0c468a/runtime/doc/motion.txt#L1200
if let Some((open_range, close_range)) = map let point = display_point.to_point(map);
.buffer_snapshot let offset = point.to_offset(&map.buffer_snapshot);
.innermost_enclosing_bracket_ranges(offset..offset)
{ // Ensure the range is contained by the current line.
if open_range.contains(&offset) { let mut line_end = map.next_line_boundary(point).0;
close_range.start.to_display_point(map) if line_end == point {
} else { line_end = map.max_point().to_point(map);
open_range.start.to_display_point(map)
} }
line_end.column = line_end.column.saturating_sub(1);
let line_range = map.prev_line_boundary(point).0..line_end;
let ranges = map.buffer_snapshot.bracket_ranges(line_range.clone());
if let Some(ranges) = ranges {
let line_range = line_range.start.to_offset(&map.buffer_snapshot)
..line_range.end.to_offset(&map.buffer_snapshot);
let mut closest_pair_destination = None;
let mut closest_distance = usize::MAX;
for (open_range, close_range) in ranges {
if open_range.start >= offset && line_range.contains(&open_range.start) {
let distance = open_range.start - offset;
if distance < closest_distance {
closest_pair_destination = Some(close_range.start);
closest_distance = distance;
continue;
}
}
if close_range.start >= offset && line_range.contains(&close_range.start) {
let distance = close_range.start - offset;
if distance < closest_distance {
closest_pair_destination = Some(open_range.start);
closest_distance = distance;
continue;
}
}
continue;
}
closest_pair_destination
.map(|destination| destination.to_display_point(map))
.unwrap_or(display_point)
} else { } else {
point display_point
} }
} }

View file

@ -824,17 +824,34 @@ mod test {
ˇ ˇ
brown fox"}) brown fox"})
.await; .await;
cx.assert(indoc! {"
cx.assert_manual(
indoc! {"
fn test() { fn test() {
println!(ˇ); println!(ˇ);
} }"},
"}) Mode::Normal,
.await; indoc! {"
cx.assert(indoc! {" fn test() {
println!();
ˇ
}"},
Mode::Insert,
);
cx.assert_manual(
indoc! {"
fn test(ˇ) { fn test(ˇ) {
println!(); println!();
}"}) }"},
.await; Mode::Normal,
indoc! {"
fn test() {
ˇ
println!();
}"},
Mode::Insert,
);
} }
#[gpui::test] #[gpui::test]
@ -857,13 +874,15 @@ mod test {
// Our indentation is smarter than vims. So we don't match here // Our indentation is smarter than vims. So we don't match here
cx.assert_manual( cx.assert_manual(
indoc! {" indoc! {"
fn test() fn test() {
println!(ˇ);"}, println!(ˇ);
}"},
Mode::Normal, Mode::Normal,
indoc! {" indoc! {"
fn test() fn test() {
ˇ ˇ
println!();"}, println!();
}"},
Mode::Insert, Mode::Insert,
); );
cx.assert_manual( cx.assert_manual(
@ -994,7 +1013,6 @@ mod test {
#[gpui::test] #[gpui::test]
async fn test_capital_f_and_capital_t(cx: &mut gpui::TestAppContext) { async fn test_capital_f_and_capital_t(cx: &mut gpui::TestAppContext) {
let mut cx = NeovimBackedTestContext::new(cx).await; let mut cx = NeovimBackedTestContext::new(cx).await;
for count in 1..=3 {
let test_case = indoc! {" let test_case = indoc! {"
ˇaaaˇbˇ ˇ ˇbˇbˇ aˇaaˇbaaa ˇaaaˇbˇ ˇ ˇbˇbˇ aˇaaˇbaaa
ˇ ˇbˇaaˇa ˇbˇbˇb ˇ ˇbˇaaˇa ˇbˇbˇb
@ -1002,6 +1020,7 @@ mod test {
ˇb ˇb
"}; "};
for count in 1..=3 {
cx.assert_binding_matches_all([&count.to_string(), "shift-f", "b"], test_case) cx.assert_binding_matches_all([&count.to_string(), "shift-f", "b"], test_case)
.await; .await;
@ -1009,4 +1028,13 @@ mod test {
.await; .await;
} }
} }
#[gpui::test]
async fn test_percent(cx: &mut gpui::TestAppContext) {
let mut cx = NeovimBackedTestContext::new(cx).await.binding(["%"]);
cx.assert_all("ˇconsole.logˇ(ˇvaˇrˇ)ˇ;").await;
cx.assert_all("ˇconsole.logˇ(ˇ'var', ˇ[ˇ1, ˇ2, 3ˇ]ˇ)ˇ;")
.await;
cx.assert_all("let result = curried_funˇ(ˇ)ˇ(ˇ)ˇ;").await;
}
} }

View file

@ -1,33 +1,29 @@
use std::ops::{Deref, DerefMut}; use std::ops::{Deref, DerefMut};
use editor::test::editor_test_context::EditorTestContext; use editor::test::{
use gpui::{json::json, AppContext, ContextHandle, ViewHandle}; editor_lsp_test_context::EditorLspTestContext, editor_test_context::EditorTestContext,
use project::Project; };
use gpui::{AppContext, ContextHandle};
use search::{BufferSearchBar, ProjectSearchBar}; use search::{BufferSearchBar, ProjectSearchBar};
use workspace::{pane, AppState, WorkspaceHandle};
use crate::{state::Operator, *}; use crate::{state::Operator, *};
use super::VimBindingTestContext; use super::VimBindingTestContext;
pub struct VimTestContext<'a> { pub struct VimTestContext<'a> {
cx: EditorTestContext<'a>, cx: EditorLspTestContext<'a>,
workspace: ViewHandle<Workspace>,
} }
impl<'a> VimTestContext<'a> { impl<'a> VimTestContext<'a> {
pub async fn new(cx: &'a mut gpui::TestAppContext, enabled: bool) -> VimTestContext<'a> { pub async fn new(cx: &'a mut gpui::TestAppContext, enabled: bool) -> VimTestContext<'a> {
cx.update(|cx| { cx.update(|cx| {
editor::init(cx);
pane::init(cx);
search::init(cx); search::init(cx);
crate::init(cx); crate::init(cx);
settings::KeymapFileContent::load("keymaps/vim.json", cx).unwrap(); settings::KeymapFileContent::load("keymaps/vim.json", cx).unwrap();
}); });
let params = cx.update(AppState::test); let mut cx = EditorLspTestContext::new_rust(Default::default(), cx).await;
let project = Project::test(params.fs.clone(), [], cx).await;
cx.update(|cx| { cx.update(|cx| {
cx.update_global(|settings: &mut Settings, _| { cx.update_global(|settings: &mut Settings, _| {
@ -35,24 +31,10 @@ impl<'a> VimTestContext<'a> {
}); });
}); });
params let window_id = cx.window_id;
.fs
.as_fake()
.insert_tree("/root", json!({ "dir": { "test.txt": "" } }))
.await;
let (window_id, workspace) = cx.add_window(|cx| {
Workspace::new(
Default::default(),
0,
project.clone(),
|_, _| unimplemented!(),
cx,
)
});
// Setup search toolbars and keypress hook // Setup search toolbars and keypress hook
workspace.update(cx, |workspace, cx| { cx.update_workspace(|workspace, cx| {
observe_keystrokes(window_id, cx); observe_keystrokes(window_id, cx);
workspace.active_pane().update(cx, |pane, cx| { workspace.active_pane().update(cx, |pane, cx| {
pane.toolbar().update(cx, |toolbar, cx| { pane.toolbar().update(cx, |toolbar, cx| {
@ -64,44 +46,14 @@ impl<'a> VimTestContext<'a> {
}); });
}); });
project Self { cx }
.update(cx, |project, cx| {
project.find_or_create_local_worktree("/root", true, cx)
})
.await
.unwrap();
cx.read(|cx| workspace.read(cx).worktree_scans_complete(cx))
.await;
let file = cx.read(|cx| workspace.file_project_paths(cx)[0].clone());
let item = workspace
.update(cx, |workspace, cx| {
workspace.open_path(file, None, true, cx)
})
.await
.expect("Could not open test file");
let editor = cx.update(|cx| {
item.act_as::<Editor>(cx)
.expect("Opened test file wasn't an editor")
});
editor.update(cx, |_, cx| cx.focus_self());
Self {
cx: EditorTestContext {
cx,
window_id,
editor,
},
workspace,
}
} }
pub fn workspace<F, T>(&mut self, read: F) -> T pub fn workspace<F, T>(&mut self, read: F) -> T
where where
F: FnOnce(&Workspace, &AppContext) -> T, F: FnOnce(&Workspace, &AppContext) -> T,
{ {
self.workspace.read_with(self.cx.cx, read) self.cx.workspace.read_with(self.cx.cx.cx, read)
} }
pub fn enable_vim(&mut self) { pub fn enable_vim(&mut self) {

View file

@ -1 +1 @@
[{"Text":"\n"},{"Mode":"Insert"},{"Selection":{"start":[1,0],"end":[1,0]}},{"Mode":"Insert"},{"Text":"The quick\n"},{"Mode":"Insert"},{"Selection":{"start":[1,0],"end":[1,0]}},{"Mode":"Insert"},{"Text":"The quick\n\nbrown fox\njumps over"},{"Mode":"Insert"},{"Selection":{"start":[1,0],"end":[1,0]}},{"Mode":"Insert"},{"Text":"The quick\nbrown fox\n\njumps over"},{"Mode":"Insert"},{"Selection":{"start":[2,0],"end":[2,0]}},{"Mode":"Insert"},{"Text":"The quick\nbrown fox\njumps over\n"},{"Mode":"Insert"},{"Selection":{"start":[3,0],"end":[3,0]}},{"Mode":"Insert"},{"Text":"The quick\n\n\nbrown fox"},{"Mode":"Insert"},{"Selection":{"start":[2,0],"end":[2,0]}},{"Mode":"Insert"},{"Text":"fn test() {\n println!();\n \n}\n"},{"Mode":"Insert"},{"Selection":{"start":[2,4],"end":[2,4]}},{"Mode":"Insert"},{"Text":"fn test() {\n\n println!();\n}"},{"Mode":"Insert"},{"Selection":{"start":[1,0],"end":[1,0]}},{"Mode":"Insert"}] [{"Text":"\n"},{"Mode":"Insert"},{"Selection":{"start":[1,0],"end":[1,0]}},{"Mode":"Insert"},{"Text":"The quick\n"},{"Mode":"Insert"},{"Selection":{"start":[1,0],"end":[1,0]}},{"Mode":"Insert"},{"Text":"The quick\n\nbrown fox\njumps over"},{"Mode":"Insert"},{"Selection":{"start":[1,0],"end":[1,0]}},{"Mode":"Insert"},{"Text":"The quick\nbrown fox\n\njumps over"},{"Mode":"Insert"},{"Selection":{"start":[2,0],"end":[2,0]}},{"Mode":"Insert"},{"Text":"The quick\nbrown fox\njumps over\n"},{"Mode":"Insert"},{"Selection":{"start":[3,0],"end":[3,0]}},{"Mode":"Insert"},{"Text":"The quick\n\n\nbrown fox"},{"Mode":"Insert"},{"Selection":{"start":[2,0],"end":[2,0]}},{"Mode":"Insert"}]

View file

@ -0,0 +1 @@
[{"Text":"console.log(var);"},{"Mode":"Normal"},{"Selection":{"start":[0,15],"end":[0,15]}},{"Mode":"Normal"},{"Text":"console.log(var);"},{"Mode":"Normal"},{"Selection":{"start":[0,15],"end":[0,15]}},{"Mode":"Normal"},{"Text":"console.log(var);"},{"Mode":"Normal"},{"Selection":{"start":[0,11],"end":[0,11]}},{"Mode":"Normal"},{"Text":"console.log(var);"},{"Mode":"Normal"},{"Selection":{"start":[0,11],"end":[0,11]}},{"Mode":"Normal"},{"Text":"console.log(var);"},{"Mode":"Normal"},{"Selection":{"start":[0,11],"end":[0,11]}},{"Mode":"Normal"},{"Text":"console.log(var);"},{"Mode":"Normal"},{"Selection":{"start":[0,16],"end":[0,16]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,28],"end":[0,28]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,28],"end":[0,28]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,27],"end":[0,27]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,27],"end":[0,27]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,19],"end":[0,19]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,19],"end":[0,19]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,19],"end":[0,19]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,11],"end":[0,11]}},{"Mode":"Normal"},{"Text":"console.log('var', [1, 2, 3]);"},{"Mode":"Normal"},{"Selection":{"start":[0,29],"end":[0,29]}},{"Mode":"Normal"},{"Text":"let result = curried_fun()();"},{"Mode":"Normal"},{"Selection":{"start":[0,25],"end":[0,25]}},{"Mode":"Normal"},{"Text":"let result = curried_fun()();"},{"Mode":"Normal"},{"Selection":{"start":[0,24],"end":[0,24]}},{"Mode":"Normal"},{"Text":"let result = curried_fun()();"},{"Mode":"Normal"},{"Selection":{"start":[0,27],"end":[0,27]}},{"Mode":"Normal"},{"Text":"let result = curried_fun()();"},{"Mode":"Normal"},{"Selection":{"start":[0,26],"end":[0,26]}},{"Mode":"Normal"},{"Text":"let result = curried_fun()();"},{"Mode":"Normal"},{"Selection":{"start":[0,28],"end":[0,28]}},{"Mode":"Normal"}]

View file

@ -1330,7 +1330,19 @@ impl Workspace {
focus_item: bool, focus_item: bool,
cx: &mut ViewContext<Self>, cx: &mut ViewContext<Self>,
) -> Task<Result<Box<dyn ItemHandle>, anyhow::Error>> { ) -> Task<Result<Box<dyn ItemHandle>, anyhow::Error>> {
let pane = pane.unwrap_or_else(|| self.active_pane().downgrade()); let pane = pane.unwrap_or_else(|| {
if !self.dock_active() {
self.active_pane().downgrade()
} else {
self.last_active_center_pane.clone().unwrap_or_else(|| {
self.panes
.first()
.expect("There must be an active pane")
.downgrade()
})
}
});
let task = self.load_path(path.into(), cx); let task = self.load_path(path.into(), cx);
cx.spawn(|this, mut cx| async move { cx.spawn(|this, mut cx| async move {
let (project_entry_id, build_item) = task.await?; let (project_entry_id, build_item) = task.await?;
@ -1637,6 +1649,10 @@ impl Workspace {
self.dock.pane() self.dock.pane()
} }
fn dock_active(&self) -> bool {
&self.active_pane == self.dock.pane()
}
fn project_remote_id_changed(&mut self, remote_id: Option<u64>, cx: &mut ViewContext<Self>) { fn project_remote_id_changed(&mut self, remote_id: Option<u64>, cx: &mut ViewContext<Self>) {
if let Some(remote_id) = remote_id { if let Some(remote_id) = remote_id {
self.remote_entity_subscription = self.remote_entity_subscription =