blob: 9338b173a87334f2006c569079f0e487dedfa3c9 [file] [log] [blame]
// Copyright (c) 2022, 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/dart/abstract_producer.dart';
import 'package:analysis_server/src/services/correction/fix.dart';
import 'package:analysis_server/src/services/correction/util.dart';
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/ast/token.dart';
import 'package:analyzer/dart/element/element.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 RemoveLeadingUnderscore extends CorrectionProducer {
@override
bool get canBeAppliedInBulk => true;
@override
bool get canBeAppliedToFile => true;
@override
FixKind get fixKind => DartFixKind.REMOVE_LEADING_UNDERSCORE;
@override
FixKind get multiFixKind => DartFixKind.REMOVE_LEADING_UNDERSCORE_MULTI;
@override
Future<void> compute(ChangeBuilder builder) async {
final node = this.node;
final Token? nameToken;
final Element? element;
if (node is SimpleIdentifier) {
nameToken = node.token;
element = node.staticElement;
} else if (node is FormalParameter) {
nameToken = node.name;
element = node.declaredElement;
} else if (node is VariableDeclaration) {
nameToken = node.name;
element = node.declaredElement2;
} else {
return;
}
if (nameToken == null || element == null) {
return;
}
final oldName = nameToken.lexeme;
if (oldName.length < 2) {
return;
}
var newName = oldName.substring(1);
// Find references to the identifier.
List<SimpleIdentifier>? references;
if (element is LocalVariableElement) {
var root = node.thisOrAncestorOfType<Block>();
if (root != null) {
references = findLocalElementReferences(root, element);
}
} else if (element is ParameterElement) {
if (!element.isNamed) {
var root = node
.thisOrAncestorMatching((node) =>
node.parent is FunctionDeclaration ||
node.parent is MethodDeclaration)
?.parent;
if (root != null) {
references = findLocalElementReferences(root, element);
}
}
} else if (element is PrefixElement) {
var root = node.thisOrAncestorOfType<CompilationUnit>();
if (root != null) {
references = findPrefixElementReferences(root, element);
}
}
if (references == null) {
return;
}
// Compute the change.
final sourceRanges = {
range.token(nameToken),
...references.map(range.node),
};
await builder.addDartFileEdit(file, (builder) {
for (var sourceRange in sourceRanges) {
builder.addSimpleReplacement(sourceRange, newName);
}
});
}
}