blob: e58c795724beba309e3bb628d61de337362b56be [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:analysis_server/src/services/correction/fix.dart';
import 'package:analyzer/dart/ast/ast.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/fixes/fixes.dart';
import 'package:analyzer_plugin/utilities/range_factory.dart';
class RemoveTypeAnnotation extends CorrectionProducer {
@override
AssistKind get assistKind => DartAssistKind.REMOVE_TYPE_ANNOTATION;
@override
FixKind get fixKind => DartFixKind.REMOVE_TYPE_ANNOTATION;
@override
FixKind get multiFixKind => DartFixKind.REMOVE_TYPE_ANNOTATION_MULTI;
@override
Future<void> compute(ChangeBuilder builder) async {
for (var node = this.node; node != null; node = node.parent) {
if (node is DeclaredIdentifier) {
return _removeFromDeclaredIdentifier(builder, node);
}
if (node is SimpleFormalParameter) {
return _removeTypeAnnotation(builder, node.type);
}
if (node is TypeAnnotation && diagnostic != null) {
return _removeTypeAnnotation(builder, node);
}
if (node is VariableDeclarationList) {
return _removeFromDeclarationList(builder, node);
}
}
}
Future<void> _removeFromDeclarationList(
ChangeBuilder builder, VariableDeclarationList declarationList) async {
// we need a type
var typeNode = declarationList.type;
if (typeNode == null) {
return;
}
// ignore if an incomplete variable declaration
if (declarationList.variables.length == 1 &&
declarationList.variables[0].name.isSynthetic) {
return;
}
// must be not after the name of the variable
var firstVariable = declarationList.variables[0];
if (selectionOffset > firstVariable.name.end) {
return;
}
// The variable must have an initializer, otherwise there is no other
// source for its type.
if (firstVariable.initializer == null) {
return;
}
var keyword = declarationList.keyword;
await builder.addDartFileEdit(file, (builder) {
var typeRange = range.startStart(typeNode, firstVariable);
if (keyword != null && keyword.lexeme != 'var') {
builder.addSimpleReplacement(typeRange, '');
} else {
builder.addSimpleReplacement(typeRange, 'var ');
}
});
}
Future<void> _removeFromDeclaredIdentifier(
ChangeBuilder builder, DeclaredIdentifier declaration) async {
var typeNode = declaration.type;
if (typeNode == null) {
return;
}
var keyword = declaration.keyword;
var variableName = declaration.identifier;
await builder.addDartFileEdit(file, (builder) {
var typeRange = range.startStart(typeNode, variableName);
if (keyword != null && keyword.lexeme != 'var') {
builder.addSimpleReplacement(typeRange, '');
} else {
builder.addSimpleReplacement(typeRange, 'var ');
}
});
}
Future<void> _removeTypeAnnotation(
ChangeBuilder builder, TypeAnnotation type) async {
if (type == null) {
return;
}
await builder.addDartFileEdit(file, (builder) {
builder.addDeletion(range.startStart(type, type.endToken.next));
});
}
/// Return an instance of this class. Used as a tear-off in `FixProcessor`.
static RemoveTypeAnnotation newInstance() => RemoveTypeAnnotation();
}