| // 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/protocol_server.dart'; |
| 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/name_suggestion.dart'; |
| import 'package:analysis_server/src/services/linter/lint_names.dart'; |
| import 'package:analysis_server/src/utilities/extensions/ast.dart'; |
| import 'package:analyzer/dart/ast/ast.dart'; |
| import 'package:analyzer/src/dart/ast/extensions.dart'; |
| import 'package:analyzer/src/dart/ast/utilities.dart'; |
| import 'package:analyzer_plugin/utilities/assist/assist.dart'; |
| import 'package:analyzer_plugin/utilities/change_builder/change_builder_core.dart'; |
| |
| class AssignToLocalVariable extends CorrectionProducer { |
| @override |
| AssistKind get assistKind => DartAssistKind.ASSIGN_TO_LOCAL_VARIABLE; |
| |
| String get _declarationKeyword { |
| if (_isLintEnabled(LintNames.prefer_final_locals)) { |
| return 'final'; |
| } else { |
| return 'var'; |
| } |
| } |
| |
| @override |
| Future<void> compute(ChangeBuilder builder) async { |
| // prepare enclosing ExpressionStatement |
| ExpressionStatement? expressionStatement; |
| for (var node in this.node.withParents) { |
| if (node is ExpressionStatement) { |
| expressionStatement = node; |
| break; |
| } |
| if (node is ArgumentList || |
| node is AssignmentExpression || |
| node is Statement || |
| node is ThrowExpression) { |
| return; |
| } |
| } |
| if (expressionStatement == null || |
| _hasPrecedingStatementRecovery(expressionStatement)) { |
| return; |
| } |
| // prepare expression |
| var expression = expressionStatement.expression; |
| var offset = expression.offset; |
| // prepare expression type |
| var type = expression.typeOrThrow; |
| if (type.isVoid) { |
| return; |
| } |
| // prepare excluded names |
| var excluded = <String>{}; |
| var scopedNameFinder = ScopedNameFinder(offset); |
| expression.accept(scopedNameFinder); |
| excluded.addAll(scopedNameFinder.locals.keys.toSet()); |
| var suggestions = |
| getVariableNameSuggestionsForExpression(type, expression, excluded); |
| |
| if (suggestions.isNotEmpty) { |
| await builder.addDartFileEdit(file, (builder) { |
| builder.addInsertion(offset, (builder) { |
| builder.write('$_declarationKeyword '); |
| builder.addSimpleLinkedEdit('NAME', suggestions[0], |
| kind: LinkedEditSuggestionKind.VARIABLE, |
| suggestions: suggestions); |
| builder.write(' = '); |
| }); |
| }); |
| } |
| } |
| |
| bool _isLintEnabled(String name) { |
| var analysisOptions = unit.declaredElement?.context.analysisOptions; |
| return analysisOptions?.isLintEnabled(name) ?? false; |
| } |
| |
| /// Return `true` if the given [statement] resulted from a recovery case that |
| /// would make the change create even worse errors than the original code. |
| static bool _hasPrecedingStatementRecovery(Statement statement) { |
| var parent = statement.parent; |
| if (parent is Block) { |
| var statements = parent.statements; |
| var index = statements.indexOf(statement); |
| if (index > 0) { |
| var precedingStatement = statements[index - 1]; |
| if (precedingStatement is ExpressionStatement) { |
| var semicolon = precedingStatement.semicolon; |
| return semicolon != null && semicolon.isSynthetic; |
| } else if (precedingStatement is VariableDeclarationStatement && |
| precedingStatement.semicolon.isSynthetic) { |
| return true; |
| } |
| } |
| } |
| return false; |
| } |
| } |