blob: 2ba09b066c791ecbe899e84e3e091c4178b689aa [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/dart/element/type.dart';
import 'package:analyzer/source/source_range.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 ConvertToSetLiteral extends CorrectionProducer {
@override
AssistKind get assistKind => DartAssistKind.CONVERT_TO_SET_LITERAL;
@override
bool get canBeAppliedInBulk => true;
@override
bool get canBeAppliedToFile => true;
@override
FixKind get fixKind => DartFixKind.CONVERT_TO_SET_LITERAL;
@override
FixKind get multiFixKind => DartFixKind.CONVERT_TO_SET_LITERAL_MULTI;
@override
Future<void> compute(ChangeBuilder builder) async {
//
// Check whether this is an invocation of `toSet` on a list literal.
//
var invocation = _findInvocationOfToSet();
if (invocation != null) {
//
// Extract the information needed to build the edit.
//
var target = invocation.target as ListLiteral;
var hasTypeArgs = target.typeArguments != null;
var openRange = range.token(target.leftBracket);
var closeRange = range.startEnd(target.rightBracket, invocation);
//
// Build the change and return the assist.
//
await builder.addDartFileEdit(file, (builder) {
if (hasTypeArgs || _listHasUnambiguousElement(target)) {
builder.addSimpleReplacement(openRange, '{');
} else {
builder.addSimpleReplacement(openRange, '<dynamic>{');
}
builder.addSimpleReplacement(closeRange, '}');
});
return;
}
//
// Check whether this is one of the constructors defined on `Set`.
//
var creation = _findSetCreation();
if (creation != null) {
//
// Extract the information needed to build the edit.
//
var name = creation.constructorName.name;
var constructorTypeArguments =
creation.constructorName.type.typeArguments;
TypeArgumentList? elementTypeArguments;
SourceRange? elementsRange;
if (name == null) {
// Handle an invocation of the default constructor `Set()`.
} else if (name.name == 'from' || name.name == 'of') {
// Handle an invocation of the constructor `Set.from()` or `Set.of()`.
var arguments = creation.argumentList.arguments;
if (arguments.length != 1) {
return;
}
if (arguments[0] is ListLiteral) {
var elements = arguments[0] as ListLiteral;
elementTypeArguments = elements.typeArguments;
elementsRange =
range.endStart(elements.leftBracket, elements.rightBracket);
} else {
// TODO(brianwilkerson) Consider handling other iterables. Literal
// sets could be treated like lists, and arbitrary iterables by using
// a spread.
return;
}
} else {
// Invocation of an unhandled constructor.
return;
}
//
// Build the edit.
//
await builder.addDartFileEdit(file, (builder) {
builder.addReplacement(range.node(creation), (builder) {
if (constructorTypeArguments != null) {
builder.write(utils.getNodeText(constructorTypeArguments));
} else if (elementTypeArguments != null) {
builder.write(utils.getNodeText(elementTypeArguments));
} else if (!_setWouldBeInferred(creation)) {
builder.write('<dynamic>');
}
builder.write('{');
if (elementsRange != null) {
builder.write(utils.getRangeText(elementsRange));
}
builder.write('}');
});
});
}
}
/// Return the invocation of `List.toSet` that is to be converted, or `null`
/// if the cursor is not inside a invocation of `List.toSet`.
MethodInvocation? _findInvocationOfToSet() {
var invocation = node.thisOrAncestorOfType<MethodInvocation>();
if (invocation == null ||
node.offset > invocation.argumentList.offset ||
invocation.methodName.name != 'toSet' ||
invocation.target is! ListLiteral) {
return null;
}
return invocation;
}
/// Return the invocation of a `Set` constructor that is to be converted, or
/// `null` if the cursor is not inside the invocation of a constructor.
InstanceCreationExpression? _findSetCreation() {
var creation = node.thisOrAncestorOfType<InstanceCreationExpression>();
if (creation == null) {
return null;
}
if (node.offset > creation.argumentList.offset) {
return null;
}
var type = creation.staticType;
if (type is! InterfaceType) {
return null;
}
// TODO(brianwilkerson) Consider also accepting uses of LinkedHashSet.
if (type.element != typeProvider.setElement) {
return null;
}
return creation;
}
/// Return `true` if the instance [creation] contains at least one unambiguous
/// element that would cause a set to be inferred.
bool _hasUnambiguousElement(InstanceCreationExpression creation) {
var arguments = creation.argumentList.arguments;
if (arguments.isEmpty) {
return false;
}
return _listHasUnambiguousElement(arguments[0]);
}
/// Return `true` if the [element] is sufficient to lexically make the
/// enclosing literal a set literal rather than a map.
bool _isUnambiguousElement(CollectionElement? element) {
if (element is ForElement) {
return _isUnambiguousElement(element.body);
} else if (element is IfElement) {
return _isUnambiguousElement(element.thenElement) ||
_isUnambiguousElement(element.elseElement);
} else if (element is Expression) {
return true;
}
return false;
}
/// Return `true` if the given [node] is a list literal whose elements, if
/// placed inside curly braces, would lexically make the resulting literal a
/// set literal rather than a map literal.
bool _listHasUnambiguousElement(AstNode node) {
if (node is ListLiteral && node.elements.isNotEmpty) {
for (var element in node.elements) {
if (_isUnambiguousElement(element)) {
return true;
}
}
}
return false;
}
/// Return `true` if a set would be inferred if the literal replacing the
/// instance [creation] did not have explicit type arguments.
bool _setWouldBeInferred(InstanceCreationExpression creation) {
var parent = creation.parent!;
if (parent is VariableDeclaration) {
var parent2 = parent.parent;
if (parent2 is VariableDeclarationList &&
parent2.type?.type?.element == typeProvider.setElement) {
return true;
}
} else if (parent.parent is InvocationExpression) {
var parameterElement = creation.staticParameterElement;
if (parameterElement?.type.element == typeProvider.setElement) {
return true;
}
}
return _hasUnambiguousElement(creation);
}
}