minor: Simplify

This commit is contained in:
Lukas Wirth 2021-10-03 12:42:00 +02:00
parent 745fd9903c
commit 0943c4be8b
7 changed files with 91 additions and 143 deletions

View file

@ -4,7 +4,6 @@ use indexmap::IndexMap;
use hir::Semantics; use hir::Semantics;
use ide_db::{ use ide_db::{
call_info::FnCallNode,
defs::{Definition, NameClass, NameRefClass}, defs::{Definition, NameClass, NameRefClass},
helpers::pick_best_token, helpers::pick_best_token,
search::FileReference, search::FileReference,
@ -101,23 +100,27 @@ pub(crate) fn outgoing_calls(db: &RootDatabase, position: FilePosition) -> Optio
_ => None, _ => None,
}) })
.flatten() .flatten()
.filter_map(|node| FnCallNode::with_node_exact(&node)) .filter_map(ast::CallableExpr::cast)
.filter_map(|call_node| { .filter_map(|call_node| {
let name_ref = call_node.name_ref()?; let (nav_target, range) = match call_node {
let func_target = match call_node { ast::CallableExpr::Call(call) => {
FnCallNode::CallExpr(expr) => { let expr = call.expr()?;
let callable = sema.type_of_expr(&expr.expr()?)?.original.as_callable(db)?; let callable = sema.type_of_expr(&expr)?.original.as_callable(db)?;
match callable.kind() { match callable.kind() {
hir::CallableKind::Function(it) => it.try_to_nav(db), hir::CallableKind::Function(it) => {
let range = expr.syntax().text_range();
it.try_to_nav(db).zip(Some(range))
}
_ => None, _ => None,
} }
} }
FnCallNode::MethodCallExpr(expr) => { ast::CallableExpr::MethodCall(expr) => {
let range = expr.name_ref()?.syntax().text_range();
let function = sema.resolve_method_call(&expr)?; let function = sema.resolve_method_call(&expr)?;
function.try_to_nav(db) function.try_to_nav(db).zip(Some(range))
} }
}?; }?;
Some((func_target, name_ref.syntax().text_range())) Some((nav_target, range))
}) })
.for_each(|(nav, range)| calls.add(nav, range)); .for_each(|(nav, range)| calls.add(nav, range));

View file

