blob: a3022fa2a5d06bd2e156464b99a018059c4a56ee [file] [log] [blame] [edit]
// Copyright (c) 2023, 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:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/ast/visitor.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/error/listener.dart';
import 'package:analyzer/src/error/codes.dart';
/// Instances of the class `RedeclareVerifier` visit all of the members of any
/// extension type declarations in a compilation unit to verify that if they
/// have a redeclare annotation it is being used correctly.
class RedeclareVerifier extends RecursiveAstVisitor<void> {
/// The error reporter used to report errors.
final DiagnosticReporter _errorReporter;
/// The current extension type.
InterfaceElement? _currentExtensionType;
RedeclareVerifier(this._errorReporter);
@override
void visitExtensionTypeDeclaration(ExtensionTypeDeclaration node) {
_currentExtensionType = node.declaredFragment!.element;
super.visitExtensionTypeDeclaration(node);
_currentExtensionType = null;
}
@override
void visitMethodDeclaration(MethodDeclaration node) {
// Only check if we're in an extension type declaration.
if (_currentExtensionType == null) return;
var element = node.declaredFragment!.element;
// Static members can't redeclare.
if (element.isStatic) return;
if (element.metadata.hasRedeclare && !_redeclaresMember(element)) {
switch (element) {
case MethodElement():
_errorReporter.atToken(
node.name,
WarningCode.redeclareOnNonRedeclaringMember,
arguments: ['method'],
);
case GetterElement():
_errorReporter.atToken(
node.name,
WarningCode.redeclareOnNonRedeclaringMember,
arguments: ['getter'],
);
case SetterElement():
_errorReporter.atToken(
node.name,
WarningCode.redeclareOnNonRedeclaringMember,
arguments: ['setter'],
);
}
}
}
/// Return `true` if the [member] redeclares a member from a superinterface.
bool _redeclaresMember(ExecutableElement member) {
var currentType = _currentExtensionType;
if (currentType != null) {
var name = Name.forElement(member);
return name != null && currentType.getInheritedMember(name) != null;
}
return false;
}
}