Skip to content

Commit f369e22

Browse files
authored
Fix class extraction followed by ( in Slim (#17278)
This PR fixes an issue where using the class shorthand in Slim templates, followed by an `(` results in the last class being ignored. E.g.: ```slim body.border-t-4.p-8(class="#{body_classes}" data-hotwire-native="#{hotwire_native_app?}" data-controller="update-time-zone") ``` This is because we will eventually extract `p-8` but it's followed by an invalid boundary character `(`. To solve this, we make sure to replace the `(` with a space. We already do a similar thing when the classes are followed by an `[`. One caveat, we _can_ have `(` in our classes, like `bg-(--my-color)`. But in my testing this is not something that can be used in the shorthand version. E.g.: ```slim div.bg-(--my-color) ``` Compiles to: ```html <div --my-color="" class="bg-"></div> ``` So I didn't add any special handling for this. Even when trying to escape the `(`, `-` and `)` characters, it still doesn't work. E.g.: ```slim div.bg-\(--my-color\) ``` Compiles to: ```html <div class="bg-">\(--my-color\)</div> ``` # Test plan 1. Added test for the issue 2. Existing tests pass 3. Verified via the extractor tool: | Before | After | | --- | --- | | <img width="958" alt="image" src="https://github.com/user-attachments/assets/f72c420e-5429-424f-a01d-12f724062bf2" /> | <img width="958" alt="image" src="https://github.com/user-attachments/assets/b0cc8f2f-97a8-4fca-8813-3bb1da8d99a8" /> | --- Fixes: #17277
1 parent d698c10 commit f369e22

File tree

2 files changed

+55
-1
lines changed

2 files changed

+55
-1
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
2929
- Remove redundant `line-height: initial` from Preflight ([#15212](https://github.com/tailwindlabs/tailwindcss/pull/15212))
3030
- Prevent segfault when loaded in a worker thread on Linux ([#17276](https://github.com/tailwindlabs/tailwindcss/pull/17276))
3131
- Ensure multiple `--value(…)` or `--modifier(…)` calls don't delete subsequent declarations ([#17273](https://github.com/tailwindlabs/tailwindcss/pull/17273))
32+
- Fix class extraction followed by `(` in Slim ([#17278](https://github.com/tailwindlabs/tailwindcss/pull/17278))
3233

3334
## [4.0.14] - 2025-03-13
3435

crates/oxide/src/extractor/pre_processors/slim.rs

+54-1
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,23 @@ impl PreProcessor for Slim {
8080
bracket_stack.push(cursor.curr);
8181
}
8282

83+
// In slim the class name shorthand can be followed by a parenthesis. E.g.:
84+
//
85+
// ```slim
86+
// body.border-t-4.p-8(attr=value)
87+
// ^ Not part of the p-8 class
88+
// ```
89+
//
90+
// This means that we need to replace all these `(` and `)` with spaces to make
91+
// sure that we can extract the `p-8`.
92+
//
93+
// However, we also need to make sure that we keep the parens that are part of the
94+
// utility class. E.g.: `bg-(--my-color)`.
95+
b'(' if bracket_stack.is_empty() && !matches!(cursor.prev, b'-' | b'/') => {
96+
result[cursor.pos] = b' ';
97+
bracket_stack.push(cursor.curr);
98+
}
99+
83100
b'(' | b'[' | b'{' => {
84101
bracket_stack.push(cursor.curr);
85102
}
@@ -116,7 +133,7 @@ mod tests {
116133
" bg-red-500 2xl:flex bg-green-200 3xl:flex",
117134
),
118135
// Keep dots in strings
119-
(r#"div(class="px-2.5")"#, r#"div(class="px-2.5")"#),
136+
(r#"div(class="px-2.5")"#, r#"div class="px-2.5")"#),
120137
// Replace top-level `(a-z0-9)[` with `$1 `. E.g.: `.flex[x]` -> `.flex x]`
121138
(".text-xl.text-red-600[", " text-xl text-red-600 "),
122139
// But keep important brackets:
@@ -194,6 +211,42 @@ mod tests {
194211
Slim::test_extract_contains(input, vec!["text-red-500", "text-3xl"]);
195212
}
196213

214+
// https://github.com/tailwindlabs/tailwindcss/issues/17277
215+
#[test]
216+
fn test_class_shorthand_followed_by_parens() {
217+
let input = r#"
218+
body.border-t-4.p-8(class="\#{body_classes}" data-hotwire-native="\#{hotwire_native_app?}" data-controller="update-time-zone")
219+
"#;
220+
Slim::test_extract_contains(input, vec!["border-t-4", "p-8"]);
221+
222+
// Additional test with CSS Variable shorthand syntax in the attribute itself because `(`
223+
// and `)` are not valid in the class shorthand version.
224+
//
225+
// Also included an arbitrary value including `(` and `)` to make sure that we don't
226+
// accidentally remove those either.
227+
let input = r#"
228+
body.p-8(class="bg-(--my-color) bg-(--my-color)/(--my-opacity) bg-[url(https://example.com)]")
229+
"#;
230+
Slim::test_extract_contains(
231+
input,
232+
vec![
233+
"p-8",
234+
"bg-(--my-color)",
235+
"bg-(--my-color)/(--my-opacity)",
236+
"bg-[url(https://example.com)]",
237+
],
238+
);
239+
240+
// Top-level class shorthand with parens
241+
let input = r#"
242+
div class="bg-(--my-color) bg-(--my-color)/(--my-opacity)"
243+
"#;
244+
Slim::test_extract_contains(
245+
input,
246+
vec!["bg-(--my-color)", "bg-(--my-color)/(--my-opacity)"],
247+
);
248+
}
249+
197250
#[test]
198251
fn test_strings_only_occur_when_nested() {
199252
let input = r#"

0 commit comments

Comments
 (0)