blob: 9d880a9355b2696c60a67f04856863b3612de3ce [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.
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/ast/token.dart';
import 'package:analyzer/dart/ast/visitor.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/src/dart/ast/extensions.dart';
import 'package:analyzer_plugin/utilities/assist/assist.dart';
import 'package:analyzer_plugin/utilities/change_builder/change_builder_core.dart';
class ShadowField extends CorrectionProducer {
@override
AssistKind get assistKind => DartAssistKind.SHADOW_FIELD;
@override
Future<void> compute(ChangeBuilder builder) async {
final node = this.node;
if (node is! SimpleIdentifier) {
return;
}
var accessor = node.writeOrReadElement;
if (accessor is! PropertyAccessorElement) {
return;
}
if (!accessor.isGetter || accessor.enclosingElement is! ClassElement) {
// TODO(brianwilkerson) Should we also require that the getter be synthetic?
return;
}
var statement = _getStatement();
if (statement == null) {
return;
}
var enclosingBlock = statement.parent;
if (enclosingBlock is! Block) {
// TODO(brianwilkerson) Support adding a block between the statement and
// its parent (where the parent will be something like a while or if
// statement). Also support the case where the parent is a case clause.
return;
}
var correspondingSetter = accessor.correspondingSetter;
if (correspondingSetter == null) {
return;
}
var finder = _ReferenceFinder(correspondingSetter);
enclosingBlock.accept(finder);
if (finder.hasSetterReference) {
return;
}
var fieldName = accessor.name;
var offset = statement.offset;
var prefix = utils.getLinePrefix(offset);
//
// Build the change.
//
await builder.addDartFileEdit(file, (builder) {
builder.addInsertion(offset, (builder) {
// TODO(brianwilkerson) Conditionally write a type annotation instead of
// 'var' when we're able to discover user preferences.
// TODO(brianwilkerson) Consider writing `final` rather than `var`.
builder.write('var ');
builder.write(fieldName);
builder.write(' = this.');
builder.write(fieldName);
builder.writeln(';');
builder.write(prefix);
});
// TODO(brianwilkerson) Consider removing unnecessary casts and null
// checks that are no longer needed because promotion works. This would
// be dependent on whether enhanced promotion is supported in the library
// being edited.
});
}
/// Return the statement immediately enclosing the [node] that would promote
/// the type of the field if it were replaced by a local variable.
Statement? _getStatement() {
var parent = node.parent;
Statement? enclosingIf(Expression expression) {
var parent = expression.parent;
while (parent is BinaryExpression) {
var opType = parent.operator.type;
if (opType != TokenType.AMPERSAND_AMPERSAND) {
break;
}
parent = parent.parent;
}
if (parent is IfStatement) {
return parent;
}
return null;
}
if (parent is IsExpression && parent.expression == node) {
return enclosingIf(parent);
} else if (parent is BinaryExpression &&
resolvedResult.libraryElement.isNonNullableByDefault) {
var opType = parent.operator.type;
if (opType == TokenType.EQ_EQ || opType == TokenType.BANG_EQ) {
return enclosingIf(parent);
}
}
return null;
}
}
/// A utility that will find any references to a setter within an AST structure.
class _ReferenceFinder extends RecursiveAstVisitor<void> {
/// The setter being searched for.
final PropertyAccessorElement setter;
/// A flag indicating whether a reference to the [setter] has been found.
bool hasSetterReference = false;
/// Initialize a newly created reference finder to find references to the
/// given [setter].
_ReferenceFinder(this.setter);
@override
void visitSimpleIdentifier(SimpleIdentifier node) {
if (node.writeOrReadElement == setter) {
hasSetterReference = true;
}
}
}