mirror of
https://github.com/rust-lang/rust-analyzer.git
synced 2025-09-30 13:51:31 +00:00
Better factoring of macro expansion machinery in goto def
This commit is contained in:
parent
1adc352587
commit
42604c673d
1 changed files with 20 additions and 11 deletions
|
@ -1,5 +1,7 @@
|
||||||
//! FIXME: write short doc here
|
//! FIXME: write short doc here
|
||||||
|
|
||||||
|
use std::iter::successors;
|
||||||
|
|
||||||
use hir::{db::AstDatabase, Source};
|
use hir::{db::AstDatabase, Source};
|
||||||
use ra_syntax::{
|
use ra_syntax::{
|
||||||
algo::find_node_at_offset,
|
algo::find_node_at_offset,
|
||||||
|
@ -18,10 +20,8 @@ pub(crate) fn goto_definition(
|
||||||
db: &RootDatabase,
|
db: &RootDatabase,
|
||||||
position: FilePosition,
|
position: FilePosition,
|
||||||
) -> Option<RangeInfo<Vec<NavigationTarget>>> {
|
) -> Option<RangeInfo<Vec<NavigationTarget>>> {
|
||||||
go(db, Source::new(position.file_id.into(), position.offset))
|
let offset = descend_into_macros(db, position);
|
||||||
}
|
|
||||||
|
|
||||||
fn go(db: &RootDatabase, offset: Source<TextUnit>) -> Option<RangeInfo<Vec<NavigationTarget>>> {
|
|
||||||
let syntax = db.parse_or_expand(offset.file_id)?;
|
let syntax = db.parse_or_expand(offset.file_id)?;
|
||||||
|
|
||||||
if let Some(name_ref) = find_node_at_offset::<ast::NameRef>(&syntax, offset.ast) {
|
if let Some(name_ref) = find_node_at_offset::<ast::NameRef>(&syntax, offset.ast) {
|
||||||
|
@ -32,16 +32,25 @@ fn go(db: &RootDatabase, offset: Source<TextUnit>) -> Option<RangeInfo<Vec<Navig
|
||||||
let navs = name_definition(db, offset.with_ast(&name))?;
|
let navs = name_definition(db, offset.with_ast(&name))?;
|
||||||
return Some(RangeInfo::new(name.syntax().text_range(), navs));
|
return Some(RangeInfo::new(name.syntax().text_range(), navs));
|
||||||
}
|
}
|
||||||
if let Some(macro_call) = find_node_at_offset::<ast::MacroCall>(&syntax, offset.ast) {
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
fn descend_into_macros(db: &RootDatabase, position: FilePosition) -> Source<TextUnit> {
|
||||||
|
successors(Some(Source::new(position.file_id.into(), position.offset)), |offset| {
|
||||||
|
let syntax = db.parse_or_expand(offset.file_id)?;
|
||||||
|
let macro_call = find_node_at_offset::<ast::MacroCall>(&syntax, offset.ast)?;
|
||||||
|
let tt = macro_call.token_tree()?;
|
||||||
|
if !tt.syntax().text_range().contains(offset.ast) {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
let source_analyzer =
|
let source_analyzer =
|
||||||
hir::SourceAnalyzer::new(db, offset.with_ast(macro_call.syntax()), None);
|
hir::SourceAnalyzer::new(db, offset.with_ast(macro_call.syntax()), None);
|
||||||
if let Some(exp) = source_analyzer.expand(db, ¯o_call) {
|
let exp = source_analyzer.expand(db, ¯o_call)?;
|
||||||
if let Some(offset) = exp.translate_offset(db, offset.ast) {
|
let next_offset = exp.translate_offset(db, offset.ast)?;
|
||||||
return go(db, Source::new(exp.file_id(), offset));
|
Some(Source::new(exp.file_id(), next_offset))
|
||||||
}
|
})
|
||||||
}
|
.last()
|
||||||
}
|
.unwrap()
|
||||||
None
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue