mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-14 06:03:58 +00:00
Merge #8932
8932: internal: even prettier itemtrees r=jonas-schievink a=jonas-schievink Extends the ItemTree pretty printer to handle all `Path`s, and to print generic parameters and where-clauses. bors r+ Co-authored-by: Jonas Schievink <jonasschievink@gmail.com>
This commit is contained in:
commit
e4722b6492
2 changed files with 343 additions and 35 deletions
|
@ -2,7 +2,12 @@
|
||||||
|
|
||||||
use std::fmt::{self, Write};
|
use std::fmt::{self, Write};
|
||||||
|
|
||||||
use crate::{attr::RawAttrs, visibility::RawVisibility};
|
use crate::{
|
||||||
|
attr::RawAttrs,
|
||||||
|
generics::{WherePredicate, WherePredicateTypeTarget},
|
||||||
|
path::GenericArg,
|
||||||
|
visibility::RawVisibility,
|
||||||
|
};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
|
@ -70,6 +75,13 @@ impl<'a> Printer<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn whitespace(&mut self) {
|
||||||
|
match self.buf.chars().next_back() {
|
||||||
|
None | Some('\n') | Some(' ') => {}
|
||||||
|
_ => self.buf.push(' '),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn print_attrs(&mut self, attrs: &RawAttrs, inner: bool) {
|
fn print_attrs(&mut self, attrs: &RawAttrs, inner: bool) {
|
||||||
let inner = if inner { "!" } else { "" };
|
let inner = if inner { "!" } else { "" };
|
||||||
for attr in &**attrs {
|
for attr in &**attrs {
|
||||||
|
@ -100,7 +112,8 @@ impl<'a> Printer<'a> {
|
||||||
fn print_fields(&mut self, fields: &Fields) {
|
fn print_fields(&mut self, fields: &Fields) {
|
||||||
match fields {
|
match fields {
|
||||||
Fields::Record(fields) => {
|
Fields::Record(fields) => {
|
||||||
w!(self, " {{");
|
self.whitespace();
|
||||||
|
w!(self, "{{");
|
||||||
self.indented(|this| {
|
self.indented(|this| {
|
||||||
for field in fields.clone() {
|
for field in fields.clone() {
|
||||||
let Field { visibility, name, type_ref } = &this.tree[field];
|
let Field { visibility, name, type_ref } = &this.tree[field];
|
||||||
|
@ -131,6 +144,25 @@ impl<'a> Printer<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn print_fields_and_where_clause(&mut self, fields: &Fields, params: &GenericParams) {
|
||||||
|
match fields {
|
||||||
|
Fields::Record(_) => {
|
||||||
|
if self.print_where_clause(params) {
|
||||||
|
wln!(self);
|
||||||
|
}
|
||||||
|
self.print_fields(fields);
|
||||||
|
}
|
||||||
|
Fields::Unit => {
|
||||||
|
self.print_where_clause(params);
|
||||||
|
self.print_fields(fields);
|
||||||
|
}
|
||||||
|
Fields::Tuple(_) => {
|
||||||
|
self.print_fields(fields);
|
||||||
|
self.print_where_clause(params);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn print_mod_item(&mut self, item: ModItem) {
|
fn print_mod_item(&mut self, item: ModItem) {
|
||||||
self.print_attrs_of(item);
|
self.print_attrs_of(item);
|
||||||
|
|
||||||
|
@ -174,7 +206,7 @@ impl<'a> Printer<'a> {
|
||||||
let Function {
|
let Function {
|
||||||
name,
|
name,
|
||||||
visibility,
|
visibility,
|
||||||
generic_params: _, // FIXME print these somehow
|
generic_params,
|
||||||
abi,
|
abi,
|
||||||
params,
|
params,
|
||||||
ret_type,
|
ret_type,
|
||||||
|
@ -188,7 +220,9 @@ 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);
|
||||||
|
self.print_generic_params(generic_params);
|
||||||
|
w!(self, "(");
|
||||||
if !params.is_empty() {
|
if !params.is_empty() {
|
||||||
self.indented(|this| {
|
self.indented(|this| {
|
||||||
for param in params.clone() {
|
for param in params.clone() {
|
||||||
|
@ -208,14 +242,15 @@ impl<'a> Printer<'a> {
|
||||||
}
|
}
|
||||||
w!(self, ") -> ");
|
w!(self, ") -> ");
|
||||||
self.print_type_ref(ret_type);
|
self.print_type_ref(ret_type);
|
||||||
|
self.print_where_clause(generic_params);
|
||||||
wln!(self, ";");
|
wln!(self, ";");
|
||||||
}
|
}
|
||||||
ModItem::Struct(it) => {
|
ModItem::Struct(it) => {
|
||||||
let Struct { visibility, name, fields, generic_params: _, ast_id: _ } =
|
let Struct { visibility, name, fields, generic_params, ast_id: _ } = &self.tree[it];
|
||||||
&self.tree[it];
|
|
||||||
self.print_visibility(*visibility);
|
self.print_visibility(*visibility);
|
||||||
w!(self, "struct {}", name);
|
w!(self, "struct {}", name);
|
||||||
self.print_fields(fields);
|
self.print_generic_params(generic_params);
|
||||||
|
self.print_fields_and_where_clause(fields, generic_params);
|
||||||
if matches!(fields, Fields::Record(_)) {
|
if matches!(fields, Fields::Record(_)) {
|
||||||
wln!(self);
|
wln!(self);
|
||||||
} else {
|
} else {
|
||||||
|
@ -223,11 +258,11 @@ impl<'a> Printer<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ModItem::Union(it) => {
|
ModItem::Union(it) => {
|
||||||
let Union { name, visibility, fields, generic_params: _, ast_id: _ } =
|
let Union { name, visibility, fields, generic_params, ast_id: _ } = &self.tree[it];
|
||||||
&self.tree[it];
|
|
||||||
self.print_visibility(*visibility);
|
self.print_visibility(*visibility);
|
||||||
w!(self, "union {}", name);
|
w!(self, "union {}", name);
|
||||||
self.print_fields(fields);
|
self.print_generic_params(generic_params);
|
||||||
|
self.print_fields_and_where_clause(fields, generic_params);
|
||||||
if matches!(fields, Fields::Record(_)) {
|
if matches!(fields, Fields::Record(_)) {
|
||||||
wln!(self);
|
wln!(self);
|
||||||
} else {
|
} else {
|
||||||
|
@ -235,10 +270,11 @@ impl<'a> Printer<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ModItem::Enum(it) => {
|
ModItem::Enum(it) => {
|
||||||
let Enum { name, visibility, variants, generic_params: _, ast_id: _ } =
|
let Enum { name, visibility, variants, generic_params, ast_id: _ } = &self.tree[it];
|
||||||
&self.tree[it];
|
|
||||||
self.print_visibility(*visibility);
|
self.print_visibility(*visibility);
|
||||||
w!(self, "enum {} {{", name);
|
w!(self, "enum {}", name);
|
||||||
|
self.print_generic_params(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 } = &this.tree[variant];
|
let Variant { name, fields } = &this.tree[variant];
|
||||||
|
@ -286,7 +322,7 @@ impl<'a> Printer<'a> {
|
||||||
is_unsafe,
|
is_unsafe,
|
||||||
bounds,
|
bounds,
|
||||||
items,
|
items,
|
||||||
generic_params: _,
|
generic_params,
|
||||||
ast_id: _,
|
ast_id: _,
|
||||||
} = &self.tree[it];
|
} = &self.tree[it];
|
||||||
self.print_visibility(*visibility);
|
self.print_visibility(*visibility);
|
||||||
|
@ -297,11 +333,12 @@ impl<'a> Printer<'a> {
|
||||||
w!(self, "auto ");
|
w!(self, "auto ");
|
||||||
}
|
}
|
||||||
w!(self, "trait {}", name);
|
w!(self, "trait {}", name);
|
||||||
|
self.print_generic_params(generic_params);
|
||||||
if !bounds.is_empty() {
|
if !bounds.is_empty() {
|
||||||
w!(self, ": ");
|
w!(self, ": ");
|
||||||
self.print_type_bounds(bounds);
|
self.print_type_bounds(bounds);
|
||||||
}
|
}
|
||||||
w!(self, " {{");
|
self.print_where_clause_and_opening_brace(generic_params);
|
||||||
self.indented(|this| {
|
self.indented(|this| {
|
||||||
for item in &**items {
|
for item in &**items {
|
||||||
this.print_mod_item((*item).into());
|
this.print_mod_item((*item).into());
|
||||||
|
@ -310,15 +347,11 @@ impl<'a> Printer<'a> {
|
||||||
wln!(self, "}}");
|
wln!(self, "}}");
|
||||||
}
|
}
|
||||||
ModItem::Impl(it) => {
|
ModItem::Impl(it) => {
|
||||||
let Impl {
|
let Impl { target_trait, self_ty, is_negative, items, generic_params, ast_id: _ } =
|
||||||
target_trait,
|
&self.tree[it];
|
||||||
self_ty,
|
w!(self, "impl");
|
||||||
is_negative,
|
self.print_generic_params(generic_params);
|
||||||
items,
|
w!(self, " ");
|
||||||
generic_params: _,
|
|
||||||
ast_id: _,
|
|
||||||
} = &self.tree[it];
|
|
||||||
w!(self, "impl ");
|
|
||||||
if *is_negative {
|
if *is_negative {
|
||||||
w!(self, "!");
|
w!(self, "!");
|
||||||
}
|
}
|
||||||
|
@ -327,7 +360,7 @@ impl<'a> Printer<'a> {
|
||||||
w!(self, " for ");
|
w!(self, " for ");
|
||||||
}
|
}
|
||||||
self.print_type_ref(self_ty);
|
self.print_type_ref(self_ty);
|
||||||
w!(self, " {{");
|
self.print_where_clause_and_opening_brace(generic_params);
|
||||||
self.indented(|this| {
|
self.indented(|this| {
|
||||||
for item in &**items {
|
for item in &**items {
|
||||||
this.print_mod_item((*item).into());
|
this.print_mod_item((*item).into());
|
||||||
|
@ -342,11 +375,12 @@ impl<'a> Printer<'a> {
|
||||||
bounds,
|
bounds,
|
||||||
type_ref,
|
type_ref,
|
||||||
is_extern,
|
is_extern,
|
||||||
generic_params: _,
|
generic_params,
|
||||||
ast_id: _,
|
ast_id: _,
|
||||||
} = &self.tree[it];
|
} = &self.tree[it];
|
||||||
self.print_visibility(*visibility);
|
self.print_visibility(*visibility);
|
||||||
w!(self, "type {}", name);
|
w!(self, "type {}", name);
|
||||||
|
self.print_generic_params(generic_params);
|
||||||
if !bounds.is_empty() {
|
if !bounds.is_empty() {
|
||||||
w!(self, ": ");
|
w!(self, ": ");
|
||||||
self.print_type_bounds(bounds);
|
self.print_type_bounds(bounds);
|
||||||
|
@ -355,6 +389,7 @@ impl<'a> Printer<'a> {
|
||||||
w!(self, " = ");
|
w!(self, " = ");
|
||||||
self.print_type_ref(ty);
|
self.print_type_ref(ty);
|
||||||
}
|
}
|
||||||
|
self.print_where_clause(generic_params);
|
||||||
w!(self, ";");
|
w!(self, ";");
|
||||||
if *is_extern {
|
if *is_extern {
|
||||||
w!(self, " // extern");
|
w!(self, " // extern");
|
||||||
|
@ -466,7 +501,7 @@ impl<'a> Printer<'a> {
|
||||||
TypeRef::Macro(_ast_id) => {
|
TypeRef::Macro(_ast_id) => {
|
||||||
w!(self, "<macro>");
|
w!(self, "<macro>");
|
||||||
}
|
}
|
||||||
TypeRef::Error => drop(write!(self, "{{unknown}}")),
|
TypeRef::Error => w!(self, "{{unknown}}"),
|
||||||
TypeRef::ImplTrait(bounds) => {
|
TypeRef::ImplTrait(bounds) => {
|
||||||
w!(self, "impl ");
|
w!(self, "impl ");
|
||||||
self.print_type_bounds(bounds);
|
self.print_type_bounds(bounds);
|
||||||
|
@ -493,14 +528,167 @@ impl<'a> Printer<'a> {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn print_path(&mut self, path: &Path) {
|
fn print_path(&mut self, path: &Path) {
|
||||||
if path.type_anchor().is_none()
|
match path.type_anchor() {
|
||||||
&& path.segments().iter().all(|seg| seg.args_and_bindings.is_none())
|
Some(anchor) => {
|
||||||
{
|
w!(self, "<");
|
||||||
w!(self, "{}", path.mod_path());
|
self.print_type_ref(anchor);
|
||||||
} else {
|
w!(self, ">::");
|
||||||
// too complicated, just use `Debug`
|
}
|
||||||
w!(self, "{:?}", path);
|
None => match path.kind() {
|
||||||
|
PathKind::Plain => {}
|
||||||
|
PathKind::Super(0) => w!(self, "self::"),
|
||||||
|
PathKind::Super(n) => {
|
||||||
|
for _ in 0..*n {
|
||||||
|
w!(self, "super::");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
PathKind::Crate => w!(self, "crate::"),
|
||||||
|
PathKind::Abs => w!(self, "::"),
|
||||||
|
PathKind::DollarCrate(_) => w!(self, "$crate::"),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (i, segment) in path.segments().iter().enumerate() {
|
||||||
|
if i != 0 {
|
||||||
|
w!(self, "::");
|
||||||
|
}
|
||||||
|
|
||||||
|
w!(self, "{}", segment.name);
|
||||||
|
if let Some(generics) = segment.args_and_bindings {
|
||||||
|
// NB: these are all in type position, so `::<` turbofish syntax is not necessary
|
||||||
|
w!(self, "<");
|
||||||
|
let mut first = true;
|
||||||
|
let args = if generics.has_self_type {
|
||||||
|
let (self_ty, args) = generics.args.split_first().unwrap();
|
||||||
|
w!(self, "Self=");
|
||||||
|
self.print_generic_arg(self_ty);
|
||||||
|
first = false;
|
||||||
|
args
|
||||||
|
} else {
|
||||||
|
&generics.args
|
||||||
|
};
|
||||||
|
for arg in args {
|
||||||
|
if !first {
|
||||||
|
w!(self, ", ");
|
||||||
|
}
|
||||||
|
first = false;
|
||||||
|
self.print_generic_arg(arg);
|
||||||
|
}
|
||||||
|
for binding in &generics.bindings {
|
||||||
|
if !first {
|
||||||
|
w!(self, ", ");
|
||||||
|
}
|
||||||
|
first = false;
|
||||||
|
w!(self, "{}", binding.name);
|
||||||
|
if !binding.bounds.is_empty() {
|
||||||
|
w!(self, ": ");
|
||||||
|
self.print_type_bounds(&binding.bounds);
|
||||||
|
}
|
||||||
|
if let Some(ty) = &binding.type_ref {
|
||||||
|
w!(self, " = ");
|
||||||
|
self.print_type_ref(ty);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
w!(self, ">");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn print_generic_arg(&mut self, arg: &GenericArg) {
|
||||||
|
match arg {
|
||||||
|
GenericArg::Type(ty) => self.print_type_ref(ty),
|
||||||
|
GenericArg::Lifetime(lt) => w!(self, "{}", lt.name),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn print_generic_params(&mut self, params: &GenericParams) {
|
||||||
|
if params.types.is_empty() && params.lifetimes.is_empty() && params.consts.is_empty() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
w!(self, "<");
|
||||||
|
let mut first = true;
|
||||||
|
for (_, lt) in params.lifetimes.iter() {
|
||||||
|
if !first {
|
||||||
|
w!(self, ", ");
|
||||||
|
}
|
||||||
|
first = false;
|
||||||
|
w!(self, "{}", lt.name);
|
||||||
|
}
|
||||||
|
for (idx, ty) in params.types.iter() {
|
||||||
|
if !first {
|
||||||
|
w!(self, ", ");
|
||||||
|
}
|
||||||
|
first = false;
|
||||||
|
match &ty.name {
|
||||||
|
Some(name) => w!(self, "{}", name),
|
||||||
|
None => w!(self, "_anon_{}", idx.into_raw()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (_, konst) in params.consts.iter() {
|
||||||
|
if !first {
|
||||||
|
w!(self, ", ");
|
||||||
|
}
|
||||||
|
first = false;
|
||||||
|
w!(self, "const {}: ", konst.name);
|
||||||
|
self.print_type_ref(&konst.ty);
|
||||||
|
}
|
||||||
|
w!(self, ">");
|
||||||
|
}
|
||||||
|
|
||||||
|
fn print_where_clause_and_opening_brace(&mut self, params: &GenericParams) {
|
||||||
|
if self.print_where_clause(params) {
|
||||||
|
w!(self, "\n{{");
|
||||||
|
} else {
|
||||||
|
self.whitespace();
|
||||||
|
w!(self, "{{");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn print_where_clause(&mut self, params: &GenericParams) -> bool {
|
||||||
|
if params.where_predicates.is_empty() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
w!(self, "\nwhere");
|
||||||
|
self.indented(|this| {
|
||||||
|
for (i, pred) in params.where_predicates.iter().enumerate() {
|
||||||
|
if i != 0 {
|
||||||
|
wln!(this, ",");
|
||||||
|
}
|
||||||
|
|
||||||
|
let (target, bound) = match pred {
|
||||||
|
WherePredicate::TypeBound { target, bound } => (target, bound),
|
||||||
|
WherePredicate::Lifetime { target, bound } => {
|
||||||
|
wln!(this, "{}: {},", target.name, bound.name);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
WherePredicate::ForLifetime { lifetimes, target, bound } => {
|
||||||
|
w!(this, "for<");
|
||||||
|
for (i, lt) in lifetimes.iter().enumerate() {
|
||||||
|
if i != 0 {
|
||||||
|
w!(this, ", ");
|
||||||
|
}
|
||||||
|
w!(this, "{}", lt);
|
||||||
|
}
|
||||||
|
w!(this, "> ");
|
||||||
|
(target, bound)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
match target {
|
||||||
|
WherePredicateTypeTarget::TypeRef(ty) => this.print_type_ref(ty),
|
||||||
|
WherePredicateTypeTarget::TypeParam(id) => match ¶ms.types[*id].name {
|
||||||
|
Some(name) => w!(this, "{}", name),
|
||||||
|
None => w!(this, "_anon_{}", id.into_raw()),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
w!(this, ": ");
|
||||||
|
this.print_type_bounds(std::slice::from_ref(bound));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -183,7 +183,11 @@ trait Tr: SuperTrait + 'lifetime {
|
||||||
_: (),
|
_: (),
|
||||||
) -> ();
|
) -> ();
|
||||||
|
|
||||||
pub(self) trait Tr: SuperTrait + 'lifetime {
|
pub(self) trait Tr<Self>: SuperTrait + 'lifetime
|
||||||
|
where
|
||||||
|
Self: SuperTrait,
|
||||||
|
Self: 'lifetime
|
||||||
|
{
|
||||||
pub(self) type Assoc: AssocBound = Default;
|
pub(self) type Assoc: AssocBound = Default;
|
||||||
|
|
||||||
// flags = 0x1
|
// flags = 0x1
|
||||||
|
@ -207,6 +211,8 @@ mod inline {
|
||||||
|
|
||||||
fn fn_in_module() {}
|
fn fn_in_module() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mod outline;
|
||||||
"#,
|
"#,
|
||||||
expect![[r##"
|
expect![[r##"
|
||||||
#[doc = " outer"] // AttrId { is_doc_comment: true, ast_index: 0 }
|
#[doc = " outer"] // AttrId { is_doc_comment: true, ast_index: 0 }
|
||||||
|
@ -217,6 +223,8 @@ mod inline {
|
||||||
// flags = 0x2
|
// flags = 0x2
|
||||||
pub(self) fn fn_in_module() -> ();
|
pub(self) fn fn_in_module() -> ();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(self) mod outline;
|
||||||
"##]],
|
"##]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -242,3 +250,115 @@ m!();
|
||||||
"#]],
|
"#]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn mod_paths() {
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S {
|
||||||
|
a: self::Ty,
|
||||||
|
b: super::SuperTy,
|
||||||
|
c: super::super::SuperSuperTy,
|
||||||
|
d: ::abs::Path,
|
||||||
|
e: crate::Crate,
|
||||||
|
f: plain::path::Ty,
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
pub(self) struct S {
|
||||||
|
pub(self) a: self::Ty,
|
||||||
|
pub(self) b: super::SuperTy,
|
||||||
|
pub(self) c: super::super::SuperSuperTy,
|
||||||
|
pub(self) d: ::abs::Path,
|
||||||
|
pub(self) e: crate::Crate,
|
||||||
|
pub(self) f: plain::path::Ty,
|
||||||
|
}
|
||||||
|
"#]],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn types() {
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S {
|
||||||
|
a: Mixed<'a, T, Item=(), OtherItem=u8>,
|
||||||
|
b: <Fully as Qualified>::Syntax,
|
||||||
|
c: <TypeAnchored>::Path::<'a>,
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
pub(self) struct S {
|
||||||
|
pub(self) a: Mixed<'a, T, Item = (), OtherItem = u8>,
|
||||||
|
pub(self) b: Qualified<Self=Fully>::Syntax,
|
||||||
|
pub(self) c: <TypeAnchored>::Path<'a>,
|
||||||
|
}
|
||||||
|
"#]],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn generics() {
|
||||||
|
check(
|
||||||
|
r#"
|
||||||
|
struct S<'a, 'b: 'a, T: Copy + 'a + 'b, const K: u8 = 0> {
|
||||||
|
field: &'a &'b T,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Tuple<T: Copy>(T);
|
||||||
|
|
||||||
|
impl<'a, 'b: 'a, T: Copy + 'a + 'b, const K: u8 = 0> S<'a, 'b, T, K> {
|
||||||
|
fn f<G: 'a>(arg: impl Copy) -> impl Copy {}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum Enum<'a, T, const U: u8> {}
|
||||||
|
union Union<'a, T, const U: u8> {}
|
||||||
|
|
||||||
|
trait Tr<'a, T: 'a>: Super {}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
pub(self) struct S<'a, 'b, T, const K: u8>
|
||||||
|
where
|
||||||
|
T: Copy,
|
||||||
|
T: 'a,
|
||||||
|
T: 'b
|
||||||
|
{
|
||||||
|
pub(self) field: &'a &'b T,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(self) struct Tuple<T>(
|
||||||
|
pub(self) 0: T,
|
||||||
|
)
|
||||||
|
where
|
||||||
|
T: Copy;
|
||||||
|
|
||||||
|
impl<'a, 'b, T, const K: u8> S<'a, 'b, T, K>
|
||||||
|
where
|
||||||
|
T: Copy,
|
||||||
|
T: 'a,
|
||||||
|
T: 'b
|
||||||
|
{
|
||||||
|
// flags = 0x2
|
||||||
|
pub(self) fn f<G, _anon_1>(
|
||||||
|
_: impl Copy,
|
||||||
|
) -> impl Copy
|
||||||
|
where
|
||||||
|
G: 'a,
|
||||||
|
_anon_1: Copy;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(self) enum Enum<'a, T, const U: u8> {
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(self) union Union<'a, T, const U: u8> {
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(self) trait Tr<'a, Self, T>: Super
|
||||||
|
where
|
||||||
|
Self: Super,
|
||||||
|
T: 'a
|
||||||
|
{
|
||||||
|
}
|
||||||
|
"#]],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue