Skip to content

Commit

Permalink
feat(linter): implement NoUnmatchableAnbSelector (#2706)
Browse files Browse the repository at this point in the history
  • Loading branch information
togami2864 authored May 4, 2024
1 parent f77ab54 commit cb0182e
Show file tree
Hide file tree
Showing 11 changed files with 458 additions and 17 deletions.
55 changes: 38 additions & 17 deletions crates/biome_configuration/src/linter/rules.rs

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 2 additions & 0 deletions crates/biome_css_analyze/src/lint/nursery.rs
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ pub mod no_important_in_keyframe;
pub mod no_unknown_function;
pub mod no_unknown_selector_pseudo_element;
pub mod no_unknown_unit;
pub mod no_unmatchable_anb_selector;
pub mod use_generic_font_names;

declare_group! {
Expand All @@ -26,6 +27,7 @@ declare_group! {
self :: no_unknown_function :: NoUnknownFunction ,
self :: no_unknown_selector_pseudo_element :: NoUnknownSelectorPseudoElement ,
self :: no_unknown_unit :: NoUnknownUnit ,
self :: no_unmatchable_anb_selector :: NoUnmatchableAnbSelector ,
self :: use_generic_font_names :: UseGenericFontNames ,
]
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
use biome_analyze::{context::RuleContext, declare_rule, Ast, Rule, RuleDiagnostic, RuleSource};
use biome_console::markup;
use biome_css_syntax::{
AnyCssPseudoClassNth, CssPseudoClassFunctionSelectorList, CssPseudoClassNthSelector,
};
use biome_rowan::{AstNode, SyntaxNodeCast};

declare_rule! {
/// Disallow unmatchable An+B selectors.
///
/// Selectors that always evaluate to 0 will not match any elements.
/// For more details about the An+B syntax, see:
/// https://www.w3.org/TR/css-syntax-3/#anb-microsyntax
///
/// ## Examples
///
/// ### Invalid
///
/// ```css,expect_diagnostic
/// a:nth-child(0) {}
/// ```
///
/// ```css,expect_diagnostic
/// a:nth-last-child(0n) {}
/// ```
///
/// ```css,expect_diagnostic
/// a:nth-of-type(0n+0) {}
/// ```
///
/// ```css,expect_diagnostic
/// a:nth-last-of-type(0 of a) {}
/// ```
///
/// ### Valid
///
/// ```css
/// a:nth-child(1) {}
/// ```
///
/// ```css
/// a:nth-last-child(1n) {}
/// ```
///
/// ```css
/// a:nth-of-type(1n+0) {}
/// ```
///
/// ```css
/// a:nth-last-of-type(1 of a) {}
/// ```
///
pub NoUnmatchableAnbSelector {
version: "next",
name: "noUnmatchableAnbSelector",
recommended: true,
sources: &[RuleSource::Stylelint("selector-anb-no-unmatchable")],
}
}

impl Rule for NoUnmatchableAnbSelector {
type Query = Ast<CssPseudoClassNthSelector>;
type State = CssPseudoClassNthSelector;
type Signals = Option<Self::State>;
type Options = ();

fn run(ctx: &RuleContext<Self>) -> Option<Self::State> {
let node = ctx.query();
let nth = node.nth().ok()?;
if is_unmatchable(&nth) && !is_within_not_pseudo_class(&nth) {
return Some(node.clone());
}
None
}

fn diagnostic(_: &RuleContext<Self>, node: &Self::State) -> Option<RuleDiagnostic> {
let span = node.range();
Some(
RuleDiagnostic::new(
rule_category!(),
span,
markup! {
"This selector will never match any elements."
},
)
.note(markup! {
"Avoid using An+B selectors that always evaluate to 0."
}).note(markup! {
"For more details, see "<Hyperlink href="https://www.w3.org/TR/css-syntax-3/#anb-microsyntax">"the official spec for An+B selectors"</Hyperlink>"."
})
)
}
}

fn is_unmatchable(nth: &AnyCssPseudoClassNth) -> bool {
match nth {
AnyCssPseudoClassNth::CssPseudoClassNthIdentifier(_) => false,
AnyCssPseudoClassNth::CssPseudoClassNth(nth) => {
let coefficient = nth.value();
let constant = nth.offset();
match (coefficient, constant) {
(Some(a), Some(b)) => a.text() == "0" && b.text() == "0",
(Some(a), None) => a.text() == "0",
_ => false,
}
}
AnyCssPseudoClassNth::CssPseudoClassNthNumber(nth) => nth.text() == "0",
}
}

// Check if the nth selector is effective within a `not` pseudo class
// Example: a:not(:nth-child(0)) returns true
// a:not(:not(:nth-child(0))) returns false
fn is_within_not_pseudo_class(node: &AnyCssPseudoClassNth) -> bool {
let number_of_not = node
.syntax()
.ancestors()
.filter_map(|n| n.cast::<CssPseudoClassFunctionSelectorList>())
.filter_map(|n| n.name().ok())
.filter(|n| n.text() == "not")
.count();
number_of_not % 2 == 1
}
1 change: 1 addition & 0 deletions crates/biome_css_analyze/src/options.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,5 +16,6 @@ pub type NoUnknownFunction =
pub type NoUnknownSelectorPseudoElement = < lint :: nursery :: no_unknown_selector_pseudo_element :: NoUnknownSelectorPseudoElement as biome_analyze :: Rule > :: Options ;
pub type NoUnknownUnit =
<lint::nursery::no_unknown_unit::NoUnknownUnit as biome_analyze::Rule>::Options;
pub type NoUnmatchableAnbSelector = < lint :: nursery :: no_unmatchable_anb_selector :: NoUnmatchableAnbSelector as biome_analyze :: Rule > :: Options ;
pub type UseGenericFontNames =
<lint::nursery::use_generic_font_names::UseGenericFontNames as biome_analyze::Rule>::Options;
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
a:nth-child(0) {}
a:nth-child(0n) {}
a:nth-child(+0n) {}
a:nth-child(-0n) {}
a:nth-child(0n+0) {}
a:nth-child(0n-0) {}
a:nth-child(-0n-0) {}
a:nth-child(0 of a) {}
a:nth-child(0), a:nth-child(1) {}
a:nth-last-child(0) {}
a:nth-of-type(0) {}
a:nth-last-of-type(0) {}
a:nth-child(0n):nth-child(-n+5) {}
a:nth-last-child(0),a:nth-last-child(n+5) ~ li {}
Loading

0 comments on commit cb0182e

Please sign in to comment.