blob: d7c68943a53025a87d177a69f0c5843188024ecf [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:_fe_analyzer_shared/src/scanner/token.dart';
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/utilities/strings.dart';
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/src/error/inheritance_override.dart';
import 'package:analyzer_plugin/protocol/protocol_common.dart' show Position;
import 'package:analyzer_plugin/utilities/change_builder/change_builder_core.dart';
import 'package:analyzer_plugin/utilities/fixes/fixes.dart';
class CreateMissingOverrides extends CorrectionProducer {
int _numElements = 0;
@override
List<Object> get fixArguments => [_numElements, _numElements == 1 ? '' : 's'];
@override
FixKind get fixKind => DartFixKind.CREATE_MISSING_OVERRIDES;
@override
Future<void> compute(ChangeBuilder builder) async {
if (node.parent is! ClassDeclaration) {
return;
}
var targetClass = node.parent as ClassDeclaration;
utils.targetClassElement = targetClass.declaredElement;
var signatures =
InheritanceOverrideVerifier.missingOverrides(targetClass).toList();
// sort by name, getters before setters
signatures.sort((ExecutableElement a, ExecutableElement b) {
var names = compareStrings(a.displayName, b.displayName);
if (names != 0) {
return names;
}
if (a.kind == ElementKind.GETTER) {
return -1;
}
return 1;
});
_numElements = signatures.length;
var location =
utils.prepareNewClassMemberLocation(targetClass, (_) => true);
if (location == null) {
return;
}
var prefix = utils.getIndent(1);
await builder.addDartFileEdit(file, (builder) {
final syntheticLeftBracket = targetClass.leftBracket.isSynthetic;
if (syntheticLeftBracket) {
var previousToLeftBracket = targetClass.leftBracket.previous!;
builder.addSimpleInsertion(previousToLeftBracket.end, ' {');
}
builder.addInsertion(location.offset, (builder) {
// Separator management.
var numOfMembersWritten = 0;
void addSeparatorBetweenDeclarations() {
if (numOfMembersWritten == 0) {
var locationPrefix = location.prefix;
if (syntheticLeftBracket && locationPrefix.startsWith(eol)) {
locationPrefix = locationPrefix.substring(eol.length);
}
builder.write(locationPrefix);
} else {
builder.write(eol); // after the previous member
builder.write(eol); // empty line separator
builder.write(prefix);
}
numOfMembersWritten++;
}
// merge getter/setter pairs into fields
for (var i = 0; i < signatures.length; i++) {
var element = signatures[i];
if (element.kind == ElementKind.GETTER && i + 1 < signatures.length) {
var nextElement = signatures[i + 1];
if (nextElement.kind == ElementKind.SETTER) {
// remove this and the next elements, adjust iterator
signatures.removeAt(i + 1);
signatures.removeAt(i);
i--;
_numElements--;
// separator
addSeparatorBetweenDeclarations();
// @override
builder.write('@override');
builder.write(eol);
// add field
builder.write(prefix);
builder.writeType(element.returnType, required: true);
builder.write(' ');
builder.write(element.name);
builder.write(';');
}
}
}
// add elements
for (var element in signatures) {
addSeparatorBetweenDeclarations();
builder.writeOverride(element);
}
builder.write(location.suffix);
if (targetClass.rightBracket.isSynthetic) {
var next = targetClass.rightBracket.next!;
if (next.type != TokenType.CLOSE_CURLY_BRACKET) {
if (!syntheticLeftBracket) {
builder.write(eol);
}
builder.write('}');
if (syntheticLeftBracket) {
builder.write(eol);
}
}
}
});
});
builder.setSelection(Position(file, location.offset));
}
}