blob: a4c849992def190dfd7a6be98c3204208f8b55f0 [file] [log] [blame]
// Copyright (c) 2025, 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/analysis_rule/rule_context.dart';
import 'package:analyzer/analysis_rule/rule_visitor_registry.dart';
import 'package:analyzer/dart/analysis/features.dart';
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/error.dart';
// ignore: implementation_imports
import 'package:analyzer/src/utilities/extensions/collection.dart';
import '../analyzer.dart';
import '../util/ascii_utils.dart';
const _desc = r'Unnecessary underscores can be removed.';
class UnnecessaryUnderscores extends LintRule {
UnnecessaryUnderscores()
: super(name: LintNames.unnecessary_underscores, description: _desc);
@override
DiagnosticCode get diagnosticCode => LinterLintCode.unnecessaryUnderscores;
@override
void registerNodeProcessors(
RuleVisitorRegistry registry,
RuleContext context,
) {
if (!context.isFeatureEnabled(Feature.wildcard_variables)) return;
var visitor = _Visitor(this);
registry.addFormalParameterList(this, visitor);
registry.addVariableDeclaration(this, visitor);
}
}
class _BodyVisitor extends RecursiveAstVisitor<void> {
final Set<Element> referencedElements = <Element>{};
_BodyVisitor();
@override
void visitSimpleIdentifier(SimpleIdentifier node) {
referencedElements.addIfNotNull(node.element);
}
}
class _Visitor extends SimpleAstVisitor<void> {
final LintRule rule;
_Visitor(this.rule);
@override
void visitFormalParameterList(FormalParameterList node) {
late Set<Element> referencedElements = collectReferences(node.parent);
for (var parameter in node.parameters) {
var parameterName = parameter.name;
if (parameterName == null) continue;
var element = parameter.declaredFragment?.element;
var name = element?.name;
if (isJustUnderscores(name)) {
if (!referencedElements.contains(element)) {
rule.reportAtToken(parameterName);
}
}
}
}
@override
void visitVariableDeclaration(VariableDeclaration node) {
var element = node.declaredFragment?.element;
if (element is FieldElement || element is TopLevelVariableElement) return;
if (isJustUnderscores(node.name.lexeme)) {
var parent = node.thisOrAncestorOfType<FunctionBody>();
if (!collectReferences(parent).contains(node.declaredFragment?.element)) {
rule.reportAtToken(node.name);
}
}
}
static Set<Element> collectReferences(AstNode? node) {
if (node == null) return {};
var visitor = _BodyVisitor();
node.accept(visitor);
return visitor.referencedElements;
}
static bool isJustUnderscores(String? name) =>
name != null && name.length > 1 && name.isJustUnderscores;
}