@ -51,27 +51,6 @@ pub(crate) fn generate_function(acc: &mut Assists, ctx: &AssistContext) -> Optio
gen_fn(acc, ctx).or_else(|| gen_method(acc, ctx)) gen_fn(acc, ctx).or_else(|| gen_method(acc, ctx))
} }
enum FuncExpr {
Func(ast::CallExpr),
Method(ast::MethodCallExpr),
}
impl FuncExpr {
fn arg_list(&self) -> Option<ast::ArgList> {
match self {
FuncExpr::Func(fn_call) => fn_call.arg_list(),
FuncExpr::Method(m_call) => m_call.arg_list(),
}
}
fn syntax(&self) -> &SyntaxNode {
match self {
FuncExpr::Func(fn_call) => fn_call.syntax(),
FuncExpr::Method(m_call) => m_call.syntax(),
}
}
}
fn gen_fn(acc: &mut Assists, ctx: &AssistContext) -> Option<()> { fn gen_fn(acc: &mut Assists, ctx: &AssistContext) -> Option<()> {
let path_expr: ast::PathExpr = ctx.find_node_at_offset()?; let path_expr: ast::PathExpr = ctx.find_node_at_offset()?;
let call = path_expr.syntax().parent().and_then(ast::CallExpr::cast)?; let call = path_expr.syntax().parent().and_then(ast::CallExpr::cast)?;
@ -254,7 +233,8 @@ impl FunctionBuilder {
let needs_pub = target_module.is_some(); let needs_pub = target_module.is_some();
let target_module = target_module.or_else(|| current_module(target.syntax(), ctx))?; let target_module = target_module.or_else(|| current_module(target.syntax(), ctx))?;
let fn_name = make::name(fn_name); let fn_name = make::name(fn_name);
let (type_params, params) = fn_args(ctx, target_module, FuncExpr::Func(call.clone()))?; let (type_params, params) =
fn_args(ctx, target_module, ast::CallableExpr::Call(call.clone()))?;
let await_expr = call.syntax().parent().and_then(ast::AwaitExpr::cast); let await_expr = call.syntax().parent().and_then(ast::AwaitExpr::cast);
let is_async = await_expr.is_some(); let is_async = await_expr.is_some();
@ -284,7 +264,8 @@ impl FunctionBuilder {
let needs_pub = let needs_pub =
!module_is_descendant(&current_module(call.syntax(), ctx)?, &target_module, ctx); !module_is_descendant(&current_module(call.syntax(), ctx)?, &target_module, ctx);
let fn_name = make::name(&name.text()); let fn_name = make::name(&name.text());
let (type_params, params) = fn_args(ctx, target_module, FuncExpr::Method(call.clone()))?; let (type_params, params) =
fn_args(ctx, target_module, ast::CallableExpr::MethodCall(call.clone()))?;
let await_expr = call.syntax().parent().and_then(ast::AwaitExpr::cast); let await_expr = call.syntax().parent().and_then(ast::AwaitExpr::cast);
let is_async = await_expr.is_some(); let is_async = await_expr.is_some();
@ -392,7 +373,7 @@ fn get_fn_target(
file = in_file; file = in_file;
target target
} }
None => next_space_for_fn_after_call_site(FuncExpr::Func(call))?, None => next_space_for_fn_after_call_site(ast::CallableExpr::Call(call))?,
}; };
Some((target.clone(), file, get_insert_offset(&target))) Some((target.clone(), file, get_insert_offset(&target)))
} }
@ -438,7 +419,7 @@ impl GeneratedFunctionTarget {
fn fn_args( fn fn_args(
ctx: &AssistContext, ctx: &AssistContext,
target_module: hir::Module, target_module: hir::Module,
call: FuncExpr, call: ast::CallableExpr,
) -> Option<(Option<ast::GenericParamList>, ast::ParamList)> { ) -> Option<(Option<ast::GenericParamList>, ast::ParamList)> {
let mut arg_names = Vec::new(); let mut arg_names = Vec::new();
let mut arg_types = Vec::new(); let mut arg_types = Vec::new();
@ -468,8 +449,8 @@ fn fn_args(
None, None,
make::param_list( make::param_list(
match call { match call {
FuncExpr::Func(_) => None, ast::CallableExpr::Call(_) => None,
FuncExpr::Method(_) => Some(make::self_param()), ast::CallableExpr::MethodCall(_) => Some(make::self_param()),
}, },
params, params,
), ),
@ -553,7 +534,7 @@ fn fn_arg_type(
/// directly after the current block /// directly after the current block
/// We want to write the generated function directly after /// We want to write the generated function directly after
/// fns, impls or macro calls, but inside mods /// fns, impls or macro calls, but inside mods
fn next_space_for_fn_after_call_site(expr: FuncExpr) -> Option<GeneratedFunctionTarget> { fn next_space_for_fn_after_call_site(expr: ast::CallableExpr) -> Option<GeneratedFunctionTarget> {
let mut ancestors = expr.syntax().ancestors().peekable(); let mut ancestors = expr.syntax().ancestors().peekable();
let mut last_ancestor: Option<SyntaxNode> = None; let mut last_ancestor: Option<SyntaxNode> = None;
while let Some(next_ancestor) = ancestors.next() { while let Some(next_ancestor) = ancestors.next() {

View file

@ -12,7 +12,7 @@ use ide_db::{
use itertools::{izip, Itertools}; use itertools::{izip, Itertools};
use syntax::{ use syntax::{
ast::{self, edit_in_place::Indent, HasArgList, PathExpr}, ast::{self, edit_in_place::Indent, HasArgList, PathExpr},
ted, AstNode, SyntaxNode, ted, AstNode,
}; };
use crate::{ use crate::{
@ -178,7 +178,7 @@ pub(crate) fn inline_call(acc: &mut Assists, ctx: &AssistContext) -> Option<()>
let name_ref: ast::NameRef = ctx.find_node_at_offset()?; let name_ref: ast::NameRef = ctx.find_node_at_offset()?;
let call_info = CallInfo::from_name_ref(name_ref.clone())?; let call_info = CallInfo::from_name_ref(name_ref.clone())?;
let (function, label) = match &call_info.node { let (function, label) = match &call_info.node {
CallExprNode::Call(call) => { ast::CallableExpr::Call(call) => {
let path = match call.expr()? { let path = match call.expr()? {
ast::Expr::PathExpr(path) => path.path(), ast::Expr::PathExpr(path) => path.path(),
_ => None, _ => None,
@ -190,7 +190,7 @@ pub(crate) fn inline_call(acc: &mut Assists, ctx: &AssistContext) -> Option<()>
}; };
(function, format!("Inline `{}`", path)) (function, format!("Inline `{}`", path))
} }
CallExprNode::MethodCallExpr(call) => { ast::CallableExpr::MethodCall(call) => {
(ctx.sema.resolve_method_call(call)?, format!("Inline `{}`", name_ref)) (ctx.sema.resolve_method_call(call)?, format!("Inline `{}`", name_ref))
} }
}; };
@ -223,8 +223,8 @@ pub(crate) fn inline_call(acc: &mut Assists, ctx: &AssistContext) -> Option<()>
builder.replace_ast( builder.replace_ast(
match call_info.node { match call_info.node {
CallExprNode::Call(it) => ast::Expr::CallExpr(it), ast::CallableExpr::Call(it) => ast::Expr::CallExpr(it),
CallExprNode::MethodCallExpr(it) => ast::Expr::MethodCallExpr(it), ast::CallableExpr::MethodCall(it) => ast::Expr::MethodCallExpr(it),
}, },
replacement, replacement,
); );
@ -232,22 +232,8 @@ pub(crate) fn inline_call(acc: &mut Assists, ctx: &AssistContext) -> Option<()>
) )
} }
enum CallExprNode {
Call(ast::CallExpr),
MethodCallExpr(ast::MethodCallExpr),
}
impl CallExprNode {
fn syntax(&self) -> &SyntaxNode {
match self {
CallExprNode::Call(it) => it.syntax(),
CallExprNode::MethodCallExpr(it) => it.syntax(),
}
}
}
struct CallInfo { struct CallInfo {
node: CallExprNode, node: ast::CallableExpr,
arguments: Vec<ast::Expr>, arguments: Vec<ast::Expr>,
generic_arg_list: Option<ast::GenericArgList>, generic_arg_list: Option<ast::GenericArgList>,
} }
@ -261,7 +247,7 @@ impl CallInfo {
arguments.extend(call.arg_list()?.args()); arguments.extend(call.arg_list()?.args());
Some(CallInfo { Some(CallInfo {
generic_arg_list: call.generic_arg_list(), generic_arg_list: call.generic_arg_list(),
node: CallExprNode::MethodCallExpr(call), node: ast::CallableExpr::MethodCall(call),
arguments, arguments,
}) })
} else if let Some(segment) = ast::PathSegment::cast(parent) { } else if let Some(segment) = ast::PathSegment::cast(parent) {
@ -271,7 +257,7 @@ impl CallInfo {
Some(CallInfo { Some(CallInfo {
arguments: call.arg_list()?.args().collect(), arguments: call.arg_list()?.args().collect(),
node: CallExprNode::Call(call), node: ast::CallableExpr::Call(call),
generic_arg_list: segment.generic_arg_list(), generic_arg_list: segment.generic_arg_list(),
}) })
} else { } else {
@ -367,8 +353,9 @@ fn inline(
ted::replace(usage.syntax(), &replacement.syntax().clone_for_update()); ted::replace(usage.syntax(), &replacement.syntax().clone_for_update());
} }
}; };
// izip confuses RA due to our lack of hygiene info currently losing us typeinfo // izip confuses RA due to our lack of hygiene info currently losing us type info causing incorrect errors
let usages: &[ast::PathExpr] = &*usages; let usages: &[ast::PathExpr] = &*usages;
let expr: &ast::Expr = expr;
match usages { match usages {
// inline single use closure arguments // inline single use closure arguments
[usage] [usage]
@ -414,8 +401,8 @@ fn inline(
} }
let original_indentation = match node { let original_indentation = match node {
CallExprNode::Call(it) => it.indent_level(), ast::CallableExpr::Call(it) => it.indent_level(),
CallExprNode::MethodCallExpr(it) => it.indent_level(), ast::CallableExpr::MethodCall(it) => it.indent_level(),
}; };
body.reindent_to(original_indentation); body.reindent_to(original_indentation);

View file

@ -6,7 +6,7 @@ use stdx::format_to;
use syntax::{ use syntax::{
algo, algo,
ast::{self, HasArgList, HasName}, ast::{self, HasArgList, HasName},
match_ast, AstNode, Direction, SyntaxNode, SyntaxToken, TextRange, TextSize, AstNode, Direction, SyntaxToken, TextRange, TextSize,
}; };
use crate::RootDatabase; use crate::RootDatabase;
@ -25,9 +25,11 @@ impl CallInfo {
pub fn parameter_labels(&self) -> impl Iterator<Item = &str> + '_ { pub fn parameter_labels(&self) -> impl Iterator<Item = &str> + '_ {
self.parameters.iter().map(move |&it| &self.signature[it]) self.parameters.iter().map(move |&it| &self.signature[it])
} }
pub fn parameter_ranges(&self) -> &[TextRange] { pub fn parameter_ranges(&self) -> &[TextRange] {
&self.parameters &self.parameters
} }
fn push_param(&mut self, param: &str) { fn push_param(&mut self, param: &str) {
if !self.signature.ends_with('(') { if !self.signature.ends_with('(') {
self.signature.push_str(", "); self.signature.push_str(", ");
@ -115,31 +117,24 @@ fn call_info_impl(
token: SyntaxToken, token: SyntaxToken,
) -> Option<(hir::Callable, Option<usize>)> { ) -> Option<(hir::Callable, Option<usize>)> {
// Find the calling expression and it's NameRef // Find the calling expression and it's NameRef
let calling_node = FnCallNode::with_node(&token.parent()?)?; let parent = token.parent()?;
let calling_node = parent.ancestors().filter_map(ast::CallableExpr::cast).find(|it| {
it.arg_list()
.map_or(false, |it| it.syntax().text_range().contains(token.text_range().start()))
})?;
let callable = match &calling_node { let callable = match &calling_node {
FnCallNode::CallExpr(call) => { ast::CallableExpr::Call(call) => {
sema.type_of_expr(&call.expr()?)?.adjusted().as_callable(sema.db)? let expr = call.expr()?;
sema.type_of_expr(&expr)?.adjusted().as_callable(sema.db)
} }
FnCallNode::MethodCallExpr(call) => sema.resolve_method_call_as_callable(call)?, ast::CallableExpr::MethodCall(call) => sema.resolve_method_call_as_callable(call),
}; }?;
let active_param = if let Some(arg_list) = calling_node.arg_list() { let active_param = if let Some(arg_list) = calling_node.arg_list() {
// Number of arguments specified at the call site let param = arg_list
let num_args_at_callsite = arg_list.args().count();
let arg_list_range = arg_list.syntax().text_range();
if !arg_list_range.contains_inclusive(token.text_range().start()) {
cov_mark::hit!(call_info_bad_offset);
return None;
}
let param = std::cmp::min(
num_args_at_callsite,
arg_list
.args() .args()
.take_while(|arg| arg.syntax().text_range().end() <= token.text_range().start()) .take_while(|arg| arg.syntax().text_range().end() <= token.text_range().start())
.count(), .count();
);
Some(param) Some(param)
} else { } else {
None None
@ -175,60 +170,5 @@ impl ActiveParameter {
} }
} }
#[derive(Debug)]
pub enum FnCallNode {
CallExpr(ast::CallExpr),
MethodCallExpr(ast::MethodCallExpr),
}
impl FnCallNode {
fn with_node(syntax: &SyntaxNode) -> Option<FnCallNode> {
syntax.ancestors().find_map(|node| {
match_ast! {
match node {
ast::CallExpr(it) => Some(FnCallNode::CallExpr(it)),
ast::MethodCallExpr(it) => {
let arg_list = it.arg_list()?;
if !arg_list.syntax().text_range().contains_range(syntax.text_range()) {
return None;
}
Some(FnCallNode::MethodCallExpr(it))
},
_ => None,
}
}
})
}
pub fn with_node_exact(node: &SyntaxNode) -> Option<FnCallNode> {
match_ast! {
match node {
ast::CallExpr(it) => Some(FnCallNode::CallExpr(it)),
ast::MethodCallExpr(it) => Some(FnCallNode::MethodCallExpr(it)),
_ => None,
}
}
}
pub fn name_ref(&self) -> Option<ast::NameRef> {
match self {
FnCallNode::CallExpr(call_expr) => Some(match call_expr.expr()? {
ast::Expr::PathExpr(path_expr) => path_expr.path()?.segment()?.name_ref()?,
_ => return None,
}),
FnCallNode::MethodCallExpr(call_expr) => {
call_expr.syntax().children().find_map(ast::NameRef::cast)
}
}
}
fn arg_list(&self) -> Option<ast::ArgList> {
match self {
FnCallNode::CallExpr(expr) => expr.arg_list(),
FnCallNode::MethodCallExpr(expr) => expr.arg_list(),
}
}
}
#[cfg(test)] #[cfg(test)]
mod tests; mod tests;

View file

@ -362,7 +362,6 @@ pub fn foo(mut r: WriteHandler<()>) {
#[test] #[test]
fn call_info_bad_offset() { fn call_info_bad_offset() {
cov_mark::check!(call_info_bad_offset);
check( check(
r#" r#"
fn foo(x: u32, y: u32) -> u32 {x + y} fn foo(x: u32, y: u32) -> u32 {x + y}

View file

@ -18,7 +18,7 @@ use crate::{
}; };
pub use self::{ pub use self::{
expr_ext::{ArrayExprKind, BlockModifier, ElseBranch, LiteralKind}, expr_ext::{ArrayExprKind, BlockModifier, CallableExpr, ElseBranch, LiteralKind},
generated::{nodes::*, tokens::*}, generated::{nodes::*, tokens::*},
node_ext::{ node_ext::{
AttrKind, FieldKind, Macro, NameLike, NameOrNameRef, PathSegmentKind, SelfParamKind, AttrKind, FieldKind, Macro, NameLike, NameOrNameRef, PathSegmentKind, SelfParamKind,

View file

@ -10,7 +10,7 @@ use crate::{
}, },
AstToken, AstToken,
SyntaxKind::*, SyntaxKind::*,
SyntaxToken, T, SyntaxNode, SyntaxToken, T,
}; };
impl ast::HasAttrs for ast::Expr {} impl ast::HasAttrs for ast::Expr {}
@ -312,3 +312,41 @@ impl ast::RecordExprField {
self.syntax().ancestors().find_map(ast::RecordExpr::cast).unwrap() self.syntax().ancestors().find_map(ast::RecordExpr::cast).unwrap()
} }
} }
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum CallableExpr {
Call(ast::CallExpr),
MethodCall(ast::MethodCallExpr),
}
impl ast::HasAttrs for CallableExpr {}
impl ast::HasArgList for CallableExpr {}
impl AstNode for CallableExpr {
fn can_cast(kind: parser::SyntaxKind) -> bool
where
Self: Sized,
{
ast::CallExpr::can_cast(kind) || ast::MethodCallExpr::can_cast(kind)
}
fn cast(syntax: SyntaxNode) -> Option<Self>
where
Self: Sized,
{
if let Some(it) = ast::CallExpr::cast(syntax.clone()) {
Some(Self::Call(it))
} else if let Some(it) = ast::MethodCallExpr::cast(syntax.clone()) {
Some(Self::MethodCall(it))
} else {
None
}
}
fn syntax(&self) -> &SyntaxNode {
match self {
Self::Call(it) => it.syntax(),
Self::MethodCall(it) => it.syntax(),
}
}
}