
org.sonar.plugins.html.checks.accessibility.LabelHasAssociatedControlCheck Maven / Gradle / Ivy
The newest version!
/*
* SonarQube HTML
* Copyright (C) 2010-2025 SonarSource SA
* mailto:info AT sonarsource DOT com
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the Sonar Source-Available License Version 1, as published by SonarSource SA.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the Sonar Source-Available License for more details.
*
* You should have received a copy of the Sonar Source-Available License
* along with this program; if not, see https://sonarsource.com/license/ssal/
*/
package org.sonar.plugins.html.checks.accessibility;
import java.util.List;
import java.util.Locale;
import java.util.Set;
import org.sonar.check.Rule;
import org.sonar.plugins.html.checks.AbstractPageCheck;
import org.sonar.plugins.html.node.DirectiveNode;
import org.sonar.plugins.html.node.ExpressionNode;
import org.sonar.plugins.html.node.Node;
import org.sonar.plugins.html.node.TagNode;
import org.sonar.plugins.html.node.TextNode;
@Rule(key = "S6853")
public class LabelHasAssociatedControlCheck extends AbstractPageCheck {
private static final String MESSAGE = "A form label must be associated with a control.";
private static final Set CONTROL_TAGS = Set.of("INPUT", "METER", "OUTPUT", "PROGRESS", "SELECT", "TEXTAREA");
private boolean foundControl;
private boolean foundAccessibleLabel;
private TagNode label;
@Override
public void startDocument(List nodes) {
label = null;
}
@Override
public void startElement(TagNode node) {
if (isLabel(node)) {
label = node;
if (hasForAttribute(label)) {
foundControl = true;
} else {
foundControl = false;
}
} else if (isControl(node)) {
foundControl = true;
}
if (hasAccessibleLabel(node)) {
foundAccessibleLabel = true;
}
}
private static boolean hasForAttribute(TagNode label) {
return label.hasProperty("for") || label.hasProperty("htmlFor");
}
private static boolean hasAccessibleLabel(TagNode node) {
return
node.hasProperty("alt") ||
node.hasProperty("aria-labelledby") ||
node.hasProperty("aria-label") ||
// see https://sonarsource.github.io/rspec/#/rspec/S1926
"FMT:MESSAGE".equalsIgnoreCase(node.getNodeName());
}
private static boolean isLabel(TagNode node) {
return "LABEL".equalsIgnoreCase(node.getNodeName());
}
private static boolean isControl(TagNode node) {
return CONTROL_TAGS.contains(node.getNodeName().toUpperCase(Locale.ROOT));
}
@Override
public void characters(TextNode textNode) {
if (!textNode.isBlank() && label != null) {
foundAccessibleLabel = true;
}
}
@Override
public void directive(DirectiveNode node) {
if (label != null) {
foundAccessibleLabel = true;
}
}
@Override
public void expression(ExpressionNode node) {
// for JSP
if (label != null) {
foundAccessibleLabel = true;
}
}
@Override
public void endElement(TagNode node) {
if (isLabel(node)) {
if ((!foundAccessibleLabel || !foundControl) && label != null) {
createViolation(label, MESSAGE);
}
foundControl = false;
foundAccessibleLabel = false;
label = null;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy