2018-12-27 11:07:21 -06:00
|
|
|
use std::fmt;
|
|
|
|
|
|
|
|
use ra_syntax::{ast, SmolStr};
|
|
|
|
|
|
|
|
/// `Name` is a wrapper around string, which is used in hir for both references
|
2019-01-08 17:47:12 -06:00
|
|
|
/// and declarations. In theory, names should also carry hygiene info, but we are
|
2018-12-27 11:07:21 -06:00
|
|
|
/// not there yet!
|
2019-05-21 05:18:30 -05:00
|
|
|
#[derive(Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
2018-12-27 11:07:21 -06:00
|
|
|
pub struct Name {
|
|
|
|
text: SmolStr,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Name {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
fmt::Display::fmt(&self.text, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-27 12:02:08 -06:00
|
|
|
impl fmt::Debug for Name {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
fmt::Debug::fmt(&self.text, f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-27 11:07:21 -06:00
|
|
|
impl Name {
|
2019-03-20 14:33:26 -05:00
|
|
|
/// Note: this is private to make creating name from random string hard.
|
|
|
|
/// Hopefully, this should allow us to integrate hygiene cleaner in the
|
|
|
|
/// future, and to switch to interned representation of names.
|
|
|
|
fn new(text: SmolStr) -> Name {
|
2018-12-28 12:34:58 -06:00
|
|
|
Name { text }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn missing() -> Name {
|
|
|
|
Name::new("[missing name]".into())
|
|
|
|
}
|
|
|
|
|
2019-01-05 15:37:59 -06:00
|
|
|
pub(crate) fn self_param() -> Name {
|
|
|
|
Name::new("self".into())
|
|
|
|
}
|
|
|
|
|
2019-01-05 18:00:34 -06:00
|
|
|
pub(crate) fn self_type() -> Name {
|
|
|
|
Name::new("Self".into())
|
|
|
|
}
|
|
|
|
|
2018-12-28 12:34:58 -06:00
|
|
|
pub(crate) fn tuple_field_name(idx: usize) -> Name {
|
|
|
|
Name::new(idx.to_string().into())
|
|
|
|
}
|
|
|
|
|
2019-05-12 11:33:47 -05:00
|
|
|
// Needed for Deref
|
|
|
|
pub(crate) fn target() -> Name {
|
|
|
|
Name::new("Target".into())
|
|
|
|
}
|
|
|
|
|
2019-04-22 07:56:28 -05:00
|
|
|
// There's should be no way to extract a string out of `Name`: `Name` in the
|
|
|
|
// future, `Name` will include hygiene information, and you can't encode
|
|
|
|
// hygiene into a String.
|
|
|
|
//
|
|
|
|
// If you need to compare something with `Name`, compare `Name`s directly.
|
|
|
|
//
|
|
|
|
// If you need to render `Name` for the user, use the `Display` impl, but be
|
|
|
|
// aware that it strips hygiene info.
|
|
|
|
#[deprecated(note = "use to_string instead")]
|
|
|
|
pub fn as_smolstr(&self) -> &SmolStr {
|
|
|
|
&self.text
|
2019-02-19 15:32:00 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 11:26:15 -06:00
|
|
|
pub(crate) fn as_known_name(&self) -> Option<KnownName> {
|
|
|
|
let name = match self.text.as_str() {
|
|
|
|
"isize" => KnownName::Isize,
|
|
|
|
"i8" => KnownName::I8,
|
|
|
|
"i16" => KnownName::I16,
|
|
|
|
"i32" => KnownName::I32,
|
|
|
|
"i64" => KnownName::I64,
|
|
|
|
"i128" => KnownName::I128,
|
|
|
|
"usize" => KnownName::Usize,
|
|
|
|
"u8" => KnownName::U8,
|
|
|
|
"u16" => KnownName::U16,
|
|
|
|
"u32" => KnownName::U32,
|
|
|
|
"u64" => KnownName::U64,
|
|
|
|
"u128" => KnownName::U128,
|
|
|
|
"f32" => KnownName::F32,
|
|
|
|
"f64" => KnownName::F64,
|
2019-01-07 13:43:41 -06:00
|
|
|
"bool" => KnownName::Bool,
|
|
|
|
"char" => KnownName::Char,
|
|
|
|
"str" => KnownName::Str,
|
2019-01-05 15:37:59 -06:00
|
|
|
"Self" => KnownName::SelfType,
|
|
|
|
"self" => KnownName::SelfParam,
|
2019-03-02 14:59:04 -06:00
|
|
|
"macro_rules" => KnownName::MacroRules,
|
2019-07-07 02:31:09 -05:00
|
|
|
|
|
|
|
"std" => KnownName::Std,
|
|
|
|
"iter" => KnownName::Iter,
|
|
|
|
"IntoIterator" => KnownName::IntoIterator,
|
|
|
|
"Item" => KnownName::Item,
|
2018-12-27 11:26:15 -06:00
|
|
|
_ => return None,
|
|
|
|
};
|
|
|
|
Some(name)
|
2018-12-27 11:07:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) trait AsName {
|
|
|
|
fn as_name(&self) -> Name;
|
|
|
|
}
|
|
|
|
|
2019-01-08 02:28:42 -06:00
|
|
|
impl AsName for ast::NameRef {
|
2018-12-27 11:07:21 -06:00
|
|
|
fn as_name(&self) -> Name {
|
2019-07-03 14:17:43 -05:00
|
|
|
let name = resolve_name(self.text());
|
2019-07-03 14:01:41 -05:00
|
|
|
Name::new(name)
|
2018-12-27 11:07:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-08 02:28:42 -06:00
|
|
|
impl AsName for ast::Name {
|
2018-12-27 11:07:21 -06:00
|
|
|
fn as_name(&self) -> Name {
|
2019-07-03 14:17:43 -05:00
|
|
|
let name = resolve_name(self.text());
|
2019-07-03 14:01:41 -05:00
|
|
|
Name::new(name)
|
2018-12-27 11:07:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-05 15:34:45 -05:00
|
|
|
impl<'a> AsName for ast::FieldKind<'a> {
|
|
|
|
fn as_name(&self) -> Name {
|
|
|
|
match self {
|
|
|
|
ast::FieldKind::Name(nr) => nr.as_name(),
|
|
|
|
ast::FieldKind::Index(idx) => Name::new(idx.text().clone()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-27 11:07:21 -06:00
|
|
|
impl AsName for ra_db::Dependency {
|
|
|
|
fn as_name(&self) -> Name {
|
|
|
|
Name::new(self.name.clone())
|
|
|
|
}
|
|
|
|
}
|
2018-12-27 11:26:15 -06:00
|
|
|
|
|
|
|
// Ideally, should be replaced with
|
|
|
|
// ```
|
|
|
|
// const ISIZE: Name = Name::new("isize")
|
|
|
|
// ```
|
|
|
|
// but const-fn is not that powerful yet.
|
2019-05-30 07:03:58 -05:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
2018-12-27 11:26:15 -06:00
|
|
|
pub(crate) enum KnownName {
|
|
|
|
Isize,
|
|
|
|
I8,
|
|
|
|
I16,
|
|
|
|
I32,
|
|
|
|
I64,
|
|
|
|
I128,
|
|
|
|
|
|
|
|
Usize,
|
|
|
|
U8,
|
|
|
|
U16,
|
|
|
|
U32,
|
|
|
|
U64,
|
|
|
|
U128,
|
|
|
|
|
|
|
|
F32,
|
|
|
|
F64,
|
2018-12-29 16:20:12 -06:00
|
|
|
|
2019-01-07 13:43:41 -06:00
|
|
|
Bool,
|
|
|
|
Char,
|
|
|
|
Str,
|
|
|
|
|
2019-01-05 15:37:59 -06:00
|
|
|
SelfType,
|
|
|
|
SelfParam,
|
2019-03-02 14:59:04 -06:00
|
|
|
|
|
|
|
MacroRules,
|
2019-07-07 02:31:09 -05:00
|
|
|
|
|
|
|
Std,
|
|
|
|
Iter,
|
|
|
|
IntoIterator,
|
|
|
|
Item,
|
2018-12-27 11:26:15 -06:00
|
|
|
}
|
2019-05-30 07:03:58 -05:00
|
|
|
|
|
|
|
impl AsName for KnownName {
|
|
|
|
fn as_name(&self) -> Name {
|
|
|
|
let s = match self {
|
|
|
|
KnownName::Isize => "isize",
|
|
|
|
KnownName::I8 => "i8",
|
|
|
|
KnownName::I16 => "i16",
|
|
|
|
KnownName::I32 => "i32",
|
|
|
|
KnownName::I64 => "i64",
|
|
|
|
KnownName::I128 => "i128",
|
|
|
|
KnownName::Usize => "usize",
|
|
|
|
KnownName::U8 => "u8",
|
|
|
|
KnownName::U16 => "u16",
|
|
|
|
KnownName::U32 => "u32",
|
|
|
|
KnownName::U64 => "u64",
|
|
|
|
KnownName::U128 => "u128",
|
|
|
|
KnownName::F32 => "f32",
|
|
|
|
KnownName::F64 => "f64",
|
|
|
|
KnownName::Bool => "bool",
|
|
|
|
KnownName::Char => "char",
|
|
|
|
KnownName::Str => "str",
|
|
|
|
KnownName::SelfType => "Self",
|
|
|
|
KnownName::SelfParam => "self",
|
|
|
|
KnownName::MacroRules => "macro_rules",
|
2019-07-07 02:31:09 -05:00
|
|
|
KnownName::Std => "std",
|
|
|
|
KnownName::Iter => "iter",
|
|
|
|
KnownName::IntoIterator => "IntoIterator",
|
|
|
|
KnownName::Item => "Item",
|
2019-05-30 07:03:58 -05:00
|
|
|
};
|
|
|
|
Name::new(s.into())
|
|
|
|
}
|
|
|
|
}
|
2019-07-03 14:01:41 -05:00
|
|
|
|
2019-07-03 14:17:43 -05:00
|
|
|
fn resolve_name(text: &SmolStr) -> SmolStr {
|
2019-07-03 14:01:41 -05:00
|
|
|
let raw_start = "r#";
|
|
|
|
if text.as_str().starts_with(raw_start) {
|
|
|
|
SmolStr::new(&text[raw_start.len()..])
|
|
|
|
} else {
|
2019-07-03 14:17:43 -05:00
|
|
|
text.clone()
|
2019-07-03 14:01:41 -05:00
|
|
|
}
|
|
|
|
}
|