blob: d5aa8b9f61f5d6cca901169b855d51955808d381 [file] [log] [blame]
// 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.
// @dart = 2.9
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;
FieldElement fieldElement = nameNode.staticElement;
// 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 parameterElement = parameter.declaredElement;
if (parameterElement is FieldFormalParameterElement &&
parameterElement.field == fieldElement) {
var identifier = parameter.identifier;
builder.addSimpleReplacement(range.node(identifier), '_$name');
}
}
}
}
// Write getter and setter.
builder.addInsertion(fieldDeclaration.end, (builder) {
String docCode;
if (fieldDeclaration.documentationComment != null) {
docCode = utils.getNodeText(fieldDeclaration.documentationComment);
}
var typeCode = '';
if (variableList.type != null) {
typeCode = utils.getNodeText(variableList.type) + ' ';
}
// 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(' }');
});
});
}
/// Return an instance of this class. Used as a tear-off in `AssistProcessor`.
static EncapsulateField newInstance() => EncapsulateField();
}