| // Copyright (c) 2020, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| import 'package:analysis_server/src/services/correction/assist.dart'; |
| import 'package:analysis_server/src/services/correction/dart/abstract_producer.dart'; |
| import 'package:analyzer/dart/ast/ast.dart'; |
| import 'package:analyzer/dart/element/element.dart'; |
| import 'package:analyzer_plugin/utilities/assist/assist.dart'; |
| import 'package:analyzer_plugin/utilities/change_builder/change_builder_core.dart'; |
| import 'package:analyzer_plugin/utilities/range_factory.dart'; |
| |
| class EncapsulateField extends CorrectionProducer { |
| @override |
| AssistKind get assistKind => DartAssistKind.ENCAPSULATE_FIELD; |
| |
| @override |
| Future<void> compute(ChangeBuilder builder) async { |
| // find FieldDeclaration |
| var fieldDeclaration = node.thisOrAncestorOfType<FieldDeclaration>(); |
| if (fieldDeclaration == null) { |
| return; |
| } |
| // not interesting for static |
| if (fieldDeclaration.isStatic) { |
| return; |
| } |
| // has a parse error |
| var variableList = fieldDeclaration.fields; |
| if (variableList.keyword == null && variableList.type == null) { |
| return; |
| } |
| // not interesting for final |
| if (variableList.isFinal) { |
| return; |
| } |
| // should have exactly one field |
| List<VariableDeclaration> fields = variableList.variables; |
| if (fields.length != 1) { |
| return; |
| } |
| var field = fields.first; |
| var nameNode = field.name; |
| var fieldElement = field.declaredElement as FieldElement; |
| // should have a public name |
| var name = nameNode.name; |
| if (Identifier.isPrivateName(name)) { |
| return; |
| } |
| // should be on the name |
| if (nameNode != node) { |
| return; |
| } |
| |
| // Should be in a class or mixin. |
| if (fieldDeclaration.parent is! ClassOrMixinDeclaration) { |
| return; |
| } |
| var classDeclaration = fieldDeclaration.parent as ClassOrMixinDeclaration; |
| |
| await builder.addDartFileEdit(file, (builder) { |
| // rename field |
| builder.addSimpleReplacement(range.node(nameNode), '_$name'); |
| // update references in constructors |
| for (var member in classDeclaration.members) { |
| if (member is ConstructorDeclaration) { |
| for (var parameter in member.parameters.parameters) { |
| var identifier = parameter.identifier; |
| var parameterElement = parameter.declaredElement; |
| if (identifier != null && |
| parameterElement is FieldFormalParameterElement && |
| parameterElement.field == fieldElement) { |
| builder.addSimpleReplacement(range.node(identifier), '_$name'); |
| } |
| } |
| } |
| } |
| |
| // Write getter and setter. |
| builder.addInsertion(fieldDeclaration.end, (builder) { |
| String? docCode; |
| var documentationComment = fieldDeclaration.documentationComment; |
| if (documentationComment != null) { |
| docCode = utils.getNodeText(documentationComment); |
| } |
| |
| var typeCode = ''; |
| var typeAnnotation = variableList.type; |
| if (typeAnnotation != null) { |
| typeCode = '${utils.getNodeText(typeAnnotation)} '; |
| } |
| |
| // Write getter. |
| builder.writeln(); |
| builder.writeln(); |
| if (docCode != null) { |
| builder.write(' '); |
| builder.writeln(docCode); |
| } |
| builder.write(' ${typeCode}get $name => _$name;'); |
| |
| // Write setter. |
| builder.writeln(); |
| builder.writeln(); |
| if (docCode != null) { |
| builder.write(' '); |
| builder.writeln(docCode); |
| } |
| builder.writeln(' set $name($typeCode$name) {'); |
| builder.writeln(' _$name = $name;'); |
| builder.write(' }'); |
| }); |
| }); |
| } |
| } |