internal: Replace Display impl for Name

This commit is contained in:
Lukas Wirth 2023-05-24 18:04:29 +02:00
parent 2f840c2236
commit c7ef6c25b7
108 changed files with 1045 additions and 656 deletions

View file

@ -2,6 +2,7 @@
use std::fmt::{self, Write}; use std::fmt::{self, Write};
use hir_expand::db::ExpandDatabase;
use syntax::ast::HasName; use syntax::ast::HasName;
use crate::{ use crate::{
@ -18,16 +19,22 @@ pub(super) fn print_body_hir(db: &dyn DefDatabase, body: &Body, owner: DefWithBo
let header = match owner { let header = match owner {
DefWithBodyId::FunctionId(it) => { DefWithBodyId::FunctionId(it) => {
let item_tree_id = it.lookup(db).id; let item_tree_id = it.lookup(db).id;
format!("fn {}", item_tree_id.item_tree(db)[item_tree_id.value].name) format!(
"fn {}",
item_tree_id.item_tree(db)[item_tree_id.value].name.display(db.upcast())
)
} }
DefWithBodyId::StaticId(it) => { DefWithBodyId::StaticId(it) => {
let item_tree_id = it.lookup(db).id; let item_tree_id = it.lookup(db).id;
format!("static {} = ", item_tree_id.item_tree(db)[item_tree_id.value].name) format!(
"static {} = ",
item_tree_id.item_tree(db)[item_tree_id.value].name.display(db.upcast())
)
} }
DefWithBodyId::ConstId(it) => { DefWithBodyId::ConstId(it) => {
let item_tree_id = it.lookup(db).id; let item_tree_id = it.lookup(db).id;
let name = match &item_tree_id.item_tree(db)[item_tree_id.value].name { let name = match &item_tree_id.item_tree(db)[item_tree_id.value].name {
Some(name) => name.to_string(), Some(name) => name.display(db.upcast()).to_string(),
None => "_".to_string(), None => "_".to_string(),
}; };
format!("const {name} = ") format!("const {name} = ")
@ -42,7 +49,8 @@ pub(super) fn print_body_hir(db: &dyn DefDatabase, body: &Body, owner: DefWithBo
} }
}; };
let mut p = Printer { body, buf: header, indent_level: 0, needs_indent: false }; let mut p =
Printer { db: db.upcast(), body, buf: header, indent_level: 0, needs_indent: false };
if let DefWithBodyId::FunctionId(it) = owner { if let DefWithBodyId::FunctionId(it) = owner {
p.buf.push('('); p.buf.push('(');
body.params.iter().zip(&db.function_data(it).params).for_each(|(&param, ty)| { body.params.iter().zip(&db.function_data(it).params).for_each(|(&param, ty)| {
@ -61,12 +69,13 @@ pub(super) fn print_body_hir(db: &dyn DefDatabase, body: &Body, owner: DefWithBo
} }
pub(super) fn print_expr_hir( pub(super) fn print_expr_hir(
_db: &dyn DefDatabase, db: &dyn DefDatabase,
body: &Body, body: &Body,
_owner: DefWithBodyId, _owner: DefWithBodyId,
expr: ExprId, expr: ExprId,
) -> String { ) -> String {
let mut p = Printer { body, buf: String::new(), indent_level: 0, needs_indent: false }; let mut p =
Printer { db: db.upcast(), body, buf: String::new(), indent_level: 0, needs_indent: false };
p.print_expr(expr); p.print_expr(expr);
p.buf p.buf
} }
@ -87,6 +96,7 @@ macro_rules! wln {
} }
struct Printer<'a> { struct Printer<'a> {
db: &'a dyn ExpandDatabase,
body: &'a Body, body: &'a Body,
buf: String, buf: String,
indent_level: usize, indent_level: usize,
@ -161,14 +171,14 @@ impl<'a> Printer<'a> {
} }
Expr::Loop { body, label } => { Expr::Loop { body, label } => {
if let Some(lbl) = label { if let Some(lbl) = label {
w!(self, "{}: ", self.body[*lbl].name); w!(self, "{}: ", self.body[*lbl].name.display(self.db));
} }
w!(self, "loop "); w!(self, "loop ");
self.print_expr(*body); self.print_expr(*body);
} }
Expr::While { condition, body, label } => { Expr::While { condition, body, label } => {
if let Some(lbl) = label { if let Some(lbl) = label {
w!(self, "{}: ", self.body[*lbl].name); w!(self, "{}: ", self.body[*lbl].name.display(self.db));
} }
w!(self, "while "); w!(self, "while ");
self.print_expr(*condition); self.print_expr(*condition);
@ -176,7 +186,7 @@ impl<'a> Printer<'a> {
} }
Expr::For { iterable, pat, body, label } => { Expr::For { iterable, pat, body, label } => {
if let Some(lbl) = label { if let Some(lbl) = label {
w!(self, "{}: ", self.body[*lbl].name); w!(self, "{}: ", self.body[*lbl].name.display(self.db));
} }
w!(self, "for "); w!(self, "for ");
self.print_pat(*pat); self.print_pat(*pat);
@ -199,10 +209,10 @@ impl<'a> Printer<'a> {
} }
Expr::MethodCall { receiver, method_name, args, generic_args } => { Expr::MethodCall { receiver, method_name, args, generic_args } => {
self.print_expr(*receiver); self.print_expr(*receiver);
w!(self, ".{}", method_name); w!(self, ".{}", method_name.display(self.db));
if let Some(args) = generic_args { if let Some(args) = generic_args {
w!(self, "::<"); w!(self, "::<");
print_generic_args(args, self).unwrap(); print_generic_args(self.db, args, self).unwrap();
w!(self, ">"); w!(self, ">");
} }
w!(self, "("); w!(self, "(");
@ -237,13 +247,13 @@ impl<'a> Printer<'a> {
Expr::Continue { label } => { Expr::Continue { label } => {
w!(self, "continue"); w!(self, "continue");
if let Some(lbl) = label { if let Some(lbl) = label {
w!(self, " {}", self.body[*lbl].name); w!(self, " {}", self.body[*lbl].name.display(self.db));
} }
} }
Expr::Break { expr, label } => { Expr::Break { expr, label } => {
w!(self, "break"); w!(self, "break");
if let Some(lbl) = label { if let Some(lbl) = label {
w!(self, " {}", self.body[*lbl].name); w!(self, " {}", self.body[*lbl].name.display(self.db));
} }
if let Some(expr) = expr { if let Some(expr) = expr {
self.whitespace(); self.whitespace();
@ -282,7 +292,7 @@ impl<'a> Printer<'a> {
w!(self, "{{"); w!(self, "{{");
self.indented(|p| { self.indented(|p| {
for field in &**fields { for field in &**fields {
w!(p, "{}: ", field.name); w!(p, "{}: ", field.name.display(self.db));
p.print_expr(field.expr); p.print_expr(field.expr);
wln!(p, ","); wln!(p, ",");
} }
@ -299,7 +309,7 @@ impl<'a> Printer<'a> {
} }
Expr::Field { expr, name } => { Expr::Field { expr, name } => {
self.print_expr(*expr); self.print_expr(*expr);
w!(self, ".{}", name); w!(self, ".{}", name.display(self.db));
} }
Expr::Await { expr } => { Expr::Await { expr } => {
self.print_expr(*expr); self.print_expr(*expr);
@ -437,7 +447,7 @@ impl<'a> Printer<'a> {
} }
Expr::Literal(lit) => self.print_literal(lit), Expr::Literal(lit) => self.print_literal(lit),
Expr::Block { id: _, statements, tail, label } => { Expr::Block { id: _, statements, tail, label } => {
let label = label.map(|lbl| format!("{}: ", self.body[lbl].name)); let label = label.map(|lbl| format!("{}: ", self.body[lbl].name.display(self.db)));
self.print_block(label.as_deref(), statements, tail); self.print_block(label.as_deref(), statements, tail);
} }
Expr::Unsafe { id: _, statements, tail } => { Expr::Unsafe { id: _, statements, tail } => {
@ -513,7 +523,7 @@ impl<'a> Printer<'a> {
w!(self, " {{"); w!(self, " {{");
self.indented(|p| { self.indented(|p| {
for arg in args.iter() { for arg in args.iter() {
w!(p, "{}: ", arg.name); w!(p, "{}: ", arg.name.display(self.db));
p.print_pat(arg.pat); p.print_pat(arg.pat);
wln!(p, ","); wln!(p, ",");
} }
@ -646,11 +656,11 @@ impl<'a> Printer<'a> {
} }
fn print_type_ref(&mut self, ty: &TypeRef) { fn print_type_ref(&mut self, ty: &TypeRef) {
print_type_ref(ty, self).unwrap(); print_type_ref(self.db, ty, self).unwrap();
} }
fn print_path(&mut self, path: &Path) { fn print_path(&mut self, path: &Path) {
print_path(path, self).unwrap(); print_path(self.db, path, self).unwrap();
} }
fn print_binding(&mut self, id: BindingId) { fn print_binding(&mut self, id: BindingId) {
@ -661,6 +671,6 @@ impl<'a> Printer<'a> {
BindingAnnotation::Ref => "ref ", BindingAnnotation::Ref => "ref ",
BindingAnnotation::RefMut => "ref mut ", BindingAnnotation::RefMut => "ref mut ",
}; };
w!(self, "{}{}", mode, name); w!(self, "{}{}", mode, name.display(self.db));
} }
} }

View file

@ -106,8 +106,14 @@ impl AsName for BuiltinType {
impl fmt::Display for BuiltinType { impl fmt::Display for BuiltinType {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let type_name = self.as_name(); match self {
type_name.fmt(f) BuiltinType::Char => f.write_str("char"),
BuiltinType::Bool => f.write_str("bool"),
BuiltinType::Str => f.write_str("str"),
BuiltinType::Int(it) => it.fmt(f),
BuiltinType::Uint(it) => it.fmt(f),
BuiltinType::Float(it) => it.fmt(f),
}
} }
} }

View file

@ -1,9 +1,11 @@
//! HIR for references to types. Paths in these are not yet resolved. They can //! HIR for references to types. Paths in these are not yet resolved. They can
//! be directly created from an ast::TypeRef, without further queries. //! be directly created from an ast::TypeRef, without further queries.
use core::fmt;
use std::fmt::Write; use std::fmt::Write;
use hir_expand::{ use hir_expand::{
db::ExpandDatabase,
name::{AsName, Name}, name::{AsName, Name},
AstId, AstId,
}; };
@ -383,15 +385,6 @@ pub enum ConstRefOrPath {
Path(Name), Path(Name),
} }
impl std::fmt::Display for ConstRefOrPath {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
ConstRefOrPath::Scalar(s) => s.fmt(f),
ConstRefOrPath::Path(n) => n.fmt(f),
}
}
}
impl ConstRefOrPath { impl ConstRefOrPath {
pub(crate) fn from_expr_opt(expr: Option<ast::Expr>) -> Self { pub(crate) fn from_expr_opt(expr: Option<ast::Expr>) -> Self {
match expr { match expr {
@ -400,6 +393,19 @@ impl ConstRefOrPath {
} }
} }
pub fn display<'a>(&'a self, db: &'a dyn ExpandDatabase) -> impl fmt::Display + 'a {
struct Display<'a>(&'a dyn ExpandDatabase, &'a ConstRefOrPath);
impl fmt::Display for Display<'_> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self.1 {
ConstRefOrPath::Scalar(s) => s.fmt(f),
ConstRefOrPath::Path(n) => n.display(self.0).fmt(f),
}
}
}
Display(db, self)
}
// FIXME: as per the comments on `TypeRef::Array`, this evaluation should not happen at this // FIXME: as per the comments on `TypeRef::Array`, this evaluation should not happen at this
// parse stage. // parse stage.
fn from_expr(expr: ast::Expr) -> Self { fn from_expr(expr: ast::Expr) -> Self {

View file

@ -33,13 +33,23 @@ pub struct ImportPath {
pub segments: Vec<Name>, pub segments: Vec<Name>,
} }
impl fmt::Display for ImportPath { impl ImportPath {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { pub fn display<'a>(&'a self, db: &'a dyn DefDatabase) -> impl fmt::Display + 'a {
fmt::Display::fmt(&self.segments.iter().format("::"), f) struct Display<'a> {
db: &'a dyn DefDatabase,
path: &'a ImportPath,
} }
impl fmt::Display for Display<'_> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Display::fmt(
&self.path.segments.iter().map(|it| it.display(self.db.upcast())).format("::"),
f,
)
}
}
Display { db, path: self }
} }
impl ImportPath {
fn len(&self) -> usize { fn len(&self) -> usize {
self.segments.len() self.segments.len()
} }
@ -76,7 +86,7 @@ impl ImportMap {
let mut importables = import_map let mut importables = import_map
.map .map
.iter() .iter()
.map(|(item, info)| (item, fst_path(&info.path))) .map(|(item, info)| (item, fst_path(db, &info.path)))
.collect::<Vec<_>>(); .collect::<Vec<_>>();
importables.sort_by(|(_, fst_path), (_, fst_path2)| fst_path.cmp(fst_path2)); importables.sort_by(|(_, fst_path), (_, fst_path2)| fst_path.cmp(fst_path2));
@ -113,6 +123,25 @@ impl ImportMap {
self.map.get(&item) self.map.get(&item)
} }
#[cfg(test)]
fn fmt_for_test(&self, db: &dyn DefDatabase) -> String {
let mut importable_paths: Vec<_> = self
.map
.iter()
.map(|(item, info)| {
let ns = match item {
ItemInNs::Types(_) => "t",
ItemInNs::Values(_) => "v",
ItemInNs::Macros(_) => "m",
};
format!("- {} ({ns})", info.path.display(db))
})
.collect();
importable_paths.sort();
importable_paths.join("\n")
}
fn collect_trait_assoc_items( fn collect_trait_assoc_items(
&mut self, &mut self,
db: &dyn DefDatabase, db: &dyn DefDatabase,
@ -238,13 +267,10 @@ impl fmt::Debug for ImportMap {
let mut importable_paths: Vec<_> = self let mut importable_paths: Vec<_> = self
.map .map
.iter() .iter()
.map(|(item, info)| { .map(|(item, _)| match item {
let ns = match item { ItemInNs::Types(it) => format!("- {it:?} (t)",),
ItemInNs::Types(_) => "t", ItemInNs::Values(it) => format!("- {it:?} (v)",),
ItemInNs::Values(_) => "v", ItemInNs::Macros(it) => format!("- {it:?} (m)",),
ItemInNs::Macros(_) => "m",
};
format!("- {} ({ns})", info.path)
}) })
.collect(); .collect();
@ -253,9 +279,9 @@ impl fmt::Debug for ImportMap {
} }
} }
fn fst_path(path: &ImportPath) -> String { fn fst_path(db: &dyn DefDatabase, path: &ImportPath) -> String {
let _p = profile::span("fst_path"); let _p = profile::span("fst_path");
let mut s = path.to_string(); let mut s = path.display(db).to_string();
s.make_ascii_lowercase(); s.make_ascii_lowercase();
s s
} }
@ -348,7 +374,12 @@ impl Query {
self self
} }
fn import_matches(&self, import: &ImportInfo, enforce_lowercase: bool) -> bool { fn import_matches(
&self,
db: &dyn DefDatabase,
import: &ImportInfo,
enforce_lowercase: bool,
) -> bool {
let _p = profile::span("import_map::Query::import_matches"); let _p = profile::span("import_map::Query::import_matches");
if import.is_trait_assoc_item { if import.is_trait_assoc_item {
if self.exclude_import_kinds.contains(&ImportKind::AssociatedItem) { if self.exclude_import_kinds.contains(&ImportKind::AssociatedItem) {
@ -359,9 +390,9 @@ impl Query {
} }
let mut input = if import.is_trait_assoc_item || self.name_only { let mut input = if import.is_trait_assoc_item || self.name_only {
import.path.segments.last().unwrap().to_string() import.path.segments.last().unwrap().display(db.upcast()).to_string()
} else { } else {
import.path.to_string() import.path.display(db).to_string()
}; };
if enforce_lowercase || !self.case_sensitive { if enforce_lowercase || !self.case_sensitive {
input.make_ascii_lowercase(); input.make_ascii_lowercase();
@ -426,25 +457,27 @@ pub fn search_dependencies(
let importables = &import_map.importables[indexed_value.value as usize..]; let importables = &import_map.importables[indexed_value.value as usize..];
let common_importable_data = &import_map.map[&importables[0]]; let common_importable_data = &import_map.map[&importables[0]];
if !query.import_matches(common_importable_data, true) { if !query.import_matches(db, common_importable_data, true) {
continue; continue;
} }
// Path shared by the importable items in this group. // Path shared by the importable items in this group.
let common_importables_path_fst = fst_path(&common_importable_data.path); let common_importables_path_fst = fst_path(db, &common_importable_data.path);
// Add the items from this `ModPath` group. Those are all subsequent items in // Add the items from this `ModPath` group. Those are all subsequent items in
// `importables` whose paths match `path`. // `importables` whose paths match `path`.
let iter = importables let iter = importables
.iter() .iter()
.copied() .copied()
.take_while(|item| common_importables_path_fst == fst_path(&import_map.map[item].path)) .take_while(|item| {
common_importables_path_fst == fst_path(db, &import_map.map[item].path)
})
.filter(|&item| match item_import_kind(item) { .filter(|&item| match item_import_kind(item) {
Some(import_kind) => !query.exclude_import_kinds.contains(&import_kind), Some(import_kind) => !query.exclude_import_kinds.contains(&import_kind),
None => true, None => true,
}) })
.filter(|item| { .filter(|item| {
!query.case_sensitive // we've already checked the common importables path case-insensitively !query.case_sensitive // we've already checked the common importables path case-insensitively
|| query.import_matches(&import_map.map[item], false) || query.import_matches(db, &import_map.map[item], false)
}); });
res.extend(iter); res.extend(iter);
@ -501,7 +534,7 @@ mod tests {
let (path, mark) = match assoc_item_path(&db, &dependency_imports, dependency) { let (path, mark) = match assoc_item_path(&db, &dependency_imports, dependency) {
Some(assoc_item_path) => (assoc_item_path, "a"), Some(assoc_item_path) => (assoc_item_path, "a"),
None => ( None => (
dependency_imports.path_of(dependency)?.to_string(), dependency_imports.path_of(dependency)?.display(&db).to_string(),
match dependency { match dependency {
ItemInNs::Types(ModuleDefId::FunctionId(_)) ItemInNs::Types(ModuleDefId::FunctionId(_))
| ItemInNs::Values(ModuleDefId::FunctionId(_)) => "f", | ItemInNs::Values(ModuleDefId::FunctionId(_)) => "f",
@ -552,7 +585,11 @@ mod tests {
None None
} }
})?; })?;
return Some(format!("{}::{assoc_item_name}", dependency_imports.path_of(trait_)?)); return Some(format!(
"{}::{}",
dependency_imports.path_of(trait_)?.display(db),
assoc_item_name.display(db.upcast())
));
} }
None None
} }
@ -592,7 +629,7 @@ mod tests {
let map = db.import_map(krate); let map = db.import_map(krate);
Some(format!("{name}:\n{map:?}\n")) Some(format!("{name}:\n{}\n", map.fmt_for_test(db.upcast())))
}) })
.sorted() .sorted()
.collect::<String>(); .collect::<String>();

View file

@ -4,7 +4,7 @@
use std::collections::hash_map::Entry; use std::collections::hash_map::Entry;
use base_db::CrateId; use base_db::CrateId;
use hir_expand::{attrs::AttrId, name::Name, AstId, MacroCallId}; use hir_expand::{attrs::AttrId, db::ExpandDatabase, name::Name, AstId, MacroCallId};
use itertools::Itertools; use itertools::Itertools;
use once_cell::sync::Lazy; use once_cell::sync::Lazy;
use profile::Count; use profile::Count;
@ -358,12 +358,16 @@ impl ItemScope {
} }
} }
pub(crate) fn dump(&self, buf: &mut String) { pub(crate) fn dump(&self, db: &dyn ExpandDatabase, buf: &mut String) {
let mut entries: Vec<_> = self.resolutions().collect(); let mut entries: Vec<_> = self.resolutions().collect();
entries.sort_by_key(|(name, _)| name.clone()); entries.sort_by_key(|(name, _)| name.clone());
for (name, def) in entries { for (name, def) in entries {
format_to!(buf, "{}:", name.map_or("_".to_string(), |name| name.to_string())); format_to!(
buf,
"{}:",
name.map_or("_".to_string(), |name| name.display(db).to_string())
);
if def.types.is_some() { if def.types.is_some() {
buf.push_str(" t"); buf.push_str(" t");

View file

@ -167,8 +167,8 @@ impl ItemTree {
Attrs::filter(db, krate, self.raw_attrs(of).clone()) Attrs::filter(db, krate, self.raw_attrs(of).clone())
} }
pub fn pretty_print(&self) -> String { pub fn pretty_print(&self, db: &dyn DefDatabase) -> String {
pretty::print_item_tree(self) pretty::print_item_tree(db.upcast(), self)
} }
fn data(&self) -> &ItemTreeData { fn data(&self) -> &ItemTreeData {

View file

@ -2,6 +2,8 @@
use std::fmt::{self, Write}; use std::fmt::{self, Write};
use hir_expand::db::ExpandDatabase;
use crate::{ use crate::{
generics::{TypeOrConstParamData, WherePredicate, WherePredicateTypeTarget}, generics::{TypeOrConstParamData, WherePredicate, WherePredicateTypeTarget},
pretty::{print_path, print_type_bounds, print_type_ref}, pretty::{print_path, print_type_bounds, print_type_ref},
@ -10,8 +12,8 @@ use crate::{
use super::*; use super::*;
pub(super) fn print_item_tree(tree: &ItemTree) -> String { pub(super) fn print_item_tree(db: &dyn ExpandDatabase, tree: &ItemTree) -> String {
let mut p = Printer { tree, buf: String::new(), indent_level: 0, needs_indent: true }; let mut p = Printer { db, tree, buf: String::new(), indent_level: 0, needs_indent: true };
if let Some(attrs) = tree.attrs.get(&AttrOwner::TopLevel) { if let Some(attrs) = tree.attrs.get(&AttrOwner::TopLevel) {
p.print_attrs(attrs, true); p.print_attrs(attrs, true);
@ -43,6 +45,7 @@ macro_rules! wln {
} }
struct Printer<'a> { struct Printer<'a> {
db: &'a dyn ExpandDatabase,
tree: &'a ItemTree, tree: &'a ItemTree,
buf: String, buf: String,
indent_level: usize, indent_level: usize,
@ -88,7 +91,7 @@ impl<'a> Printer<'a> {
self, self,
"#{}[{}{}]", "#{}[{}{}]",
inner, inner,
attr.path, attr.path.display(self.db),
attr.input.as_ref().map(|it| it.to_string()).unwrap_or_default(), attr.input.as_ref().map(|it| it.to_string()).unwrap_or_default(),
); );
} }
@ -102,7 +105,7 @@ impl<'a> Printer<'a> {
fn print_visibility(&mut self, vis: RawVisibilityId) { fn print_visibility(&mut self, vis: RawVisibilityId) {
match &self.tree[vis] { match &self.tree[vis] {
RawVisibility::Module(path) => w!(self, "pub({}) ", path), RawVisibility::Module(path) => w!(self, "pub({}) ", path.display(self.db)),
RawVisibility::Public => w!(self, "pub "), RawVisibility::Public => w!(self, "pub "),
}; };
} }
@ -117,7 +120,7 @@ impl<'a> Printer<'a> {
let Field { visibility, name, type_ref, ast_id: _ } = &this.tree[field]; let Field { visibility, name, type_ref, ast_id: _ } = &this.tree[field];
this.print_attrs_of(field); this.print_attrs_of(field);
this.print_visibility(*visibility); this.print_visibility(*visibility);
w!(this, "{}: ", name); w!(this, "{}: ", name.display(self.db));
this.print_type_ref(type_ref); this.print_type_ref(type_ref);
wln!(this, ","); wln!(this, ",");
} }
@ -131,7 +134,7 @@ impl<'a> Printer<'a> {
let Field { visibility, name, type_ref, ast_id: _ } = &this.tree[field]; let Field { visibility, name, type_ref, ast_id: _ } = &this.tree[field];
this.print_attrs_of(field); this.print_attrs_of(field);
this.print_visibility(*visibility); this.print_visibility(*visibility);
w!(this, "{}: ", name); w!(this, "{}: ", name.display(self.db));
this.print_type_ref(type_ref); this.print_type_ref(type_ref);
wln!(this, ","); wln!(this, ",");
} }
@ -164,20 +167,20 @@ impl<'a> Printer<'a> {
fn print_use_tree(&mut self, use_tree: &UseTree) { fn print_use_tree(&mut self, use_tree: &UseTree) {
match &use_tree.kind { match &use_tree.kind {
UseTreeKind::Single { path, alias } => { UseTreeKind::Single { path, alias } => {
w!(self, "{}", path); w!(self, "{}", path.display(self.db));
if let Some(alias) = alias { if let Some(alias) = alias {
w!(self, " as {}", alias); w!(self, " as {}", alias);
} }
} }
UseTreeKind::Glob { path } => { UseTreeKind::Glob { path } => {
if let Some(path) = path { if let Some(path) = path {
w!(self, "{}::", path); w!(self, "{}::", path.display(self.db));
} }
w!(self, "*"); w!(self, "*");
} }
UseTreeKind::Prefixed { prefix, list } => { UseTreeKind::Prefixed { prefix, list } => {
if let Some(prefix) = prefix { if let Some(prefix) = prefix {
w!(self, "{}::", prefix); w!(self, "{}::", prefix.display(self.db));
} }
w!(self, "{{"); w!(self, "{{");
for (i, tree) in list.iter().enumerate() { for (i, tree) in list.iter().enumerate() {
@ -205,7 +208,7 @@ impl<'a> Printer<'a> {
ModItem::ExternCrate(it) => { ModItem::ExternCrate(it) => {
let ExternCrate { name, alias, visibility, ast_id: _ } = &self.tree[it]; let ExternCrate { name, alias, visibility, ast_id: _ } = &self.tree[it];
self.print_visibility(*visibility); self.print_visibility(*visibility);
w!(self, "extern crate {}", name); w!(self, "extern crate {}", name.display(self.db));
if let Some(alias) = alias { if let Some(alias) = alias {
w!(self, " as {}", alias); w!(self, " as {}", alias);
} }
@ -252,7 +255,7 @@ impl<'a> Printer<'a> {
if let Some(abi) = abi { if let Some(abi) = abi {
w!(self, "extern \"{}\" ", abi); w!(self, "extern \"{}\" ", abi);
} }
w!(self, "fn {}", name); w!(self, "fn {}", name.display(self.db));
self.print_generic_params(explicit_generic_params); self.print_generic_params(explicit_generic_params);
w!(self, "("); w!(self, "(");
if !params.is_empty() { if !params.is_empty() {
@ -286,7 +289,7 @@ impl<'a> Printer<'a> {
ModItem::Struct(it) => { ModItem::Struct(it) => {
let Struct { visibility, name, fields, generic_params, ast_id: _ } = &self.tree[it]; let Struct { visibility, name, fields, generic_params, ast_id: _ } = &self.tree[it];
self.print_visibility(*visibility); self.print_visibility(*visibility);
w!(self, "struct {}", name); w!(self, "struct {}", name.display(self.db));
self.print_generic_params(generic_params); self.print_generic_params(generic_params);
self.print_fields_and_where_clause(fields, generic_params); self.print_fields_and_where_clause(fields, generic_params);
if matches!(fields, Fields::Record(_)) { if matches!(fields, Fields::Record(_)) {
@ -298,7 +301,7 @@ impl<'a> Printer<'a> {
ModItem::Union(it) => { ModItem::Union(it) => {
let Union { name, visibility, fields, generic_params, ast_id: _ } = &self.tree[it]; let Union { name, visibility, fields, generic_params, ast_id: _ } = &self.tree[it];
self.print_visibility(*visibility); self.print_visibility(*visibility);
w!(self, "union {}", name); w!(self, "union {}", name.display(self.db));
self.print_generic_params(generic_params); self.print_generic_params(generic_params);
self.print_fields_and_where_clause(fields, generic_params); self.print_fields_and_where_clause(fields, generic_params);
if matches!(fields, Fields::Record(_)) { if matches!(fields, Fields::Record(_)) {
@ -310,14 +313,14 @@ impl<'a> Printer<'a> {
ModItem::Enum(it) => { ModItem::Enum(it) => {
let Enum { name, visibility, variants, generic_params, ast_id: _ } = &self.tree[it]; let Enum { name, visibility, variants, generic_params, ast_id: _ } = &self.tree[it];
self.print_visibility(*visibility); self.print_visibility(*visibility);
w!(self, "enum {}", name); w!(self, "enum {}", name.display(self.db));
self.print_generic_params(generic_params); self.print_generic_params(generic_params);
self.print_where_clause_and_opening_brace(generic_params); self.print_where_clause_and_opening_brace(generic_params);
self.indented(|this| { self.indented(|this| {
for variant in variants.clone() { for variant in variants.clone() {
let Variant { name, fields, ast_id: _ } = &this.tree[variant]; let Variant { name, fields, ast_id: _ } = &this.tree[variant];
this.print_attrs_of(variant); this.print_attrs_of(variant);
w!(this, "{}", name); w!(this, "{}", name.display(self.db));
this.print_fields(fields); this.print_fields(fields);
wln!(this, ","); wln!(this, ",");
} }
@ -329,7 +332,7 @@ impl<'a> Printer<'a> {
self.print_visibility(*visibility); self.print_visibility(*visibility);
w!(self, "const "); w!(self, "const ");
match name { match name {
Some(name) => w!(self, "{}", name), Some(name) => w!(self, "{}", name.display(self.db)),
None => w!(self, "_"), None => w!(self, "_"),
} }
w!(self, ": "); w!(self, ": ");
@ -343,7 +346,7 @@ impl<'a> Printer<'a> {
if *mutable { if *mutable {
w!(self, "mut "); w!(self, "mut ");
} }
w!(self, "{}: ", name); w!(self, "{}: ", name.display(self.db));
self.print_type_ref(type_ref); self.print_type_ref(type_ref);
w!(self, " = _;"); w!(self, " = _;");
wln!(self); wln!(self);
@ -365,7 +368,7 @@ impl<'a> Printer<'a> {
if *is_auto { if *is_auto {
w!(self, "auto "); w!(self, "auto ");
} }
w!(self, "trait {}", name); w!(self, "trait {}", name.display(self.db));
self.print_generic_params(generic_params); self.print_generic_params(generic_params);
self.print_where_clause_and_opening_brace(generic_params); self.print_where_clause_and_opening_brace(generic_params);
self.indented(|this| { self.indented(|this| {
@ -378,7 +381,7 @@ impl<'a> Printer<'a> {
ModItem::TraitAlias(it) => { ModItem::TraitAlias(it) => {
let TraitAlias { name, visibility, generic_params, ast_id: _ } = &self.tree[it]; let TraitAlias { name, visibility, generic_params, ast_id: _ } = &self.tree[it];
self.print_visibility(*visibility); self.print_visibility(*visibility);
w!(self, "trait {}", name); w!(self, "trait {}", name.display(self.db));
self.print_generic_params(generic_params); self.print_generic_params(generic_params);
w!(self, " = "); w!(self, " = ");
self.print_where_clause(generic_params); self.print_where_clause(generic_params);
@ -411,7 +414,7 @@ impl<'a> Printer<'a> {
let TypeAlias { name, visibility, bounds, type_ref, generic_params, ast_id: _ } = let TypeAlias { name, visibility, bounds, type_ref, generic_params, ast_id: _ } =
&self.tree[it]; &self.tree[it];
self.print_visibility(*visibility); self.print_visibility(*visibility);
w!(self, "type {}", name); w!(self, "type {}", name.display(self.db));
self.print_generic_params(generic_params); self.print_generic_params(generic_params);
if !bounds.is_empty() { if !bounds.is_empty() {
w!(self, ": "); w!(self, ": ");
@ -428,7 +431,7 @@ impl<'a> Printer<'a> {
ModItem::Mod(it) => { ModItem::Mod(it) => {
let Mod { name, visibility, kind, ast_id: _ } = &self.tree[it]; let Mod { name, visibility, kind, ast_id: _ } = &self.tree[it];
self.print_visibility(*visibility); self.print_visibility(*visibility);
w!(self, "mod {}", name); w!(self, "mod {}", name.display(self.db));
match kind { match kind {
ModKind::Inline { items } => { ModKind::Inline { items } => {
w!(self, " {{"); w!(self, " {{");
@ -446,16 +449,16 @@ impl<'a> Printer<'a> {
} }
ModItem::MacroCall(it) => { ModItem::MacroCall(it) => {
let MacroCall { path, ast_id: _, expand_to: _ } = &self.tree[it]; let MacroCall { path, ast_id: _, expand_to: _ } = &self.tree[it];
wln!(self, "{}!(...);", path); wln!(self, "{}!(...);", path.display(self.db));
} }
ModItem::MacroRules(it) => { ModItem::MacroRules(it) => {
let MacroRules { name, ast_id: _ } = &self.tree[it]; let MacroRules { name, ast_id: _ } = &self.tree[it];
wln!(self, "macro_rules! {} {{ ... }}", name); wln!(self, "macro_rules! {} {{ ... }}", name.display(self.db));
} }
ModItem::MacroDef(it) => { ModItem::MacroDef(it) => {
let MacroDef { name, visibility, ast_id: _ } = &self.tree[it]; let MacroDef { name, visibility, ast_id: _ } = &self.tree[it];
self.print_visibility(*visibility); self.print_visibility(*visibility);
wln!(self, "macro {} {{ ... }}", name); wln!(self, "macro {} {{ ... }}", name.display(self.db));
} }
} }
@ -463,15 +466,15 @@ impl<'a> Printer<'a> {
} }
fn print_type_ref(&mut self, type_ref: &TypeRef) { fn print_type_ref(&mut self, type_ref: &TypeRef) {
print_type_ref(type_ref, self).unwrap(); print_type_ref(self.db, type_ref, self).unwrap();
} }
fn print_type_bounds(&mut self, bounds: &[Interned<TypeBound>]) { fn print_type_bounds(&mut self, bounds: &[Interned<TypeBound>]) {
print_type_bounds(bounds, self).unwrap(); print_type_bounds(self.db, bounds, self).unwrap();
} }
fn print_path(&mut self, path: &Path) { fn print_path(&mut self, path: &Path) {
print_path(path, self).unwrap(); print_path(self.db, path, self).unwrap();
} }
fn print_generic_params(&mut self, params: &GenericParams) { fn print_generic_params(&mut self, params: &GenericParams) {
@ -486,7 +489,7 @@ impl<'a> Printer<'a> {
w!(self, ", "); w!(self, ", ");
} }
first = false; first = false;
w!(self, "{}", lt.name); w!(self, "{}", lt.name.display(self.db));
} }
for (idx, x) in params.type_or_consts.iter() { for (idx, x) in params.type_or_consts.iter() {
if !first { if !first {
@ -495,11 +498,11 @@ impl<'a> Printer<'a> {
first = false; first = false;
match x { match x {
TypeOrConstParamData::TypeParamData(ty) => match &ty.name { TypeOrConstParamData::TypeParamData(ty) => match &ty.name {
Some(name) => w!(self, "{}", name), Some(name) => w!(self, "{}", name.display(self.db)),
None => w!(self, "_anon_{}", idx.into_raw()), None => w!(self, "_anon_{}", idx.into_raw()),
}, },
TypeOrConstParamData::ConstParamData(konst) => { TypeOrConstParamData::ConstParamData(konst) => {
w!(self, "const {}: ", konst.name); w!(self, "const {}: ", konst.name.display(self.db));
self.print_type_ref(&konst.ty); self.print_type_ref(&konst.ty);
} }
} }
@ -531,7 +534,12 @@ impl<'a> Printer<'a> {
let (target, bound) = match pred { let (target, bound) = match pred {
WherePredicate::TypeBound { target, bound } => (target, bound), WherePredicate::TypeBound { target, bound } => (target, bound),
WherePredicate::Lifetime { target, bound } => { WherePredicate::Lifetime { target, bound } => {
wln!(this, "{}: {},", target.name, bound.name); wln!(
this,
"{}: {},",
target.name.display(self.db),
bound.name.display(self.db)
);
continue; continue;
} }
WherePredicate::ForLifetime { lifetimes, target, bound } => { WherePredicate::ForLifetime { lifetimes, target, bound } => {
@ -540,7 +548,7 @@ impl<'a> Printer<'a> {
if i != 0 { if i != 0 {
w!(this, ", "); w!(this, ", ");
} }
w!(this, "{}", lt); w!(this, "{}", lt.display(self.db));
} }
w!(this, "> "); w!(this, "> ");
(target, bound) (target, bound)
@ -551,7 +559,7 @@ impl<'a> Printer<'a> {
WherePredicateTypeTarget::TypeRef(ty) => this.print_type_ref(ty), WherePredicateTypeTarget::TypeRef(ty) => this.print_type_ref(ty),
WherePredicateTypeTarget::TypeOrConstParam(id) => { WherePredicateTypeTarget::TypeOrConstParam(id) => {
match &params.type_or_consts[*id].name() { match &params.type_or_consts[*id].name() {
Some(name) => w!(this, "{}", name), Some(name) => w!(this, "{}", name.display(self.db)),
None => w!(this, "_anon_{}", id.into_raw()), None => w!(this, "_anon_{}", id.into_raw()),
} }
} }

View file

@ -6,7 +6,7 @@ use crate::{db::DefDatabase, test_db::TestDB};
fn check(ra_fixture: &str, expect: Expect) { fn check(ra_fixture: &str, expect: Expect) {
let (db, file_id) = TestDB::with_single_file(ra_fixture); let (db, file_id) = TestDB::with_single_file(ra_fixture);
let item_tree = db.file_item_tree(file_id.into()); let item_tree = db.file_item_tree(file_id.into());
let pretty = item_tree.pretty_print(); let pretty = item_tree.pretty_print(&db);
expect.assert_eq(&pretty); expect.assert_eq(&pretty);
} }

View file

@ -463,25 +463,31 @@ impl DefMap {
let mut arc; let mut arc;
let mut current_map = self; let mut current_map = self;
while let Some(block) = current_map.block { while let Some(block) = current_map.block {
go(&mut buf, current_map, "block scope", current_map.root); go(&mut buf, db, current_map, "block scope", current_map.root);
buf.push('\n'); buf.push('\n');
arc = block.parent.def_map(db); arc = block.parent.def_map(db);
current_map = &arc; current_map = &arc;
} }
go(&mut buf, current_map, "crate", current_map.root); go(&mut buf, db, current_map, "crate", current_map.root);
return buf; return buf;
fn go(buf: &mut String, map: &DefMap, path: &str, module: LocalModuleId) { fn go(
buf: &mut String,
db: &dyn DefDatabase,
map: &DefMap,
path: &str,
module: LocalModuleId,
) {
format_to!(buf, "{}\n", path); format_to!(buf, "{}\n", path);
map.modules[module].scope.dump(buf); map.modules[module].scope.dump(db.upcast(), buf);
for (name, child) in for (name, child) in
map.modules[module].children.iter().sorted_by(|a, b| Ord::cmp(&a.0, &b.0)) map.modules[module].children.iter().sorted_by(|a, b| Ord::cmp(&a.0, &b.0))
{ {
let path = format!("{path}::{name}"); let path = format!("{path}::{}", name.display(db.upcast()));
buf.push('\n'); buf.push('\n');
go(buf, map, &path, *child); go(buf, db, map, &path, *child);
} }
} }
} }

View file

@ -565,7 +565,7 @@ impl DefCollector<'_> {
types => { types => {
tracing::debug!( tracing::debug!(
"could not resolve prelude path `{}` to module (resolved to {:?})", "could not resolve prelude path `{}` to module (resolved to {:?})",
path, path.display(self.db.upcast()),
types types
); );
} }
@ -766,7 +766,8 @@ impl DefCollector<'_> {
} }
fn resolve_import(&self, module_id: LocalModuleId, import: &Import) -> PartialResolvedImport { fn resolve_import(&self, module_id: LocalModuleId, import: &Import) -> PartialResolvedImport {
let _p = profile::span("resolve_import").detail(|| format!("{}", import.path)); let _p = profile::span("resolve_import")
.detail(|| format!("{}", import.path.display(self.db.upcast())));
tracing::debug!("resolving import: {:?} ({:?})", import, self.def_map.edition); tracing::debug!("resolving import: {:?} ({:?})", import, self.def_map.edition);
if import.is_extern_crate { if import.is_extern_crate {
let name = import let name = import
@ -1985,7 +1986,10 @@ impl ModCollector<'_, '_> {
if self.def_collector.def_map.is_builtin_or_registered_attr(&attr.path) { if self.def_collector.def_map.is_builtin_or_registered_attr(&attr.path) {
continue; continue;
} }
tracing::debug!("non-builtin attribute {}", attr.path); tracing::debug!(
"non-builtin attribute {}",
attr.path.display(self.def_collector.db.upcast())
);
let ast_id = AstIdWithPath::new( let ast_id = AstIdWithPath::new(
self.file_id(), self.file_id(),
@ -2119,8 +2123,8 @@ impl ModCollector<'_, '_> {
stdx::always!( stdx::always!(
name == mac.name, name == mac.name,
"built-in macro {} has #[rustc_builtin_macro] which declares different name {}", "built-in macro {} has #[rustc_builtin_macro] which declares different name {}",
mac.name, mac.name.display(self.def_collector.db.upcast()),
name name.display(self.def_collector.db.upcast())
); );
helpers_opt = Some(helpers); helpers_opt = Some(helpers);
} }

View file

@ -74,12 +74,20 @@ impl ModDir {
candidate_files.push(self.dir_path.join_attr(attr_path, self.root_non_dir_owner)) candidate_files.push(self.dir_path.join_attr(attr_path, self.root_non_dir_owner))
} }
None if file_id.is_include_macro(db.upcast()) => { None if file_id.is_include_macro(db.upcast()) => {
candidate_files.push(format!("{name}.rs")); candidate_files.push(format!("{}.rs", name.display(db.upcast())));
candidate_files.push(format!("{name}/mod.rs")); candidate_files.push(format!("{}/mod.rs", name.display(db.upcast())));
} }
None => { None => {
candidate_files.push(format!("{}{name}.rs", self.dir_path.0)); candidate_files.push(format!(
candidate_files.push(format!("{}{name}/mod.rs", self.dir_path.0)); "{}{}.rs",
self.dir_path.0,
name.display(db.upcast())
));
candidate_files.push(format!(
"{}{}/mod.rs",
self.dir_path.0,
name.display(db.upcast())
));
} }
}; };
@ -91,7 +99,7 @@ impl ModDir {
let (dir_path, root_non_dir_owner) = if is_mod_rs || attr_path.is_some() { let (dir_path, root_non_dir_owner) = if is_mod_rs || attr_path.is_some() {
(DirPath::empty(), false) (DirPath::empty(), false)
} else { } else {
(DirPath::new(format!("{name}/")), true) (DirPath::new(format!("{}/", name.display(db.upcast()))), true)
}; };
if let Some(mod_dir) = self.child(dir_path, root_non_dir_owner) { if let Some(mod_dir) = self.child(dir_path, root_non_dir_owner) {
return Ok((file_id, is_mod_rs, mod_dir)); return Ok((file_id, is_mod_rs, mod_dir));

View file

@ -192,8 +192,11 @@ impl DefMap {
) -> ResolvePathResult { ) -> ResolvePathResult {
let graph = db.crate_graph(); let graph = db.crate_graph();
let _cx = stdx::panic_context::enter(format!( let _cx = stdx::panic_context::enter(format!(
"DefMap {:?} crate_name={:?} block={:?} path={path}", "DefMap {:?} crate_name={:?} block={:?} path={}",
self.krate, graph[self.krate].display_name, self.block self.krate,
graph[self.krate].display_name,
self.block,
path.display(db.upcast())
)); ));
let mut segments = path.segments().iter().enumerate(); let mut segments = path.segments().iter().enumerate();
@ -262,8 +265,8 @@ impl DefMap {
); );
tracing::debug!( tracing::debug!(
"`super` path: {} -> {} in parent map", "`super` path: {} -> {} in parent map",
path, path.display(db.upcast()),
new_path new_path.display(db.upcast())
); );
return block.parent.def_map(db).resolve_path_fp_with_macro( return block.parent.def_map(db).resolve_path_fp_with_macro(
db, db,

View file

@ -1080,7 +1080,7 @@ macro_rules! mbe {
#[test] #[test]
fn collects_derive_helpers() { fn collects_derive_helpers() {
let def_map = compute_crate_def_map( let db = TestDB::with_files(
r#" r#"
#![crate_type="proc-macro"] #![crate_type="proc-macro"]
struct TokenStream; struct TokenStream;
@ -1091,11 +1091,13 @@ pub fn derive_macro_2(_item: TokenStream) -> TokenStream {
} }
"#, "#,
); );
let krate = db.crate_graph().iter().next().unwrap();
let def_map = db.crate_def_map(krate);
assert_eq!(def_map.exported_derives.len(), 1); assert_eq!(def_map.exported_derives.len(), 1);
match def_map.exported_derives.values().next() { match def_map.exported_derives.values().next() {
Some(helpers) => match &**helpers { Some(helpers) => match &**helpers {
[attr] => assert_eq!(attr.to_string(), "helper_attr"), [attr] => assert_eq!(attr.display(&db).to_string(), "helper_attr"),
_ => unreachable!(), _ => unreachable!(),
}, },
_ => unreachable!(), _ => unreachable!(),
@ -1258,7 +1260,7 @@ struct A;
#[test] #[test]
fn macro_use_imports_all_macro_types() { fn macro_use_imports_all_macro_types() {
let def_map = compute_crate_def_map( let db = TestDB::with_files(
r#" r#"
//- /main.rs crate:main deps:lib //- /main.rs crate:main deps:lib
#[macro_use] #[macro_use]
@ -1281,6 +1283,8 @@ struct TokenStream;
fn proc_attr(a: TokenStream, b: TokenStream) -> TokenStream { a } fn proc_attr(a: TokenStream, b: TokenStream) -> TokenStream { a }
"#, "#,
); );
let krate = db.crate_graph().iter().next().unwrap();
let def_map = db.crate_def_map(krate);
let root_module = &def_map[def_map.root()].scope; let root_module = &def_map[def_map.root()].scope;
assert!( assert!(
@ -1288,7 +1292,12 @@ fn proc_attr(a: TokenStream, b: TokenStream) -> TokenStream { a }
"`#[macro_use]` shouldn't bring macros into textual macro scope", "`#[macro_use]` shouldn't bring macros into textual macro scope",
); );
let actual = def_map.macro_use_prelude.iter().map(|(name, _)| name).sorted().join("\n"); let actual = def_map
.macro_use_prelude
.iter()
.map(|(name, _)| name.display(&db).to_string())
.sorted()
.join("\n");
expect![[r#" expect![[r#"
legacy legacy

View file

@ -2,7 +2,7 @@
use std::fmt::{self, Write}; use std::fmt::{self, Write};
use hir_expand::mod_path::PathKind; use hir_expand::{db::ExpandDatabase, mod_path::PathKind};
use intern::Interned; use intern::Interned;
use itertools::Itertools; use itertools::Itertools;
@ -11,14 +11,14 @@ use crate::{
type_ref::{Mutability, TraitBoundModifier, TypeBound, TypeRef}, type_ref::{Mutability, TraitBoundModifier, TypeBound, TypeRef},
}; };
pub(crate) fn print_path(path: &Path, buf: &mut dyn Write) -> fmt::Result { pub(crate) fn print_path(db: &dyn ExpandDatabase, path: &Path, buf: &mut dyn Write) -> fmt::Result {
if let Path::LangItem(x) = path { if let Path::LangItem(x) = path {
return write!(buf, "$lang_item::{x:?}"); return write!(buf, "$lang_item::{x:?}");
} }
match path.type_anchor() { match path.type_anchor() {
Some(anchor) => { Some(anchor) => {
write!(buf, "<")?; write!(buf, "<")?;
print_type_ref(anchor, buf)?; print_type_ref(db, anchor, buf)?;
write!(buf, ">::")?; write!(buf, ">::")?;
} }
None => match path.kind() { None => match path.kind() {
@ -44,10 +44,10 @@ pub(crate) fn print_path(path: &Path, buf: &mut dyn Write) -> fmt::Result {
write!(buf, "::")?; write!(buf, "::")?;
} }
write!(buf, "{}", segment.name)?; write!(buf, "{}", segment.name.display(db))?;
if let Some(generics) = segment.args_and_bindings { if let Some(generics) = segment.args_and_bindings {
write!(buf, "::<")?; write!(buf, "::<")?;
print_generic_args(generics, buf)?; print_generic_args(db, generics, buf)?;
write!(buf, ">")?; write!(buf, ">")?;
} }
@ -56,12 +56,16 @@ pub(crate) fn print_path(path: &Path, buf: &mut dyn Write) -> fmt::Result {
Ok(()) Ok(())
} }
pub(crate) fn print_generic_args(generics: &GenericArgs, buf: &mut dyn Write) -> fmt::Result { pub(crate) fn print_generic_args(
db: &dyn ExpandDatabase,
generics: &GenericArgs,
buf: &mut dyn Write,
) -> fmt::Result {
let mut first = true; let mut first = true;
let args = if generics.has_self_type { let args = if generics.has_self_type {
let (self_ty, args) = generics.args.split_first().unwrap(); let (self_ty, args) = generics.args.split_first().unwrap();
write!(buf, "Self=")?; write!(buf, "Self=")?;
print_generic_arg(self_ty, buf)?; print_generic_arg(db, self_ty, buf)?;
first = false; first = false;
args args
} else { } else {
@ -72,35 +76,43 @@ pub(crate) fn print_generic_args(generics: &GenericArgs, buf: &mut dyn Write) ->
write!(buf, ", ")?; write!(buf, ", ")?;
} }
first = false; first = false;
print_generic_arg(arg, buf)?; print_generic_arg(db, arg, buf)?;
} }
for binding in generics.bindings.iter() { for binding in generics.bindings.iter() {
if !first { if !first {
write!(buf, ", ")?; write!(buf, ", ")?;
} }
first = false; first = false;
write!(buf, "{}", binding.name)?; write!(buf, "{}", binding.name.display(db))?;
if !binding.bounds.is_empty() { if !binding.bounds.is_empty() {
write!(buf, ": ")?; write!(buf, ": ")?;
print_type_bounds(&binding.bounds, buf)?; print_type_bounds(db, &binding.bounds, buf)?;
} }
if let Some(ty) = &binding.type_ref { if let Some(ty) = &binding.type_ref {
write!(buf, " = ")?; write!(buf, " = ")?;
print_type_ref(ty, buf)?; print_type_ref(db, ty, buf)?;
} }
} }
Ok(()) Ok(())
} }
pub(crate) fn print_generic_arg(arg: &GenericArg, buf: &mut dyn Write) -> fmt::Result { pub(crate) fn print_generic_arg(
db: &dyn ExpandDatabase,
arg: &GenericArg,
buf: &mut dyn Write,
) -> fmt::Result {
match arg { match arg {
GenericArg::Type(ty) => print_type_ref(ty, buf), GenericArg::Type(ty) => print_type_ref(db, ty, buf),
GenericArg::Const(c) => write!(buf, "{c}"), GenericArg::Const(c) => write!(buf, "{}", c.display(db)),
GenericArg::Lifetime(lt) => write!(buf, "{}", lt.name), GenericArg::Lifetime(lt) => write!(buf, "{}", lt.name.display(db)),
} }
} }
pub(crate) fn print_type_ref(type_ref: &TypeRef, buf: &mut dyn Write) -> fmt::Result { pub(crate) fn print_type_ref(
db: &dyn ExpandDatabase,
type_ref: &TypeRef,
buf: &mut dyn Write,
) -> fmt::Result {
// FIXME: deduplicate with `HirDisplay` impl // FIXME: deduplicate with `HirDisplay` impl
match type_ref { match type_ref {
TypeRef::Never => write!(buf, "!")?, TypeRef::Never => write!(buf, "!")?,
@ -111,18 +123,18 @@ pub(crate) fn print_type_ref(type_ref: &TypeRef, buf: &mut dyn Write) -> fmt::Re
if i != 0 { if i != 0 {
write!(buf, ", ")?; write!(buf, ", ")?;
} }
print_type_ref(field, buf)?; print_type_ref(db, field, buf)?;
} }
write!(buf, ")")?; write!(buf, ")")?;
} }
TypeRef::Path(path) => print_path(path, buf)?, TypeRef::Path(path) => print_path(db, path, buf)?,
TypeRef::RawPtr(pointee, mtbl) => { TypeRef::RawPtr(pointee, mtbl) => {
let mtbl = match mtbl { let mtbl = match mtbl {
Mutability::Shared => "*const", Mutability::Shared => "*const",
Mutability::Mut => "*mut", Mutability::Mut => "*mut",
}; };
write!(buf, "{mtbl} ")?; write!(buf, "{mtbl} ")?;
print_type_ref(pointee, buf)?; print_type_ref(db, pointee, buf)?;
} }
TypeRef::Reference(pointee, lt, mtbl) => { TypeRef::Reference(pointee, lt, mtbl) => {
let mtbl = match mtbl { let mtbl = match mtbl {
@ -131,19 +143,19 @@ pub(crate) fn print_type_ref(type_ref: &TypeRef, buf: &mut dyn Write) -> fmt::Re
}; };
write!(buf, "&")?; write!(buf, "&")?;
if let Some(lt) = lt { if let Some(lt) = lt {
write!(buf, "{} ", lt.name)?; write!(buf, "{} ", lt.name.display(db))?;
} }
write!(buf, "{mtbl}")?; write!(buf, "{mtbl}")?;
print_type_ref(pointee, buf)?; print_type_ref(db, pointee, buf)?;
} }
TypeRef::Array(elem, len) => { TypeRef::Array(elem, len) => {
write!(buf, "[")?; write!(buf, "[")?;
print_type_ref(elem, buf)?; print_type_ref(db, elem, buf)?;
write!(buf, "; {len}]")?; write!(buf, "; {}]", len.display(db))?;
} }
TypeRef::Slice(elem) => { TypeRef::Slice(elem) => {
write!(buf, "[")?; write!(buf, "[")?;
print_type_ref(elem, buf)?; print_type_ref(db, elem, buf)?;
write!(buf, "]")?; write!(buf, "]")?;
} }
TypeRef::Fn(args_and_ret, varargs, is_unsafe) => { TypeRef::Fn(args_and_ret, varargs, is_unsafe) => {
@ -157,7 +169,7 @@ pub(crate) fn print_type_ref(type_ref: &TypeRef, buf: &mut dyn Write) -> fmt::Re
if i != 0 { if i != 0 {
write!(buf, ", ")?; write!(buf, ", ")?;
} }
print_type_ref(typeref, buf)?; print_type_ref(db, typeref, buf)?;
} }
if *varargs { if *varargs {
if !args.is_empty() { if !args.is_empty() {
@ -166,7 +178,7 @@ pub(crate) fn print_type_ref(type_ref: &TypeRef, buf: &mut dyn Write) -> fmt::Re
write!(buf, "...")?; write!(buf, "...")?;
} }
write!(buf, ") -> ")?; write!(buf, ") -> ")?;
print_type_ref(return_type, buf)?; print_type_ref(db, return_type, buf)?;
} }
TypeRef::Macro(_ast_id) => { TypeRef::Macro(_ast_id) => {
write!(buf, "<macro>")?; write!(buf, "<macro>")?;
@ -174,11 +186,11 @@ pub(crate) fn print_type_ref(type_ref: &TypeRef, buf: &mut dyn Write) -> fmt::Re
TypeRef::Error => write!(buf, "{{unknown}}")?, TypeRef::Error => write!(buf, "{{unknown}}")?,
TypeRef::ImplTrait(bounds) => { TypeRef::ImplTrait(bounds) => {
write!(buf, "impl ")?; write!(buf, "impl ")?;
print_type_bounds(bounds, buf)?; print_type_bounds(db, bounds, buf)?;
} }
TypeRef::DynTrait(bounds) => { TypeRef::DynTrait(bounds) => {
write!(buf, "dyn ")?; write!(buf, "dyn ")?;
print_type_bounds(bounds, buf)?; print_type_bounds(db, bounds, buf)?;
} }
} }
@ -186,6 +198,7 @@ pub(crate) fn print_type_ref(type_ref: &TypeRef, buf: &mut dyn Write) -> fmt::Re
} }
pub(crate) fn print_type_bounds( pub(crate) fn print_type_bounds(
db: &dyn ExpandDatabase,
bounds: &[Interned<TypeBound>], bounds: &[Interned<TypeBound>],
buf: &mut dyn Write, buf: &mut dyn Write,
) -> fmt::Result { ) -> fmt::Result {
@ -200,13 +213,13 @@ pub(crate) fn print_type_bounds(
TraitBoundModifier::None => (), TraitBoundModifier::None => (),
TraitBoundModifier::Maybe => write!(buf, "?")?, TraitBoundModifier::Maybe => write!(buf, "?")?,
} }
print_path(path, buf)?; print_path(db, path, buf)?;
} }
TypeBound::ForLifetime(lifetimes, path) => { TypeBound::ForLifetime(lifetimes, path) => {
write!(buf, "for<{}> ", lifetimes.iter().format(", "))?; write!(buf, "for<{}> ", lifetimes.iter().map(|it| it.display(db)).format(", "))?;
print_path(path, buf)?; print_path(db, path, buf)?;
} }
TypeBound::Lifetime(lt) => write!(buf, "{}", lt.name)?, TypeBound::Lifetime(lt) => write!(buf, "{}", lt.name.display(db))?,
TypeBound::Error => write!(buf, "{{unknown}}")?, TypeBound::Error => write!(buf, "{{unknown}}")?,
} }
} }

View file

@ -1,7 +1,7 @@
//! A lowering for `use`-paths (more generally, paths without angle-bracketed segments). //! A lowering for `use`-paths (more generally, paths without angle-bracketed segments).
use std::{ use std::{
fmt::{self, Display}, fmt::{self, Display as _},
iter, iter,
}; };
@ -24,6 +24,12 @@ pub struct ModPath {
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] #[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct UnescapedModPath<'a>(&'a ModPath); pub struct UnescapedModPath<'a>(&'a ModPath);
impl<'a> UnescapedModPath<'a> {
pub fn display(&'a self, db: &'a dyn crate::db::ExpandDatabase) -> impl fmt::Display + 'a {
UnescapedDisplay { db, path: self }
}
}
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum PathKind { pub enum PathKind {
Plain, Plain,
@ -110,7 +116,44 @@ impl ModPath {
UnescapedModPath(self) UnescapedModPath(self)
} }
fn _fmt(&self, f: &mut fmt::Formatter<'_>, escaped: bool) -> fmt::Result { pub fn display<'a>(&'a self, db: &'a dyn crate::db::ExpandDatabase) -> impl fmt::Display + 'a {
Display { db, path: self }
}
}
struct Display<'a> {
db: &'a dyn ExpandDatabase,
path: &'a ModPath,
}
impl<'a> fmt::Display for Display<'a> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
display_fmt_path(self.db, self.path, f, true)
}
}
struct UnescapedDisplay<'a> {
db: &'a dyn ExpandDatabase,
path: &'a UnescapedModPath<'a>,
}
impl<'a> fmt::Display for UnescapedDisplay<'a> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
display_fmt_path(self.db, self.path.0, f, false)
}
}
impl From<Name> for ModPath {
fn from(name: Name) -> ModPath {
ModPath::from_segments(PathKind::Plain, iter::once(name))
}
}
fn display_fmt_path(
db: &dyn ExpandDatabase,
path: &ModPath,
f: &mut fmt::Formatter<'_>,
escaped: bool,
) -> fmt::Result {
let mut first_segment = true; let mut first_segment = true;
let mut add_segment = |s| -> fmt::Result { let mut add_segment = |s| -> fmt::Result {
if !first_segment { if !first_segment {
@ -120,7 +163,7 @@ impl ModPath {
f.write_str(s)?; f.write_str(s)?;
Ok(()) Ok(())
}; };
match self.kind { match path.kind {
PathKind::Plain => {} PathKind::Plain => {}
PathKind::Super(0) => add_segment("self")?, PathKind::Super(0) => add_segment("self")?,
PathKind::Super(n) => { PathKind::Super(n) => {
@ -132,38 +175,19 @@ impl ModPath {
PathKind::Abs => add_segment("")?, PathKind::Abs => add_segment("")?,
PathKind::DollarCrate(_) => add_segment("$crate")?, PathKind::DollarCrate(_) => add_segment("$crate")?,
} }
for segment in &self.segments { for segment in &path.segments {
if !first_segment { if !first_segment {
f.write_str("::")?; f.write_str("::")?;
} }
first_segment = false; first_segment = false;
if escaped { if escaped {
segment.fmt(f)? segment.display(db).fmt(f)?;
} else { } else {
segment.unescaped().fmt(f)? segment.unescaped().display(db).fmt(f)?;
}; }
} }
Ok(()) Ok(())
} }
}
impl Display for ModPath {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self._fmt(f, true)
}
}
impl<'a> Display for UnescapedModPath<'a> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.0._fmt(f, false)
}
}
impl From<Name> for ModPath {
fn from(name: Name) -> ModPath {
ModPath::from_segments(PathKind::Plain, iter::once(name))
}
}
fn convert_path( fn convert_path(
db: &dyn ExpandDatabase, db: &dyn ExpandDatabase,

View file

@ -24,27 +24,6 @@ enum Repr {
TupleField(usize), TupleField(usize),
} }
impl fmt::Display for Name {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match &self.0 {
Repr::Text(text) => fmt::Display::fmt(&text, f),
Repr::TupleField(idx) => fmt::Display::fmt(&idx, f),
}
}
}
impl<'a> fmt::Display for UnescapedName<'a> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match &self.0 .0 {
Repr::Text(text) => {
let text = text.strip_prefix("r#").unwrap_or(text);
fmt::Display::fmt(&text, f)
}
Repr::TupleField(idx) => fmt::Display::fmt(&idx, f),
}
}
}
impl<'a> UnescapedName<'a> { impl<'a> UnescapedName<'a> {
/// Returns the textual representation of this name as a [`SmolStr`]. Prefer using this over /// Returns the textual representation of this name as a [`SmolStr`]. Prefer using this over
/// [`ToString::to_string`] if possible as this conversion is cheaper in the general case. /// [`ToString::to_string`] if possible as this conversion is cheaper in the general case.
@ -60,6 +39,11 @@ impl<'a> UnescapedName<'a> {
Repr::TupleField(it) => SmolStr::new(it.to_string()), Repr::TupleField(it) => SmolStr::new(it.to_string()),
} }
} }
pub fn display(&'a self, db: &dyn crate::db::ExpandDatabase) -> impl fmt::Display + 'a {
_ = db;
UnescapedDisplay { name: self }
}
} }
impl Name { impl Name {
@ -167,6 +151,40 @@ impl Name {
Repr::TupleField(_) => false, Repr::TupleField(_) => false,
} }
} }
pub fn display<'a>(&'a self, db: &dyn crate::db::ExpandDatabase) -> impl fmt::Display + 'a {
_ = db;
Display { name: self }
}
}
struct Display<'a> {
name: &'a Name,
}
impl<'a> fmt::Display for Display<'a> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match &self.name.0 {
Repr::Text(text) => fmt::Display::fmt(&text, f),
Repr::TupleField(idx) => fmt::Display::fmt(&idx, f),
}
}
}
struct UnescapedDisplay<'a> {
name: &'a UnescapedName<'a>,
}
impl<'a> fmt::Display for UnescapedDisplay<'a> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match &self.name.0 .0 {
Repr::Text(text) => {
let text = text.strip_prefix("r#").unwrap_or(text);
fmt::Display::fmt(&text, f)
}
Repr::TupleField(idx) => fmt::Display::fmt(&idx, f),
}
}
} }
pub trait AsName { pub trait AsName {

View file

@ -365,13 +365,19 @@ impl<'a> chalk_solve::RustIrDatabase<Interner> for ChalkContext<'a> {
fn trait_name(&self, trait_id: chalk_ir::TraitId<Interner>) -> String { fn trait_name(&self, trait_id: chalk_ir::TraitId<Interner>) -> String {
let id = from_chalk_trait_id(trait_id); let id = from_chalk_trait_id(trait_id);
self.db.trait_data(id).name.to_string() self.db.trait_data(id).name.display(self.db.upcast()).to_string()
} }
fn adt_name(&self, chalk_ir::AdtId(adt_id): AdtId) -> String { fn adt_name(&self, chalk_ir::AdtId(adt_id): AdtId) -> String {
match adt_id { match adt_id {
hir_def::AdtId::StructId(id) => self.db.struct_data(id).name.to_string(), hir_def::AdtId::StructId(id) => {
hir_def::AdtId::EnumId(id) => self.db.enum_data(id).name.to_string(), self.db.struct_data(id).name.display(self.db.upcast()).to_string()
hir_def::AdtId::UnionId(id) => self.db.union_data(id).name.to_string(), }
hir_def::AdtId::EnumId(id) => {
self.db.enum_data(id).name.display(self.db.upcast()).to_string()
}
hir_def::AdtId::UnionId(id) => {
self.db.union_data(id).name.display(self.db.upcast()).to_string()
}
} }
} }
fn adt_size_align(&self, _id: chalk_ir::AdtId<Interner>) -> Arc<rust_ir::AdtSizeAlign> { fn adt_size_align(&self, _id: chalk_ir::AdtId<Interner>) -> Arc<rust_ir::AdtSizeAlign> {
@ -380,7 +386,7 @@ impl<'a> chalk_solve::RustIrDatabase<Interner> for ChalkContext<'a> {
} }
fn assoc_type_name(&self, assoc_ty_id: chalk_ir::AssocTypeId<Interner>) -> String { fn assoc_type_name(&self, assoc_ty_id: chalk_ir::AssocTypeId<Interner>) -> String {
let id = self.db.associated_ty_data(assoc_ty_id).name; let id = self.db.associated_ty_data(assoc_ty_id).name;
self.db.type_alias_data(id).name.to_string() self.db.type_alias_data(id).name.display(self.db.upcast()).to_string()
} }
fn opaque_type_name(&self, opaque_ty_id: chalk_ir::OpaqueTyId<Interner>) -> String { fn opaque_type_name(&self, opaque_ty_id: chalk_ir::OpaqueTyId<Interner>) -> String {
format!("Opaque_{}", opaque_ty_id.0) format!("Opaque_{}", opaque_ty_id.0)

View file

@ -79,7 +79,7 @@ fn eval_goal(db: &TestDB, file_id: FileId) -> Result<Const, ConstEvalError> {
.declarations() .declarations()
.find_map(|x| match x { .find_map(|x| match x {
hir_def::ModuleDefId::ConstId(x) => { hir_def::ModuleDefId::ConstId(x) => {
if db.const_data(x).name.as_ref()?.to_string() == "GOAL" { if db.const_data(x).name.as_ref()?.display(db).to_string() == "GOAL" {
Some(x) Some(x)
} else { } else {
None None

View file

@ -243,13 +243,19 @@ pub trait HirDatabase: DefDatabase + Upcast<dyn DefDatabase> {
fn infer_wait(db: &dyn HirDatabase, def: DefWithBodyId) -> Arc<InferenceResult> { fn infer_wait(db: &dyn HirDatabase, def: DefWithBodyId) -> Arc<InferenceResult> {
let _p = profile::span("infer:wait").detail(|| match def { let _p = profile::span("infer:wait").detail(|| match def {
DefWithBodyId::FunctionId(it) => db.function_data(it).name.to_string(), DefWithBodyId::FunctionId(it) => db.function_data(it).name.display(db.upcast()).to_string(),
DefWithBodyId::StaticId(it) => db.static_data(it).name.clone().to_string(), DefWithBodyId::StaticId(it) => {
DefWithBodyId::ConstId(it) => { db.static_data(it).name.clone().display(db.upcast()).to_string()
db.const_data(it).name.clone().unwrap_or_else(Name::missing).to_string()
} }
DefWithBodyId::ConstId(it) => db
.const_data(it)
.name
.clone()
.unwrap_or_else(Name::missing)
.display(db.upcast())
.to_string(),
DefWithBodyId::VariantId(it) => { DefWithBodyId::VariantId(it) => {
db.enum_data(it.parent).variants[it.local_id].name.to_string() db.enum_data(it.parent).variants[it.local_id].name.display(db.upcast()).to_string()
} }
}); });
db.infer_query(def) db.infer_query(def)

View file

@ -223,7 +223,7 @@ impl<'a> DeclValidator<'a> {
} }
// Check the function name. // Check the function name.
let function_name = data.name.to_string(); let function_name = data.name.display(self.db.upcast()).to_string();
let fn_name_replacement = to_lower_snake_case(&function_name).map(|new_name| Replacement { let fn_name_replacement = to_lower_snake_case(&function_name).map(|new_name| Replacement {
current_name: data.name.clone(), current_name: data.name.clone(),
suggested_text: new_name, suggested_text: new_name,
@ -244,7 +244,9 @@ impl<'a> DeclValidator<'a> {
id, id,
Replacement { Replacement {
current_name: bind_name.clone(), current_name: bind_name.clone(),
suggested_text: to_lower_snake_case(&bind_name.to_string())?, suggested_text: to_lower_snake_case(
&bind_name.display(self.db.upcast()).to_string(),
)?,
expected_case: CaseType::LowerSnakeCase, expected_case: CaseType::LowerSnakeCase,
}, },
)) ))
@ -287,7 +289,7 @@ impl<'a> DeclValidator<'a> {
ident_type: IdentType::Function, ident_type: IdentType::Function,
ident: AstPtr::new(&ast_ptr), ident: AstPtr::new(&ast_ptr),
expected_case: fn_name_replacement.expected_case, expected_case: fn_name_replacement.expected_case,
ident_text: fn_name_replacement.current_name.to_string(), ident_text: fn_name_replacement.current_name.display(self.db.upcast()).to_string(),
suggested_text: fn_name_replacement.suggested_text, suggested_text: fn_name_replacement.suggested_text,
}; };
@ -343,7 +345,10 @@ impl<'a> DeclValidator<'a> {
ident_type, ident_type,
ident: AstPtr::new(&name_ast), ident: AstPtr::new(&name_ast),
expected_case: replacement.expected_case, expected_case: replacement.expected_case,
ident_text: replacement.current_name.to_string(), ident_text: replacement
.current_name
.display(self.db.upcast())
.to_string(),
suggested_text: replacement.suggested_text, suggested_text: replacement.suggested_text,
}; };
@ -362,7 +367,7 @@ impl<'a> DeclValidator<'a> {
let non_snake_case_allowed = self.allowed(struct_id.into(), allow::NON_SNAKE_CASE, false); let non_snake_case_allowed = self.allowed(struct_id.into(), allow::NON_SNAKE_CASE, false);
// Check the structure name. // Check the structure name.
let struct_name = data.name.to_string(); let struct_name = data.name.display(self.db.upcast()).to_string();
let struct_name_replacement = if !non_camel_case_allowed { let struct_name_replacement = if !non_camel_case_allowed {
to_camel_case(&struct_name).map(|new_name| Replacement { to_camel_case(&struct_name).map(|new_name| Replacement {
current_name: data.name.clone(), current_name: data.name.clone(),
@ -379,7 +384,7 @@ impl<'a> DeclValidator<'a> {
if !non_snake_case_allowed { if !non_snake_case_allowed {
if let VariantData::Record(fields) = data.variant_data.as_ref() { if let VariantData::Record(fields) = data.variant_data.as_ref() {
for (_, field) in fields.iter() { for (_, field) in fields.iter() {
let field_name = field.name.to_string(); let field_name = field.name.display(self.db.upcast()).to_string();
if let Some(new_name) = to_lower_snake_case(&field_name) { if let Some(new_name) = to_lower_snake_case(&field_name) {
let replacement = Replacement { let replacement = Replacement {
current_name: field.name.clone(), current_name: field.name.clone(),
@ -434,7 +439,7 @@ impl<'a> DeclValidator<'a> {
ident_type: IdentType::Structure, ident_type: IdentType::Structure,
ident: AstPtr::new(&ast_ptr), ident: AstPtr::new(&ast_ptr),
expected_case: replacement.expected_case, expected_case: replacement.expected_case,
ident_text: replacement.current_name.to_string(), ident_text: replacement.current_name.display(self.db.upcast()).to_string(),
suggested_text: replacement.suggested_text, suggested_text: replacement.suggested_text,
}; };
@ -479,7 +484,7 @@ impl<'a> DeclValidator<'a> {
ident_type: IdentType::Field, ident_type: IdentType::Field,
ident: AstPtr::new(&ast_ptr), ident: AstPtr::new(&ast_ptr),
expected_case: field_to_rename.expected_case, expected_case: field_to_rename.expected_case,
ident_text: field_to_rename.current_name.to_string(), ident_text: field_to_rename.current_name.display(self.db.upcast()).to_string(),
suggested_text: field_to_rename.suggested_text, suggested_text: field_to_rename.suggested_text,
}; };
@ -496,7 +501,7 @@ impl<'a> DeclValidator<'a> {
} }
// Check the enum name. // Check the enum name.
let enum_name = data.name.to_string(); let enum_name = data.name.display(self.db.upcast()).to_string();
let enum_name_replacement = to_camel_case(&enum_name).map(|new_name| Replacement { let enum_name_replacement = to_camel_case(&enum_name).map(|new_name| Replacement {
current_name: data.name.clone(), current_name: data.name.clone(),
suggested_text: new_name, suggested_text: new_name,
@ -510,7 +515,9 @@ impl<'a> DeclValidator<'a> {
.filter_map(|(_, variant)| { .filter_map(|(_, variant)| {
Some(Replacement { Some(Replacement {
current_name: variant.name.clone(), current_name: variant.name.clone(),
suggested_text: to_camel_case(&variant.name.to_string())?, suggested_text: to_camel_case(
&variant.name.display(self.db.upcast()).to_string(),
)?,
expected_case: CaseType::UpperCamelCase, expected_case: CaseType::UpperCamelCase,
}) })
}) })
@ -558,7 +565,7 @@ impl<'a> DeclValidator<'a> {
ident_type: IdentType::Enum, ident_type: IdentType::Enum,
ident: AstPtr::new(&ast_ptr), ident: AstPtr::new(&ast_ptr),
expected_case: replacement.expected_case, expected_case: replacement.expected_case,
ident_text: replacement.current_name.to_string(), ident_text: replacement.current_name.display(self.db.upcast()).to_string(),
suggested_text: replacement.suggested_text, suggested_text: replacement.suggested_text,
}; };
@ -603,7 +610,7 @@ impl<'a> DeclValidator<'a> {
ident_type: IdentType::Variant, ident_type: IdentType::Variant,
ident: AstPtr::new(&ast_ptr), ident: AstPtr::new(&ast_ptr),
expected_case: variant_to_rename.expected_case, expected_case: variant_to_rename.expected_case,
ident_text: variant_to_rename.current_name.to_string(), ident_text: variant_to_rename.current_name.display(self.db.upcast()).to_string(),
suggested_text: variant_to_rename.suggested_text, suggested_text: variant_to_rename.suggested_text,
}; };
@ -623,7 +630,7 @@ impl<'a> DeclValidator<'a> {
None => return, None => return,
}; };
let const_name = name.to_string(); let const_name = name.display(self.db.upcast()).to_string();
let replacement = if let Some(new_name) = to_upper_snake_case(&const_name) { let replacement = if let Some(new_name) = to_upper_snake_case(&const_name) {
Replacement { Replacement {
current_name: name.clone(), current_name: name.clone(),
@ -648,7 +655,7 @@ impl<'a> DeclValidator<'a> {
ident_type: IdentType::Constant, ident_type: IdentType::Constant,
ident: AstPtr::new(&ast_ptr), ident: AstPtr::new(&ast_ptr),
expected_case: replacement.expected_case, expected_case: replacement.expected_case,
ident_text: replacement.current_name.to_string(), ident_text: replacement.current_name.display(self.db.upcast()).to_string(),
suggested_text: replacement.suggested_text, suggested_text: replacement.suggested_text,
}; };
@ -668,7 +675,7 @@ impl<'a> DeclValidator<'a> {
let name = &data.name; let name = &data.name;
let static_name = name.to_string(); let static_name = name.display(self.db.upcast()).to_string();
let replacement = if let Some(new_name) = to_upper_snake_case(&static_name) { let replacement = if let Some(new_name) = to_upper_snake_case(&static_name) {
Replacement { Replacement {
current_name: name.clone(), current_name: name.clone(),
@ -693,7 +700,7 @@ impl<'a> DeclValidator<'a> {
ident_type: IdentType::StaticVariable, ident_type: IdentType::StaticVariable,
ident: AstPtr::new(&ast_ptr), ident: AstPtr::new(&ast_ptr),
expected_case: replacement.expected_case, expected_case: replacement.expected_case,
ident_text: replacement.current_name.to_string(), ident_text: replacement.current_name.display(self.db.upcast()).to_string(),
suggested_text: replacement.suggested_text, suggested_text: replacement.suggested_text,
}; };

View file

@ -153,7 +153,7 @@ impl<'a> PatCtxt<'a> {
match (bm, ty.kind(Interner)) { match (bm, ty.kind(Interner)) {
(BindingMode::Ref(_), TyKind::Ref(.., rty)) => ty = rty, (BindingMode::Ref(_), TyKind::Ref(.., rty)) => ty = rty,
(BindingMode::Ref(_), _) => { (BindingMode::Ref(_), _) => {
never!("`ref {}` has wrong type {:?}", name, ty); never!("`ref {}` has wrong type {:?}", name.display(self.db.upcast()), ty);
self.errors.push(PatternError::UnexpectedType); self.errors.push(PatternError::UnexpectedType);
return Pat { ty: ty.clone(), kind: PatKind::Wild.into() }; return Pat { ty: ty.clone(), kind: PatKind::Wild.into() };
} }
@ -298,7 +298,7 @@ impl HirDisplay for Pat {
match &*self.kind { match &*self.kind {
PatKind::Wild => write!(f, "_"), PatKind::Wild => write!(f, "_"),
PatKind::Binding { name, subpattern } => { PatKind::Binding { name, subpattern } => {
write!(f, "{name}")?; write!(f, "{}", name.display(f.db.upcast()))?;
if let Some(subpattern) = subpattern { if let Some(subpattern) = subpattern {
write!(f, " @ ")?; write!(f, " @ ")?;
subpattern.hir_fmt(f)?; subpattern.hir_fmt(f)?;
@ -319,10 +319,14 @@ impl HirDisplay for Pat {
match variant { match variant {
VariantId::EnumVariantId(v) => { VariantId::EnumVariantId(v) => {
let data = f.db.enum_data(v.parent); let data = f.db.enum_data(v.parent);
write!(f, "{}", data.variants[v.local_id].name)?; write!(f, "{}", data.variants[v.local_id].name.display(f.db.upcast()))?;
}
VariantId::StructId(s) => {
write!(f, "{}", f.db.struct_data(s).name.display(f.db.upcast()))?
}
VariantId::UnionId(u) => {
write!(f, "{}", f.db.union_data(u).name.display(f.db.upcast()))?
} }
VariantId::StructId(s) => write!(f, "{}", f.db.struct_data(s).name)?,
VariantId::UnionId(u) => write!(f, "{}", f.db.union_data(u).name)?,
}; };
let variant_data = variant.variant_data(f.db.upcast()); let variant_data = variant.variant_data(f.db.upcast());
@ -336,7 +340,11 @@ impl HirDisplay for Pat {
.map(|p| { .map(|p| {
printed += 1; printed += 1;
WriteWith(move |f| { WriteWith(move |f| {
write!(f, "{}: ", rec_fields[p.field].name)?; write!(
f,
"{}: ",
rec_fields[p.field].name.display(f.db.upcast())
)?;
p.pattern.hir_fmt(f) p.pattern.hir_fmt(f)
}) })
}); });

View file

@ -372,7 +372,13 @@ impl HirDisplay for ProjectionTy {
let trait_ref = self.trait_ref(f.db); let trait_ref = self.trait_ref(f.db);
write!(f, "<")?; write!(f, "<")?;
fmt_trait_ref(f, &trait_ref, true)?; fmt_trait_ref(f, &trait_ref, true)?;
write!(f, ">::{}", f.db.type_alias_data(from_assoc_type_id(self.associated_ty_id)).name)?; write!(
f,
">::{}",
f.db.type_alias_data(from_assoc_type_id(self.associated_ty_id))
.name
.display(f.db.upcast())
)?;
let proj_params_count = let proj_params_count =
self.substitution.len(Interner) - trait_ref.substitution.len(Interner); self.substitution.len(Interner) - trait_ref.substitution.len(Interner);
let proj_params = &self.substitution.as_slice(Interner)[..proj_params_count]; let proj_params = &self.substitution.as_slice(Interner)[..proj_params_count];
@ -415,7 +421,8 @@ impl HirDisplay for Const {
let id = from_placeholder_idx(f.db, *idx); let id = from_placeholder_idx(f.db, *idx);
let generics = generics(f.db.upcast(), id.parent); let generics = generics(f.db.upcast(), id.parent);
let param_data = &generics.params.type_or_consts[id.local_id]; let param_data = &generics.params.type_or_consts[id.local_id];
write!(f, "{}", param_data.name().unwrap()) write!(f, "{}", param_data.name().unwrap().display(f.db.upcast()))?;
Ok(())
} }
ConstValue::Concrete(c) => match &c.interned { ConstValue::Concrete(c) => match &c.interned {
ConstScalar::Bytes(b, m) => render_const_scalar(f, &b, m, &data.ty), ConstScalar::Bytes(b, m) => render_const_scalar(f, &b, m, &data.ty),
@ -546,19 +553,19 @@ fn render_const_scalar(
}; };
let mut it = fields.iter(); let mut it = fields.iter();
if matches!(data.variant_data.as_ref(), VariantData::Record(_)) { if matches!(data.variant_data.as_ref(), VariantData::Record(_)) {
write!(f, "{} {{", data.name)?; write!(f, "{} {{", data.name.display(f.db.upcast()))?;
if let Some((id, data)) = it.next() { if let Some((id, data)) = it.next() {
write!(f, " {}: ", data.name)?; write!(f, " {}: ", data.name.display(f.db.upcast()))?;
render_field(f, id)?; render_field(f, id)?;
} }
for (id, data) in it { for (id, data) in it {
write!(f, ", {}: ", data.name)?; write!(f, ", {}: ", data.name.display(f.db.upcast()))?;
render_field(f, id)?; render_field(f, id)?;
} }
write!(f, " }}")?; write!(f, " }}")?;
} else { } else {
let mut it = it.map(|x| x.0); let mut it = it.map(|x| x.0);
write!(f, "{}(", data.name)?; write!(f, "{}(", data.name.display(f.db.upcast()))?;
if let Some(id) = it.next() { if let Some(id) = it.next() {
render_field(f, id)?; render_field(f, id)?;
} }
@ -570,10 +577,12 @@ fn render_const_scalar(
} }
return Ok(()); return Ok(());
} }
VariantData::Unit => write!(f, "{}", data.name), VariantData::Unit => write!(f, "{}", data.name.display(f.db.upcast())),
} }
} }
hir_def::AdtId::UnionId(u) => write!(f, "{}", f.db.union_data(u).name), hir_def::AdtId::UnionId(u) => {
write!(f, "{}", f.db.union_data(u).name.display(f.db.upcast()))
}
hir_def::AdtId::EnumId(_) => f.write_str("<enum-not-supported>"), hir_def::AdtId::EnumId(_) => f.write_str("<enum-not-supported>"),
}, },
chalk_ir::TyKind::FnDef(..) => ty.hir_fmt(f), chalk_ir::TyKind::FnDef(..) => ty.hir_fmt(f),
@ -739,11 +748,17 @@ impl HirDisplay for Ty {
let sig = db.callable_item_signature(def).substitute(Interner, parameters); let sig = db.callable_item_signature(def).substitute(Interner, parameters);
f.start_location_link(def.into()); f.start_location_link(def.into());
match def { match def {
CallableDefId::FunctionId(ff) => write!(f, "fn {}", db.function_data(ff).name)?, CallableDefId::FunctionId(ff) => {
CallableDefId::StructId(s) => write!(f, "{}", db.struct_data(s).name)?, write!(f, "fn {}", db.function_data(ff).name.display(f.db.upcast()))?
CallableDefId::EnumVariantId(e) => {
write!(f, "{}", db.enum_data(e.parent).variants[e.local_id].name)?
} }
CallableDefId::StructId(s) => {
write!(f, "{}", db.struct_data(s).name.display(f.db.upcast()))?
}
CallableDefId::EnumVariantId(e) => write!(
f,
"{}",
db.enum_data(e.parent).variants[e.local_id].name.display(f.db.upcast())
)?,
}; };
f.end_location_link(); f.end_location_link();
if parameters.len(Interner) > 0 { if parameters.len(Interner) > 0 {
@ -783,7 +798,7 @@ impl HirDisplay for Ty {
hir_def::AdtId::UnionId(it) => db.union_data(it).name.clone(), hir_def::AdtId::UnionId(it) => db.union_data(it).name.clone(),
hir_def::AdtId::EnumId(it) => db.enum_data(it).name.clone(), hir_def::AdtId::EnumId(it) => db.enum_data(it).name.clone(),
}; };
write!(f, "{name}")?; write!(f, "{}", name.display(f.db.upcast()))?;
} }
DisplayTarget::SourceCode { module_id, allow_opaque: _ } => { DisplayTarget::SourceCode { module_id, allow_opaque: _ } => {
if let Some(path) = find_path::find_path( if let Some(path) = find_path::find_path(
@ -792,7 +807,7 @@ impl HirDisplay for Ty {
module_id, module_id,
false, false,
) { ) {
write!(f, "{path}")?; write!(f, "{}", path.display(f.db.upcast()))?;
} else { } else {
return Err(HirDisplayError::DisplaySourceCodeError( return Err(HirDisplayError::DisplaySourceCodeError(
DisplaySourceCodeError::PathNotFound, DisplaySourceCodeError::PathNotFound,
@ -818,12 +833,12 @@ impl HirDisplay for Ty {
// Use placeholder associated types when the target is test (https://rust-lang.github.io/chalk/book/clauses/type_equality.html#placeholder-associated-types) // Use placeholder associated types when the target is test (https://rust-lang.github.io/chalk/book/clauses/type_equality.html#placeholder-associated-types)
if f.display_target.is_test() { if f.display_target.is_test() {
f.start_location_link(trait_.into()); f.start_location_link(trait_.into());
write!(f, "{}", trait_data.name)?; write!(f, "{}", trait_data.name.display(f.db.upcast()))?;
f.end_location_link(); f.end_location_link();
write!(f, "::")?; write!(f, "::")?;
f.start_location_link(type_alias.into()); f.start_location_link(type_alias.into());
write!(f, "{}", type_alias_data.name)?; write!(f, "{}", type_alias_data.name.display(f.db.upcast()))?;
f.end_location_link(); f.end_location_link();
// Note that the generic args for the associated type come before those for the // Note that the generic args for the associated type come before those for the
// trait (including the self type). // trait (including the self type).
@ -846,7 +861,7 @@ impl HirDisplay for Ty {
let alias = from_foreign_def_id(*type_alias); let alias = from_foreign_def_id(*type_alias);
let type_alias = db.type_alias_data(alias); let type_alias = db.type_alias_data(alias);
f.start_location_link(alias.into()); f.start_location_link(alias.into());
write!(f, "{}", type_alias.name)?; write!(f, "{}", type_alias.name.display(f.db.upcast()))?;
f.end_location_link(); f.end_location_link();
} }
TyKind::OpaqueType(opaque_ty_id, parameters) => { TyKind::OpaqueType(opaque_ty_id, parameters) => {
@ -954,7 +969,11 @@ impl HirDisplay for Ty {
match param_data { match param_data {
TypeOrConstParamData::TypeParamData(p) => match p.provenance { TypeOrConstParamData::TypeParamData(p) => match p.provenance {
TypeParamProvenance::TypeParamList | TypeParamProvenance::TraitSelf => { TypeParamProvenance::TypeParamList | TypeParamProvenance::TraitSelf => {
write!(f, "{}", p.name.clone().unwrap_or_else(Name::missing))? write!(
f,
"{}",
p.name.clone().unwrap_or_else(Name::missing).display(f.db.upcast())
)?
} }
TypeParamProvenance::ArgumentImplTrait => { TypeParamProvenance::ArgumentImplTrait => {
let substs = generics.placeholder_subst(db); let substs = generics.placeholder_subst(db);
@ -983,7 +1002,7 @@ impl HirDisplay for Ty {
} }
}, },
TypeOrConstParamData::ConstParamData(p) => { TypeOrConstParamData::ConstParamData(p) => {
write!(f, "{}", p.name)?; write!(f, "{}", p.name.display(f.db.upcast()))?;
} }
} }
} }
@ -1258,7 +1277,7 @@ fn write_bounds_like_dyn_trait(
// existential) here, which is the only thing that's // existential) here, which is the only thing that's
// possible in actual Rust, and hence don't print it // possible in actual Rust, and hence don't print it
f.start_location_link(trait_.into()); f.start_location_link(trait_.into());
write!(f, "{}", f.db.trait_data(trait_).name)?; write!(f, "{}", f.db.trait_data(trait_).name.display(f.db.upcast()))?;
f.end_location_link(); f.end_location_link();
if let [_, params @ ..] = &*trait_ref.substitution.as_slice(Interner) { if let [_, params @ ..] = &*trait_ref.substitution.as_slice(Interner) {
if is_fn_trait { if is_fn_trait {
@ -1297,7 +1316,7 @@ fn write_bounds_like_dyn_trait(
let assoc_ty_id = from_assoc_type_id(proj.associated_ty_id); let assoc_ty_id = from_assoc_type_id(proj.associated_ty_id);
let type_alias = f.db.type_alias_data(assoc_ty_id); let type_alias = f.db.type_alias_data(assoc_ty_id);
f.start_location_link(assoc_ty_id.into()); f.start_location_link(assoc_ty_id.into());
write!(f, "{}", type_alias.name)?; write!(f, "{}", type_alias.name.display(f.db.upcast()))?;
f.end_location_link(); f.end_location_link();
let proj_arg_count = generics(f.db.upcast(), assoc_ty_id.into()).len_self(); let proj_arg_count = generics(f.db.upcast(), assoc_ty_id.into()).len_self();
@ -1364,7 +1383,7 @@ fn fmt_trait_ref(
} }
let trait_ = tr.hir_trait_id(); let trait_ = tr.hir_trait_id();
f.start_location_link(trait_.into()); f.start_location_link(trait_.into());
write!(f, "{}", f.db.trait_data(trait_).name)?; write!(f, "{}", f.db.trait_data(trait_).name.display(f.db.upcast()))?;
f.end_location_link(); f.end_location_link();
if tr.substitution.len(Interner) > 1 { if tr.substitution.len(Interner) > 1 {
write!(f, "<")?; write!(f, "<")?;
@ -1394,7 +1413,7 @@ impl HirDisplay for WhereClause {
write!(f, ">::",)?; write!(f, ">::",)?;
let type_alias = from_assoc_type_id(projection_ty.associated_ty_id); let type_alias = from_assoc_type_id(projection_ty.associated_ty_id);
f.start_location_link(type_alias.into()); f.start_location_link(type_alias.into());
write!(f, "{}", f.db.type_alias_data(type_alias).name,)?; write!(f, "{}", f.db.type_alias_data(type_alias).name.display(f.db.upcast()),)?;
f.end_location_link(); f.end_location_link();
write!(f, " = ")?; write!(f, " = ")?;
ty.hir_fmt(f)?; ty.hir_fmt(f)?;
@ -1432,7 +1451,8 @@ impl HirDisplay for LifetimeData {
let id = lt_from_placeholder_idx(f.db, *idx); let id = lt_from_placeholder_idx(f.db, *idx);
let generics = generics(f.db.upcast(), id.parent); let generics = generics(f.db.upcast(), id.parent);
let param_data = &generics.params.lifetimes[id.local_id]; let param_data = &generics.params.lifetimes[id.local_id];
write!(f, "{}", param_data.name) write!(f, "{}", param_data.name.display(f.db.upcast()))?;
Ok(())
} }
LifetimeData::Static => write!(f, "'static"), LifetimeData::Static => write!(f, "'static"),
LifetimeData::Erased => Ok(()), LifetimeData::Erased => Ok(()),
@ -1508,7 +1528,7 @@ impl HirDisplay for TypeRef {
}; };
write!(f, "&")?; write!(f, "&")?;
if let Some(lifetime) = lifetime { if let Some(lifetime) = lifetime {
write!(f, "{} ", lifetime.name)?; write!(f, "{} ", lifetime.name.display(f.db.upcast()))?;
} }
write!(f, "{mutability}")?; write!(f, "{mutability}")?;
inner.hir_fmt(f)?; inner.hir_fmt(f)?;
@ -1516,7 +1536,7 @@ impl HirDisplay for TypeRef {
TypeRef::Array(inner, len) => { TypeRef::Array(inner, len) => {
write!(f, "[")?; write!(f, "[")?;
inner.hir_fmt(f)?; inner.hir_fmt(f)?;
write!(f, "; {len}]")?; write!(f, "; {}]", len.display(f.db.upcast()))?;
} }
TypeRef::Slice(inner) => { TypeRef::Slice(inner) => {
write!(f, "[")?; write!(f, "[")?;
@ -1533,7 +1553,7 @@ impl HirDisplay for TypeRef {
for index in 0..function_parameters.len() { for index in 0..function_parameters.len() {
let (param_name, param_type) = &function_parameters[index]; let (param_name, param_type) = &function_parameters[index];
if let Some(name) = param_name { if let Some(name) = param_name {
write!(f, "{name}: ")?; write!(f, "{}: ", name.display(f.db.upcast()))?;
} }
param_type.hir_fmt(f)?; param_type.hir_fmt(f)?;
@ -1594,9 +1614,13 @@ impl HirDisplay for TypeBound {
} }
path.hir_fmt(f) path.hir_fmt(f)
} }
TypeBound::Lifetime(lifetime) => write!(f, "{}", lifetime.name), TypeBound::Lifetime(lifetime) => write!(f, "{}", lifetime.name.display(f.db.upcast())),
TypeBound::ForLifetime(lifetimes, path) => { TypeBound::ForLifetime(lifetimes, path) => {
write!(f, "for<{}> ", lifetimes.iter().format(", "))?; write!(
f,
"for<{}> ",
lifetimes.iter().map(|it| it.display(f.db.upcast())).format(", ")
)?;
path.hir_fmt(f) path.hir_fmt(f)
} }
TypeBound::Error => write!(f, "{{error}}"), TypeBound::Error => write!(f, "{{error}}"),
@ -1642,7 +1666,7 @@ impl HirDisplay for Path {
if !matches!(self.kind(), PathKind::Plain) || seg_idx > 0 { if !matches!(self.kind(), PathKind::Plain) || seg_idx > 0 {
write!(f, "::")?; write!(f, "::")?;
} }
write!(f, "{}", segment.name)?; write!(f, "{}", segment.name.display(f.db.upcast()))?;
if let Some(generic_args) = segment.args_and_bindings { if let Some(generic_args) = segment.args_and_bindings {
// We should be in type context, so format as `Foo<Bar>` instead of `Foo::<Bar>`. // We should be in type context, so format as `Foo<Bar>` instead of `Foo::<Bar>`.
// Do we actually format expressions? // Do we actually format expressions?
@ -1689,7 +1713,7 @@ impl HirDisplay for Path {
} else { } else {
write!(f, ", ")?; write!(f, ", ")?;
} }
write!(f, "{}", binding.name)?; write!(f, "{}", binding.name.display(f.db.upcast()))?;
match &binding.type_ref { match &binding.type_ref {
Some(ty) => { Some(ty) => {
write!(f, " = ")?; write!(f, " = ")?;
@ -1712,8 +1736,10 @@ impl HirDisplay for hir_def::path::GenericArg {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
match self { match self {
hir_def::path::GenericArg::Type(ty) => ty.hir_fmt(f), hir_def::path::GenericArg::Type(ty) => ty.hir_fmt(f),
hir_def::path::GenericArg::Const(c) => write!(f, "{c}"), hir_def::path::GenericArg::Const(c) => write!(f, "{}", c.display(f.db.upcast())),
hir_def::path::GenericArg::Lifetime(lifetime) => write!(f, "{}", lifetime.name), hir_def::path::GenericArg::Lifetime(lifetime) => {
write!(f, "{}", lifetime.name.display(f.db.upcast()))
}
} }
} }
} }

View file

@ -181,7 +181,7 @@ impl CapturedItem {
pub fn display_place(&self, owner: DefWithBodyId, db: &dyn HirDatabase) -> String { pub fn display_place(&self, owner: DefWithBodyId, db: &dyn HirDatabase) -> String {
let body = db.body(owner); let body = db.body(owner);
let mut result = body[self.place.local].name.to_string(); let mut result = body[self.place.local].name.display(db.upcast()).to_string();
let mut field_need_paren = false; let mut field_need_paren = false;
for proj in &self.place.projections { for proj in &self.place.projections {
match proj { match proj {

View file

@ -309,7 +309,12 @@ impl MirEvalError {
match func { match func {
Either::Left(func) => { Either::Left(func) => {
let function_name = db.function_data(*func); let function_name = db.function_data(*func);
writeln!(f, "In function {} ({:?})", function_name.name, func)?; writeln!(
f,
"In function {} ({:?})",
function_name.name.display(db.upcast()),
func
)?;
} }
Either::Right(clos) => { Either::Right(clos) => {
writeln!(f, "In {:?}", clos)?; writeln!(f, "In {:?}", clos)?;
@ -349,7 +354,7 @@ impl MirEvalError {
writeln!( writeln!(
f, f,
"Generic arg not provided for {}", "Generic arg not provided for {}",
param.name().unwrap_or(&Name::missing()) param.name().unwrap_or(&Name::missing()).display(db.upcast())
)?; )?;
writeln!(f, "Provided args: [")?; writeln!(f, "Provided args: [")?;
for g in subst.iter(Interner) { for g in subst.iter(Interner) {
@ -362,7 +367,8 @@ impl MirEvalError {
writeln!( writeln!(
f, f,
"MIR lowering for function `{}` ({:?}) failed due:", "MIR lowering for function `{}` ({:?}) failed due:",
function_name.name, func function_name.name.display(db.upcast()),
func
)?; )?;
err.pretty_print(f, db, span_formatter)?; err.pretty_print(f, db, span_formatter)?;
} }
@ -2070,7 +2076,11 @@ impl Evaluator<'_> {
Ok(r) => Ok(r), Ok(r) => Ok(r),
Err(e) => { Err(e) => {
let data = self.db.enum_data(variant.parent); let data = self.db.enum_data(variant.parent);
let name = format!("{}::{}", data.name, data.variants[variant.local_id].name); let name = format!(
"{}::{}",
data.name.display(self.db.upcast()),
data.variants[variant.local_id].name.display(self.db.upcast())
);
Err(MirEvalError::ConstEvalError(name, Box::new(e))) Err(MirEvalError::ConstEvalError(name, Box::new(e)))
} }
} }

View file

@ -13,7 +13,7 @@ fn eval_main(db: &TestDB, file_id: FileId) -> Result<(String, String), MirEvalEr
.declarations() .declarations()
.find_map(|x| match x { .find_map(|x| match x {
hir_def::ModuleDefId::FunctionId(x) => { hir_def::ModuleDefId::FunctionId(x) => {
if db.function_data(x).name.to_string() == "main" { if db.function_data(x).name.display(db).to_string() == "main" {
Some(x) Some(x)
} else { } else {
None None

View file

@ -619,7 +619,7 @@ impl<'ctx> MirLowerCtx<'ctx> {
} }
Expr::MethodCall { receiver, args, method_name, .. } => { Expr::MethodCall { receiver, args, method_name, .. } => {
let (func_id, generic_args) = let (func_id, generic_args) =
self.infer.method_resolution(expr_id).ok_or_else(|| MirLowerError::UnresolvedMethod(format!("{}", method_name)))?; self.infer.method_resolution(expr_id).ok_or_else(|| MirLowerError::UnresolvedMethod(method_name.display(self.db.upcast()).to_string()))?;
let func = Operand::from_fn(self.db, func_id, generic_args); let func = Operand::from_fn(self.db, func_id, generic_args);
self.lower_call_and_args( self.lower_call_and_args(
func, func,
@ -1614,7 +1614,11 @@ impl<'ctx> MirLowerCtx<'ctx> {
Ok(r) => Ok(r), Ok(r) => Ok(r),
Err(e) => { Err(e) => {
let data = self.db.enum_data(variant.parent); let data = self.db.enum_data(variant.parent);
let name = format!("{}::{}", data.name, data.variants[variant.local_id].name); let name = format!(
"{}::{}",
data.name.display(self.db.upcast()),
data.variants[variant.local_id].name.display(self.db.upcast())
);
Err(MirLowerError::ConstEvalError(name, Box::new(e))) Err(MirLowerError::ConstEvalError(name, Box::new(e)))
} }
} }
@ -1766,13 +1770,17 @@ pub fn mir_body_for_closure_query(
pub fn mir_body_query(db: &dyn HirDatabase, def: DefWithBodyId) -> Result<Arc<MirBody>> { pub fn mir_body_query(db: &dyn HirDatabase, def: DefWithBodyId) -> Result<Arc<MirBody>> {
let _p = profile::span("mir_body_query").detail(|| match def { let _p = profile::span("mir_body_query").detail(|| match def {
DefWithBodyId::FunctionId(it) => db.function_data(it).name.to_string(), DefWithBodyId::FunctionId(it) => db.function_data(it).name.display(db.upcast()).to_string(),
DefWithBodyId::StaticId(it) => db.static_data(it).name.clone().to_string(), DefWithBodyId::StaticId(it) => db.static_data(it).name.display(db.upcast()).to_string(),
DefWithBodyId::ConstId(it) => { DefWithBodyId::ConstId(it) => db
db.const_data(it).name.clone().unwrap_or_else(Name::missing).to_string() .const_data(it)
} .name
.clone()
.unwrap_or_else(Name::missing)
.display(db.upcast())
.to_string(),
DefWithBodyId::VariantId(it) => { DefWithBodyId::VariantId(it) => {
db.enum_data(it.parent).variants[it.local_id].name.to_string() db.enum_data(it.parent).variants[it.local_id].name.display(db.upcast()).to_string()
} }
}); });
let body = db.body(def); let body = db.body(def);

View file

@ -42,19 +42,23 @@ impl MirBody {
ctx.for_body(|this| match ctx.body.owner { ctx.for_body(|this| match ctx.body.owner {
hir_def::DefWithBodyId::FunctionId(id) => { hir_def::DefWithBodyId::FunctionId(id) => {
let data = db.function_data(id); let data = db.function_data(id);
w!(this, "fn {}() ", data.name); w!(this, "fn {}() ", data.name.display(db.upcast()));
} }
hir_def::DefWithBodyId::StaticId(id) => { hir_def::DefWithBodyId::StaticId(id) => {
let data = db.static_data(id); let data = db.static_data(id);
w!(this, "static {}: _ = ", data.name); w!(this, "static {}: _ = ", data.name.display(db.upcast()));
} }
hir_def::DefWithBodyId::ConstId(id) => { hir_def::DefWithBodyId::ConstId(id) => {
let data = db.const_data(id); let data = db.const_data(id);
w!(this, "const {}: _ = ", data.name.as_ref().unwrap_or(&Name::missing())); w!(
this,
"const {}: _ = ",
data.name.as_ref().unwrap_or(&Name::missing()).display(db.upcast())
);
} }
hir_def::DefWithBodyId::VariantId(id) => { hir_def::DefWithBodyId::VariantId(id) => {
let data = db.enum_data(id.parent); let data = db.enum_data(id.parent);
w!(this, "enum {} = ", data.name); w!(this, "enum {} = ", data.name.display(db.upcast()));
} }
}); });
ctx.result ctx.result
@ -99,11 +103,16 @@ enum LocalName {
Binding(Name, LocalId), Binding(Name, LocalId),
} }
impl Display for LocalName { impl HirDisplay for LocalName {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { fn hir_fmt(
&self,
f: &mut crate::display::HirFormatter<'_>,
) -> Result<(), crate::display::HirDisplayError> {
match self { match self {
LocalName::Unknown(l) => write!(f, "_{}", u32::from(l.into_raw())), LocalName::Unknown(l) => write!(f, "_{}", u32::from(l.into_raw())),
LocalName::Binding(n, l) => write!(f, "{n}_{}", u32::from(l.into_raw())), LocalName::Binding(n, l) => {
write!(f, "{}_{}", n.display(f.db.upcast()), u32::from(l.into_raw()))
}
} }
} }
} }
@ -177,7 +186,12 @@ impl<'a> MirPrettyCtx<'a> {
fn locals(&mut self) { fn locals(&mut self) {
for (id, local) in self.body.locals.iter() { for (id, local) in self.body.locals.iter() {
wln!(self, "let {}: {};", self.local_name(id), self.hir_display(&local.ty)); wln!(
self,
"let {}: {};",
self.local_name(id).display(self.db),
self.hir_display(&local.ty)
);
} }
} }
@ -206,10 +220,10 @@ impl<'a> MirPrettyCtx<'a> {
wln!(this, ";"); wln!(this, ";");
} }
StatementKind::StorageDead(p) => { StatementKind::StorageDead(p) => {
wln!(this, "StorageDead({})", this.local_name(*p)); wln!(this, "StorageDead({})", this.local_name(*p).display(self.db));
} }
StatementKind::StorageLive(p) => { StatementKind::StorageLive(p) => {
wln!(this, "StorageLive({})", this.local_name(*p)); wln!(this, "StorageLive({})", this.local_name(*p).display(self.db));
} }
StatementKind::Deinit(p) => { StatementKind::Deinit(p) => {
w!(this, "Deinit("); w!(this, "Deinit(");
@ -267,7 +281,7 @@ impl<'a> MirPrettyCtx<'a> {
fn f(this: &mut MirPrettyCtx<'_>, local: LocalId, projections: &[PlaceElem]) { fn f(this: &mut MirPrettyCtx<'_>, local: LocalId, projections: &[PlaceElem]) {
let Some((last, head)) = projections.split_last() else { let Some((last, head)) = projections.split_last() else {
// no projection // no projection
w!(this, "{}", this.local_name(local)); w!(this, "{}", this.local_name(local).display(this.db));
return; return;
}; };
match last { match last {
@ -285,11 +299,16 @@ impl<'a> MirPrettyCtx<'a> {
f(this, local, head); f(this, local, head);
let variant_name = let variant_name =
&this.db.enum_data(e.parent).variants[e.local_id].name; &this.db.enum_data(e.parent).variants[e.local_id].name;
w!(this, " as {}).{}", variant_name, name); w!(
this,
" as {}).{}",
variant_name.display(this.db.upcast()),
name.display(this.db.upcast())
);
} }
hir_def::VariantId::StructId(_) | hir_def::VariantId::UnionId(_) => { hir_def::VariantId::StructId(_) | hir_def::VariantId::UnionId(_) => {
f(this, local, head); f(this, local, head);
w!(this, ".{name}"); w!(this, ".{}", name.display(this.db.upcast()));
} }
} }
} }
@ -299,7 +318,7 @@ impl<'a> MirPrettyCtx<'a> {
} }
ProjectionElem::Index(l) => { ProjectionElem::Index(l) => {
f(this, local, head); f(this, local, head);
w!(this, "[{}]", this.local_name(*l)); w!(this, "[{}]", this.local_name(*l).display(this.db));
} }
x => { x => {
f(this, local, head); f(this, local, head);

View file

@ -24,7 +24,8 @@ impl DebugContext<'_> {
AdtId::UnionId(it) => self.0.union_data(it).name.clone(), AdtId::UnionId(it) => self.0.union_data(it).name.clone(),
AdtId::EnumId(it) => self.0.enum_data(it).name.clone(), AdtId::EnumId(it) => self.0.enum_data(it).name.clone(),
}; };
name.fmt(f) name.display(self.0.upcast()).fmt(f)?;
Ok(())
} }
pub(crate) fn debug_trait_id( pub(crate) fn debug_trait_id(
@ -34,7 +35,8 @@ impl DebugContext<'_> {
) -> Result<(), fmt::Error> { ) -> Result<(), fmt::Error> {
let trait_: hir_def::TraitId = from_chalk_trait_id(id); let trait_: hir_def::TraitId = from_chalk_trait_id(id);
let trait_data = self.0.trait_data(trait_); let trait_data = self.0.trait_data(trait_);
trait_data.name.fmt(f) trait_data.name.display(self.0.upcast()).fmt(f)?;
Ok(())
} }
pub(crate) fn debug_assoc_type_id( pub(crate) fn debug_assoc_type_id(
@ -49,7 +51,13 @@ impl DebugContext<'_> {
_ => panic!("associated type not in trait"), _ => panic!("associated type not in trait"),
}; };
let trait_data = self.0.trait_data(trait_); let trait_data = self.0.trait_data(trait_);
write!(fmt, "{}::{}", trait_data.name, type_alias_data.name) write!(
fmt,
"{}::{}",
trait_data.name.display(self.0.upcast()),
type_alias_data.name.display(self.0.upcast())
)?;
Ok(())
} }
pub(crate) fn debug_projection_ty( pub(crate) fn debug_projection_ty(
@ -67,7 +75,7 @@ impl DebugContext<'_> {
let trait_ref = projection_ty.trait_ref(self.0); let trait_ref = projection_ty.trait_ref(self.0);
let trait_params = trait_ref.substitution.as_slice(Interner); let trait_params = trait_ref.substitution.as_slice(Interner);
let self_ty = trait_ref.self_type_parameter(Interner); let self_ty = trait_ref.self_type_parameter(Interner);
write!(fmt, "<{self_ty:?} as {trait_name}")?; write!(fmt, "<{self_ty:?} as {}", trait_name.display(self.0.upcast()))?;
if trait_params.len() > 1 { if trait_params.len() > 1 {
write!( write!(
fmt, fmt,
@ -75,7 +83,7 @@ impl DebugContext<'_> {
trait_params[1..].iter().format_with(", ", |x, f| f(&format_args!("{x:?}"))), trait_params[1..].iter().format_with(", ", |x, f| f(&format_args!("{x:?}"))),
)?; )?;
} }
write!(fmt, ">::{}", type_alias_data.name)?; write!(fmt, ">::{}", type_alias_data.name.display(self.0.upcast()))?;
let proj_params_count = projection_ty.substitution.len(Interner) - trait_params.len(); let proj_params_count = projection_ty.substitution.len(Interner) - trait_params.len();
let proj_params = &projection_ty.substitution.as_slice(Interner)[..proj_params_count]; let proj_params = &projection_ty.substitution.as_slice(Interner)[..proj_params_count];
@ -105,9 +113,9 @@ impl DebugContext<'_> {
} }
}; };
match def { match def {
CallableDefId::FunctionId(_) => write!(fmt, "{{fn {name}}}"), CallableDefId::FunctionId(_) => write!(fmt, "{{fn {}}}", name.display(self.0.upcast())),
CallableDefId::StructId(_) | CallableDefId::EnumVariantId(_) => { CallableDefId::StructId(_) | CallableDefId::EnumVariantId(_) => {
write!(fmt, "{{ctor {name}}}") write!(fmt, "{{ctor {}}}", name.display(self.0.upcast()))
} }
} }
} }

View file

@ -88,7 +88,7 @@ pub(crate) fn trait_solve_query(
) -> Option<Solution> { ) -> Option<Solution> {
let _p = profile::span("trait_solve_query").detail(|| match &goal.value.goal.data(Interner) { let _p = profile::span("trait_solve_query").detail(|| match &goal.value.goal.data(Interner) {
GoalData::DomainGoal(DomainGoal::Holds(WhereClause::Implemented(it))) => { GoalData::DomainGoal(DomainGoal::Holds(WhereClause::Implemented(it))) => {
db.trait_data(it.hir_trait_id()).name.to_string() db.trait_data(it.hir_trait_id()).name.display(db.upcast()).to_string()
} }
GoalData::DomainGoal(DomainGoal::Holds(WhereClause::AliasEq(_))) => "alias_eq".to_string(), GoalData::DomainGoal(DomainGoal::Holds(WhereClause::AliasEq(_))) => "alias_eq".to_string(),
_ => "??".to_string(), _ => "??".to_string(),

View file

@ -51,7 +51,7 @@ impl HirDisplay for Function {
// FIXME: String escape? // FIXME: String escape?
write!(f, "extern \"{}\" ", &**abi)?; write!(f, "extern \"{}\" ", &**abi)?;
} }
write!(f, "fn {}", data.name)?; write!(f, "fn {}", data.name.display(f.db.upcast()))?;
write_generic_params(GenericDefId::FunctionId(self.id), f)?; write_generic_params(GenericDefId::FunctionId(self.id), f)?;
@ -63,7 +63,7 @@ impl HirDisplay for Function {
{ {
f.write_char('&')?; f.write_char('&')?;
if let Some(lifetime) = lifetime { if let Some(lifetime) = lifetime {
write!(f, "{} ", lifetime.name)?; write!(f, "{} ", lifetime.name.display(f.db.upcast()))?;
} }
if let hir_def::type_ref::Mutability::Mut = mut_ { if let hir_def::type_ref::Mutability::Mut = mut_ {
f.write_str("mut ")?; f.write_str("mut ")?;
@ -90,7 +90,7 @@ impl HirDisplay for Function {
} }
} }
match local { match local {
Some(name) => write!(f, "{name}: ")?, Some(name) => write!(f, "{}: ", name.display(f.db.upcast()))?,
None => f.write_str("_: ")?, None => f.write_str("_: ")?,
} }
type_ref.hir_fmt(f)?; type_ref.hir_fmt(f)?;
@ -151,7 +151,7 @@ impl HirDisplay for Struct {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write_visibility(self.module(f.db).id, self.visibility(f.db), f)?; write_visibility(self.module(f.db).id, self.visibility(f.db), f)?;
f.write_str("struct ")?; f.write_str("struct ")?;
write!(f, "{}", self.name(f.db))?; write!(f, "{}", self.name(f.db).display(f.db.upcast()))?;
let def_id = GenericDefId::AdtId(AdtId::StructId(self.id)); let def_id = GenericDefId::AdtId(AdtId::StructId(self.id));
write_generic_params(def_id, f)?; write_generic_params(def_id, f)?;
write_where_clause(def_id, f)?; write_where_clause(def_id, f)?;
@ -163,7 +163,7 @@ impl HirDisplay for Enum {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write_visibility(self.module(f.db).id, self.visibility(f.db), f)?; write_visibility(self.module(f.db).id, self.visibility(f.db), f)?;
f.write_str("enum ")?; f.write_str("enum ")?;
write!(f, "{}", self.name(f.db))?; write!(f, "{}", self.name(f.db).display(f.db.upcast()))?;
let def_id = GenericDefId::AdtId(AdtId::EnumId(self.id)); let def_id = GenericDefId::AdtId(AdtId::EnumId(self.id));
write_generic_params(def_id, f)?; write_generic_params(def_id, f)?;
write_where_clause(def_id, f)?; write_where_clause(def_id, f)?;
@ -175,7 +175,7 @@ impl HirDisplay for Union {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write_visibility(self.module(f.db).id, self.visibility(f.db), f)?; write_visibility(self.module(f.db).id, self.visibility(f.db), f)?;
f.write_str("union ")?; f.write_str("union ")?;
write!(f, "{}", self.name(f.db))?; write!(f, "{}", self.name(f.db).display(f.db.upcast()))?;
let def_id = GenericDefId::AdtId(AdtId::UnionId(self.id)); let def_id = GenericDefId::AdtId(AdtId::UnionId(self.id));
write_generic_params(def_id, f)?; write_generic_params(def_id, f)?;
write_where_clause(def_id, f)?; write_where_clause(def_id, f)?;
@ -186,14 +186,14 @@ impl HirDisplay for Union {
impl HirDisplay for Field { impl HirDisplay for Field {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write_visibility(self.parent.module(f.db).id, self.visibility(f.db), f)?; write_visibility(self.parent.module(f.db).id, self.visibility(f.db), f)?;
write!(f, "{}: ", self.name(f.db))?; write!(f, "{}: ", self.name(f.db).display(f.db.upcast()))?;
self.ty(f.db).hir_fmt(f) self.ty(f.db).hir_fmt(f)
} }
} }
impl HirDisplay for Variant { impl HirDisplay for Variant {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write!(f, "{}", self.name(f.db))?; write!(f, "{}", self.name(f.db).display(f.db.upcast()))?;
let data = self.variant_data(f.db); let data = self.variant_data(f.db);
match &*data { match &*data {
VariantData::Unit => {} VariantData::Unit => {}
@ -222,7 +222,7 @@ impl HirDisplay for Variant {
f.write_str(", ")?; f.write_str(", ")?;
} }
// Enum variant fields must be pub. // Enum variant fields must be pub.
write!(f, "{}: ", field.name)?; write!(f, "{}: ", field.name.display(f.db.upcast()))?;
field.type_ref.hir_fmt(f)?; field.type_ref.hir_fmt(f)?;
} }
f.write_str(" }")?; f.write_str(" }")?;
@ -259,7 +259,7 @@ impl HirDisplay for TypeOrConstParam {
impl HirDisplay for TypeParam { impl HirDisplay for TypeParam {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write!(f, "{}", self.name(f.db))?; write!(f, "{}", self.name(f.db).display(f.db.upcast()))?;
if f.omit_verbose_types() { if f.omit_verbose_types() {
return Ok(()); return Ok(());
} }
@ -286,13 +286,13 @@ impl HirDisplay for TypeParam {
impl HirDisplay for LifetimeParam { impl HirDisplay for LifetimeParam {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write!(f, "{}", self.name(f.db)) write!(f, "{}", self.name(f.db).display(f.db.upcast()))
} }
} }
impl HirDisplay for ConstParam { impl HirDisplay for ConstParam {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write!(f, "const {}: ", self.name(f.db))?; write!(f, "const {}: ", self.name(f.db).display(f.db.upcast()))?;
self.ty(f.db).hir_fmt(f) self.ty(f.db).hir_fmt(f)
} }
} }
@ -325,7 +325,7 @@ fn write_generic_params(
}; };
for (_, lifetime) in params.lifetimes.iter() { for (_, lifetime) in params.lifetimes.iter() {
delim(f)?; delim(f)?;
write!(f, "{}", lifetime.name)?; write!(f, "{}", lifetime.name.display(f.db.upcast()))?;
} }
for (_, ty) in params.type_or_consts.iter() { for (_, ty) in params.type_or_consts.iter() {
if let Some(name) = &ty.name() { if let Some(name) = &ty.name() {
@ -335,7 +335,7 @@ fn write_generic_params(
continue; continue;
} }
delim(f)?; delim(f)?;
write!(f, "{name}")?; write!(f, "{}", name.display(f.db.upcast()))?;
if let Some(default) = &ty.default { if let Some(default) = &ty.default {
f.write_str(" = ")?; f.write_str(" = ")?;
default.hir_fmt(f)?; default.hir_fmt(f)?;
@ -343,7 +343,7 @@ fn write_generic_params(
} }
TypeOrConstParamData::ConstParamData(c) => { TypeOrConstParamData::ConstParamData(c) => {
delim(f)?; delim(f)?;
write!(f, "const {name}: ")?; write!(f, "const {}: ", name.display(f.db.upcast()))?;
c.ty.hir_fmt(f)?; c.ty.hir_fmt(f)?;
} }
} }
@ -380,7 +380,7 @@ fn write_where_clause(def: GenericDefId, f: &mut HirFormatter<'_>) -> Result<(),
WherePredicateTypeTarget::TypeRef(ty) => ty.hir_fmt(f), WherePredicateTypeTarget::TypeRef(ty) => ty.hir_fmt(f),
WherePredicateTypeTarget::TypeOrConstParam(id) => { WherePredicateTypeTarget::TypeOrConstParam(id) => {
match &params.type_or_consts[*id].name() { match &params.type_or_consts[*id].name() {
Some(name) => write!(f, "{name}"), Some(name) => write!(f, "{}", name.display(f.db.upcast())),
None => f.write_str("{unnamed}"), None => f.write_str("{unnamed}"),
} }
} }
@ -412,10 +412,15 @@ fn write_where_clause(def: GenericDefId, f: &mut HirFormatter<'_>) -> Result<(),
WherePredicate::Lifetime { target, bound } => { WherePredicate::Lifetime { target, bound } => {
if matches!(prev_pred, Some(WherePredicate::Lifetime { target: target_, .. }) if target_ == target) if matches!(prev_pred, Some(WherePredicate::Lifetime { target: target_, .. }) if target_ == target)
{ {
write!(f, " + {}", bound.name)?; write!(f, " + {}", bound.name.display(f.db.upcast()))?;
} else { } else {
new_predicate(f)?; new_predicate(f)?;
write!(f, "{}: {}", target.name, bound.name)?; write!(
f,
"{}: {}",
target.name.display(f.db.upcast()),
bound.name.display(f.db.upcast())
)?;
} }
} }
WherePredicate::ForLifetime { lifetimes, target, bound } => { WherePredicate::ForLifetime { lifetimes, target, bound } => {
@ -432,7 +437,7 @@ fn write_where_clause(def: GenericDefId, f: &mut HirFormatter<'_>) -> Result<(),
if idx != 0 { if idx != 0 {
f.write_str(", ")?; f.write_str(", ")?;
} }
write!(f, "{lifetime}")?; write!(f, "{}", lifetime.display(f.db.upcast()))?;
} }
f.write_str("> ")?; f.write_str("> ")?;
write_target(target, f)?; write_target(target, f)?;
@ -462,7 +467,7 @@ impl HirDisplay for Const {
let data = db.const_data(self.id); let data = db.const_data(self.id);
f.write_str("const ")?; f.write_str("const ")?;
match &data.name { match &data.name {
Some(name) => write!(f, "{name}: ")?, Some(name) => write!(f, "{}: ", name.display(f.db.upcast()))?,
None => f.write_str("_: ")?, None => f.write_str("_: ")?,
} }
data.type_ref.hir_fmt(f)?; data.type_ref.hir_fmt(f)?;
@ -478,7 +483,7 @@ impl HirDisplay for Static {
if data.mutable { if data.mutable {
f.write_str("mut ")?; f.write_str("mut ")?;
} }
write!(f, "{}: ", &data.name)?; write!(f, "{}: ", data.name.display(f.db.upcast()))?;
data.type_ref.hir_fmt(f)?; data.type_ref.hir_fmt(f)?;
Ok(()) Ok(())
} }
@ -494,7 +499,7 @@ impl HirDisplay for Trait {
if data.is_auto { if data.is_auto {
f.write_str("auto ")?; f.write_str("auto ")?;
} }
write!(f, "trait {}", data.name)?; write!(f, "trait {}", data.name.display(f.db.upcast()))?;
let def_id = GenericDefId::TraitId(self.id); let def_id = GenericDefId::TraitId(self.id);
write_generic_params(def_id, f)?; write_generic_params(def_id, f)?;
write_where_clause(def_id, f)?; write_where_clause(def_id, f)?;
@ -506,7 +511,7 @@ impl HirDisplay for TraitAlias {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write_visibility(self.module(f.db).id, self.visibility(f.db), f)?; write_visibility(self.module(f.db).id, self.visibility(f.db), f)?;
let data = f.db.trait_alias_data(self.id); let data = f.db.trait_alias_data(self.id);
write!(f, "trait {}", data.name)?; write!(f, "trait {}", data.name.display(f.db.upcast()))?;
let def_id = GenericDefId::TraitAliasId(self.id); let def_id = GenericDefId::TraitAliasId(self.id);
write_generic_params(def_id, f)?; write_generic_params(def_id, f)?;
f.write_str(" = ")?; f.write_str(" = ")?;
@ -522,7 +527,7 @@ impl HirDisplay for TypeAlias {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
write_visibility(self.module(f.db).id, self.visibility(f.db), f)?; write_visibility(self.module(f.db).id, self.visibility(f.db), f)?;
let data = f.db.type_alias_data(self.id); let data = f.db.type_alias_data(self.id);
write!(f, "type {}", data.name)?; write!(f, "type {}", data.name.display(f.db.upcast()))?;
let def_id = GenericDefId::TypeAliasId(self.id); let def_id = GenericDefId::TypeAliasId(self.id);
write_generic_params(def_id, f)?; write_generic_params(def_id, f)?;
write_where_clause(def_id, f)?; write_where_clause(def_id, f)?;
@ -542,7 +547,7 @@ impl HirDisplay for Module {
fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> { fn hir_fmt(&self, f: &mut HirFormatter<'_>) -> Result<(), HirDisplayError> {
// FIXME: Module doesn't have visibility saved in data. // FIXME: Module doesn't have visibility saved in data.
match self.name(f.db) { match self.name(f.db) {
Some(name) => write!(f, "mod {name}"), Some(name) => write!(f, "mod {}", name.display(f.db.upcast())),
None if self.is_crate_root(f.db) => match self.krate(f.db).display_name(f.db) { None if self.is_crate_root(f.db) => match self.krate(f.db).display_name(f.db) {
Some(name) => write!(f, "extern crate {name}"), Some(name) => write!(f, "extern crate {name}"),
None => f.write_str("extern crate {unknown}"), None => f.write_str("extern crate {unknown}"),
@ -559,6 +564,6 @@ impl HirDisplay for Macro {
hir_def::MacroId::MacroRulesId(_) => f.write_str("macro_rules!"), hir_def::MacroId::MacroRulesId(_) => f.write_str("macro_rules!"),
hir_def::MacroId::ProcMacroId(_) => f.write_str("proc_macro"), hir_def::MacroId::ProcMacroId(_) => f.write_str("proc_macro"),
}?; }?;
write!(f, " {}", self.name(f.db)) write!(f, " {}", self.name(f.db).display(f.db.upcast()))
} }
} }

View file

@ -329,7 +329,7 @@ impl ModuleDef {
segments.extend(m.name(db)) segments.extend(m.name(db))
} }
segments.reverse(); segments.reverse();
Some(segments.into_iter().join("::")) Some(segments.iter().map(|it| it.display(db.upcast())).join("::"))
} }
pub fn canonical_module_path( pub fn canonical_module_path(
@ -555,7 +555,11 @@ impl Module {
/// Fills `acc` with the module's diagnostics. /// Fills `acc` with the module's diagnostics.
pub fn diagnostics(self, db: &dyn HirDatabase, acc: &mut Vec<AnyDiagnostic>) { pub fn diagnostics(self, db: &dyn HirDatabase, acc: &mut Vec<AnyDiagnostic>) {
let _p = profile::span("Module::diagnostics").detail(|| { let _p = profile::span("Module::diagnostics").detail(|| {
format!("{:?}", self.name(db).map_or("<unknown>".into(), |name| name.to_string())) format!(
"{:?}",
self.name(db)
.map_or("<unknown>".into(), |name| name.display(db.upcast()).to_string())
)
}); });
let def_map = self.id.def_map(db.upcast()); let def_map = self.id.def_map(db.upcast());
for diag in def_map.diagnostics() { for diag in def_map.diagnostics() {

View file

@ -184,7 +184,8 @@ fn try_gen_trait_body(
trait_ref: hir::TraitRef, trait_ref: hir::TraitRef,
impl_def: &ast::Impl, impl_def: &ast::Impl,
) -> Option<()> { ) -> Option<()> {
let trait_path = make::ext::ident_path(&trait_ref.trait_().name(ctx.db()).to_string()); let trait_path =
make::ext::ident_path(&trait_ref.trait_().name(ctx.db()).display(ctx.db()).to_string());
let hir_ty = ctx.sema.resolve_type(&impl_def.self_ty()?)?; let hir_ty = ctx.sema.resolve_type(&impl_def.self_ty()?)?;
let adt = hir_ty.as_adt()?.source(ctx.db())?; let adt = hir_ty.as_adt()?.source(ctx.db())?;
gen_trait_fn_body(func, &trait_path, &adt.value, Some(trait_ref)) gen_trait_fn_body(func, &trait_path, &adt.value, Some(trait_ref))

View file

@ -132,7 +132,7 @@ pub(crate) fn auto_import(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option<
acc.add_group( acc.add_group(
&group_label, &group_label,
AssistId("auto_import", AssistKind::QuickFix), AssistId("auto_import", AssistKind::QuickFix),
format!("Import `{import_path}`"), format!("Import `{}`", import_path.display(ctx.db())),
range, range,
|builder| { |builder| {
let scope = match scope.clone() { let scope = match scope.clone() {

View file

@ -160,7 +160,7 @@ pub(crate) fn convert_bool_then_to_if(acc: &mut Assists, ctx: &AssistContext<'_>
}; };
// Verify this is `bool::then` that is being called. // Verify this is `bool::then` that is being called.
let func = ctx.sema.resolve_method_call(&mcall)?; let func = ctx.sema.resolve_method_call(&mcall)?;
if func.name(ctx.sema.db).to_string() != "then" { if func.name(ctx.sema.db).display(ctx.db()).to_string() != "then" {
return None; return None;
} }
let assoc = func.as_assoc_item(ctx.sema.db)?; let assoc = func.as_assoc_item(ctx.sema.db)?;

View file

@ -119,7 +119,7 @@ pub(crate) fn convert_for_loop_with_for_each(
{ {
// We have either "for x in &col" and col implements a method called iter // We have either "for x in &col" and col implements a method called iter
// or "for x in &mut col" and col implements a method called iter_mut // or "for x in &mut col" and col implements a method called iter_mut
format_to!(buf, "{expr_behind_ref}.{method}()"); format_to!(buf, "{expr_behind_ref}.{}()", method.display(ctx.db()));
} else if let ast::Expr::RangeExpr(..) = iterable { } else if let ast::Expr::RangeExpr(..) = iterable {
// range expressions need to be parenthesized for the syntax to be correct // range expressions need to be parenthesized for the syntax to be correct
format_to!(buf, "({iterable})"); format_to!(buf, "({iterable})");

View file

@ -65,7 +65,7 @@ pub(crate) fn expand_glob_import(acc: &mut Assists, ctx: &AssistContext<'_>) ->
let names_to_import = find_names_to_import(ctx, refs_in_target, imported_defs); let names_to_import = find_names_to_import(ctx, refs_in_target, imported_defs);
let expanded = make::use_tree_list(names_to_import.iter().map(|n| { let expanded = make::use_tree_list(names_to_import.iter().map(|n| {
let path = make::ext::ident_path(&n.to_string()); let path = make::ext::ident_path(&n.display(ctx.db()).to_string());
make::use_tree(path, None, None, false) make::use_tree(path, None, None, false)
})) }))
.clone_for_update(); .clone_for_update();

View file

@ -183,7 +183,9 @@ pub(crate) fn extract_function(acc: &mut Assists, ctx: &AssistContext<'_>) -> Op
fn make_function_name(semantics_scope: &hir::SemanticsScope<'_>) -> ast::NameRef { fn make_function_name(semantics_scope: &hir::SemanticsScope<'_>) -> ast::NameRef {
let mut names_in_scope = vec![]; let mut names_in_scope = vec![];
semantics_scope.process_all_names(&mut |name, _| names_in_scope.push(name.to_string())); semantics_scope.process_all_names(&mut |name, _| {
names_in_scope.push(name.display(semantics_scope.db.upcast()).to_string())
});
let default_name = "fun_name"; let default_name = "fun_name";
@ -443,7 +445,7 @@ impl Param {
} }
fn to_param(&self, ctx: &AssistContext<'_>, module: hir::Module) -> ast::Param { fn to_param(&self, ctx: &AssistContext<'_>, module: hir::Module) -> ast::Param {
let var = self.var.name(ctx.db()).to_string(); let var = self.var.name(ctx.db()).display(ctx.db()).to_string();
let var_name = make::name(&var); let var_name = make::name(&var);
let pat = match self.kind() { let pat = match self.kind() {
ParamKind::MutValue => make::ident_pat(false, true, var_name), ParamKind::MutValue => make::ident_pat(false, true, var_name),
@ -473,7 +475,8 @@ impl TryKind {
let name = adt.name(ctx.db()); let name = adt.name(ctx.db());
// FIXME: use lang items to determine if it is std type or user defined // FIXME: use lang items to determine if it is std type or user defined
// E.g. if user happens to define type named `Option`, we would have false positive // E.g. if user happens to define type named `Option`, we would have false positive
match name.to_string().as_str() { let name = &name.display(ctx.db()).to_string();
match name.as_str() {
"Option" => Some(TryKind::Option), "Option" => Some(TryKind::Option),
"Result" => Some(TryKind::Result { ty }), "Result" => Some(TryKind::Result { ty }),
_ => None, _ => None,
@ -1341,14 +1344,15 @@ fn make_call(ctx: &AssistContext<'_>, fun: &Function, indent: IndentLevel) -> St
[var] => { [var] => {
let modifier = mut_modifier(var); let modifier = mut_modifier(var);
let name = var.local.name(ctx.db()); let name = var.local.name(ctx.db());
format_to!(buf, "let {modifier}{name} = ") format_to!(buf, "let {modifier}{} = ", name.display(ctx.db()))
} }
vars => { vars => {
buf.push_str("let ("); buf.push_str("let (");
let bindings = vars.iter().format_with(", ", |local, f| { let bindings = vars.iter().format_with(", ", |local, f| {
let modifier = mut_modifier(local); let modifier = mut_modifier(local);
let name = local.local.name(ctx.db()); let name = local.local.name(ctx.db());
f(&format_args!("{modifier}{name}")) f(&format_args!("{modifier}{}", name.display(ctx.db())))?;
Ok(())
}); });
format_to!(buf, "{bindings}"); format_to!(buf, "{bindings}");
buf.push_str(") = "); buf.push_str(") = ");
@ -1487,7 +1491,7 @@ impl FlowHandler {
} }
fn path_expr_from_local(ctx: &AssistContext<'_>, var: Local) -> ast::Expr { fn path_expr_from_local(ctx: &AssistContext<'_>, var: Local) -> ast::Expr {
let name = var.name(ctx.db()).to_string(); let name = var.name(ctx.db()).display(ctx.db()).to_string();
make::expr_path(make::ext::ident_path(&name)) make::expr_path(make::ext::ident_path(&name))
} }

View file

@ -904,7 +904,7 @@ fn compare_hir_and_ast_module(
) -> Option<()> { ) -> Option<()> {
let hir_mod_name = hir_module.name(ctx.db())?; let hir_mod_name = hir_module.name(ctx.db())?;
let ast_mod_name = ast_module.name()?; let ast_mod_name = ast_module.name()?;
if hir_mod_name.to_string() != ast_mod_name.to_string() { if hir_mod_name.display(ctx.db()).to_string() != ast_mod_name.to_string() {
return None; return None;
} }

View file

@ -158,7 +158,7 @@ fn existing_definition(db: &RootDatabase, variant_name: &ast::Name, variant: &Va
), ),
_ => false, _ => false,
}) })
.any(|(name, _)| name.to_string() == variant_name.to_string()) .any(|(name, _)| name.display(db).to_string() == variant_name.to_string())
} }
fn extract_generic_params( fn extract_generic_params(

View file

@ -62,7 +62,9 @@ fn add_vis_to_referenced_module_def(acc: &mut Assists, ctx: &AssistContext<'_>)
let assist_label = match target_name { let assist_label = match target_name {
None => format!("Change visibility to {missing_visibility}"), None => format!("Change visibility to {missing_visibility}"),
Some(name) => format!("Change visibility of {name} to {missing_visibility}"), Some(name) => {
format!("Change visibility of {} to {missing_visibility}", name.display(ctx.db()))
}
}; };
acc.add(AssistId("fix_visibility", AssistKind::QuickFix), assist_label, target, |builder| { acc.add(AssistId("fix_visibility", AssistKind::QuickFix), assist_label, target, |builder| {
@ -117,8 +119,11 @@ fn add_vis_to_referenced_record_field(acc: &mut Assists, ctx: &AssistContext<'_>
let target_file = in_file_source.file_id.original_file(ctx.db()); let target_file = in_file_source.file_id.original_file(ctx.db());
let target_name = record_field_def.name(ctx.db()); let target_name = record_field_def.name(ctx.db());
let assist_label = let assist_label = format!(
format!("Change visibility of {parent_name}.{target_name} to {missing_visibility}"); "Change visibility of {}.{} to {missing_visibility}",
parent_name.display(ctx.db()),
target_name.display(ctx.db())
);
acc.add(AssistId("fix_visibility", AssistKind::QuickFix), assist_label, target, |builder| { acc.add(AssistId("fix_visibility", AssistKind::QuickFix), assist_label, target, |builder| {
builder.edit_file(target_file); builder.edit_file(target_file);

View file

@ -85,13 +85,16 @@ pub(crate) fn generate_delegate_methods(acc: &mut Assists, ctx: &AssistContext<'
for method in methods { for method in methods {
let adt = ast::Adt::Struct(strukt.clone()); let adt = ast::Adt::Struct(strukt.clone());
let name = method.name(ctx.db()).to_string(); let name = method.name(ctx.db()).display(ctx.db()).to_string();
// if `find_struct_impl` returns None, that means that a function named `name` already exists. // if `find_struct_impl` returns None, that means that a function named `name` already exists.
let Some(impl_def) = find_struct_impl(ctx, &adt, &[name]) else { continue; }; let Some(impl_def) = find_struct_impl(ctx, &adt, &[name]) else { continue; };
acc.add_group( acc.add_group(
&GroupLabel("Generate delegate methods…".to_owned()), &GroupLabel("Generate delegate methods…".to_owned()),
AssistId("generate_delegate_methods", AssistKind::Generate), AssistId("generate_delegate_methods", AssistKind::Generate),
format!("Generate delegate for `{field_name}.{}()`", method.name(ctx.db())), format!(
"Generate delegate for `{field_name}.{}()`",
method.name(ctx.db()).display(ctx.db())
),
target, target,
|builder| { |builder| {
// Create the function // Create the function
@ -101,7 +104,7 @@ pub(crate) fn generate_delegate_methods(acc: &mut Assists, ctx: &AssistContext<'
}; };
let method_name = method.name(ctx.db()); let method_name = method.name(ctx.db());
let vis = method_source.visibility(); let vis = method_source.visibility();
let name = make::name(&method.name(ctx.db()).to_string()); let name = make::name(&method.name(ctx.db()).display(ctx.db()).to_string());
let params = let params =
method_source.param_list().unwrap_or_else(|| make::param_list(None, [])); method_source.param_list().unwrap_or_else(|| make::param_list(None, []));
let type_params = method_source.generic_param_list(); let type_params = method_source.generic_param_list();
@ -111,7 +114,7 @@ pub(crate) fn generate_delegate_methods(acc: &mut Assists, ctx: &AssistContext<'
}; };
let tail_expr = make::expr_method_call( let tail_expr = make::expr_method_call(
make::ext::field_from_idents(["self", &field_name]).unwrap(), // This unwrap is ok because we have at least 1 arg in the list make::ext::field_from_idents(["self", &field_name]).unwrap(), // This unwrap is ok because we have at least 1 arg in the list
make::name_ref(&method_name.to_string()), make::name_ref(&method_name.display(ctx.db()).to_string()),
arg_list, arg_list,
); );
let ret_type = method_source.ret_type(); let ret_type = method_source.ret_type();

View file

@ -70,6 +70,7 @@ fn generate_record_deref(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option<(
target, target,
|edit| { |edit| {
generate_edit( generate_edit(
ctx.db(),
edit, edit,
strukt, strukt,
field_type.syntax(), field_type.syntax(),
@ -109,6 +110,7 @@ fn generate_tuple_deref(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option<()
target, target,
|edit| { |edit| {
generate_edit( generate_edit(
ctx.db(),
edit, edit,
strukt, strukt,
field_type.syntax(), field_type.syntax(),
@ -121,6 +123,7 @@ fn generate_tuple_deref(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option<()
} }
fn generate_edit( fn generate_edit(
db: &RootDatabase,
edit: &mut SourceChangeBuilder, edit: &mut SourceChangeBuilder,
strukt: ast::Struct, strukt: ast::Struct,
field_type_syntax: &SyntaxNode, field_type_syntax: &SyntaxNode,
@ -144,7 +147,8 @@ fn generate_edit(
), ),
}; };
let strukt_adt = ast::Adt::Struct(strukt); let strukt_adt = ast::Adt::Struct(strukt);
let deref_impl = generate_trait_impl_text(&strukt_adt, &trait_path.to_string(), &impl_code); let deref_impl =
generate_trait_impl_text(&strukt_adt, &trait_path.display(db).to_string(), &impl_code);
edit.insert(start_offset, deref_impl); edit.insert(start_offset, deref_impl);
} }

View file

@ -196,7 +196,7 @@ fn add_func_to_accumulator(
let mut func = function_template.to_string(ctx.config.snippet_cap); let mut func = function_template.to_string(ctx.config.snippet_cap);
if let Some(name) = adt_name { if let Some(name) = adt_name {
// FIXME: adt may have generic params. // FIXME: adt may have generic params.
func = format!("\n{indent}impl {name} {{\n{func}\n{indent}}}"); func = format!("\n{indent}impl {} {{\n{func}\n{indent}}}", name.display(ctx.db()));
} }
builder.edit_file(file); builder.edit_file(file);
match ctx.config.snippet_cap { match ctx.config.snippet_cap {

View file

@ -98,7 +98,7 @@ pub(crate) fn move_const_to_impl(acc: &mut Assists, ctx: &AssistContext<'_>) ->
}; };
builder.delete(range_to_delete); builder.delete(range_to_delete);
let const_ref = format!("Self::{name}"); let const_ref = format!("Self::{}", name.display(ctx.db()));
for range in usages.all().file_ranges().map(|it| it.range) { for range in usages.all().file_ranges().map(|it| it.range) {
builder.replace(range, const_ref.clone()); builder.replace(range, const_ref.clone());
} }

View file

@ -39,7 +39,7 @@ pub(crate) fn move_from_mod_rs(acc: &mut Assists, ctx: &AssistContext<'_>) -> Op
} }
let target = source_file.syntax().text_range(); let target = source_file.syntax().text_range();
let module_name = module.name(ctx.db())?.to_string(); let module_name = module.name(ctx.db())?.display(ctx.db()).to_string();
let path = format!("../{module_name}.rs"); let path = format!("../{module_name}.rs");
let dst = AnchoredPathBuf { anchor: ctx.file_id(), path }; let dst = AnchoredPathBuf { anchor: ctx.file_id(), path };
acc.add( acc.add(

View file

@ -52,7 +52,7 @@ pub(crate) fn move_module_to_file(acc: &mut Assists, ctx: &AssistContext<'_>) ->
let mut buf = String::from("./"); let mut buf = String::from("./");
match parent_module.name(ctx.db()) { match parent_module.name(ctx.db()) {
Some(name) if !parent_module.is_mod_rs(ctx.db()) => { Some(name) if !parent_module.is_mod_rs(ctx.db()) => {
format_to!(buf, "{name}/") format_to!(buf, "{}/", name.display(ctx.db()))
} }
_ => (), _ => (),
} }

View file

@ -39,7 +39,7 @@ pub(crate) fn move_to_mod_rs(acc: &mut Assists, ctx: &AssistContext<'_>) -> Opti
} }
let target = source_file.syntax().text_range(); let target = source_file.syntax().text_range();
let module_name = module.name(ctx.db())?.to_string(); let module_name = module.name(ctx.db())?.display(ctx.db()).to_string();
let path = format!("./{module_name}/mod.rs"); let path = format!("./{module_name}/mod.rs");
let dst = AnchoredPathBuf { anchor: ctx.file_id(), path }; let dst = AnchoredPathBuf { anchor: ctx.file_id(), path };
acc.add( acc.add(

View file

@ -86,7 +86,7 @@ pub(crate) fn qualify_path(acc: &mut Assists, ctx: &AssistContext<'_>) -> Option
acc.add_group( acc.add_group(
&group_label, &group_label,
AssistId("qualify_path", AssistKind::QuickFix), AssistId("qualify_path", AssistKind::QuickFix),
label(candidate, &import), label(ctx.db(), candidate, &import),
range, range,
|builder| { |builder| {
qualify_candidate.qualify( qualify_candidate.qualify(
@ -186,7 +186,7 @@ fn find_trait_method(
if let Some(hir::AssocItem::Function(method)) = if let Some(hir::AssocItem::Function(method)) =
trait_.items(db).into_iter().find(|item: &hir::AssocItem| { trait_.items(db).into_iter().find(|item: &hir::AssocItem| {
item.name(db) item.name(db)
.map(|name| name.to_string() == trait_method_name.to_string()) .map(|name| name.display(db).to_string() == trait_method_name.to_string())
.unwrap_or(false) .unwrap_or(false)
}) })
{ {
@ -216,14 +216,14 @@ fn group_label(candidate: &ImportCandidate) -> GroupLabel {
GroupLabel(format!("Qualify {name}")) GroupLabel(format!("Qualify {name}"))
} }
fn label(candidate: &ImportCandidate, import: &LocatedImport) -> String { fn label(db: &RootDatabase, candidate: &ImportCandidate, import: &LocatedImport) -> String {
let import_path = &import.import_path; let import_path = &import.import_path;
match candidate { match candidate {
ImportCandidate::Path(candidate) if candidate.qualifier.is_none() => { ImportCandidate::Path(candidate) if candidate.qualifier.is_none() => {
format!("Qualify as `{import_path}`") format!("Qualify as `{}`", import_path.display(db))
} }
_ => format!("Qualify with `{import_path}`"), _ => format!("Qualify with `{}`", import_path.display(db)),
} }
} }

View file

@ -97,7 +97,7 @@ fn compute_fields_ranks(
.fields(ctx.db()) .fields(ctx.db())
.into_iter() .into_iter()
.enumerate() .enumerate()
.map(|(idx, field)| (field.name(ctx.db()).to_string(), idx)) .map(|(idx, field)| (field.name(ctx.db()).display(ctx.db()).to_string(), idx))
.collect(); .collect();
Some(res) Some(res)

View file

@ -114,7 +114,7 @@ fn compute_item_ranks(
.iter() .iter()
.flat_map(|i| i.name(ctx.db())) .flat_map(|i| i.name(ctx.db()))
.enumerate() .enumerate()
.map(|(idx, name)| (name.to_string(), idx)) .map(|(idx, name)| (name.display(ctx.db()).to_string(), idx))
.collect(), .collect(),
) )
} }

View file

@ -234,7 +234,7 @@ fn from_type(expr: &ast::Expr, sema: &Semantics<'_, RootDatabase>) -> Option<Str
fn name_of_type(ty: &hir::Type, db: &RootDatabase) -> Option<String> { fn name_of_type(ty: &hir::Type, db: &RootDatabase) -> Option<String> {
let name = if let Some(adt) = ty.as_adt() { let name = if let Some(adt) = ty.as_adt() {
let name = adt.name(db).to_string(); let name = adt.name(db).display(db).to_string();
if WRAPPER_TYPES.contains(&name.as_str()) { if WRAPPER_TYPES.contains(&name.as_str()) {
let inner_ty = ty.type_arguments().next()?; let inner_ty = ty.type_arguments().next()?;
@ -258,7 +258,7 @@ fn name_of_type(ty: &hir::Type, db: &RootDatabase) -> Option<String> {
} }
fn trait_name(trait_: &hir::Trait, db: &RootDatabase) -> Option<String> { fn trait_name(trait_: &hir::Trait, db: &RootDatabase) -> Option<String> {
let name = trait_.name(db).to_string(); let name = trait_.name(db).display(db).to_string();
if USELESS_TRAITS.contains(&name.as_str()) { if USELESS_TRAITS.contains(&name.as_str()) {
return None; return None;
} }

View file

@ -24,7 +24,7 @@ pub(crate) mod env_vars;
use std::iter; use std::iter;
use hir::{known, HasAttrs, ScopeDef, Variant}; use hir::{known, HasAttrs, ScopeDef, Variant};
use ide_db::{imports::import_assets::LocatedImport, SymbolKind}; use ide_db::{imports::import_assets::LocatedImport, RootDatabase, SymbolKind};
use syntax::ast; use syntax::ast;
use crate::{ use crate::{
@ -62,8 +62,8 @@ impl From<Completions> for Vec<CompletionItem> {
impl Builder { impl Builder {
/// Convenience method, which allows to add a freshly created completion into accumulator /// Convenience method, which allows to add a freshly created completion into accumulator
/// without binding it to the variable. /// without binding it to the variable.
pub(crate) fn add_to(self, acc: &mut Completions) { pub(crate) fn add_to(self, acc: &mut Completions, db: &RootDatabase) {
acc.add(self.build()) acc.add(self.build(db))
} }
} }
@ -80,7 +80,7 @@ impl Completions {
pub(crate) fn add_keyword(&mut self, ctx: &CompletionContext<'_>, keyword: &'static str) { pub(crate) fn add_keyword(&mut self, ctx: &CompletionContext<'_>, keyword: &'static str) {
let item = CompletionItem::new(CompletionItemKind::Keyword, ctx.source_range(), keyword); let item = CompletionItem::new(CompletionItemKind::Keyword, ctx.source_range(), keyword);
item.add_to(self); item.add_to(self, ctx.db);
} }
pub(crate) fn add_nameref_keywords_with_colon(&mut self, ctx: &CompletionContext<'_>) { pub(crate) fn add_nameref_keywords_with_colon(&mut self, ctx: &CompletionContext<'_>) {
@ -134,7 +134,7 @@ impl Completions {
item.insert_text(if snippet.contains('$') { kw } else { snippet }); item.insert_text(if snippet.contains('$') { kw } else { snippet });
} }
}; };
item.add_to(self); item.add_to(self, ctx.db);
} }
pub(crate) fn add_keyword_snippet( pub(crate) fn add_keyword_snippet(
@ -149,7 +149,7 @@ impl Completions {
Some(cap) => item.insert_snippet(cap, snippet), Some(cap) => item.insert_snippet(cap, snippet),
None => item.insert_text(if snippet.contains('$') { kw } else { snippet }), None => item.insert_text(if snippet.contains('$') { kw } else { snippet }),
}; };
item.add_to(self); item.add_to(self, ctx.db);
} }
pub(crate) fn add_crate_roots( pub(crate) fn add_crate_roots(
@ -190,7 +190,7 @@ impl Completions {
local_name, local_name,
resolution, resolution,
) )
.build(), .build(ctx.db),
); );
} }
@ -216,7 +216,7 @@ impl Completions {
local_name, local_name,
resolution, resolution,
) )
.build(), .build(ctx.db),
); );
} }
@ -276,7 +276,7 @@ impl Completions {
local_name, local_name,
mac, mac,
) )
.build(), .build(ctx.db),
); );
} }
@ -305,7 +305,7 @@ impl Completions {
local_name, local_name,
func, func,
) )
.build(), .build(ctx.db),
); );
} }
@ -336,7 +336,7 @@ impl Completions {
local_name, local_name,
func, func,
) )
.build(), .build(ctx.db),
); );
} }
@ -367,7 +367,7 @@ impl Completions {
None, None,
func, func,
) )
.build(), .build(ctx.db),
); );
} }
@ -429,7 +429,7 @@ impl Completions {
if let Some(builder) = if let Some(builder) =
render_variant_lit(RenderContext::new(ctx), path_ctx, None, variant, Some(path)) render_variant_lit(RenderContext::new(ctx), path_ctx, None, variant, Some(path))
{ {
self.add(builder.build()); self.add(builder.build(ctx.db));
} }
} }
@ -452,7 +452,7 @@ impl Completions {
if let Some(builder) = if let Some(builder) =
render_variant_lit(RenderContext::new(ctx), path_ctx, local_name, variant, None) render_variant_lit(RenderContext::new(ctx), path_ctx, local_name, variant, None)
{ {
self.add(builder.build()); self.add(builder.build(ctx.db));
} }
} }
@ -497,7 +497,7 @@ impl Completions {
if let Some(builder) = if let Some(builder) =
render_struct_literal(RenderContext::new(ctx), path_ctx, strukt, path, local_name) render_struct_literal(RenderContext::new(ctx), path_ctx, strukt, path, local_name)
{ {
self.add(builder.build()); self.add(builder.build(ctx.db));
} }
} }
@ -530,11 +530,12 @@ impl Completions {
pub(crate) fn add_lifetime(&mut self, ctx: &CompletionContext<'_>, name: hir::Name) { pub(crate) fn add_lifetime(&mut self, ctx: &CompletionContext<'_>, name: hir::Name) {
CompletionItem::new(SymbolKind::LifetimeParam, ctx.source_range(), name.to_smol_str()) CompletionItem::new(SymbolKind::LifetimeParam, ctx.source_range(), name.to_smol_str())
.add_to(self) .add_to(self, ctx.db)
} }
pub(crate) fn add_label(&mut self, ctx: &CompletionContext<'_>, name: hir::Name) { pub(crate) fn add_label(&mut self, ctx: &CompletionContext<'_>, name: hir::Name) {
CompletionItem::new(SymbolKind::Label, ctx.source_range(), name.to_smol_str()).add_to(self) CompletionItem::new(SymbolKind::Label, ctx.source_range(), name.to_smol_str())
.add_to(self, ctx.db)
} }
pub(crate) fn add_variant_pat( pub(crate) fn add_variant_pat(

View file

@ -139,7 +139,7 @@ pub(crate) fn complete_attribute_path(
} }
if is_inner || !attr_completion.prefer_inner { if is_inner || !attr_completion.prefer_inner {
item.add_to(acc); item.add_to(acc, ctx.db);
} }
}; };

View file

@ -12,7 +12,7 @@ pub(crate) fn complete_cfg(acc: &mut Completions, ctx: &CompletionContext<'_>) {
let add_completion = |item: &str| { let add_completion = |item: &str| {
let mut completion = CompletionItem::new(SymbolKind::BuiltinAttr, ctx.source_range(), item); let mut completion = CompletionItem::new(SymbolKind::BuiltinAttr, ctx.source_range(), item);
completion.insert_text(format!(r#""{item}""#)); completion.insert_text(format!(r#""{item}""#));
acc.add(completion.build()); acc.add(completion.build(ctx.db));
}; };
let previous = iter::successors(ctx.original_token.prev_token(), |t| { let previous = iter::successors(ctx.original_token.prev_token(), |t| {
@ -33,11 +33,11 @@ pub(crate) fn complete_cfg(acc: &mut Completions, ctx: &CompletionContext<'_>) {
let mut item = CompletionItem::new(SymbolKind::BuiltinAttr, ctx.source_range(), s); let mut item = CompletionItem::new(SymbolKind::BuiltinAttr, ctx.source_range(), s);
item.insert_text(insert_text); item.insert_text(insert_text);
acc.add(item.build()); acc.add(item.build(ctx.db));
}), }),
None => ctx.krate.potential_cfg(ctx.db).get_cfg_keys().cloned().unique().for_each(|s| { None => ctx.krate.potential_cfg(ctx.db).get_cfg_keys().cloned().unique().for_each(|s| {
let item = CompletionItem::new(SymbolKind::BuiltinAttr, ctx.source_range(), s); let item = CompletionItem::new(SymbolKind::BuiltinAttr, ctx.source_range(), s);
acc.add(item.build()); acc.add(item.build(ctx.db));
}), }),
}; };
} }

View file

@ -90,7 +90,7 @@ pub(crate) fn complete_derive_path(
item.documentation(docs); item.documentation(docs);
} }
item.lookup_by(lookup); item.lookup_by(lookup);
item.add_to(acc); item.add_to(acc, ctx.db);
} }
None => acc.add_macro(ctx, path_ctx, mac, name), None => acc.add_macro(ctx, path_ctx, mac, name),
} }

View file

@ -56,6 +56,6 @@ pub(super) fn complete_lint(
}; };
let mut item = CompletionItem::new(SymbolKind::Attribute, ctx.source_range(), label); let mut item = CompletionItem::new(SymbolKind::Attribute, ctx.source_range(), label);
item.documentation(hir::Documentation::new(description.to_owned())); item.documentation(hir::Documentation::new(description.to_owned()));
item.add_to(acc) item.add_to(acc, ctx.db)
} }
} }

View file

@ -37,7 +37,7 @@ pub(super) fn complete_repr(
if let Some((snippet, cap)) = snippet.zip(ctx.config.snippet_cap) { if let Some((snippet, cap)) = snippet.zip(ctx.config.snippet_cap) {
item.insert_snippet(cap, snippet); item.insert_snippet(cap, snippet);
} }
item.add_to(acc); item.add_to(acc, ctx.db);
} }
} }
} }

View file

@ -23,7 +23,7 @@ pub(crate) fn complete_dot(
let mut item = let mut item =
CompletionItem::new(CompletionItemKind::Keyword, ctx.source_range(), "await"); CompletionItem::new(CompletionItemKind::Keyword, ctx.source_range(), "await");
item.detail("expr.await"); item.detail("expr.await");
item.add_to(acc); item.add_to(acc, ctx.db);
} }
if let DotAccessKind::Method { .. } = dot_access.kind { if let DotAccessKind::Method { .. } = dot_access.kind {

View file

@ -40,7 +40,7 @@ pub(crate) fn complete_cargo_env_vars(
CARGO_DEFINED_VARS.into_iter().for_each(|&(var, detail)| { CARGO_DEFINED_VARS.into_iter().for_each(|&(var, detail)| {
let mut item = CompletionItem::new(CompletionItemKind::Keyword, range, var); let mut item = CompletionItem::new(CompletionItemKind::Keyword, range, var);
item.detail(detail); item.detail(detail);
item.add_to(acc); item.add_to(acc, ctx.db);
}); });
Some(()) Some(())

View file

@ -42,7 +42,7 @@ const SUPPORTED_CALLING_CONVENTIONS: &[&str] = &[
pub(crate) fn complete_extern_abi( pub(crate) fn complete_extern_abi(
acc: &mut Completions, acc: &mut Completions,
_ctx: &CompletionContext<'_>, ctx: &CompletionContext<'_>,
expanded: &ast::String, expanded: &ast::String,
) -> Option<()> { ) -> Option<()> {
if !expanded.syntax().parent().map_or(false, |it| ast::Abi::can_cast(it.kind())) { if !expanded.syntax().parent().map_or(false, |it| ast::Abi::can_cast(it.kind())) {
@ -51,7 +51,7 @@ pub(crate) fn complete_extern_abi(
let abi_str = expanded; let abi_str = expanded;
let source_range = abi_str.text_range_between_quotes()?; let source_range = abi_str.text_range_between_quotes()?;
for &abi in SUPPORTED_CALLING_CONVENTIONS { for &abi in SUPPORTED_CALLING_CONVENTIONS {
CompletionItem::new(CompletionItemKind::Keyword, source_range, abi).add_to(acc); CompletionItem::new(CompletionItemKind::Keyword, source_range, abi).add_to(acc, ctx.db);
} }
Some(()) Some(())
} }

View file

@ -273,7 +273,7 @@ fn import_on_the_fly(
.filter_map(|import| { .filter_map(|import| {
render_resolution_with_import(RenderContext::new(ctx), path_ctx, import) render_resolution_with_import(RenderContext::new(ctx), path_ctx, import)
}) })
.map(|builder| builder.build()) .map(|builder| builder.build(ctx.db))
.for_each(|item| acc.add(item)); .for_each(|item| acc.add(item));
Some(()) Some(())
} }
@ -315,7 +315,7 @@ fn import_on_the_fly_pat_(
.filter_map(|import| { .filter_map(|import| {
render_resolution_with_import_pat(RenderContext::new(ctx), pattern_ctx, import) render_resolution_with_import_pat(RenderContext::new(ctx), pattern_ctx, import)
}) })
.map(|builder| builder.build()) .map(|builder| builder.build(ctx.db))
.for_each(|item| acc.add(item)); .for_each(|item| acc.add(item));
Some(()) Some(())
} }

View file

@ -40,7 +40,7 @@ pub(crate) fn complete_fn_param(
}; };
// Completion lookup is omitted intentionally here. // Completion lookup is omitted intentionally here.
// See the full discussion: https://github.com/rust-lang/rust-analyzer/issues/12073 // See the full discussion: https://github.com/rust-lang/rust-analyzer/issues/12073
item.add_to(acc) item.add_to(acc, ctx.db)
}; };
match kind { match kind {
@ -50,7 +50,7 @@ pub(crate) fn complete_fn_param(
ParamKind::Closure(closure) => { ParamKind::Closure(closure) => {
let stmt_list = closure.syntax().ancestors().find_map(ast::StmtList::cast)?; let stmt_list = closure.syntax().ancestors().find_map(ast::StmtList::cast)?;
params_from_stmt_list_scope(ctx, stmt_list, |name, ty| { params_from_stmt_list_scope(ctx, stmt_list, |name, ty| {
add_new_item_to_acc(&format!("{name}: {ty}")); add_new_item_to_acc(&format!("{}: {ty}", name.display(ctx.db)));
}); });
} }
} }
@ -100,7 +100,9 @@ fn fill_fn_params(
if let Some(stmt_list) = function.syntax().parent().and_then(ast::StmtList::cast) { if let Some(stmt_list) = function.syntax().parent().and_then(ast::StmtList::cast) {
params_from_stmt_list_scope(ctx, stmt_list, |name, ty| { params_from_stmt_list_scope(ctx, stmt_list, |name, ty| {
file_params.entry(format!("{name}: {ty}")).or_insert(name.to_string()); file_params
.entry(format!("{}: {ty}", name.display(ctx.db)))
.or_insert(name.display(ctx.db).to_string());
}); });
} }
remove_duplicated(&mut file_params, param_list.params()); remove_duplicated(&mut file_params, param_list.params());

View file

@ -32,7 +32,7 @@ pub(crate) fn format_string(
let source_range = TextRange::new(brace_offset, cursor); let source_range = TextRange::new(brace_offset, cursor);
ctx.locals.iter().for_each(|(name, _)| { ctx.locals.iter().for_each(|(name, _)| {
CompletionItem::new(CompletionItemKind::Binding, source_range, name.to_smol_str()) CompletionItem::new(CompletionItemKind::Binding, source_range, name.to_smol_str())
.add_to(acc); .add_to(acc, ctx.db);
}) })
} }

View file

@ -182,7 +182,7 @@ fn add_function_impl(
let label = format!( let label = format!(
"fn {}({})", "fn {}({})",
fn_name, fn_name.display(ctx.db),
if func.assoc_fn_params(ctx.db).is_empty() { "" } else { ".." } if func.assoc_fn_params(ctx.db).is_empty() { "" } else { ".." }
); );
@ -193,7 +193,7 @@ fn add_function_impl(
}; };
let mut item = CompletionItem::new(completion_kind, replacement_range, label); let mut item = CompletionItem::new(completion_kind, replacement_range, label);
item.lookup_by(format!("fn {fn_name}")) item.lookup_by(format!("fn {}", fn_name.display(ctx.db)))
.set_documentation(func.docs(ctx.db)) .set_documentation(func.docs(ctx.db))
.set_relevance(CompletionRelevance { is_item_from_trait: true, ..Default::default() }); .set_relevance(CompletionRelevance { is_item_from_trait: true, ..Default::default() });
@ -216,7 +216,7 @@ fn add_function_impl(
item.text_edit(TextEdit::replace(replacement_range, header)); item.text_edit(TextEdit::replace(replacement_range, header));
} }
}; };
item.add_to(acc); item.add_to(acc, ctx.db);
} }
} }
} }
@ -300,7 +300,7 @@ fn add_type_alias_impl(
item.text_edit(TextEdit::replace(replacement_range, decl)); item.text_edit(TextEdit::replace(replacement_range, decl));
} }
}; };
item.add_to(acc); item.add_to(acc, ctx.db);
} }
} }
} }
@ -340,7 +340,7 @@ fn add_const_impl(
), ),
None => item.text_edit(TextEdit::replace(replacement_range, replacement)), None => item.text_edit(TextEdit::replace(replacement_range, replacement)),
}; };
item.add_to(acc); item.add_to(acc, ctx.db);
} }
} }
} }

View file

@ -52,7 +52,7 @@ pub(crate) fn complete_mod(
let existing_mod_declarations = current_module let existing_mod_declarations = current_module
.children(ctx.db) .children(ctx.db)
.filter_map(|module| Some(module.name(ctx.db)?.to_string())) .filter_map(|module| Some(module.name(ctx.db)?.display(ctx.db).to_string()))
.filter(|module| module != ctx.original_token.text()) .filter(|module| module != ctx.original_token.text())
.collect::<FxHashSet<_>>(); .collect::<FxHashSet<_>>();
@ -99,7 +99,7 @@ pub(crate) fn complete_mod(
label.push(';'); label.push(';');
} }
let item = CompletionItem::new(SymbolKind::Module, ctx.source_range(), &label); let item = CompletionItem::new(SymbolKind::Module, ctx.source_range(), &label);
item.add_to(acc) item.add_to(acc, ctx.db)
}); });
Some(()) Some(())

View file

@ -64,7 +64,7 @@ pub(crate) fn complete_postfix(
&format!("drop($0{receiver_text})"), &format!("drop($0{receiver_text})"),
); );
item.set_documentation(drop_fn.docs(ctx.db)); item.set_documentation(drop_fn.docs(ctx.db));
item.add_to(acc); item.add_to(acc, ctx.db);
} }
} }
} }
@ -78,14 +78,14 @@ pub(crate) fn complete_postfix(
"if let Ok {}", "if let Ok {}",
&format!("if let Ok($1) = {receiver_text} {{\n $0\n}}"), &format!("if let Ok($1) = {receiver_text} {{\n $0\n}}"),
) )
.add_to(acc); .add_to(acc, ctx.db);
postfix_snippet( postfix_snippet(
"while", "while",
"while let Ok {}", "while let Ok {}",
&format!("while let Ok($1) = {receiver_text} {{\n $0\n}}"), &format!("while let Ok($1) = {receiver_text} {{\n $0\n}}"),
) )
.add_to(acc); .add_to(acc, ctx.db);
} }
TryEnum::Option => { TryEnum::Option => {
postfix_snippet( postfix_snippet(
@ -93,22 +93,22 @@ pub(crate) fn complete_postfix(
"if let Some {}", "if let Some {}",
&format!("if let Some($1) = {receiver_text} {{\n $0\n}}"), &format!("if let Some($1) = {receiver_text} {{\n $0\n}}"),
) )
.add_to(acc); .add_to(acc, ctx.db);
postfix_snippet( postfix_snippet(
"while", "while",
"while let Some {}", "while let Some {}",
&format!("while let Some($1) = {receiver_text} {{\n $0\n}}"), &format!("while let Some($1) = {receiver_text} {{\n $0\n}}"),
) )
.add_to(acc); .add_to(acc, ctx.db);
} }
} }
} else if receiver_ty.is_bool() || receiver_ty.is_unknown() { } else if receiver_ty.is_bool() || receiver_ty.is_unknown() {
postfix_snippet("if", "if expr {}", &format!("if {receiver_text} {{\n $0\n}}")) postfix_snippet("if", "if expr {}", &format!("if {receiver_text} {{\n $0\n}}"))
.add_to(acc); .add_to(acc, ctx.db);
postfix_snippet("while", "while expr {}", &format!("while {receiver_text} {{\n $0\n}}")) postfix_snippet("while", "while expr {}", &format!("while {receiver_text} {{\n $0\n}}"))
.add_to(acc); .add_to(acc, ctx.db);
postfix_snippet("not", "!expr", &format!("!{receiver_text}")).add_to(acc); postfix_snippet("not", "!expr", &format!("!{receiver_text}")).add_to(acc, ctx.db);
} else if let Some(trait_) = ctx.famous_defs().core_iter_IntoIterator() { } else if let Some(trait_) = ctx.famous_defs().core_iter_IntoIterator() {
if receiver_ty.impls_trait(ctx.db, trait_, &[]) { if receiver_ty.impls_trait(ctx.db, trait_, &[]) {
postfix_snippet( postfix_snippet(
@ -116,12 +116,12 @@ pub(crate) fn complete_postfix(
"for ele in expr {}", "for ele in expr {}",
&format!("for ele in {receiver_text} {{\n $0\n}}"), &format!("for ele in {receiver_text} {{\n $0\n}}"),
) )
.add_to(acc); .add_to(acc, ctx.db);
} }
} }
postfix_snippet("ref", "&expr", &format!("&{receiver_text}")).add_to(acc); postfix_snippet("ref", "&expr", &format!("&{receiver_text}")).add_to(acc, ctx.db);
postfix_snippet("refm", "&mut expr", &format!("&mut {receiver_text}")).add_to(acc); postfix_snippet("refm", "&mut expr", &format!("&mut {receiver_text}")).add_to(acc, ctx.db);
let mut unsafe_should_be_wrapped = true; let mut unsafe_should_be_wrapped = true;
if dot_receiver.syntax().kind() == BLOCK_EXPR { if dot_receiver.syntax().kind() == BLOCK_EXPR {
@ -137,7 +137,7 @@ pub(crate) fn complete_postfix(
} else { } else {
format!("unsafe {receiver_text}") format!("unsafe {receiver_text}")
}; };
postfix_snippet("unsafe", "unsafe {}", &unsafe_completion_string).add_to(acc); postfix_snippet("unsafe", "unsafe {}", &unsafe_completion_string).add_to(acc, ctx.db);
// The rest of the postfix completions create an expression that moves an argument, // The rest of the postfix completions create an expression that moves an argument,
// so it's better to consider references now to avoid breaking the compilation // so it's better to consider references now to avoid breaking the compilation
@ -162,7 +162,7 @@ pub(crate) fn complete_postfix(
"match expr {}", "match expr {}",
&format!("match {receiver_text} {{\n Ok(${{1:_}}) => {{$2}},\n Err(${{3:_}}) => {{$0}},\n}}"), &format!("match {receiver_text} {{\n Ok(${{1:_}}) => {{$2}},\n Err(${{3:_}}) => {{$0}},\n}}"),
) )
.add_to(acc); .add_to(acc, ctx.db);
} }
TryEnum::Option => { TryEnum::Option => {
postfix_snippet( postfix_snippet(
@ -172,7 +172,7 @@ pub(crate) fn complete_postfix(
"match {receiver_text} {{\n Some(${{1:_}}) => {{$2}},\n None => {{$0}},\n}}" "match {receiver_text} {{\n Some(${{1:_}}) => {{$2}},\n None => {{$0}},\n}}"
), ),
) )
.add_to(acc); .add_to(acc, ctx.db);
} }
}, },
None => { None => {
@ -181,20 +181,23 @@ pub(crate) fn complete_postfix(
"match expr {}", "match expr {}",
&format!("match {receiver_text} {{\n ${{1:_}} => {{$0}},\n}}"), &format!("match {receiver_text} {{\n ${{1:_}} => {{$0}},\n}}"),
) )
.add_to(acc); .add_to(acc, ctx.db);
} }
} }
postfix_snippet("box", "Box::new(expr)", &format!("Box::new({receiver_text})")).add_to(acc); postfix_snippet("box", "Box::new(expr)", &format!("Box::new({receiver_text})"))
postfix_snippet("dbg", "dbg!(expr)", &format!("dbg!({receiver_text})")).add_to(acc); // fixme .add_to(acc, ctx.db);
postfix_snippet("dbgr", "dbg!(&expr)", &format!("dbg!(&{receiver_text})")).add_to(acc); postfix_snippet("dbg", "dbg!(expr)", &format!("dbg!({receiver_text})")).add_to(acc, ctx.db); // fixme
postfix_snippet("call", "function(expr)", &format!("${{1}}({receiver_text})")).add_to(acc); postfix_snippet("dbgr", "dbg!(&expr)", &format!("dbg!(&{receiver_text})")).add_to(acc, ctx.db);
postfix_snippet("call", "function(expr)", &format!("${{1}}({receiver_text})"))
.add_to(acc, ctx.db);
if let Some(parent) = dot_receiver.syntax().parent().and_then(|p| p.parent()) { if let Some(parent) = dot_receiver.syntax().parent().and_then(|p| p.parent()) {
if matches!(parent.kind(), STMT_LIST | EXPR_STMT) { if matches!(parent.kind(), STMT_LIST | EXPR_STMT) {
postfix_snippet("let", "let", &format!("let $0 = {receiver_text};")).add_to(acc); postfix_snippet("let", "let", &format!("let $0 = {receiver_text};"))
.add_to(acc, ctx.db);
postfix_snippet("letm", "let mut", &format!("let mut $0 = {receiver_text};")) postfix_snippet("letm", "let mut", &format!("let mut $0 = {receiver_text};"))
.add_to(acc); .add_to(acc, ctx.db);
} }
} }
@ -315,7 +318,7 @@ fn add_custom_postfix_completions(
for import in imports.into_iter() { for import in imports.into_iter() {
builder.add_import(import); builder.add_import(import);
} }
builder.add_to(acc); builder.add_to(acc, ctx.db);
}, },
); );
None None

View file

@ -60,7 +60,7 @@ pub(crate) fn add_format_like_completions(
format!(r#"{}({}, {})"#, macro_name, out, exprs.join(", ")) format!(r#"{}({}, {})"#, macro_name, out, exprs.join(", "))
}; };
postfix_snippet(label, macro_name, &snippet).add_to(acc); postfix_snippet(label, macro_name, &snippet).add_to(acc, ctx.db);
} }
} }
} }

View file

@ -69,7 +69,7 @@ pub(crate) fn complete_record_expr_fields(
let mut item = let mut item =
CompletionItem::new(CompletionItemKind::Snippet, ctx.source_range(), ".."); CompletionItem::new(CompletionItemKind::Snippet, ctx.source_range(), "..");
item.insert_text("."); item.insert_text(".");
item.add_to(acc); item.add_to(acc, ctx.db);
return; return;
} }
missing_fields missing_fields
@ -98,7 +98,7 @@ pub(crate) fn add_default_update(
postfix_match: Some(CompletionRelevancePostfixMatch::Exact), postfix_match: Some(CompletionRelevancePostfixMatch::Exact),
..Default::default() ..Default::default()
}); });
item.add_to(acc); item.add_to(acc, ctx.db);
} }
} }

View file

@ -32,8 +32,8 @@ pub(crate) fn complete_expr_snippet(
} }
if in_block_expr { if in_block_expr {
snippet(ctx, cap, "pd", "eprintln!(\"$0 = {:?}\", $0);").add_to(acc); snippet(ctx, cap, "pd", "eprintln!(\"$0 = {:?}\", $0);").add_to(acc, ctx.db);
snippet(ctx, cap, "ppd", "eprintln!(\"$0 = {:#?}\", $0);").add_to(acc); snippet(ctx, cap, "ppd", "eprintln!(\"$0 = {:#?}\", $0);").add_to(acc, ctx.db);
let item = snippet( let item = snippet(
ctx, ctx,
cap, cap,
@ -45,7 +45,7 @@ macro_rules! $1 {
}; };
}", }",
); );
item.add_to(acc); item.add_to(acc, ctx.db);
} }
} }
@ -88,7 +88,7 @@ mod tests {
}", }",
); );
item.lookup_by("tmod"); item.lookup_by("tmod");
item.add_to(acc); item.add_to(acc, ctx.db);
let mut item = snippet( let mut item = snippet(
ctx, ctx,
@ -101,7 +101,7 @@ fn ${1:feature}() {
}", }",
); );
item.lookup_by("tfn"); item.lookup_by("tfn");
item.add_to(acc); item.add_to(acc, ctx.db);
let item = snippet( let item = snippet(
ctx, ctx,
@ -114,7 +114,7 @@ macro_rules! $1 {
}; };
}", }",
); );
item.add_to(acc); item.add_to(acc, ctx.db);
} }
} }
@ -146,7 +146,7 @@ fn add_custom_completions(
builder.add_import(import); builder.add_import(import);
} }
builder.set_detail(snip.description.clone()); builder.set_detail(snip.description.clone());
builder.add_to(acc); builder.add_to(acc, ctx.db);
}, },
); );
None None

View file

@ -75,7 +75,7 @@ pub(crate) fn complete_use_path(
is_name_already_imported, is_name_already_imported,
..Default::default() ..Default::default()
}); });
acc.add(builder.build()); acc.add(builder.build(ctx.db));
} }
} }
} }
@ -108,9 +108,9 @@ pub(crate) fn complete_use_path(
let item = CompletionItem::new( let item = CompletionItem::new(
CompletionItemKind::SymbolKind(SymbolKind::Enum), CompletionItemKind::SymbolKind(SymbolKind::Enum),
ctx.source_range(), ctx.source_range(),
format!("{}::", e.name(ctx.db)), format!("{}::", e.name(ctx.db).display(ctx.db)),
); );
acc.add(item.build()); acc.add(item.build(ctx.db));
} }
} }
_ => {} _ => {}

View file

@ -1190,7 +1190,7 @@ fn pattern_context_for(
}) })
}).and_then(|variants| { }).and_then(|variants| {
Some(variants.iter().filter_map(|variant| { Some(variants.iter().filter_map(|variant| {
let variant_name = variant.name(sema.db).to_string(); let variant_name = variant.name(sema.db).display(sema.db).to_string();
let variant_already_present = match_arm_list.arms().any(|arm| { let variant_already_present = match_arm_list.arms().any(|arm| {
arm.pat().and_then(|pat| { arm.pat().and_then(|pat| {

View file

@ -3,7 +3,7 @@
use std::fmt; use std::fmt;
use hir::{Documentation, Mutability}; use hir::{Documentation, Mutability};
use ide_db::{imports::import_assets::LocatedImport, SnippetCap, SymbolKind}; use ide_db::{imports::import_assets::LocatedImport, RootDatabase, SnippetCap, SymbolKind};
use itertools::Itertools; use itertools::Itertools;
use smallvec::SmallVec; use smallvec::SmallVec;
use stdx::{impl_from, never}; use stdx::{impl_from, never};
@ -76,7 +76,8 @@ pub struct CompletionItem {
pub ref_match: Option<(Mutability, TextSize)>, pub ref_match: Option<(Mutability, TextSize)>,
/// The import data to add to completion's edits. /// The import data to add to completion's edits.
pub import_to_add: SmallVec<[LocatedImport; 1]>, /// (ImportPath, LastSegment)
pub import_to_add: SmallVec<[(String, String); 1]>,
} }
// We use custom debug for CompletionItem to make snapshot tests more readable. // We use custom debug for CompletionItem to make snapshot tests more readable.
@ -418,7 +419,7 @@ impl Builder {
) )
} }
pub(crate) fn build(self) -> CompletionItem { pub(crate) fn build(self, db: &RootDatabase) -> CompletionItem {
let _p = profile::span("item::Builder::build"); let _p = profile::span("item::Builder::build");
let mut label = self.label; let mut label = self.label;
@ -433,7 +434,7 @@ impl Builder {
if let [import_edit] = &*self.imports_to_add { if let [import_edit] = &*self.imports_to_add {
// snippets can have multiple imports, but normal completions only have up to one // snippets can have multiple imports, but normal completions only have up to one
if let Some(original_path) = import_edit.original_path.as_ref() { if let Some(original_path) = import_edit.original_path.as_ref() {
label = SmolStr::from(format!("{label} (use {original_path})")); label = SmolStr::from(format!("{label} (use {})", original_path.display(db)));
} }
} else if let Some(trait_name) = self.trait_name { } else if let Some(trait_name) = self.trait_name {
label = SmolStr::from(format!("{label} (as {trait_name})")); label = SmolStr::from(format!("{label} (as {trait_name})"));
@ -444,6 +445,17 @@ impl Builder {
None => TextEdit::replace(self.source_range, insert_text), None => TextEdit::replace(self.source_range, insert_text),
}; };
let import_to_add = self
.imports_to_add
.into_iter()
.filter_map(|import| {
Some((
import.import_path.display(db).to_string(),
import.import_path.segments().last()?.display(db).to_string(),
))
})
.collect();
CompletionItem { CompletionItem {
source_range: self.source_range, source_range: self.source_range,
label, label,
@ -457,7 +469,7 @@ impl Builder {
trigger_call_info: self.trigger_call_info, trigger_call_info: self.trigger_call_info,
relevance: self.relevance, relevance: self.relevance,
ref_match: self.ref_match, ref_match: self.ref_match,
import_to_add: self.imports_to_add, import_to_add,
} }
} }
pub(crate) fn lookup_by(&mut self, lookup: impl Into<SmolStr>) -> &mut Builder { pub(crate) fn lookup_by(&mut self, lookup: impl Into<SmolStr>) -> &mut Builder {

View file

@ -243,7 +243,7 @@ pub fn resolve_completion_edits(
config.prefer_no_std, config.prefer_no_std,
) )
}) })
.find(|mod_path| mod_path.to_string() == full_import_path); .find(|mod_path| mod_path.display(db).to_string() == full_import_path);
if let Some(import_path) = import { if let Some(import_path) = import {
insert_use::insert_use(&new_ast, mod_path_to_ast(&import_path), &config.insert_use); insert_use::insert_use(&new_ast, mod_path_to_ast(&import_path), &config.insert_use);
} }

View file

@ -126,24 +126,25 @@ pub(crate) fn render_field(
field: hir::Field, field: hir::Field,
ty: &hir::Type, ty: &hir::Type,
) -> CompletionItem { ) -> CompletionItem {
let db = ctx.db();
let is_deprecated = ctx.is_deprecated(field); let is_deprecated = ctx.is_deprecated(field);
let name = field.name(ctx.db()); let name = field.name(db);
let (name, escaped_name) = (name.unescaped().to_smol_str(), name.to_smol_str()); let (name, escaped_name) = (name.unescaped().to_smol_str(), name.to_smol_str());
let mut item = CompletionItem::new( let mut item = CompletionItem::new(
SymbolKind::Field, SymbolKind::Field,
ctx.source_range(), ctx.source_range(),
field_with_receiver(receiver.as_ref(), &name), field_with_receiver(db, receiver.as_ref(), &name),
); );
item.set_relevance(CompletionRelevance { item.set_relevance(CompletionRelevance {
type_match: compute_type_match(ctx.completion, ty), type_match: compute_type_match(ctx.completion, ty),
exact_name_match: compute_exact_name_match(ctx.completion, name.as_str()), exact_name_match: compute_exact_name_match(ctx.completion, name.as_str()),
..CompletionRelevance::default() ..CompletionRelevance::default()
}); });
item.detail(ty.display(ctx.db()).to_string()) item.detail(ty.display(db).to_string())
.set_documentation(field.docs(ctx.db())) .set_documentation(field.docs(db))
.set_deprecated(is_deprecated) .set_deprecated(is_deprecated)
.lookup_by(name); .lookup_by(name);
item.insert_text(field_with_receiver(receiver.as_ref(), &escaped_name)); item.insert_text(field_with_receiver(db, receiver.as_ref(), &escaped_name));
if let Some(receiver) = &dot_access.receiver { if let Some(receiver) = &dot_access.receiver {
if let Some(original) = ctx.completion.sema.original_ast_node(receiver.clone()) { if let Some(original) = ctx.completion.sema.original_ast_node(receiver.clone()) {
if let Some(ref_match) = compute_ref_match(ctx.completion, ty) { if let Some(ref_match) = compute_ref_match(ctx.completion, ty) {
@ -152,11 +153,18 @@ pub(crate) fn render_field(
} }
} }
item.doc_aliases(ctx.doc_aliases); item.doc_aliases(ctx.doc_aliases);
item.build() item.build(db)
} }
fn field_with_receiver(receiver: Option<&hir::Name>, field_name: &str) -> SmolStr { fn field_with_receiver(
receiver.map_or_else(|| field_name.into(), |receiver| format!("{receiver}.{field_name}").into()) db: &RootDatabase,
receiver: Option<&hir::Name>,
field_name: &str,
) -> SmolStr {
receiver.map_or_else(
|| field_name.into(),
|receiver| format!("{}.{field_name}", receiver.display(db)).into(),
)
} }
pub(crate) fn render_tuple_field( pub(crate) fn render_tuple_field(
@ -168,10 +176,10 @@ pub(crate) fn render_tuple_field(
let mut item = CompletionItem::new( let mut item = CompletionItem::new(
SymbolKind::Field, SymbolKind::Field,
ctx.source_range(), ctx.source_range(),
field_with_receiver(receiver.as_ref(), &field.to_string()), field_with_receiver(ctx.db(), receiver.as_ref(), &field.to_string()),
); );
item.detail(ty.display(ctx.db()).to_string()).lookup_by(field.to_string()); item.detail(ty.display(ctx.db()).to_string()).lookup_by(field.to_string());
item.build() item.build(ctx.db())
} }
pub(crate) fn render_type_inference( pub(crate) fn render_type_inference(
@ -181,7 +189,7 @@ pub(crate) fn render_type_inference(
let mut builder = let mut builder =
CompletionItem::new(CompletionItemKind::InferredType, ctx.source_range(), ty_string); CompletionItem::new(CompletionItemKind::InferredType, ctx.source_range(), ty_string);
builder.set_relevance(CompletionRelevance { is_definite: true, ..Default::default() }); builder.set_relevance(CompletionRelevance { is_definite: true, ..Default::default() });
builder.build() builder.build(ctx.db)
} }
pub(crate) fn render_path_resolution( pub(crate) fn render_path_resolution(
@ -319,7 +327,7 @@ fn render_resolution_path(
item.lookup_by(name.clone()) item.lookup_by(name.clone())
.label(SmolStr::from_iter([&name, "<…>"])) .label(SmolStr::from_iter([&name, "<…>"]))
.trigger_call_info() .trigger_call_info()
.insert_snippet(cap, format!("{local_name}<$0>")); .insert_snippet(cap, format!("{}<$0>", local_name.display(db)));
} }
} }
} }

View file

@ -29,5 +29,5 @@ fn render(ctx: RenderContext<'_>, const_: hir::Const) -> Option<CompletionItem>
} }
item.insert_text(escaped_name); item.insert_text(escaped_name);
Some(item.build()) Some(item.build(ctx.db()))
} }

View file

@ -52,8 +52,13 @@ fn render(
let (call, escaped_call) = match &func_kind { let (call, escaped_call) = match &func_kind {
FuncKind::Method(_, Some(receiver)) => ( FuncKind::Method(_, Some(receiver)) => (
format!("{}.{}", receiver.unescaped(), name.unescaped()).into(), format!(
format!("{receiver}.{name}").into(), "{}.{}",
receiver.unescaped().display(ctx.db()),
name.unescaped().display(ctx.db())
)
.into(),
format!("{}.{}", receiver.display(ctx.db()), name.display(ctx.db())).into(),
), ),
_ => (name.unescaped().to_smol_str(), name.to_smol_str()), _ => (name.unescaped().to_smol_str(), name.to_smol_str()),
}; };

View file

@ -71,8 +71,10 @@ fn render(
} }
None => (name.clone().into(), name.into(), false), None => (name.clone().into(), name.into(), false),
}; };
let (qualified_name, escaped_qualified_name) = let (qualified_name, escaped_qualified_name) = (
(qualified_name.unescaped().to_string(), qualified_name.to_string()); qualified_name.unescaped().display(ctx.db()).to_string(),
qualified_name.display(ctx.db()).to_string(),
);
let snippet_cap = ctx.snippet_cap(); let snippet_cap = ctx.snippet_cap();
let mut rendered = match kind { let mut rendered = match kind {
@ -98,7 +100,7 @@ fn render(
} }
let label = format_literal_label(&qualified_name, kind, snippet_cap); let label = format_literal_label(&qualified_name, kind, snippet_cap);
let lookup = if qualified { let lookup = if qualified {
format_literal_lookup(&short_qualified_name.to_string(), kind) format_literal_lookup(&short_qualified_name.display(ctx.db()).to_string(), kind)
} else { } else {
format_literal_lookup(&qualified_name, kind) format_literal_lookup(&qualified_name, kind)
}; };

View file

@ -57,7 +57,10 @@ pub(crate) fn render_variant_pat(
let enum_ty = variant.parent_enum(ctx.db()).ty(ctx.db()); let enum_ty = variant.parent_enum(ctx.db()).ty(ctx.db());
let (name, escaped_name) = match path { let (name, escaped_name) = match path {
Some(path) => (path.unescaped().to_string().into(), path.to_string().into()), Some(path) => (
path.unescaped().display(ctx.db()).to_string().into(),
path.display(ctx.db()).to_string().into(),
),
None => { None => {
let name = local_name.unwrap_or_else(|| variant.name(ctx.db())); let name = local_name.unwrap_or_else(|| variant.name(ctx.db()));
(name.unescaped().to_smol_str(), name.to_smol_str()) (name.unescaped().to_smol_str(), name.to_smol_str())
@ -121,7 +124,7 @@ fn build_completion(
Some(snippet_cap) => item.insert_snippet(snippet_cap, pat), Some(snippet_cap) => item.insert_snippet(snippet_cap, pat),
None => item.insert_text(pat), None => item.insert_text(pat),
}; };
item.build() item.build(ctx.db())
} }
fn render_pat( fn render_pat(
@ -172,7 +175,7 @@ fn render_record_as_pat(
format!( format!(
"{name} {{ {}{} }}", "{name} {{ {}{} }}",
fields.enumerate().format_with(", ", |(idx, field), f| { fields.enumerate().format_with(", ", |(idx, field), f| {
f(&format_args!("{}${}", field.name(db), idx + 1)) f(&format_args!("{}${}", field.name(db).display(db.upcast()), idx + 1))
}), }),
if fields_omitted { ", .." } else { "" }, if fields_omitted { ", .." } else { "" },
name = name name = name

View file

@ -53,5 +53,5 @@ fn render(
} }
item.insert_text(escaped_name); item.insert_text(escaped_name);
Some(item.build()) Some(item.build(ctx.db()))
} }

View file

@ -21,8 +21,10 @@ pub(crate) fn render_union_literal(
let name = local_name.unwrap_or_else(|| un.name(ctx.db())); let name = local_name.unwrap_or_else(|| un.name(ctx.db()));
let (qualified_name, escaped_qualified_name) = match path { let (qualified_name, escaped_qualified_name) = match path {
Some(p) => (p.unescaped().to_string(), p.to_string()), Some(p) => (p.unescaped().display(ctx.db()).to_string(), p.display(ctx.db()).to_string()),
None => (name.unescaped().to_string(), name.to_string()), None => {
(name.unescaped().display(ctx.db()).to_string(), name.display(ctx.db()).to_string())
}
}; };
let label = format_literal_label(&name.to_smol_str(), StructKind::Record, ctx.snippet_cap()); let label = format_literal_label(&name.to_smol_str(), StructKind::Record, ctx.snippet_cap());
let lookup = format_literal_lookup(&name.to_smol_str(), StructKind::Record); let lookup = format_literal_lookup(&name.to_smol_str(), StructKind::Record);
@ -51,9 +53,9 @@ pub(crate) fn render_union_literal(
format!( format!(
"{} {{ {} }}", "{} {{ {} }}",
escaped_qualified_name, escaped_qualified_name,
fields fields.iter().format_with(", ", |field, f| {
.iter() f(&format_args!("{}: ()", field.name(ctx.db()).display(ctx.db())))
.format_with(", ", |field, f| { f(&format_args!("{}: ()", field.name(ctx.db()))) }) })
) )
}; };
@ -61,7 +63,11 @@ pub(crate) fn render_union_literal(
"{} {{ {}{} }}", "{} {{ {}{} }}",
qualified_name, qualified_name,
fields.iter().format_with(", ", |field, f| { fields.iter().format_with(", ", |field, f| {
f(&format_args!("{}: {}", field.name(ctx.db()), field.ty(ctx.db()).display(ctx.db()))) f(&format_args!(
"{}: {}",
field.name(ctx.db()).display(ctx.db()),
field.ty(ctx.db()).display(ctx.db())
))
}), }),
if fields_omitted { ", .." } else { "" } if fields_omitted { ", .." } else { "" }
); );
@ -76,5 +82,5 @@ pub(crate) fn render_union_literal(
None => item.insert_text(literal), None => item.insert_text(literal),
}; };
Some(item.build()) Some(item.build(ctx.db()))
} }

View file

@ -27,14 +27,14 @@ pub(crate) fn render_record_lit(
} }
let completions = fields.iter().enumerate().format_with(", ", |(idx, field), f| { let completions = fields.iter().enumerate().format_with(", ", |(idx, field), f| {
if snippet_cap.is_some() { if snippet_cap.is_some() {
f(&format_args!("{}: ${{{}:()}}", field.name(db), idx + 1)) f(&format_args!("{}: ${{{}:()}}", field.name(db).display(db.upcast()), idx + 1))
} else { } else {
f(&format_args!("{}: ()", field.name(db))) f(&format_args!("{}: ()", field.name(db).display(db.upcast())))
} }
}); });
let types = fields.iter().format_with(", ", |field, f| { let types = fields.iter().format_with(", ", |field, f| {
f(&format_args!("{}: {}", field.name(db), field.ty(db).display(db))) f(&format_args!("{}: {}", field.name(db).display(db.upcast()), field.ty(db).display(db)))
}); });
RenderedLiteral { RenderedLiteral {

View file

@ -198,11 +198,11 @@ pub(crate) fn check_edit_with_config(
&db, &db,
&config, &config,
position, position,
completion.import_to_add.iter().filter_map(|import_edit| { completion
let import_path = &import_edit.import_path; .import_to_add
let import_name = import_path.segments().last()?; .iter()
Some((import_path.to_string(), import_name.to_string())) .cloned()
}), .filter_map(|(import_path, import_name)| Some((import_path, import_name))),
) )
.into_iter() .into_iter()
.flatten() .flatten()

View file

@ -362,12 +362,12 @@ fn import_for_item(
let original_item_candidate = item_for_path_search(db, original_item)?; let original_item_candidate = item_for_path_search(db, original_item)?;
let import_path_candidate = mod_path(original_item_candidate)?; let import_path_candidate = mod_path(original_item_candidate)?;
let import_path_string = import_path_candidate.to_string(); let import_path_string = import_path_candidate.display(db).to_string();
let expected_import_end = if item_as_assoc(db, original_item).is_some() { let expected_import_end = if item_as_assoc(db, original_item).is_some() {
unresolved_qualifier.to_string() unresolved_qualifier.to_string()
} else { } else {
format!("{unresolved_qualifier}::{}", item_name(db, original_item)?) format!("{unresolved_qualifier}::{}", item_name(db, original_item)?.display(db))
}; };
if !import_path_string.contains(unresolved_first_segment) if !import_path_string.contains(unresolved_first_segment)
|| !import_path_string.ends_with(&expected_import_end) || !import_path_string.ends_with(&expected_import_end)

View file

@ -202,12 +202,13 @@ fn rename_mod(
// - Module has submodules defined in separate files // - Module has submodules defined in separate files
let dir_paths = match (is_mod_rs, has_detached_child, module.name(sema.db)) { let dir_paths = match (is_mod_rs, has_detached_child, module.name(sema.db)) {
// Go up one level since the anchor is inside the dir we're trying to rename // Go up one level since the anchor is inside the dir we're trying to rename
(true, _, Some(mod_name)) => { (true, _, Some(mod_name)) => Some((
Some((format!("../{}", mod_name.unescaped()), format!("../{new_name}"))) format!("../{}", mod_name.unescaped().display(sema.db)),
} format!("../{new_name}"),
)),
// The anchor is on the same level as target dir // The anchor is on the same level as target dir
(false, true, Some(mod_name)) => { (false, true, Some(mod_name)) => {
Some((mod_name.unescaped().to_string(), new_name.to_owned())) Some((mod_name.unescaped().display(sema.db).to_string(), new_name.to_owned()))
} }
_ => None, _ => None,
}; };

View file

@ -38,15 +38,15 @@ pub fn get_missing_assoc_items(
for item in imp.items(sema.db) { for item in imp.items(sema.db) {
match item { match item {
hir::AssocItem::Function(it) => { hir::AssocItem::Function(it) => {
impl_fns_consts.insert(it.name(sema.db).to_string()); impl_fns_consts.insert(it.name(sema.db).display(sema.db).to_string());
} }
hir::AssocItem::Const(it) => { hir::AssocItem::Const(it) => {
if let Some(name) = it.name(sema.db) { if let Some(name) = it.name(sema.db) {
impl_fns_consts.insert(name.to_string()); impl_fns_consts.insert(name.display(sema.db).to_string());
} }
} }
hir::AssocItem::TypeAlias(it) => { hir::AssocItem::TypeAlias(it) => {
impl_type.insert(it.name(sema.db).to_string()); impl_type.insert(it.name(sema.db).display(sema.db).to_string());
} }
} }
} }
@ -57,12 +57,14 @@ pub fn get_missing_assoc_items(
.into_iter() .into_iter()
.filter(|i| match i { .filter(|i| match i {
hir::AssocItem::Function(f) => { hir::AssocItem::Function(f) => {
!impl_fns_consts.contains(&f.name(sema.db).to_string()) !impl_fns_consts.contains(&f.name(sema.db).display(sema.db).to_string())
}
hir::AssocItem::TypeAlias(t) => {
!impl_type.contains(&t.name(sema.db).display(sema.db).to_string())
} }
hir::AssocItem::TypeAlias(t) => !impl_type.contains(&t.name(sema.db).to_string()),
hir::AssocItem::Const(c) => c hir::AssocItem::Const(c) => c
.name(sema.db) .name(sema.db)
.map(|n| !impl_fns_consts.contains(&n.to_string())) .map(|n| !impl_fns_consts.contains(&n.display(sema.db).to_string()))
.unwrap_or_default(), .unwrap_or_default(),
}) })
.collect() .collect()
@ -137,7 +139,7 @@ mod tests {
sema.find_node_at_offset_with_descend(file.syntax(), position.offset).unwrap(); sema.find_node_at_offset_with_descend(file.syntax(), position.offset).unwrap();
let trait_ = crate::traits::resolve_target_trait(&sema, &impl_block); let trait_ = crate::traits::resolve_target_trait(&sema, &impl_block);
let actual = match trait_ { let actual = match trait_ {
Some(trait_) => trait_.name(&db).to_string(), Some(trait_) => trait_.name(&db).display(&db).to_string(),
None => String::new(), None => String::new(),
}; };
expect.assert_eq(&actual); expect.assert_eq(&actual);
@ -152,7 +154,7 @@ mod tests {
let items = crate::traits::get_missing_assoc_items(&sema, &impl_block); let items = crate::traits::get_missing_assoc_items(&sema, &impl_block);
let actual = items let actual = items
.into_iter() .into_iter()
.map(|item| item.name(&db).unwrap().to_string()) .map(|item| item.name(&db).unwrap().display(&db).to_string())
.collect::<Vec<_>>() .collect::<Vec<_>>()
.join("\n"); .join("\n");
expect.assert_eq(&actual); expect.assert_eq(&actual);

View file

@ -31,7 +31,7 @@ use crate::{fix, Diagnostic, DiagnosticsContext};
pub(crate) fn missing_fields(ctx: &DiagnosticsContext<'_>, d: &hir::MissingFields) -> Diagnostic { pub(crate) fn missing_fields(ctx: &DiagnosticsContext<'_>, d: &hir::MissingFields) -> Diagnostic {
let mut message = String::from("missing structure fields:\n"); let mut message = String::from("missing structure fields:\n");
for field in &d.missed_fields { for field in &d.missed_fields {
format_to!(message, "- {}\n", field); format_to!(message, "- {}\n", field.display(ctx.sema.db));
} }
let ptr = InFile::new( let ptr = InFile::new(
@ -175,7 +175,7 @@ fn fixes(ctx: &DiagnosticsContext<'_>, d: &hir::MissingFields) -> Option<Vec<Ass
fn make_ty(ty: &hir::Type, db: &dyn HirDatabase, module: hir::Module) -> ast::Type { fn make_ty(ty: &hir::Type, db: &dyn HirDatabase, module: hir::Module) -> ast::Type {
let ty_str = match ty.as_adt() { let ty_str = match ty.as_adt() {
Some(adt) => adt.name(db).to_string(), Some(adt) => adt.name(db).display(db.upcast()).to_string(),
None => { None => {
ty.display_source_code(db, module.into(), false).ok().unwrap_or_else(|| "_".to_string()) ty.display_source_code(db, module.into(), false).ok().unwrap_or_else(|| "_".to_string())
} }

View file

@ -30,7 +30,10 @@ pub(crate) fn need_mut(ctx: &DiagnosticsContext<'_>, d: &hir::NeedMut) -> Diagno
})(); })();
Diagnostic::new( Diagnostic::new(
"need-mut", "need-mut",
format!("cannot mutate immutable variable `{}`", d.local.name(ctx.sema.db)), format!(
"cannot mutate immutable variable `{}`",
d.local.name(ctx.sema.db).display(ctx.sema.db)
),
ctx.sema.diagnostics_display_range(d.span.clone()).range, ctx.sema.diagnostics_display_range(d.span.clone()).range,
) )
.with_fixes(fixes) .with_fixes(fixes)

View file

@ -11,7 +11,11 @@ pub(crate) fn private_assoc_item(
d: &hir::PrivateAssocItem, d: &hir::PrivateAssocItem,
) -> Diagnostic { ) -> Diagnostic {
// FIXME: add quickfix // FIXME: add quickfix
let name = d.item.name(ctx.sema.db).map(|name| format!("`{name}` ")).unwrap_or_default(); let name = d
.item
.name(ctx.sema.db)
.map(|name| format!("`{}` ", name.display(ctx.sema.db)))
.unwrap_or_default();
Diagnostic::new( Diagnostic::new(
"private-assoc-item", "private-assoc-item",
format!( format!(

View file

@ -9,8 +9,8 @@ pub(crate) fn private_field(ctx: &DiagnosticsContext<'_>, d: &hir::PrivateField)
"private-field", "private-field",
format!( format!(
"field `{}` of `{}` is private", "field `{}` of `{}` is private",
d.field.name(ctx.sema.db), d.field.name(ctx.sema.db).display(ctx.sema.db),
d.field.parent_def(ctx.sema.db).name(ctx.sema.db) d.field.parent_def(ctx.sema.db).name(ctx.sema.db).display(ctx.sema.db)
), ),
ctx.sema.diagnostics_display_range(d.expr.clone().map(|it| it.into())).range, ctx.sema.diagnostics_display_range(d.expr.clone().map(|it| it.into())).range,
) )

View file

@ -8,7 +8,7 @@ pub(crate) fn undeclared_label(
let name = &d.name; let name = &d.name;
Diagnostic::new( Diagnostic::new(
"undeclared-label", "undeclared-label",
format!("use of undeclared label `{name}`"), format!("use of undeclared label `{}`", name.display(ctx.sema.db)),
ctx.sema.diagnostics_display_range(d.node.clone().map(|it| it.into())).range, ctx.sema.diagnostics_display_range(d.node.clone().map(|it| it.into())).range,
) )
} }

View file

@ -8,7 +8,7 @@ pub(crate) fn unreachable_label(
let name = &d.name; let name = &d.name;
Diagnostic::new( Diagnostic::new(
"unreachable-label", "unreachable-label",
format!("use of unreachable label `{name}`"), format!("use of unreachable label `{}`", name.display(ctx.sema.db)),
ctx.sema.diagnostics_display_range(d.node.clone().map(|it| it.into())).range, ctx.sema.diagnostics_display_range(d.node.clone().map(|it| it.into())).range,
) )
} }

View file

@ -26,7 +26,7 @@ pub(crate) fn unresolved_field(
"unresolved-field", "unresolved-field",
format!( format!(
"no field `{}` on type `{}`{method_suffix}", "no field `{}` on type `{}`{method_suffix}",
d.name, d.name.display(ctx.sema.db),
d.receiver.display(ctx.sema.db) d.receiver.display(ctx.sema.db)
), ),
ctx.sema.diagnostics_display_range(d.expr.clone().map(|it| it.into())).range, ctx.sema.diagnostics_display_range(d.expr.clone().map(|it| it.into())).range,

View file

@ -13,7 +13,7 @@ pub(crate) fn unresolved_macro_call(
let bang = if d.is_bang { "!" } else { "" }; let bang = if d.is_bang { "!" } else { "" };
Diagnostic::new( Diagnostic::new(
"unresolved-macro-call", "unresolved-macro-call",
format!("unresolved macro `{}{bang}`", d.path), format!("unresolved macro `{}{bang}`", d.path.display(ctx.sema.db)),
display_range, display_range,
) )
.experimental() .experimental()

View file

@ -26,7 +26,7 @@ pub(crate) fn unresolved_method(
"unresolved-method", "unresolved-method",
format!( format!(
"no method `{}` on type `{}`{field_suffix}", "no method `{}` on type `{}`{field_suffix}",
d.name, d.name.display(ctx.sema.db),
d.receiver.display(ctx.sema.db) d.receiver.display(ctx.sema.db)
), ),
ctx.sema.diagnostics_display_range(d.expr.clone().map(|it| it.into())).range, ctx.sema.diagnostics_display_range(d.expr.clone().map(|it| it.into())).range,

View file

@ -184,6 +184,7 @@ impl<'db> MatchFinder<'db> {
( (
file_id, file_id,
replacing::matches_to_edit( replacing::matches_to_edit(
self.sema.db,
&matches, &matches,
&self.sema.db.file_text(file_id), &self.sema.db.file_text(file_id),
&self.rules, &self.rules,

View file

@ -14,14 +14,16 @@ use crate::{fragments, resolving::ResolvedRule, Match, SsrMatches};
/// template. Placeholders in the template will have been substituted with whatever they matched to /// template. Placeholders in the template will have been substituted with whatever they matched to
/// in the original code. /// in the original code.
pub(crate) fn matches_to_edit( pub(crate) fn matches_to_edit(
db: &dyn hir::db::ExpandDatabase,
matches: &SsrMatches, matches: &SsrMatches,
file_src: &str, file_src: &str,
rules: &[ResolvedRule], rules: &[ResolvedRule],
) -> TextEdit { ) -> TextEdit {
matches_to_edit_at_offset(matches, file_src, 0.into(), rules) matches_to_edit_at_offset(db, matches, file_src, 0.into(), rules)
} }
fn matches_to_edit_at_offset( fn matches_to_edit_at_offset(
db: &dyn hir::db::ExpandDatabase,
matches: &SsrMatches, matches: &SsrMatches,
file_src: &str, file_src: &str,
relative_start: TextSize, relative_start: TextSize,
@ -31,13 +33,14 @@ fn matches_to_edit_at_offset(
for m in &matches.matches { for m in &matches.matches {
edit_builder.replace( edit_builder.replace(
m.range.range.checked_sub(relative_start).unwrap(), m.range.range.checked_sub(relative_start).unwrap(),
render_replace(m, file_src, rules), render_replace(db, m, file_src, rules),
); );
} }
edit_builder.finish() edit_builder.finish()
} }
struct ReplacementRenderer<'a> { struct ReplacementRenderer<'a> {
db: &'a dyn hir::db::ExpandDatabase,
match_info: &'a Match, match_info: &'a Match,
file_src: &'a str, file_src: &'a str,
rules: &'a [ResolvedRule], rules: &'a [ResolvedRule],
@ -53,13 +56,19 @@ struct ReplacementRenderer<'a> {
placeholder_tokens_requiring_parenthesis: FxHashSet<SyntaxToken>, placeholder_tokens_requiring_parenthesis: FxHashSet<SyntaxToken>,
} }
fn render_replace(match_info: &Match, file_src: &str, rules: &[ResolvedRule]) -> String { fn render_replace(
db: &dyn hir::db::ExpandDatabase,
match_info: &Match,
file_src: &str,
rules: &[ResolvedRule],
) -> String {
let rule = &rules[match_info.rule_index]; let rule = &rules[match_info.rule_index];
let template = rule let template = rule
.template .template
.as_ref() .as_ref()
.expect("You called MatchFinder::edits after calling MatchFinder::add_search_pattern"); .expect("You called MatchFinder::edits after calling MatchFinder::add_search_pattern");
let mut renderer = ReplacementRenderer { let mut renderer = ReplacementRenderer {
db,
match_info, match_info,
file_src, file_src,
rules, rules,
@ -96,7 +105,7 @@ impl ReplacementRenderer<'_> {
fn render_node(&mut self, node: &SyntaxNode) { fn render_node(&mut self, node: &SyntaxNode) {
if let Some(mod_path) = self.match_info.rendered_template_paths.get(node) { if let Some(mod_path) = self.match_info.rendered_template_paths.get(node) {
self.out.push_str(&mod_path.to_string()); self.out.push_str(&mod_path.display(self.db).to_string());
// Emit everything except for the segment's name-ref, since we already effectively // Emit everything except for the segment's name-ref, since we already effectively
// emitted that as part of `mod_path`. // emitted that as part of `mod_path`.
if let Some(path) = ast::Path::cast(node.clone()) { if let Some(path) = ast::Path::cast(node.clone()) {
@ -144,6 +153,7 @@ impl ReplacementRenderer<'_> {
); );
} }
let edit = matches_to_edit_at_offset( let edit = matches_to_edit_at_offset(
self.db,
&placeholder_value.inner_matches, &placeholder_value.inner_matches,
self.file_src, self.file_src,
range.start(), range.start(),

View file

@ -394,7 +394,7 @@ fn rewrite_url_link(db: &RootDatabase, def: Definition, target: &str) -> Option<
fn mod_path_of_def(db: &RootDatabase, def: Definition) -> Option<String> { fn mod_path_of_def(db: &RootDatabase, def: Definition) -> Option<String> {
def.canonical_module_path(db).map(|it| { def.canonical_module_path(db).map(|it| {
let mut path = String::new(); let mut path = String::new();
it.flat_map(|it| it.name(db)).for_each(|name| format_to!(path, "{}/", name)); it.flat_map(|it| it.name(db)).for_each(|name| format_to!(path, "{}/", name.display(db)));
path path
}) })
} }
@ -567,9 +567,9 @@ fn filename_and_frag_for_def(
let res = match def { let res = match def {
Definition::Adt(adt) => match adt { Definition::Adt(adt) => match adt {
Adt::Struct(s) => format!("struct.{}.html", s.name(db)), Adt::Struct(s) => format!("struct.{}.html", s.name(db).display(db.upcast())),
Adt::Enum(e) => format!("enum.{}.html", e.name(db)), Adt::Enum(e) => format!("enum.{}.html", e.name(db).display(db.upcast())),
Adt::Union(u) => format!("union.{}.html", u.name(db)), Adt::Union(u) => format!("union.{}.html", u.name(db).display(db.upcast())),
}, },
Definition::Module(m) => match m.name(db) { Definition::Module(m) => match m.name(db) {
// `#[doc(keyword = "...")]` is internal used only by rust compiler // `#[doc(keyword = "...")]` is internal used only by rust compiler
@ -577,21 +577,25 @@ fn filename_and_frag_for_def(
Some(kw) => { Some(kw) => {
format!("keyword.{}.html", kw.trim_matches('"')) format!("keyword.{}.html", kw.trim_matches('"'))
} }
None => format!("{name}/index.html"), None => format!("{}/index.html", name.display(db.upcast())),
}, },
None => String::from("index.html"), None => String::from("index.html"),
}, },
Definition::Trait(t) => format!("trait.{}.html", t.name(db)), Definition::Trait(t) => format!("trait.{}.html", t.name(db).display(db.upcast())),
Definition::TraitAlias(t) => format!("traitalias.{}.html", t.name(db)), Definition::TraitAlias(t) => format!("traitalias.{}.html", t.name(db).display(db.upcast())),
Definition::TypeAlias(t) => format!("type.{}.html", t.name(db)), Definition::TypeAlias(t) => format!("type.{}.html", t.name(db).display(db.upcast())),
Definition::BuiltinType(t) => format!("primitive.{}.html", t.name()), Definition::BuiltinType(t) => format!("primitive.{}.html", t.name().display(db.upcast())),
Definition::Function(f) => format!("fn.{}.html", f.name(db)), Definition::Function(f) => format!("fn.{}.html", f.name(db).display(db.upcast())),
Definition::Variant(ev) => { Definition::Variant(ev) => {
format!("enum.{}.html#variant.{}", ev.parent_enum(db).name(db), ev.name(db)) format!(
"enum.{}.html#variant.{}",
ev.parent_enum(db).name(db).display(db.upcast()),
ev.name(db).display(db.upcast())
)
} }
Definition::Const(c) => format!("const.{}.html", c.name(db)?), Definition::Const(c) => format!("const.{}.html", c.name(db)?.display(db.upcast())),
Definition::Static(s) => format!("static.{}.html", s.name(db)), Definition::Static(s) => format!("static.{}.html", s.name(db).display(db.upcast())),
Definition::Macro(mac) => format!("macro.{}.html", mac.name(db)), Definition::Macro(mac) => format!("macro.{}.html", mac.name(db).display(db.upcast())),
Definition::Field(field) => { Definition::Field(field) => {
let def = match field.parent_def(db) { let def = match field.parent_def(db) {
hir::VariantDef::Struct(it) => Definition::Adt(it.into()), hir::VariantDef::Struct(it) => Definition::Adt(it.into()),
@ -599,7 +603,11 @@ fn filename_and_frag_for_def(
hir::VariantDef::Variant(it) => Definition::Variant(it), hir::VariantDef::Variant(it) => Definition::Variant(it),
}; };
let (_, file, _) = filename_and_frag_for_def(db, def)?; let (_, file, _) = filename_and_frag_for_def(db, def)?;
return Some((def, file, Some(format!("structfield.{}", field.name(db))))); return Some((
def,
file,
Some(format!("structfield.{}", field.name(db).display(db.upcast()))),
));
} }
Definition::SelfType(impl_) => { Definition::SelfType(impl_) => {
let adt = impl_.self_ty(db).as_adt()?.into(); let adt = impl_.self_ty(db).as_adt()?.into();
@ -633,12 +641,14 @@ fn get_assoc_item_fragment(db: &dyn HirDatabase, assoc_item: hir::AssocItem) ->
// Rustdoc makes this decision based on whether a method 'has defaultness'. // Rustdoc makes this decision based on whether a method 'has defaultness'.
// Currently this is only the case for provided trait methods. // Currently this is only the case for provided trait methods.
if is_trait_method && !function.has_body(db) { if is_trait_method && !function.has_body(db) {
format!("tymethod.{}", function.name(db)) format!("tymethod.{}", function.name(db).display(db.upcast()))
} else { } else {
format!("method.{}", function.name(db)) format!("method.{}", function.name(db).display(db.upcast()))
} }
} }
AssocItem::Const(constant) => format!("associatedconstant.{}", constant.name(db)?), AssocItem::Const(constant) => {
AssocItem::TypeAlias(ty) => format!("associatedtype.{}", ty.name(db)), format!("associatedconstant.{}", constant.name(db)?.display(db.upcast()))
}
AssocItem::TypeAlias(ty) => format!("associatedtype.{}", ty.name(db).display(db.upcast())),
}) })
} }

View file

@ -76,7 +76,7 @@ pub(crate) fn expand_macro(db: &RootDatabase, position: FilePosition) -> Option<
if let Some(item) = ast::Item::cast(node.clone()) { if let Some(item) = ast::Item::cast(node.clone()) {
if let Some(def) = sema.resolve_attr_macro_call(&item) { if let Some(def) = sema.resolve_attr_macro_call(&item) {
break ( break (
def.name(db).to_string(), def.name(db).display(db).to_string(),
expand_attr_macro_recur(&sema, &item)?, expand_attr_macro_recur(&sema, &item)?,
SyntaxKind::MACRO_ITEMS, SyntaxKind::MACRO_ITEMS,
); );

View file

@ -56,7 +56,7 @@ impl HoverAction {
mod_path: render::path( mod_path: render::path(
db, db,
it.module(db)?, it.module(db)?,
it.name(db).map(|name| name.to_string()), it.name(db).map(|name| name.display(db).to_string()),
), ),
nav: it.try_to_nav(db)?, nav: it.try_to_nav(db)?,
}) })

View file

@ -370,7 +370,7 @@ fn definition_owner_name(db: &RootDatabase, def: &Definition) -> Option<String>
Definition::Variant(e) => Some(e.parent_enum(db).name(db)), Definition::Variant(e) => Some(e.parent_enum(db).name(db)),
_ => None, _ => None,
} }
.map(|name| name.to_string()) .map(|name| name.display(db).to_string())
} }
pub(super) fn path(db: &RootDatabase, module: hir::Module, item_name: Option<String>) -> String { pub(super) fn path(db: &RootDatabase, module: hir::Module, item_name: Option<String>) -> String {
@ -380,7 +380,7 @@ pub(super) fn path(db: &RootDatabase, module: hir::Module, item_name: Option<Str
.path_to_root(db) .path_to_root(db)
.into_iter() .into_iter()
.rev() .rev()
.flat_map(|it| it.name(db).map(|name| name.to_string())); .flat_map(|it| it.name(db).map(|name| name.display(db).to_string()));
crate_name.into_iter().chain(module_path).chain(item_name).join("::") crate_name.into_iter().chain(module_path).chain(item_name).join("::")
} }
@ -468,18 +468,20 @@ pub(super) fn definition(
Definition::BuiltinType(it) => { Definition::BuiltinType(it) => {
return famous_defs return famous_defs
.and_then(|fd| builtin(fd, it)) .and_then(|fd| builtin(fd, it))
.or_else(|| Some(Markup::fenced_block(&it.name()))) .or_else(|| Some(Markup::fenced_block(&it.name().display(db))))
} }
Definition::Local(it) => return local(db, it, config), Definition::Local(it) => return local(db, it, config),
Definition::SelfType(impl_def) => { Definition::SelfType(impl_def) => {
impl_def.self_ty(db).as_adt().map(|adt| label_and_docs(db, adt))? impl_def.self_ty(db).as_adt().map(|adt| label_and_docs(db, adt))?
} }
Definition::GenericParam(it) => label_and_docs(db, it), Definition::GenericParam(it) => label_and_docs(db, it),
Definition::Label(it) => return Some(Markup::fenced_block(&it.name(db))), Definition::Label(it) => return Some(Markup::fenced_block(&it.name(db).display(db))),
// FIXME: We should be able to show more info about these // FIXME: We should be able to show more info about these
Definition::BuiltinAttr(it) => return render_builtin_attr(db, it), Definition::BuiltinAttr(it) => return render_builtin_attr(db, it),
Definition::ToolModule(it) => return Some(Markup::fenced_block(&it.name(db))), Definition::ToolModule(it) => return Some(Markup::fenced_block(&it.name(db))),
Definition::DeriveHelper(it) => (format!("derive_helper {}", it.name(db)), None), Definition::DeriveHelper(it) => {
(format!("derive_helper {}", it.name(db).display(db)), None)
}
}; };
let docs = docs let docs = docs
@ -717,19 +719,19 @@ fn markup(docs: Option<String>, desc: String, mod_path: Option<String>) -> Optio
fn builtin(famous_defs: &FamousDefs<'_, '_>, builtin: hir::BuiltinType) -> Option<Markup> { fn builtin(famous_defs: &FamousDefs<'_, '_>, builtin: hir::BuiltinType) -> Option<Markup> {
// std exposes prim_{} modules with docstrings on the root to document the builtins // std exposes prim_{} modules with docstrings on the root to document the builtins
let primitive_mod = format!("prim_{}", builtin.name()); let primitive_mod = format!("prim_{}", builtin.name().display(famous_defs.0.db));
let doc_owner = find_std_module(famous_defs, &primitive_mod)?; let doc_owner = find_std_module(famous_defs, &primitive_mod)?;
let docs = doc_owner.attrs(famous_defs.0.db).docs()?; let docs = doc_owner.attrs(famous_defs.0.db).docs()?;
markup(Some(docs.into()), builtin.name().to_string(), None) markup(Some(docs.into()), builtin.name().display(famous_defs.0.db).to_string(), None)
} }
fn find_std_module(famous_defs: &FamousDefs<'_, '_>, name: &str) -> Option<hir::Module> { fn find_std_module(famous_defs: &FamousDefs<'_, '_>, name: &str) -> Option<hir::Module> {
let db = famous_defs.0.db; let db = famous_defs.0.db;
let std_crate = famous_defs.std()?; let std_crate = famous_defs.std()?;
let std_root_module = std_crate.root_module(db); let std_root_module = std_crate.root_module(db);
std_root_module std_root_module.children(db).find(|module| {
.children(db) module.name(db).map_or(false, |module| module.display(db).to_string() == name)
.find(|module| module.name(db).map_or(false, |module| module.to_string() == name)) })
} }
fn local(db: &RootDatabase, it: hir::Local, config: &HoverConfig) -> Option<Markup> { fn local(db: &RootDatabase, it: hir::Local, config: &HoverConfig) -> Option<Markup> {
@ -748,7 +750,7 @@ fn local(db: &RootDatabase, it: hir::Local, config: &HoverConfig) -> Option<Mark
} else { } else {
"" ""
}; };
format!("{let_kw}{is_mut}{name}: {ty}") format!("{let_kw}{is_mut}{}: {ty}", name.display(db))
} }
None => format!("{is_mut}self: {ty}"), None => format!("{is_mut}self: {ty}"),
}; };

Some files were not shown because too many files have changed in this diff Show more