summaryrefslogtreecommitdiff
path: root/helix-core
diff options
context:
space:
mode:
authorMichael Davis2024-01-15 06:34:38 +0000
committerGitHub2024-01-15 06:34:38 +0000
commiteca3ccff76b9f45dd2b266215764fe7630dee884 (patch)
tree78572b21340458dab26dcf887782488d2d823e3c /helix-core
parent3011df4f35e43f9f7690b236c85ab54f210c8b3a (diff)
Select subtree within injections in :tree-sitter-subtree (#9309)
`:tree-sitter-subtree` could previously only print subtrees of nodes in the root injection layer. We can improve on that by finding the layer that contains the given byte range and printing the subtree within that layer. That gives more useful results when a selection is within an injection layer.
Diffstat (limited to 'helix-core')
-rw-r--r--helix-core/src/syntax.rs43
1 files changed, 43 insertions, 0 deletions
diff --git a/helix-core/src/syntax.rs b/helix-core/src/syntax.rs
index 4e44c486..83bd09b4 100644
--- a/helix-core/src/syntax.rs
+++ b/helix-core/src/syntax.rs
@@ -1338,6 +1338,23 @@ impl Syntax {
result
}
+ pub fn descendant_for_byte_range(&self, start: usize, end: usize) -> Option<Node<'_>> {
+ let mut container_id = self.root;
+
+ for (layer_id, layer) in self.layers.iter() {
+ if layer.depth > self.layers[container_id].depth
+ && layer.contains_byte_range(start, end)
+ {
+ container_id = layer_id;
+ }
+ }
+
+ self.layers[container_id]
+ .tree()
+ .root_node()
+ .descendant_for_byte_range(start, end)
+ }
+
// Commenting
// comment_strings_for_pos
// is_commented
@@ -1434,6 +1451,32 @@ impl LanguageLayer {
self.tree = Some(tree);
Ok(())
}
+
+ /// Whether the layer contains the given byte range.
+ ///
+ /// If the layer has multiple ranges (i.e. combined injections), the
+ /// given range is considered contained if it is within the start and
+ /// end bytes of the first and last ranges **and** if the given range
+ /// starts or ends within any of the layer's ranges.
+ fn contains_byte_range(&self, start: usize, end: usize) -> bool {
+ let layer_start = self
+ .ranges
+ .first()
+ .expect("ranges should not be empty")
+ .start_byte;
+ let layer_end = self
+ .ranges
+ .last()
+ .expect("ranges should not be empty")
+ .end_byte;
+
+ layer_start <= start
+ && layer_end >= end
+ && self.ranges.iter().any(|range| {
+ let byte_range = range.start_byte..range.end_byte;
+ byte_range.contains(&start) || byte_range.contains(&end)
+ })
+ }
}
pub(crate) fn generate_edits(