mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-15 06:33:58 +00:00
Add text edit to adjustment hints
This commit is contained in:
parent
3ae93bcb82
commit
f086fa9c02
2 changed files with 42 additions and 21 deletions
|
@ -3,6 +3,8 @@
|
||||||
//! let _: u32 = /* <never-to-any> */ loop {};
|
//! let _: u32 = /* <never-to-any> */ loop {};
|
||||||
//! let _: &u32 = /* &* */ &mut 0;
|
//! let _: &u32 = /* &* */ &mut 0;
|
||||||
//! ```
|
//! ```
|
||||||
|
use std::ops::Not;
|
||||||
|
|
||||||
use either::Either;
|
use either::Either;
|
||||||
use hir::{
|
use hir::{
|
||||||
Adjust, Adjustment, AutoBorrow, HirDisplay, Mutability, OverloadedDeref, PointerCast, Safety,
|
Adjust, Adjustment, AutoBorrow, HirDisplay, Mutability, OverloadedDeref, PointerCast, Safety,
|
||||||
|
@ -15,6 +17,7 @@ use syntax::{
|
||||||
ast::{self, make, AstNode},
|
ast::{self, make, AstNode},
|
||||||
ted,
|
ted,
|
||||||
};
|
};
|
||||||
|
use text_edit::TextEditBuilder;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
AdjustmentHints, AdjustmentHintsMode, InlayHint, InlayHintLabel, InlayHintLabelPart,
|
AdjustmentHints, AdjustmentHintsMode, InlayHint, InlayHintLabel, InlayHintLabelPart,
|
||||||
|
@ -51,15 +54,15 @@ pub(super) fn hints(
|
||||||
let adjustments = sema.expr_adjustments(desc_expr).filter(|it| !it.is_empty())?;
|
let adjustments = sema.expr_adjustments(desc_expr).filter(|it| !it.is_empty())?;
|
||||||
|
|
||||||
if let ast::Expr::BlockExpr(_) | ast::Expr::IfExpr(_) | ast::Expr::MatchExpr(_) = desc_expr {
|
if let ast::Expr::BlockExpr(_) | ast::Expr::IfExpr(_) | ast::Expr::MatchExpr(_) = desc_expr {
|
||||||
if let [Adjustment { kind: Adjust::Deref(_), source, .. }, Adjustment { kind: Adjust::Borrow(_), source: _, target }] =
|
|
||||||
&*adjustments
|
|
||||||
{
|
|
||||||
// Don't show unnecessary reborrows for these, they will just repeat the inner ones again
|
// Don't show unnecessary reborrows for these, they will just repeat the inner ones again
|
||||||
if source == target {
|
if matches!(
|
||||||
|
&*adjustments,
|
||||||
|
[Adjustment { kind: Adjust::Deref(_), source, .. }, Adjustment { kind: Adjust::Borrow(_), target, .. }]
|
||||||
|
if source == target
|
||||||
|
) {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
let (postfix, needs_outer_parens, needs_inner_parens) =
|
let (postfix, needs_outer_parens, needs_inner_parens) =
|
||||||
mode_and_needs_parens_for_adjustment_hints(expr, config.adjustment_hints_mode);
|
mode_and_needs_parens_for_adjustment_hints(expr, config.adjustment_hints_mode);
|
||||||
|
@ -170,12 +173,39 @@ pub(super) fn hints(
|
||||||
if needs_outer_parens || (!postfix && needs_inner_parens) {
|
if needs_outer_parens || (!postfix && needs_inner_parens) {
|
||||||
post.label.append_str(")");
|
post.label.append_str(")");
|
||||||
}
|
}
|
||||||
if !pre.label.parts.is_empty() {
|
|
||||||
acc.push(pre);
|
let mut pre = pre.label.parts.is_empty().not().then_some(pre);
|
||||||
|
let mut post = post.label.parts.is_empty().not().then_some(post);
|
||||||
|
if pre.is_none() && post.is_none() {
|
||||||
|
return None;
|
||||||
}
|
}
|
||||||
if !post.label.parts.is_empty() {
|
let edit = {
|
||||||
acc.push(post);
|
let mut b = TextEditBuilder::default();
|
||||||
|
if let Some(pre) = &pre {
|
||||||
|
b.insert(
|
||||||
|
pre.range.start(),
|
||||||
|
pre.label.parts.iter().map(|part| &*part.text).collect::<String>(),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
if let Some(post) = &post {
|
||||||
|
b.insert(
|
||||||
|
post.range.end(),
|
||||||
|
post.label.parts.iter().map(|part| &*part.text).collect::<String>(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
b.finish()
|
||||||
|
};
|
||||||
|
match (&mut pre, &mut post) {
|
||||||
|
(Some(pre), Some(post)) => {
|
||||||
|
pre.text_edit = Some(edit.clone());
|
||||||
|
post.text_edit = Some(edit);
|
||||||
|
}
|
||||||
|
(Some(pre), None) => pre.text_edit = Some(edit),
|
||||||
|
(None, Some(post)) => post.text_edit = Some(edit),
|
||||||
|
(None, None) => (),
|
||||||
|
}
|
||||||
|
acc.extend(pre);
|
||||||
|
acc.extend(post);
|
||||||
Some(())
|
Some(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -23,16 +23,7 @@ pub(super) fn hints(
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
let outer_paren_pat = pat
|
let outer_paren_pat = pat.syntax().ancestors().skip(1).map_while(ast::ParenPat::cast).last();
|
||||||
.syntax()
|
|
||||||
.ancestors()
|
|
||||||
.skip(1)
|
|
||||||
.map_while(ast::Pat::cast)
|
|
||||||
.map_while(|pat| match pat {
|
|
||||||
ast::Pat::ParenPat(pat) => Some(pat),
|
|
||||||
_ => None,
|
|
||||||
})
|
|
||||||
.last();
|
|
||||||
let range = outer_paren_pat.as_ref().map_or_else(
|
let range = outer_paren_pat.as_ref().map_or_else(
|
||||||
|| match pat {
|
|| match pat {
|
||||||
// for ident patterns that @ bind a name, render the un-ref patterns in front of the inner pattern
|
// for ident patterns that @ bind a name, render the un-ref patterns in front of the inner pattern
|
||||||
|
|
Loading…
Reference in a new issue