2024-09-17 20:31:36 +00:00
|
|
|
//! Pretty printing of macros output.
|
|
|
|
|
|
|
|
use base_db::CrateId;
|
|
|
|
use rustc_hash::FxHashMap;
|
|
|
|
use syntax::NodeOrToken;
|
|
|
|
use syntax::{ast::make, SyntaxNode};
|
|
|
|
|
|
|
|
use crate::{db::ExpandDatabase, span_map::ExpansionSpanMap};
|
|
|
|
|
|
|
|
/// Inserts whitespace and replaces `$crate` in macro expansions.
|
|
|
|
#[expect(deprecated)]
|
|
|
|
pub fn prettify_macro_expansion(
|
|
|
|
db: &dyn ExpandDatabase,
|
|
|
|
syn: SyntaxNode,
|
|
|
|
span_map: &ExpansionSpanMap,
|
|
|
|
target_crate_id: CrateId,
|
|
|
|
) -> SyntaxNode {
|
2024-10-05 19:39:33 +00:00
|
|
|
// Because `syntax_bridge::prettify_macro_expansion::prettify_macro_expansion()` clones subtree for `syn`,
|
|
|
|
// that means it will be offsetted to the beginning.
|
|
|
|
let span_offset = syn.text_range().start();
|
2024-09-17 20:31:36 +00:00
|
|
|
let crate_graph = db.crate_graph();
|
|
|
|
let target_crate = &crate_graph[target_crate_id];
|
|
|
|
let mut syntax_ctx_id_to_dollar_crate_replacement = FxHashMap::default();
|
|
|
|
syntax_bridge::prettify_macro_expansion::prettify_macro_expansion(syn, &mut |dollar_crate| {
|
2024-10-05 19:39:33 +00:00
|
|
|
let ctx = span_map.span_at(dollar_crate.text_range().start() + span_offset).ctx;
|
2024-09-17 20:31:36 +00:00
|
|
|
let replacement =
|
|
|
|
syntax_ctx_id_to_dollar_crate_replacement.entry(ctx).or_insert_with(|| {
|
|
|
|
let ctx_data = db.lookup_intern_syntax_context(ctx);
|
|
|
|
let macro_call_id =
|
|
|
|
ctx_data.outer_expn.expect("`$crate` cannot come from `SyntaxContextId::ROOT`");
|
|
|
|
let macro_call = db.lookup_intern_macro_call(macro_call_id);
|
|
|
|
let macro_def_crate = macro_call.def.krate;
|
|
|
|
// First, if this is the same crate as the macro, nothing will work but `crate`.
|
|
|
|
// If not, if the target trait has the macro's crate as a dependency, using the dependency name
|
|
|
|
// will work in inserted code and match the user's expectation.
|
|
|
|
// If not, the crate's display name is what the dependency name is likely to be once such dependency
|
|
|
|
// is inserted, and also understandable to the user.
|
|
|
|
// Lastly, if nothing else found, resort to leaving `$crate`.
|
|
|
|
if target_crate_id == macro_def_crate {
|
|
|
|
make::tokens::crate_kw()
|
|
|
|
} else if let Some(dep) =
|
|
|
|
target_crate.dependencies.iter().find(|dep| dep.crate_id == macro_def_crate)
|
|
|
|
{
|
|
|
|
make::tokens::ident(&dep.name)
|
|
|
|
} else if let Some(crate_name) = &crate_graph[macro_def_crate].display_name {
|
|
|
|
make::tokens::ident(crate_name.crate_name())
|
|
|
|
} else {
|
|
|
|
return dollar_crate.clone();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if replacement.text() == "$crate" {
|
|
|
|
// The parent may have many children, and looking for the token may yield incorrect results.
|
|
|
|
return dollar_crate.clone();
|
|
|
|
}
|
|
|
|
// We need to `clone_subtree()` but rowan doesn't provide such operation for tokens.
|
|
|
|
let parent = replacement.parent().unwrap().clone_subtree().clone_for_update();
|
|
|
|
parent
|
|
|
|
.children_with_tokens()
|
|
|
|
.filter_map(NodeOrToken::into_token)
|
|
|
|
.find(|it| it.kind() == replacement.kind())
|
|
|
|
.unwrap()
|
|
|
|
})
|
|
|
|
}
|