Skip to content

Commit c254537

Browse files
committed
refactor: editor for destructure_struct_binding
Signed-off-by: Prajwal S N <[email protected]>
1 parent bee9998 commit c254537

File tree

4 files changed

+76
-99
lines changed

4 files changed

+76
-99
lines changed

crates/ide-assists/src/handlers/destructure_struct_binding.rs

Lines changed: 51 additions & 93 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
use hir::{HasVisibility, sym};
2-
use ide_db::text_edit::TextRange;
32
use ide_db::{
43
FxHashMap, FxHashSet,
54
assists::AssistId,
@@ -8,7 +7,9 @@ use ide_db::{
87
search::{FileReference, SearchScope},
98
};
109
use itertools::Itertools;
11-
use syntax::{AstNode, Edition, SmolStr, SyntaxNode, ToSmolStr, ast, ted};
10+
use syntax::ast::syntax_factory::SyntaxFactory;
11+
use syntax::syntax_editor::SyntaxEditor;
12+
use syntax::{AstNode, Edition, SmolStr, SyntaxNode, ToSmolStr, ast};
1213

1314
use crate::{
1415
assist_context::{AssistContext, Assists, SourceChangeBuilder},
@@ -62,13 +63,10 @@ fn destructure_struct_binding_impl(
6263
data: &StructEditData,
6364
) {
6465
let field_names = generate_field_names(ctx, data);
65-
let assignment_edit = build_assignment_edit(ctx, builder, data, &field_names);
66-
let usage_edits = build_usage_edits(ctx, builder, data, &field_names.into_iter().collect());
67-
68-
assignment_edit.apply();
69-
for edit in usage_edits {
70-
edit.apply(builder);
71-
}
66+
let mut editor = builder.make_editor(data.ident_pat.syntax());
67+
destructure_pat(ctx, &mut editor, data, &field_names);
68+
update_usages(ctx, &mut editor, data, &field_names.into_iter().collect());
69+
builder.add_file_edits(ctx.file_id(), editor);
7270
}
7371

7472
struct StructEditData {
@@ -173,64 +171,57 @@ fn get_names_in_scope(
173171
Some(names)
174172
}
175173

176-
fn build_assignment_edit(
174+
fn destructure_pat(
177175
_ctx: &AssistContext<'_>,
178-
builder: &mut SourceChangeBuilder,
176+
editor: &mut SyntaxEditor,
179177
data: &StructEditData,
180178
field_names: &[(SmolStr, SmolStr)],
181-
) -> AssignmentEdit {
182-
let ident_pat = builder.make_mut(data.ident_pat.clone());
179+
) {
180+
let ident_pat = &data.ident_pat;
183181

184182
let struct_path = mod_path_to_ast(&data.struct_def_path, data.edition);
185183
let is_ref = ident_pat.ref_token().is_some();
186184
let is_mut = ident_pat.mut_token().is_some();
187185

186+
let make = SyntaxFactory::with_mappings();
188187
let new_pat = match data.kind {
189188
hir::StructKind::Tuple => {
190189
let ident_pats = field_names.iter().map(|(_, new_name)| {
191-
let name = ast::make::name(new_name);
192-
ast::Pat::from(ast::make::ident_pat(is_ref, is_mut, name))
190+
let name = make.name(new_name);
191+
ast::Pat::from(make.ident_pat(is_ref, is_mut, name))
193192
});
194-
ast::Pat::TupleStructPat(ast::make::tuple_struct_pat(struct_path, ident_pats))
193+
ast::Pat::TupleStructPat(make.tuple_struct_pat(struct_path, ident_pats))
195194
}
196195
hir::StructKind::Record => {
197196
let fields = field_names.iter().map(|(old_name, new_name)| {
198197
// Use shorthand syntax if possible
199198
if old_name == new_name && !is_mut {
200-
ast::make::record_pat_field_shorthand(ast::make::name_ref(old_name))
199+
make.record_pat_field_shorthand(make.name_ref(old_name))
201200
} else {
202-
ast::make::record_pat_field(
203-
ast::make::name_ref(old_name),
204-
ast::Pat::IdentPat(ast::make::ident_pat(
205-
is_ref,
206-
is_mut,
207-
ast::make::name(new_name),
208-
)),
201+
make.record_pat_field(
202+
make.name_ref(old_name),
203+
ast::Pat::IdentPat(make.ident_pat(is_ref, is_mut, make.name(new_name))),
209204
)
210205
}
211206
});
207+
let field_list = make
208+
.record_pat_field_list(fields, data.has_private_members.then_some(make.rest_pat()));
212209

213-
let field_list = ast::make::record_pat_field_list(
214-
fields,
215-
data.has_private_members.then_some(ast::make::rest_pat()),
216-
);
217-
ast::Pat::RecordPat(ast::make::record_pat_with_fields(struct_path, field_list))
210+
ast::Pat::RecordPat(make.record_pat_with_fields(struct_path, field_list))
218211
}
219-
hir::StructKind::Unit => ast::make::path_pat(struct_path),
212+
hir::StructKind::Unit => make.path_pat(struct_path),
220213
};
221214

222215
// If the binding is nested inside a record, we need to wrap the new
223216
// destructured pattern in a non-shorthand record field
224-
let new_pat = if data.is_nested {
225-
let record_pat_field =
226-
ast::make::record_pat_field(ast::make::name_ref(&ident_pat.to_string()), new_pat)
227-
.clone_for_update();
228-
NewPat::RecordPatField(record_pat_field)
217+
let destructured_pat = if data.is_nested {
218+
make.record_pat_field(make.name_ref(&ident_pat.to_string()), new_pat).syntax().clone()
229219
} else {
230-
NewPat::Pat(new_pat.clone_for_update())
220+
new_pat.syntax().clone()
231221
};
232222

233-
AssignmentEdit { old_pat: ident_pat, new_pat }
223+
editor.add_mappings(make.finish_with_mappings());
224+
editor.replace(data.ident_pat.syntax(), destructured_pat);
234225
}
235226

236227
fn generate_field_names(ctx: &AssistContext<'_>, data: &StructEditData) -> Vec<(SmolStr, SmolStr)> {
@@ -267,85 +258,52 @@ fn new_field_name(base_name: SmolStr, names_in_scope: &FxHashSet<SmolStr>) -> Sm
267258
name
268259
}
269260

270-
struct AssignmentEdit {
271-
old_pat: ast::IdentPat,
272-
new_pat: NewPat,
273-
}
274-
275-
enum NewPat {
276-
Pat(ast::Pat),
277-
RecordPatField(ast::RecordPatField),
278-
}
279-
280-
impl AssignmentEdit {
281-
fn apply(self) {
282-
match self.new_pat {
283-
NewPat::Pat(pat) => ted::replace(self.old_pat.syntax(), pat.syntax()),
284-
NewPat::RecordPatField(record_pat_field) => {
285-
ted::replace(self.old_pat.syntax(), record_pat_field.syntax())
286-
}
287-
}
288-
}
289-
}
290-
291-
fn build_usage_edits(
261+
fn update_usages(
292262
ctx: &AssistContext<'_>,
293-
builder: &mut SourceChangeBuilder,
263+
editor: &mut SyntaxEditor,
294264
data: &StructEditData,
295265
field_names: &FxHashMap<SmolStr, SmolStr>,
296-
) -> Vec<StructUsageEdit> {
297-
data.usages
266+
) {
267+
let make = SyntaxFactory::with_mappings();
268+
let edits = data
269+
.usages
298270
.iter()
299-
.filter_map(|r| build_usage_edit(ctx, builder, data, r, field_names))
300-
.collect_vec()
271+
.filter_map(|r| build_usage_edit(ctx, &make, data, r, field_names))
272+
.collect_vec();
273+
editor.add_mappings(make.finish_with_mappings());
274+
for (old, new) in edits {
275+
editor.replace(old, new);
276+
}
301277
}
302278

303279
fn build_usage_edit(
304280
ctx: &AssistContext<'_>,
305-
builder: &mut SourceChangeBuilder,
281+
make: &SyntaxFactory,
306282
data: &StructEditData,
307283
usage: &FileReference,
308284
field_names: &FxHashMap<SmolStr, SmolStr>,
309-
) -> Option<StructUsageEdit> {
285+
) -> Option<(SyntaxNode, SyntaxNode)> {
310286
match usage.name.syntax().ancestors().find_map(ast::FieldExpr::cast) {
311287
Some(field_expr) => Some({
312288
let field_name: SmolStr = field_expr.name_ref()?.to_string().into();
313289
let new_field_name = field_names.get(&field_name)?;
314-
let new_expr = ast::make::expr_path(ast::make::ext::ident_path(new_field_name));
290+
let new_expr = make.expr_path(ast::make::ext::ident_path(new_field_name));
315291

316292
// If struct binding is a reference, we might need to deref field usages
317293
if data.is_ref {
318294
let (replace_expr, ref_data) = determine_ref_and_parens(ctx, &field_expr);
319-
StructUsageEdit::IndexField(
320-
builder.make_mut(replace_expr),
321-
ref_data.wrap_expr(new_expr).clone_for_update(),
295+
(
296+
replace_expr.syntax().clone_for_update(),
297+
ref_data.wrap_expr(new_expr).syntax().clone_for_update(),
322298
)
323299
} else {
324-
StructUsageEdit::IndexField(
325-
builder.make_mut(field_expr).into(),
326-
new_expr.clone_for_update(),
327-
)
300+
(field_expr.syntax().clone(), new_expr.syntax().clone())
328301
}
329302
}),
330-
None => Some(StructUsageEdit::Path(usage.range)),
331-
}
332-
}
333-
334-
enum StructUsageEdit {
335-
Path(TextRange),
336-
IndexField(ast::Expr, ast::Expr),
337-
}
338-
339-
impl StructUsageEdit {
340-
fn apply(self, edit: &mut SourceChangeBuilder) {
341-
match self {
342-
StructUsageEdit::Path(target_expr) => {
343-
edit.replace(target_expr, "todo!()");
344-
}
345-
StructUsageEdit::IndexField(target_expr, replace_with) => {
346-
ted::replace(target_expr.syntax(), replace_with.syntax())
347-
}
348-
}
303+
None => Some((
304+
usage.name.syntax().as_node().unwrap().clone(),
305+
make.expr_macro(ast::make::ext::ident_path("todo"), make.arg_list([])).syntax().clone(),
306+
)),
349307
}
350308
}
351309

crates/ide-assists/src/utils/gen_trait_fn_body.rs

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -228,11 +228,11 @@ fn gen_debug_impl(adt: &ast::Adt, func: &ast::Fn) -> Option<()> {
228228
None => {
229229
let fmt_string = make::expr_literal(&(format!("\"{name}\""))).into();
230230
let args = make::arg_list([target, fmt_string]);
231-
let macro_name = make::expr_path(make::ext::ident_path("write"));
232-
let macro_call = make::expr_macro_call(macro_name, args);
231+
let macro_name = make::ext::ident_path("write");
232+
let macro_call = make::expr_macro(macro_name, args);
233233

234234
let variant_name = make::path_pat(variant_name);
235-
arms.push(make::match_arm(variant_name, None, macro_call));
235+
arms.push(make::match_arm(variant_name, None, macro_call.into()));
236236
}
237237
}
238238
}

crates/syntax/src/ast/make.rs

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -643,8 +643,8 @@ pub fn expr_method_call(
643643
) -> ast::Expr {
644644
expr_from_text(&format!("{receiver}.{method}{arg_list}"))
645645
}
646-
pub fn expr_macro_call(f: ast::Expr, arg_list: ast::ArgList) -> ast::Expr {
647-
expr_from_text(&format!("{f}!{arg_list}"))
646+
pub fn expr_macro(path: ast::Path, arg_list: ast::ArgList) -> ast::MacroExpr {
647+
expr_from_text(&format!("{path}!{arg_list}"))
648648
}
649649
pub fn expr_ref(expr: ast::Expr, exclusive: bool) -> ast::Expr {
650650
expr_from_text(&if exclusive { format!("&mut {expr}") } else { format!("&{expr}") })

crates/syntax/src/ast/syntax_factory/constructors.rs

Lines changed: 20 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -241,7 +241,7 @@ impl SyntaxFactory {
241241
ast
242242
}
243243

244-
pub fn record_pat_field(self, name_ref: ast::NameRef, pat: ast::Pat) -> ast::RecordPatField {
244+
pub fn record_pat_field(&self, name_ref: ast::NameRef, pat: ast::Pat) -> ast::RecordPatField {
245245
let ast = make::record_pat_field(name_ref.clone(), pat.clone()).clone_for_update();
246246

247247
if let Some(mut mapping) = self.mappings() {
@@ -290,6 +290,10 @@ impl SyntaxFactory {
290290
ast
291291
}
292292

293+
pub fn rest_pat(&self) -> ast::RestPat {
294+
make::rest_pat().clone_for_update()
295+
}
296+
293297
pub fn block_expr(
294298
&self,
295299
statements: impl IntoIterator<Item = ast::Stmt>,
@@ -597,6 +601,21 @@ impl SyntaxFactory {
597601
ast
598602
}
599603

604+
pub fn expr_macro(&self, path: ast::Path, args: ast::ArgList) -> ast::MacroExpr {
605+
let ast = make::expr_macro(path.clone(), args.clone()).clone_for_update();
606+
607+
if let Some(mut mapping) = self.mappings() {
608+
let macro_call = ast.macro_call().unwrap();
609+
let mut builder = SyntaxMappingBuilder::new(macro_call.syntax().clone());
610+
builder.map_node(path.syntax().clone(), macro_call.path().unwrap().syntax().clone());
611+
builder
612+
.map_node(args.syntax().clone(), macro_call.token_tree().unwrap().syntax().clone());
613+
builder.finish(&mut mapping);
614+
}
615+
616+
ast
617+
}
618+
600619
pub fn match_arm(
601620
&self,
602621
pat: ast::Pat,

0 commit comments

Comments
 (0)