mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-13 05:38:46 +00:00
Add derive handles cursor
This commit is contained in:
parent
7094291573
commit
a5515d9d6f
5 changed files with 51 additions and 20 deletions
|
@ -1,6 +1,6 @@
|
|||
use {TextUnit, File, EditBuilder, Edit};
|
||||
use libsyntax2::{
|
||||
ast::{self, AstNode},
|
||||
ast::{self, AstNode, AttrsOwner},
|
||||
SyntaxKind::COMMA,
|
||||
SyntaxNodeRef,
|
||||
SyntaxRoot,
|
||||
|
@ -39,18 +39,28 @@ pub fn flip_comma<'a>(file: &'a File, offset: TextUnit) -> Option<impl FnOnce()
|
|||
}
|
||||
|
||||
pub fn add_derive<'a>(file: &'a File, offset: TextUnit) -> Option<impl FnOnce() -> ActionResult + 'a> {
|
||||
let syntax = file.syntax();
|
||||
let syntax = syntax.as_ref();
|
||||
let nominal = find_node::<ast::NominalDef<_>>(syntax, offset)?;
|
||||
let nominal = find_node::<ast::NominalDef<_>>(file.syntax_ref(), offset)?;
|
||||
Some(move || {
|
||||
let derive_attr = nominal
|
||||
.attrs()
|
||||
.filter_map(|x| x.as_call())
|
||||
.filter(|(name, _arg)| name == "derive")
|
||||
.map(|(_name, arg)| arg)
|
||||
.next();
|
||||
let mut edit = EditBuilder::new();
|
||||
let offset = match derive_attr {
|
||||
None => {
|
||||
let node_start = nominal.syntax().range().start();
|
||||
edit.insert(node_start, "#[derive()]\n".to_string());
|
||||
node_start + TextUnit::of_str("#[derive(")
|
||||
}
|
||||
Some(tt) => {
|
||||
tt.syntax().range().end() - TextUnit::of_char(')')
|
||||
}
|
||||
};
|
||||
ActionResult {
|
||||
edit: edit.finish(),
|
||||
cursor_position: CursorPosition::Offset(
|
||||
node_start + TextUnit::of_str("#[derive(")
|
||||
),
|
||||
cursor_position: CursorPosition::Offset(offset),
|
||||
}
|
||||
})
|
||||
}
|
||||
|
|
|
@ -116,7 +116,12 @@ fn test_add_derive() {
|
|||
"struct Foo { a: i32, <|>}",
|
||||
"#[derive(<|>)]\nstruct Foo { a: i32, }",
|
||||
|file, off| add_derive(file, off).map(|f| f()),
|
||||
)
|
||||
);
|
||||
check_action(
|
||||
"#[derive(Clone)]\nstruct Foo { a: i32<|>, }",
|
||||
"#[derive(Clone<|>)]\nstruct Foo { a: i32, }",
|
||||
|file, off| add_derive(file, off).map(|f| f()),
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
|
|
|
@ -339,6 +339,7 @@ impl<R: TreeRoot> AstNode<R> for NominalDef<R> {
|
|||
}
|
||||
}
|
||||
|
||||
impl<R: TreeRoot> ast::AttrsOwner<R> for NominalDef<R> {}
|
||||
impl<R: TreeRoot> NominalDef<R> {}
|
||||
|
||||
// ParenType
|
||||
|
|
|
@ -52,20 +52,32 @@ impl<R: TreeRoot> File<R> {
|
|||
impl<R: TreeRoot> FnDef<R> {
|
||||
pub fn has_atom_attr(&self, atom: &str) -> bool {
|
||||
self.attrs()
|
||||
.filter_map(|x| x.value())
|
||||
.filter_map(|x| as_atom(x))
|
||||
.filter_map(|x| x.as_atom())
|
||||
.any(|x| x == atom)
|
||||
}
|
||||
}
|
||||
|
||||
fn as_atom<R: TreeRoot>(tt: TokenTree<R>) -> Option<SmolStr> {
|
||||
let syntax = tt.syntax_ref();
|
||||
let (_bra, attr, _ket) = syntax.children().collect_tuple()?;
|
||||
impl<R: TreeRoot> Attr<R> {
|
||||
pub fn as_atom(&self) -> Option<SmolStr> {
|
||||
let tt = self.value()?;
|
||||
let (_bra, attr, _ket) = tt.syntax().children().collect_tuple()?;
|
||||
if attr.kind() == IDENT {
|
||||
Some(attr.leaf_text().unwrap())
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
pub fn as_call(&self) -> Option<(SmolStr, TokenTree<R>)> {
|
||||
let tt = self.value()?;
|
||||
let (_bra, attr, args, _ket) = tt.syntax().children().collect_tuple()?;
|
||||
let args = TokenTree::cast(args)?;
|
||||
if attr.kind() == IDENT {
|
||||
Some((attr.leaf_text().unwrap(), args))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<R: TreeRoot> Name<R> {
|
||||
|
|
|
@ -272,6 +272,9 @@ Grammar(
|
|||
"DynTraitType",
|
||||
]),
|
||||
|
||||
"NominalDef": ( enum: ["StructDef", "EnumDef"]),
|
||||
"NominalDef": (
|
||||
enum: ["StructDef", "EnumDef"],
|
||||
traits: [ "AttrsOwner" ],
|
||||
),
|
||||
},
|
||||
)
|
||||
|
|
Loading…
Reference in a new issue