7975: Provide regions in file structure r=ivan770 a=ivan770

Closes #7913 

https://user-images.githubusercontent.com/14003886/110819163-96b3c080-8296-11eb-993e-a7cdb574a12d.mp4



Co-authored-by: ivan770 <leshenko.ivan770@gmail.com>
This commit is contained in:
bors[bot] 2021-03-15 09:26:58 +00:00 committed by GitHub
commit cec676d082
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 207 additions and 50 deletions

View File

@ -1,7 +1,7 @@
use hir::Semantics; use hir::Semantics;
use ide_db::{ use ide_db::{
base_db::{FileId, FilePosition, FileRange, SourceDatabase}, base_db::{FileId, FilePosition, FileRange, SourceDatabase},
RootDatabase, SymbolKind, RootDatabase, StructureNodeKind, SymbolKind,
}; };
use syntax::TextRange; use syntax::TextRange;
@ -80,15 +80,17 @@ pub(crate) fn annotations(
.filter(|node| { .filter(|node| {
matches!( matches!(
node.kind, node.kind,
SymbolKind::Trait StructureNodeKind::SymbolKind(SymbolKind::Trait)
| SymbolKind::Struct | StructureNodeKind::SymbolKind(SymbolKind::Struct)
| SymbolKind::Enum | StructureNodeKind::SymbolKind(SymbolKind::Enum)
| SymbolKind::Union | StructureNodeKind::SymbolKind(SymbolKind::Union)
| SymbolKind::Const | StructureNodeKind::SymbolKind(SymbolKind::Const)
) )
}) })
.for_each(|node| { .for_each(|node| {
if config.annotate_impls && node.kind != SymbolKind::Const { if config.annotate_impls
&& node.kind != StructureNodeKind::SymbolKind(SymbolKind::Const)
{
annotations.push(Annotation { annotations.push(Annotation {
range: node.node_range, range: node.node_range,
kind: AnnotationKind::HasImpls { kind: AnnotationKind::HasImpls {

View File

@ -1,7 +1,8 @@
use ide_db::SymbolKind; use ide_db::{StructureNodeKind, SymbolKind};
use syntax::{ use syntax::{
ast::{self, AttrsOwner, GenericParamsOwner, NameOwner}, ast::{self, AttrsOwner, GenericParamsOwner, NameOwner},
match_ast, AstNode, SourceFile, SyntaxNode, TextRange, WalkEvent, match_ast, AstNode, AstToken, NodeOrToken, SourceFile, SyntaxNode, SyntaxToken, TextRange,
WalkEvent,
}; };
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
@ -10,7 +11,7 @@ pub struct StructureNode {
pub label: String, pub label: String,
pub navigation_range: TextRange, pub navigation_range: TextRange,
pub node_range: TextRange, pub node_range: TextRange,
pub kind: SymbolKind, pub kind: StructureNodeKind,
pub detail: Option<String>, pub detail: Option<String>,
pub deprecated: bool, pub deprecated: bool,
} }
@ -32,34 +33,46 @@ pub(crate) fn file_structure(file: &SourceFile) -> Vec<StructureNode> {
let mut res = Vec::new(); let mut res = Vec::new();
let mut stack = Vec::new(); let mut stack = Vec::new();
for event in file.syntax().preorder() { for event in file.syntax().preorder_with_tokens() {
match event { match event {
WalkEvent::Enter(node) => { WalkEvent::Enter(NodeOrToken::Node(node)) => {
if let Some(mut symbol) = structure_node(&node) { if let Some(mut symbol) = structure_node(&node) {
symbol.parent = stack.last().copied(); symbol.parent = stack.last().copied();
stack.push(res.len()); stack.push(res.len());
res.push(symbol); res.push(symbol);
} }
} }
WalkEvent::Leave(node) => { WalkEvent::Leave(NodeOrToken::Node(node)) => {
if structure_node(&node).is_some() { if structure_node(&node).is_some() {
stack.pop().unwrap(); stack.pop().unwrap();
} }
} }
WalkEvent::Enter(NodeOrToken::Token(token)) => {
if let Some(mut symbol) = structure_token(token) {
symbol.parent = stack.last().copied();
stack.push(res.len());
res.push(symbol);
}
}
WalkEvent::Leave(NodeOrToken::Token(token)) => {
if structure_token(token).is_some() {
stack.pop().unwrap();
}
}
} }
} }
res res
} }
fn structure_node(node: &SyntaxNode) -> Option<StructureNode> { fn structure_node(node: &SyntaxNode) -> Option<StructureNode> {
fn decl<N: NameOwner + AttrsOwner>(node: N, kind: SymbolKind) -> Option<StructureNode> { fn decl<N: NameOwner + AttrsOwner>(node: N, kind: StructureNodeKind) -> Option<StructureNode> {
decl_with_detail(&node, None, kind) decl_with_detail(&node, None, kind)
} }
fn decl_with_type_ref<N: NameOwner + AttrsOwner>( fn decl_with_type_ref<N: NameOwner + AttrsOwner>(
node: &N, node: &N,
type_ref: Option<ast::Type>, type_ref: Option<ast::Type>,
kind: SymbolKind, kind: StructureNodeKind,
) -> Option<StructureNode> { ) -> Option<StructureNode> {
let detail = type_ref.map(|type_ref| { let detail = type_ref.map(|type_ref| {
let mut detail = String::new(); let mut detail = String::new();
@ -72,7 +85,7 @@ fn structure_node(node: &SyntaxNode) -> Option<StructureNode> {
fn decl_with_detail<N: NameOwner + AttrsOwner>( fn decl_with_detail<N: NameOwner + AttrsOwner>(
node: &N, node: &N,
detail: Option<String>, detail: Option<String>,
kind: SymbolKind, kind: StructureNodeKind,
) -> Option<StructureNode> { ) -> Option<StructureNode> {
let name = node.name()?; let name = node.name()?;
@ -120,18 +133,18 @@ fn structure_node(node: &SyntaxNode) -> Option<StructureNode> {
collapse_ws(ret_type.syntax(), &mut detail); collapse_ws(ret_type.syntax(), &mut detail);
} }
decl_with_detail(&it, Some(detail), SymbolKind::Function) decl_with_detail(&it, Some(detail), StructureNodeKind::SymbolKind(SymbolKind::Function))
}, },
ast::Struct(it) => decl(it, SymbolKind::Struct), ast::Struct(it) => decl(it, StructureNodeKind::SymbolKind(SymbolKind::Struct)),
ast::Union(it) => decl(it, SymbolKind::Union), ast::Union(it) => decl(it, StructureNodeKind::SymbolKind(SymbolKind::Union)),
ast::Enum(it) => decl(it, SymbolKind::Enum), ast::Enum(it) => decl(it, StructureNodeKind::SymbolKind(SymbolKind::Enum)),
ast::Variant(it) => decl(it, SymbolKind::Variant), ast::Variant(it) => decl(it, StructureNodeKind::SymbolKind(SymbolKind::Variant)),
ast::Trait(it) => decl(it, SymbolKind::Trait), ast::Trait(it) => decl(it, StructureNodeKind::SymbolKind(SymbolKind::Trait)),
ast::Module(it) => decl(it, SymbolKind::Module), ast::Module(it) => decl(it, StructureNodeKind::SymbolKind(SymbolKind::Module)),
ast::TypeAlias(it) => decl_with_type_ref(&it, it.ty(), SymbolKind::TypeAlias), ast::TypeAlias(it) => decl_with_type_ref(&it, it.ty(), StructureNodeKind::SymbolKind(SymbolKind::TypeAlias)),
ast::RecordField(it) => decl_with_type_ref(&it, it.ty(), SymbolKind::Field), ast::RecordField(it) => decl_with_type_ref(&it, it.ty(), StructureNodeKind::SymbolKind(SymbolKind::Field)),
ast::Const(it) => decl_with_type_ref(&it, it.ty(), SymbolKind::Const), ast::Const(it) => decl_with_type_ref(&it, it.ty(), StructureNodeKind::SymbolKind(SymbolKind::Const)),
ast::Static(it) => decl_with_type_ref(&it, it.ty(), SymbolKind::Static), ast::Static(it) => decl_with_type_ref(&it, it.ty(), StructureNodeKind::SymbolKind(SymbolKind::Static)),
ast::Impl(it) => { ast::Impl(it) => {
let target_type = it.self_ty()?; let target_type = it.self_ty()?;
let target_trait = it.trait_(); let target_trait = it.trait_();
@ -147,18 +160,38 @@ fn structure_node(node: &SyntaxNode) -> Option<StructureNode> {
label, label,
navigation_range: target_type.syntax().text_range(), navigation_range: target_type.syntax().text_range(),
node_range: it.syntax().text_range(), node_range: it.syntax().text_range(),
kind: SymbolKind::Impl, kind: StructureNodeKind::SymbolKind(SymbolKind::Impl),
detail: None, detail: None,
deprecated: false, deprecated: false,
}; };
Some(node) Some(node)
}, },
ast::MacroRules(it) => decl(it, SymbolKind::Macro), ast::MacroRules(it) => decl(it, StructureNodeKind::SymbolKind(SymbolKind::Macro)),
_ => None, _ => None,
} }
} }
} }
fn structure_token(token: SyntaxToken) -> Option<StructureNode> {
if let Some(comment) = ast::Comment::cast(token) {
let text = comment.text().trim();
if let Some(region_name) = text.strip_prefix("// region:").map(str::trim) {
return Some(StructureNode {
parent: None,
label: region_name.to_string(),
navigation_range: comment.syntax().text_range(),
node_range: comment.syntax().text_range(),
kind: StructureNodeKind::Region,
detail: None,
deprecated: false,
});
}
}
None
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use expect_test::{expect, Expect}; use expect_test::{expect, Expect};
@ -217,6 +250,16 @@ fn obsolete() {}
#[deprecated(note = "for awhile")] #[deprecated(note = "for awhile")]
fn very_obsolete() {} fn very_obsolete() {}
// region: Some region name
// endregion
// region: dontpanic
mod m {
fn f() {}
// endregion
fn g() {}
}
"#, "#,
expect![[r#" expect![[r#"
[ [
@ -225,7 +268,9 @@ fn very_obsolete() {}
label: "Foo", label: "Foo",
navigation_range: 8..11, navigation_range: 8..11,
node_range: 1..26, node_range: 1..26,
kind: Struct, kind: SymbolKind(
Struct,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -236,7 +281,9 @@ fn very_obsolete() {}
label: "x", label: "x",
navigation_range: 18..19, navigation_range: 18..19,
node_range: 18..24, node_range: 18..24,
kind: Field, kind: SymbolKind(
Field,
),
detail: Some( detail: Some(
"i32", "i32",
), ),
@ -247,7 +294,9 @@ fn very_obsolete() {}
label: "m", label: "m",
navigation_range: 32..33, navigation_range: 32..33,
node_range: 28..158, node_range: 28..158,
kind: Module, kind: SymbolKind(
Module,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -258,7 +307,9 @@ fn very_obsolete() {}
label: "bar1", label: "bar1",
navigation_range: 43..47, navigation_range: 43..47,
node_range: 40..52, node_range: 40..52,
kind: Function, kind: SymbolKind(
Function,
),
detail: Some( detail: Some(
"fn()", "fn()",
), ),
@ -271,7 +322,9 @@ fn very_obsolete() {}
label: "bar2", label: "bar2",
navigation_range: 60..64, navigation_range: 60..64,
node_range: 57..81, node_range: 57..81,
kind: Function, kind: SymbolKind(
Function,
),
detail: Some( detail: Some(
"fn<T>(t: T) -> T", "fn<T>(t: T) -> T",
), ),
@ -284,7 +337,9 @@ fn very_obsolete() {}
label: "bar3", label: "bar3",
navigation_range: 89..93, navigation_range: 89..93,
node_range: 86..156, node_range: 86..156,
kind: Function, kind: SymbolKind(
Function,
),
detail: Some( detail: Some(
"fn<A, B>(a: A, b: B) -> Vec< u32 >", "fn<A, B>(a: A, b: B) -> Vec< u32 >",
), ),
@ -295,7 +350,9 @@ fn very_obsolete() {}
label: "E", label: "E",
navigation_range: 165..166, navigation_range: 165..166,
node_range: 160..180, node_range: 160..180,
kind: Enum, kind: SymbolKind(
Enum,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -306,7 +363,9 @@ fn very_obsolete() {}
label: "X", label: "X",
navigation_range: 169..170, navigation_range: 169..170,
node_range: 169..170, node_range: 169..170,
kind: Variant, kind: SymbolKind(
Variant,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -317,7 +376,9 @@ fn very_obsolete() {}
label: "Y", label: "Y",
navigation_range: 172..173, navigation_range: 172..173,
node_range: 172..178, node_range: 172..178,
kind: Variant, kind: SymbolKind(
Variant,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -326,7 +387,9 @@ fn very_obsolete() {}
label: "T", label: "T",
navigation_range: 186..187, navigation_range: 186..187,
node_range: 181..193, node_range: 181..193,
kind: TypeAlias, kind: SymbolKind(
TypeAlias,
),
detail: Some( detail: Some(
"()", "()",
), ),
@ -337,7 +400,9 @@ fn very_obsolete() {}
label: "S", label: "S",
navigation_range: 201..202, navigation_range: 201..202,
node_range: 194..213, node_range: 194..213,
kind: Static, kind: SymbolKind(
Static,
),
detail: Some( detail: Some(
"i32", "i32",
), ),
@ -348,7 +413,9 @@ fn very_obsolete() {}
label: "C", label: "C",
navigation_range: 220..221, navigation_range: 220..221,
node_range: 214..232, node_range: 214..232,
kind: Const, kind: SymbolKind(
Const,
),
detail: Some( detail: Some(
"i32", "i32",
), ),
@ -359,7 +426,9 @@ fn very_obsolete() {}
label: "impl E", label: "impl E",
navigation_range: 239..240, navigation_range: 239..240,
node_range: 234..243, node_range: 234..243,
kind: Impl, kind: SymbolKind(
Impl,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -368,7 +437,9 @@ fn very_obsolete() {}
label: "impl fmt::Debug for E", label: "impl fmt::Debug for E",
navigation_range: 265..266, navigation_range: 265..266,
node_range: 245..269, node_range: 245..269,
kind: Impl, kind: SymbolKind(
Impl,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -377,7 +448,9 @@ fn very_obsolete() {}
label: "mc", label: "mc",
navigation_range: 284..286, navigation_range: 284..286,
node_range: 271..303, node_range: 271..303,
kind: Macro, kind: SymbolKind(
Macro,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -386,7 +459,9 @@ fn very_obsolete() {}
label: "mcexp", label: "mcexp",
navigation_range: 334..339, navigation_range: 334..339,
node_range: 305..356, node_range: 305..356,
kind: Macro, kind: SymbolKind(
Macro,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -395,7 +470,9 @@ fn very_obsolete() {}
label: "mcexp", label: "mcexp",
navigation_range: 387..392, navigation_range: 387..392,
node_range: 358..409, node_range: 358..409,
kind: Macro, kind: SymbolKind(
Macro,
),
detail: None, detail: None,
deprecated: false, deprecated: false,
}, },
@ -404,7 +481,9 @@ fn very_obsolete() {}
label: "obsolete", label: "obsolete",
navigation_range: 428..436, navigation_range: 428..436,
node_range: 411..441, node_range: 411..441,
kind: Function, kind: SymbolKind(
Function,
),
detail: Some( detail: Some(
"fn()", "fn()",
), ),
@ -415,12 +494,75 @@ fn very_obsolete() {}
label: "very_obsolete", label: "very_obsolete",
navigation_range: 481..494, navigation_range: 481..494,
node_range: 443..499, node_range: 443..499,
kind: Function, kind: SymbolKind(
Function,
),
detail: Some( detail: Some(
"fn()", "fn()",
), ),
deprecated: true, deprecated: true,
}, },
StructureNode {
parent: None,
label: "Some region name",
navigation_range: 501..528,
node_range: 501..528,
kind: Region,
detail: None,
deprecated: false,
},
StructureNode {
parent: None,
label: "m",
navigation_range: 568..569,
node_range: 543..606,
kind: SymbolKind(
Module,
),
detail: None,
deprecated: false,
},
StructureNode {
parent: Some(
20,
),
label: "dontpanic",
navigation_range: 543..563,
node_range: 543..563,
kind: Region,
detail: None,
deprecated: false,
},
StructureNode {
parent: Some(
20,
),
label: "f",
navigation_range: 575..576,
node_range: 572..581,
kind: SymbolKind(
Function,
),
detail: Some(
"fn()",
),
deprecated: false,
},
StructureNode {
parent: Some(
20,
),
label: "g",
navigation_range: 598..599,
node_range: 582..604,
kind: SymbolKind(
Function,
),
detail: Some(
"fn()",
),
deprecated: false,
},
] ]
"#]], "#]],
); );

View File

@ -135,6 +135,12 @@ fn line_index(db: &dyn LineIndexDatabase, file_id: FileId) -> Arc<LineIndex> {
Arc::new(LineIndex::new(&*text)) Arc::new(LineIndex::new(&*text))
} }
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum StructureNodeKind {
SymbolKind(SymbolKind),
Region,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)] #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum SymbolKind { pub enum SymbolKind {
Const, Const,

View File

@ -289,7 +289,7 @@ pub(crate) fn handle_document_symbol(
let doc_symbol = lsp_types::DocumentSymbol { let doc_symbol = lsp_types::DocumentSymbol {
name: symbol.label, name: symbol.label,
detail: symbol.detail, detail: symbol.detail,
kind: to_proto::symbol_kind(symbol.kind), kind: to_proto::structure_node_kind(symbol.kind),
tags: Some(tags), tags: Some(tags),
deprecated: Some(symbol.deprecated), deprecated: Some(symbol.deprecated),
range: to_proto::range(&line_index, symbol.node_range), range: to_proto::range(&line_index, symbol.node_range),

View File

@ -11,7 +11,7 @@ use ide::{
Markup, NavigationTarget, ReferenceAccess, RenameError, Runnable, Severity, SourceChange, Markup, NavigationTarget, ReferenceAccess, RenameError, Runnable, Severity, SourceChange,
TextEdit, TextRange, TextSize, TextEdit, TextRange, TextSize,
}; };
use ide_db::SymbolKind; use ide_db::{StructureNodeKind, SymbolKind};
use itertools::Itertools; use itertools::Itertools;
use serde_json::to_value; use serde_json::to_value;
@ -63,6 +63,13 @@ pub(crate) fn symbol_kind(symbol_kind: SymbolKind) -> lsp_types::SymbolKind {
} }
} }
pub(crate) fn structure_node_kind(kind: StructureNodeKind) -> lsp_types::SymbolKind {
match kind {
StructureNodeKind::SymbolKind(symbol) => symbol_kind(symbol),
StructureNodeKind::Region => lsp_types::SymbolKind::Namespace,
}
}
pub(crate) fn document_highlight_kind( pub(crate) fn document_highlight_kind(
reference_access: ReferenceAccess, reference_access: ReferenceAccess,
) -> lsp_types::DocumentHighlightKind { ) -> lsp_types::DocumentHighlightKind {