mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-27 04:15:08 +00:00
scope based comletion
This commit is contained in:
parent
4c121bfa2f
commit
ac226021cf
8 changed files with 475 additions and 41 deletions
|
@ -9,7 +9,7 @@ use libsyntax2::{
|
||||||
SyntaxNodeRef,
|
SyntaxNodeRef,
|
||||||
algo::{
|
algo::{
|
||||||
Direction, siblings,
|
Direction, siblings,
|
||||||
find_leaf_at_offset, ancestors,
|
find_leaf_at_offset,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,11 @@
|
||||||
use libsyntax2::{
|
use libsyntax2::{
|
||||||
File, TextUnit,
|
File, TextUnit, AstNode, SyntaxNodeRef,
|
||||||
ast,
|
ast::{self, NameOwner},
|
||||||
algo::find_leaf_at_offset,
|
algo::{
|
||||||
|
ancestors,
|
||||||
|
visit::{visitor_ctx, VisitorCtx},
|
||||||
|
walk::preorder,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
use {
|
use {
|
||||||
|
@ -25,7 +29,33 @@ pub fn scope_completion(file: &File, offset: TextUnit) -> Option<Vec<CompletionI
|
||||||
}
|
}
|
||||||
|
|
||||||
fn complete(name_ref: ast::NameRef) -> Vec<CompletionItem> {
|
fn complete(name_ref: ast::NameRef) -> Vec<CompletionItem> {
|
||||||
vec![CompletionItem {
|
let mut res = Vec::new();
|
||||||
name: "foo".to_string()
|
for node in ancestors(name_ref.syntax()) {
|
||||||
}]
|
process_scope(node, &mut res);
|
||||||
|
}
|
||||||
|
res
|
||||||
|
}
|
||||||
|
|
||||||
|
fn process_scope(node: SyntaxNodeRef, sink: &mut Vec<CompletionItem>) {
|
||||||
|
let _ = visitor_ctx(sink)
|
||||||
|
.visit::<ast::Block, _>(|block, sink| {
|
||||||
|
block.let_stmts()
|
||||||
|
.filter_map(|it| it.pat())
|
||||||
|
.for_each(move |it| process_pat(it, sink))
|
||||||
|
})
|
||||||
|
.visit::<ast::FnDef, _>(|fn_def, sink| {
|
||||||
|
fn_def.param_list().into_iter()
|
||||||
|
.flat_map(|it| it.params())
|
||||||
|
.filter_map(|it| it.pat())
|
||||||
|
.for_each(move |it| process_pat(it, sink))
|
||||||
|
})
|
||||||
|
.accept(node);
|
||||||
|
|
||||||
|
fn process_pat(pat: ast::Pat, sink: &mut Vec<CompletionItem>) {
|
||||||
|
let items = preorder(pat.syntax())
|
||||||
|
.filter_map(ast::BindPat::cast)
|
||||||
|
.filter_map(ast::BindPat::name)
|
||||||
|
.map(|name| CompletionItem { name: name.text().to_string() });
|
||||||
|
sink.extend(items);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -256,25 +256,23 @@ struct Foo { f: u32 }
|
||||||
");
|
");
|
||||||
}
|
}
|
||||||
|
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_completion() {
|
fn test_completion() {
|
||||||
// fn do_check(code: &str, expected_completions: &str) {
|
fn do_check(code: &str, expected_completions: &str) {
|
||||||
// let (off, code) = extract_offset(&code);
|
let (off, code) = extract_offset(&code);
|
||||||
// let file = file(&code);
|
let file = file(&code);
|
||||||
// let completions = scope_completion(&file, off).unwrap();
|
let completions = scope_completion(&file, off).unwrap();
|
||||||
// assert_eq_dbg(expected_completions, &completions);
|
assert_eq_dbg(expected_completions, &completions);
|
||||||
// }
|
}
|
||||||
|
|
||||||
// do_check(r"
|
do_check(r"
|
||||||
// fn foo(foo: i32) {
|
fn quux(x: i32) {
|
||||||
// let bar = 92;
|
let y = 92;
|
||||||
// 1 + <|>
|
1 + <|>
|
||||||
// }
|
}
|
||||||
// ", r#"
|
", r#"[CompletionItem { name: "y" },
|
||||||
// CompletionItem { name: "bar" },
|
CompletionItem { name: "x" }]"#);
|
||||||
// CompletionItem { name: "foo" },
|
}
|
||||||
// "#);
|
|
||||||
// }
|
|
||||||
|
|
||||||
fn file(text: &str) -> File {
|
fn file(text: &str) -> File {
|
||||||
File::parse(text)
|
File::parse(text)
|
||||||
|
|
|
@ -6,6 +6,10 @@ pub fn visitor<'a, T>() -> impl Visitor<'a, Output=T> {
|
||||||
EmptyVisitor { ph: PhantomData }
|
EmptyVisitor { ph: PhantomData }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn visitor_ctx<'a, T, C>(ctx: C) -> impl VisitorCtx<'a, Output=T, Ctx=C> {
|
||||||
|
EmptyVisitorCtx { ph: PhantomData, ctx }
|
||||||
|
}
|
||||||
|
|
||||||
pub trait Visitor<'a>: Sized {
|
pub trait Visitor<'a>: Sized {
|
||||||
type Output;
|
type Output;
|
||||||
fn accept(self, node: SyntaxNodeRef<'a>) -> Option<Self::Output>;
|
fn accept(self, node: SyntaxNodeRef<'a>) -> Option<Self::Output>;
|
||||||
|
@ -17,6 +21,18 @@ pub trait Visitor<'a>: Sized {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait VisitorCtx<'a>: Sized {
|
||||||
|
type Output;
|
||||||
|
type Ctx;
|
||||||
|
fn accept(self, node: SyntaxNodeRef<'a>) -> Result<Self::Output, Self::Ctx>;
|
||||||
|
fn visit<N, F>(self, f: F) -> VisCtx<Self, N, F>
|
||||||
|
where N: AstNode<'a>,
|
||||||
|
F: FnOnce(N, Self::Ctx) -> Self::Output,
|
||||||
|
{
|
||||||
|
VisCtx { inner: self, f, ph: PhantomData }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct EmptyVisitor<T> {
|
struct EmptyVisitor<T> {
|
||||||
ph: PhantomData<fn() -> T>
|
ph: PhantomData<fn() -> T>
|
||||||
|
@ -30,6 +46,21 @@ impl<'a, T> Visitor<'a> for EmptyVisitor<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct EmptyVisitorCtx<T, C> {
|
||||||
|
ctx: C,
|
||||||
|
ph: PhantomData<fn() -> T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T, C> VisitorCtx<'a> for EmptyVisitorCtx<T, C> {
|
||||||
|
type Output = T;
|
||||||
|
type Ctx = C;
|
||||||
|
|
||||||
|
fn accept(self, _node: SyntaxNodeRef<'a>) -> Result<T, C> {
|
||||||
|
Err(self.ctx)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Vis<V, N, F> {
|
pub struct Vis<V, N, F> {
|
||||||
inner: V,
|
inner: V,
|
||||||
|
@ -50,3 +81,30 @@ impl<'a, V, N, F> Visitor<'a> for Vis<V, N, F>
|
||||||
inner.accept(node).or_else(|| N::cast(node).map(f))
|
inner.accept(node).or_else(|| N::cast(node).map(f))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct VisCtx<V, N, F> {
|
||||||
|
inner: V,
|
||||||
|
f: F,
|
||||||
|
ph: PhantomData<fn(N)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, V, N, F> VisitorCtx<'a> for VisCtx<V, N, F>
|
||||||
|
where
|
||||||
|
V: VisitorCtx<'a>,
|
||||||
|
N: AstNode<'a>,
|
||||||
|
F: FnOnce(N, <V as VisitorCtx<'a>>::Ctx) -> <V as VisitorCtx<'a>>::Output,
|
||||||
|
{
|
||||||
|
type Output = <V as VisitorCtx<'a>>::Output;
|
||||||
|
type Ctx = <V as VisitorCtx<'a>>::Ctx;
|
||||||
|
|
||||||
|
fn accept(self, node: SyntaxNodeRef<'a>) -> Result<Self::Output, Self::Ctx> {
|
||||||
|
let VisCtx { inner, f, .. } = self;
|
||||||
|
inner.accept(node).or_else(|ctx|
|
||||||
|
match N::cast(node) {
|
||||||
|
None => Err(ctx),
|
||||||
|
Some(node) => Ok(f(node, ctx))
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -17,19 +17,19 @@ pub enum WalkEvent<'a> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn walk<'a>(root: SyntaxNodeRef<'a>) -> impl Iterator<Item = WalkEvent<'a>> {
|
pub fn walk<'a>(root: SyntaxNodeRef<'a>) -> impl Iterator<Item = WalkEvent<'a>> {
|
||||||
generate(Some(WalkEvent::Enter(root)), |pos| {
|
generate(Some(WalkEvent::Enter(root)), move |pos| {
|
||||||
let next = match *pos {
|
let next = match *pos {
|
||||||
WalkEvent::Enter(node) => match node.first_child() {
|
WalkEvent::Enter(node) => match node.first_child() {
|
||||||
Some(child) => WalkEvent::Enter(child),
|
Some(child) => WalkEvent::Enter(child),
|
||||||
None => WalkEvent::Exit(node),
|
None => WalkEvent::Exit(node),
|
||||||
},
|
},
|
||||||
WalkEvent::Exit(node) => {
|
WalkEvent::Exit(node) => {
|
||||||
|
if node == root {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
match node.next_sibling() {
|
match node.next_sibling() {
|
||||||
Some(sibling) => WalkEvent::Enter(sibling),
|
Some(sibling) => WalkEvent::Enter(sibling),
|
||||||
None => match node.parent() {
|
None => WalkEvent::Exit(node.parent().unwrap()),
|
||||||
Some(node) => WalkEvent::Exit(node),
|
|
||||||
None => return None,
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -80,6 +80,25 @@ impl<'a> AstNode<'a> for BinExpr<'a> {
|
||||||
|
|
||||||
impl<'a> BinExpr<'a> {}
|
impl<'a> BinExpr<'a> {}
|
||||||
|
|
||||||
|
// BindPat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct BindPat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for BindPat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
BIND_PAT => Some(BindPat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> ast::NameOwner<'a> for BindPat<'a> {}
|
||||||
|
impl<'a> BindPat<'a> {}
|
||||||
|
|
||||||
// Block
|
// Block
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct Block<'a> {
|
pub struct Block<'a> {
|
||||||
|
@ -96,7 +115,11 @@ impl<'a> AstNode<'a> for Block<'a> {
|
||||||
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Block<'a> {}
|
impl<'a> Block<'a> {
|
||||||
|
pub fn let_stmts(self) -> impl Iterator<Item = LetStmt<'a>> + 'a {
|
||||||
|
super::children(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// BlockExpr
|
// BlockExpr
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
@ -378,6 +401,24 @@ impl<'a> AstNode<'a> for FieldExpr<'a> {
|
||||||
|
|
||||||
impl<'a> FieldExpr<'a> {}
|
impl<'a> FieldExpr<'a> {}
|
||||||
|
|
||||||
|
// FieldPatList
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct FieldPatList<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for FieldPatList<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
FIELD_PAT_LIST => Some(FieldPatList { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> FieldPatList<'a> {}
|
||||||
|
|
||||||
// FnDef
|
// FnDef
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct FnDef<'a> {
|
pub struct FnDef<'a> {
|
||||||
|
@ -397,7 +438,11 @@ impl<'a> AstNode<'a> for FnDef<'a> {
|
||||||
impl<'a> ast::NameOwner<'a> for FnDef<'a> {}
|
impl<'a> ast::NameOwner<'a> for FnDef<'a> {}
|
||||||
impl<'a> ast::TypeParamsOwner<'a> for FnDef<'a> {}
|
impl<'a> ast::TypeParamsOwner<'a> for FnDef<'a> {}
|
||||||
impl<'a> ast::AttrsOwner<'a> for FnDef<'a> {}
|
impl<'a> ast::AttrsOwner<'a> for FnDef<'a> {}
|
||||||
impl<'a> FnDef<'a> {}
|
impl<'a> FnDef<'a> {
|
||||||
|
pub fn param_list(self) -> Option<ParamList<'a>> {
|
||||||
|
super::child_opt(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// FnPointerType
|
// FnPointerType
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
@ -561,6 +606,28 @@ impl<'a> AstNode<'a> for LambdaExpr<'a> {
|
||||||
|
|
||||||
impl<'a> LambdaExpr<'a> {}
|
impl<'a> LambdaExpr<'a> {}
|
||||||
|
|
||||||
|
// LetStmt
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct LetStmt<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for LetStmt<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
LET_STMT => Some(LetStmt { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> LetStmt<'a> {
|
||||||
|
pub fn pat(self) -> Option<Pat<'a>> {
|
||||||
|
super::child_opt(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// LoopExpr
|
// LoopExpr
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct LoopExpr<'a> {
|
pub struct LoopExpr<'a> {
|
||||||
|
@ -831,6 +898,50 @@ impl<'a> ast::TypeParamsOwner<'a> for NominalDef<'a> {}
|
||||||
impl<'a> ast::AttrsOwner<'a> for NominalDef<'a> {}
|
impl<'a> ast::AttrsOwner<'a> for NominalDef<'a> {}
|
||||||
impl<'a> NominalDef<'a> {}
|
impl<'a> NominalDef<'a> {}
|
||||||
|
|
||||||
|
// Param
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct Param<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for Param<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
PARAM => Some(Param { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> Param<'a> {
|
||||||
|
pub fn pat(self) -> Option<Pat<'a>> {
|
||||||
|
super::child_opt(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ParamList
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct ParamList<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for ParamList<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
PARAM_LIST => Some(ParamList { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> ParamList<'a> {
|
||||||
|
pub fn params(self) -> impl Iterator<Item = Param<'a>> + 'a {
|
||||||
|
super::children(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// ParenExpr
|
// ParenExpr
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct ParenExpr<'a> {
|
pub struct ParenExpr<'a> {
|
||||||
|
@ -867,6 +978,55 @@ impl<'a> AstNode<'a> for ParenType<'a> {
|
||||||
|
|
||||||
impl<'a> ParenType<'a> {}
|
impl<'a> ParenType<'a> {}
|
||||||
|
|
||||||
|
// Pat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub enum Pat<'a> {
|
||||||
|
RefPat(RefPat<'a>),
|
||||||
|
BindPat(BindPat<'a>),
|
||||||
|
PlaceholderPat(PlaceholderPat<'a>),
|
||||||
|
PathPat(PathPat<'a>),
|
||||||
|
StructPat(StructPat<'a>),
|
||||||
|
FieldPatList(FieldPatList<'a>),
|
||||||
|
TupleStructPat(TupleStructPat<'a>),
|
||||||
|
TuplePat(TuplePat<'a>),
|
||||||
|
SlicePat(SlicePat<'a>),
|
||||||
|
RangePat(RangePat<'a>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for Pat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
REF_PAT => Some(Pat::RefPat(RefPat { syntax })),
|
||||||
|
BIND_PAT => Some(Pat::BindPat(BindPat { syntax })),
|
||||||
|
PLACEHOLDER_PAT => Some(Pat::PlaceholderPat(PlaceholderPat { syntax })),
|
||||||
|
PATH_PAT => Some(Pat::PathPat(PathPat { syntax })),
|
||||||
|
STRUCT_PAT => Some(Pat::StructPat(StructPat { syntax })),
|
||||||
|
FIELD_PAT_LIST => Some(Pat::FieldPatList(FieldPatList { syntax })),
|
||||||
|
TUPLE_STRUCT_PAT => Some(Pat::TupleStructPat(TupleStructPat { syntax })),
|
||||||
|
TUPLE_PAT => Some(Pat::TuplePat(TuplePat { syntax })),
|
||||||
|
SLICE_PAT => Some(Pat::SlicePat(SlicePat { syntax })),
|
||||||
|
RANGE_PAT => Some(Pat::RangePat(RangePat { syntax })),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> {
|
||||||
|
match self {
|
||||||
|
Pat::RefPat(inner) => inner.syntax(),
|
||||||
|
Pat::BindPat(inner) => inner.syntax(),
|
||||||
|
Pat::PlaceholderPat(inner) => inner.syntax(),
|
||||||
|
Pat::PathPat(inner) => inner.syntax(),
|
||||||
|
Pat::StructPat(inner) => inner.syntax(),
|
||||||
|
Pat::FieldPatList(inner) => inner.syntax(),
|
||||||
|
Pat::TupleStructPat(inner) => inner.syntax(),
|
||||||
|
Pat::TuplePat(inner) => inner.syntax(),
|
||||||
|
Pat::SlicePat(inner) => inner.syntax(),
|
||||||
|
Pat::RangePat(inner) => inner.syntax(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> Pat<'a> {}
|
||||||
|
|
||||||
// PathExpr
|
// PathExpr
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct PathExpr<'a> {
|
pub struct PathExpr<'a> {
|
||||||
|
@ -885,6 +1045,24 @@ impl<'a> AstNode<'a> for PathExpr<'a> {
|
||||||
|
|
||||||
impl<'a> PathExpr<'a> {}
|
impl<'a> PathExpr<'a> {}
|
||||||
|
|
||||||
|
// PathPat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct PathPat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for PathPat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
PATH_PAT => Some(PathPat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> PathPat<'a> {}
|
||||||
|
|
||||||
// PathType
|
// PathType
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct PathType<'a> {
|
pub struct PathType<'a> {
|
||||||
|
@ -903,6 +1081,24 @@ impl<'a> AstNode<'a> for PathType<'a> {
|
||||||
|
|
||||||
impl<'a> PathType<'a> {}
|
impl<'a> PathType<'a> {}
|
||||||
|
|
||||||
|
// PlaceholderPat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct PlaceholderPat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for PlaceholderPat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
PLACEHOLDER_PAT => Some(PlaceholderPat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> PlaceholderPat<'a> {}
|
||||||
|
|
||||||
// PlaceholderType
|
// PlaceholderType
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct PlaceholderType<'a> {
|
pub struct PlaceholderType<'a> {
|
||||||
|
@ -975,6 +1171,24 @@ impl<'a> AstNode<'a> for RangeExpr<'a> {
|
||||||
|
|
||||||
impl<'a> RangeExpr<'a> {}
|
impl<'a> RangeExpr<'a> {}
|
||||||
|
|
||||||
|
// RangePat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct RangePat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for RangePat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
RANGE_PAT => Some(RangePat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> RangePat<'a> {}
|
||||||
|
|
||||||
// RefExpr
|
// RefExpr
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct RefExpr<'a> {
|
pub struct RefExpr<'a> {
|
||||||
|
@ -993,6 +1207,24 @@ impl<'a> AstNode<'a> for RefExpr<'a> {
|
||||||
|
|
||||||
impl<'a> RefExpr<'a> {}
|
impl<'a> RefExpr<'a> {}
|
||||||
|
|
||||||
|
// RefPat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct RefPat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for RefPat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
REF_PAT => Some(RefPat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> RefPat<'a> {}
|
||||||
|
|
||||||
// ReferenceType
|
// ReferenceType
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct ReferenceType<'a> {
|
pub struct ReferenceType<'a> {
|
||||||
|
@ -1055,6 +1287,24 @@ impl<'a> Root<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SlicePat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct SlicePat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for SlicePat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
SLICE_PAT => Some(SlicePat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> SlicePat<'a> {}
|
||||||
|
|
||||||
// SliceType
|
// SliceType
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct SliceType<'a> {
|
pub struct SliceType<'a> {
|
||||||
|
@ -1137,6 +1387,24 @@ impl<'a> AstNode<'a> for StructLit<'a> {
|
||||||
|
|
||||||
impl<'a> StructLit<'a> {}
|
impl<'a> StructLit<'a> {}
|
||||||
|
|
||||||
|
// StructPat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct StructPat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for StructPat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
STRUCT_PAT => Some(StructPat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> StructPat<'a> {}
|
||||||
|
|
||||||
// TokenTree
|
// TokenTree
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct TokenTree<'a> {
|
pub struct TokenTree<'a> {
|
||||||
|
@ -1211,6 +1479,42 @@ impl<'a> AstNode<'a> for TupleExpr<'a> {
|
||||||
|
|
||||||
impl<'a> TupleExpr<'a> {}
|
impl<'a> TupleExpr<'a> {}
|
||||||
|
|
||||||
|
// TuplePat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct TuplePat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for TuplePat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
TUPLE_PAT => Some(TuplePat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> TuplePat<'a> {}
|
||||||
|
|
||||||
|
// TupleStructPat
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub struct TupleStructPat<'a> {
|
||||||
|
syntax: SyntaxNodeRef<'a>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AstNode<'a> for TupleStructPat<'a> {
|
||||||
|
fn cast(syntax: SyntaxNodeRef<'a>) -> Option<Self> {
|
||||||
|
match syntax.kind() {
|
||||||
|
TUPLE_STRUCT_PAT => Some(TupleStructPat { syntax }),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> TupleStructPat<'a> {}
|
||||||
|
|
||||||
// TupleType
|
// TupleType
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct TupleType<'a> {
|
pub struct TupleType<'a> {
|
||||||
|
|
|
@ -241,11 +241,16 @@ Grammar(
|
||||||
["modules", "Module"],
|
["modules", "Module"],
|
||||||
]
|
]
|
||||||
),
|
),
|
||||||
"FnDef": ( traits: [
|
"FnDef": (
|
||||||
"NameOwner",
|
traits: [
|
||||||
"TypeParamsOwner",
|
"NameOwner",
|
||||||
"AttrsOwner",
|
"TypeParamsOwner",
|
||||||
] ),
|
"AttrsOwner",
|
||||||
|
],
|
||||||
|
options: [
|
||||||
|
["param_list", "ParamList"]
|
||||||
|
],
|
||||||
|
),
|
||||||
"StructDef": (
|
"StructDef": (
|
||||||
traits: [
|
traits: [
|
||||||
"NameOwner",
|
"NameOwner",
|
||||||
|
@ -393,13 +398,52 @@ Grammar(
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|
||||||
|
"RefPat": (),
|
||||||
|
"BindPat": ( traits: ["NameOwner"] ),
|
||||||
|
"PlaceholderPat": (),
|
||||||
|
"PathPat": (),
|
||||||
|
"StructPat": (),
|
||||||
|
"FieldPatList": (),
|
||||||
|
"TupleStructPat": (),
|
||||||
|
"TuplePat": (),
|
||||||
|
"SlicePat": (),
|
||||||
|
"RangePat": (),
|
||||||
|
|
||||||
|
"Pat": (
|
||||||
|
enum: [
|
||||||
|
"RefPat",
|
||||||
|
"BindPat",
|
||||||
|
"PlaceholderPat",
|
||||||
|
"PathPat",
|
||||||
|
"StructPat",
|
||||||
|
"FieldPatList",
|
||||||
|
"TupleStructPat",
|
||||||
|
"TuplePat",
|
||||||
|
"SlicePat",
|
||||||
|
"RangePat",
|
||||||
|
],
|
||||||
|
),
|
||||||
|
|
||||||
"Name": (),
|
"Name": (),
|
||||||
"NameRef": (),
|
"NameRef": (),
|
||||||
"Attr": ( options: [ ["value", "TokenTree"] ] ),
|
"Attr": ( options: [ ["value", "TokenTree"] ] ),
|
||||||
"TokenTree": (),
|
"TokenTree": (),
|
||||||
"TypeParamList": ( collections: [ ["type_params", "TypeParam" ] ]),
|
"TypeParamList": ( collections: [ ["type_params", "TypeParam" ] ]),
|
||||||
"TypeParam": ( traits: ["NameOwner"]),
|
"TypeParam": ( traits: ["NameOwner"] ),
|
||||||
"WhereClause": (),
|
"WhereClause": (),
|
||||||
"Block": (),
|
"LetStmt": ( options: [ ["pat", "Pat"] ]),
|
||||||
|
"Block": (
|
||||||
|
collections: [
|
||||||
|
["let_stmts", "LetStmt"],
|
||||||
|
]
|
||||||
|
),
|
||||||
|
"ParamList": (
|
||||||
|
collections: [
|
||||||
|
["params", "Param"]
|
||||||
|
]
|
||||||
|
),
|
||||||
|
"Param": (
|
||||||
|
options: [["pat", "Pat"]],
|
||||||
|
)
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
|
@ -115,7 +115,7 @@ pub(super) fn maybe_item(p: &mut Parser, flavor: ItemFlavor) -> MaybeItem {
|
||||||
// test unsafe_fn
|
// test unsafe_fn
|
||||||
// unsafe fn foo() {}
|
// unsafe fn foo() {}
|
||||||
FN_KW => {
|
FN_KW => {
|
||||||
function(p, flavor);
|
fn_def(p, flavor);
|
||||||
FN_DEF
|
FN_DEF
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -227,7 +227,7 @@ fn extern_item_list(p: &mut Parser) {
|
||||||
m.complete(p, EXTERN_ITEM_LIST);
|
m.complete(p, EXTERN_ITEM_LIST);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn function(p: &mut Parser, flavor: ItemFlavor) {
|
fn fn_def(p: &mut Parser, flavor: ItemFlavor) {
|
||||||
assert!(p.at(FN_KW));
|
assert!(p.at(FN_KW));
|
||||||
p.bump();
|
p.bump();
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue