-
-
Notifications
You must be signed in to change notification settings - Fork 2.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Reimplement tree motions in terms of syntax::TreeCursor
This uses the new TreeCursor type from the parent commit to reimplement the tree-sitter motions (`A-p/o/i/n`). Other tree-sitter related features like textobjects are not touched with this change and will need a different, unrelated approach to solve.
- Loading branch information
1 parent
c009448
commit 8b5e751
Showing
3 changed files
with
103 additions
and
50 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,76 +1,74 @@ | ||
use crate::{Range, RopeSlice, Selection, Syntax}; | ||
use tree_sitter::Node; | ||
use crate::{syntax::TreeCursor, Range, RopeSlice, Selection, Syntax}; | ||
|
||
pub fn expand_selection(syntax: &Syntax, text: RopeSlice, selection: Selection) -> Selection { | ||
select_node_impl(syntax, text, selection, |mut node, from, to| { | ||
while node.start_byte() == from && node.end_byte() == to { | ||
node = node.parent()?; | ||
let cursor = &mut syntax.walk(); | ||
|
||
selection.transform(|range| { | ||
let from = text.char_to_byte(range.from()); | ||
let to = text.char_to_byte(range.to()); | ||
|
||
let mut byte_range = from..to; | ||
cursor.reset_to_byte_range(from, to); | ||
|
||
loop { | ||
if cursor.node().byte_range() != byte_range { | ||
break; | ||
} | ||
byte_range = cursor.node().byte_range(); | ||
if !cursor.goto_parent() { | ||
break; | ||
} | ||
} | ||
Some(node) | ||
|
||
let node = cursor.node(); | ||
let from = text.byte_to_char(node.start_byte()); | ||
let to = text.byte_to_char(node.end_byte()); | ||
|
||
Range::new(to, from).with_direction(range.direction()) | ||
}) | ||
} | ||
|
||
pub fn shrink_selection(syntax: &Syntax, text: RopeSlice, selection: Selection) -> Selection { | ||
select_node_impl(syntax, text, selection, |descendant, _from, _to| { | ||
descendant.child(0).or(Some(descendant)) | ||
select_node_impl(syntax, text, selection, |cursor| { | ||
cursor.goto_first_child(); | ||
}) | ||
} | ||
|
||
pub fn select_sibling<F>( | ||
syntax: &Syntax, | ||
text: RopeSlice, | ||
selection: Selection, | ||
sibling_fn: &F, | ||
) -> Selection | ||
where | ||
F: Fn(Node) -> Option<Node>, | ||
{ | ||
select_node_impl(syntax, text, selection, |descendant, _from, _to| { | ||
find_sibling_recursive(descendant, sibling_fn) | ||
pub fn select_next_sibling(syntax: &Syntax, text: RopeSlice, selection: Selection) -> Selection { | ||
select_node_impl(syntax, text, selection, |cursor| { | ||
let _ = cursor.goto_next_sibling() || cursor.goto_parent() && cursor.goto_next_sibling(); | ||
}) | ||
} | ||
|
||
fn find_sibling_recursive<F>(node: Node, sibling_fn: F) -> Option<Node> | ||
where | ||
F: Fn(Node) -> Option<Node>, | ||
{ | ||
sibling_fn(node).or_else(|| { | ||
node.parent() | ||
.and_then(|node| find_sibling_recursive(node, sibling_fn)) | ||
pub fn select_prev_sibling(syntax: &Syntax, text: RopeSlice, selection: Selection) -> Selection { | ||
select_node_impl(syntax, text, selection, |cursor| { | ||
let _ = cursor.goto_prev_sibling() || cursor.goto_parent() && cursor.goto_prev_sibling(); | ||
}) | ||
} | ||
|
||
fn select_node_impl<F>( | ||
syntax: &Syntax, | ||
text: RopeSlice, | ||
selection: Selection, | ||
select_fn: F, | ||
motion: F, | ||
) -> Selection | ||
where | ||
F: Fn(Node, usize, usize) -> Option<Node>, | ||
F: Fn(&mut TreeCursor), | ||
{ | ||
let tree = syntax.tree(); | ||
let cursor = &mut syntax.walk(); | ||
|
||
selection.transform(|range| { | ||
let from = text.char_to_byte(range.from()); | ||
let to = text.char_to_byte(range.to()); | ||
|
||
let node = match tree | ||
.root_node() | ||
.descendant_for_byte_range(from, to) | ||
.and_then(|node| select_fn(node, from, to)) | ||
{ | ||
Some(node) => node, | ||
None => return range, | ||
}; | ||
cursor.reset_to_byte_range(from, to); | ||
|
||
motion(cursor); | ||
|
||
let node = cursor.node(); | ||
let from = text.byte_to_char(node.start_byte()); | ||
let to = text.byte_to_char(node.end_byte()); | ||
|
||
if range.head < range.anchor { | ||
Range::new(to, from) | ||
} else { | ||
Range::new(from, to) | ||
} | ||
Range::new(to, from).with_direction(range.direction()) | ||
}) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters