14
14
* the License.
15
15
*/
16
16
17
+ import type { AnyNode , ChildNode , Rule } from 'postcss'
18
+ import type Root_ from 'postcss/lib/root'
17
19
import { parse , stringify } from 'postcss'
18
- import mediaParser from 'postcss-media-query-parser'
20
+ import mediaParser , { type Child , type Root } from 'postcss-media-query-parser'
19
21
20
22
/**
21
23
* Parse a textual CSS Stylesheet into a Stylesheet instance.
22
24
* Stylesheet is a mutable postcss AST with format similar to CSSOM.
23
25
* @see https://github.com/postcss/postcss/
24
26
* @private
25
- * @param {string } stylesheet
26
- * @returns {css.Stylesheet } ast
27
27
*/
28
- export function parseStylesheet ( stylesheet ) {
28
+ export function parseStylesheet ( stylesheet : string ) {
29
29
return parse ( stylesheet )
30
30
}
31
31
32
+ /**
33
+ * Options used by the stringify logic
34
+ */
35
+ interface SerializeStylesheetOptions {
36
+ /** Compress CSS output (removes comments, whitespace, etc) */
37
+ compress ?: boolean
38
+ }
39
+
32
40
/**
33
41
* Serialize a postcss Stylesheet to a String of CSS.
34
42
* @private
35
- * @param {css.Stylesheet } ast A Stylesheet to serialize, such as one returned from `parseStylesheet()`
36
- * @param {object } options Options used by the stringify logic
37
- * @param {boolean } [options.compress] Compress CSS output (removes comments, whitespace, etc)
43
+ * @param ast A Stylesheet to serialize, such as one returned from `parseStylesheet()`
38
44
*/
39
- export function serializeStylesheet ( ast , options ) {
45
+ export function serializeStylesheet ( ast : AnyNode , options : SerializeStylesheetOptions ) {
40
46
let cssStr = ''
41
47
42
48
stringify ( ast , ( result , node , type ) => {
@@ -61,7 +67,7 @@ export function serializeStylesheet(ast, options) {
61
67
}
62
68
63
69
if ( type === 'start' ) {
64
- if ( node . type === 'rule' && node . selectors ) {
70
+ if ( node ? .type === 'rule' && node . selectors ) {
65
71
cssStr += `${ node . selectors . join ( ',' ) } {`
66
72
}
67
73
else {
@@ -80,33 +86,36 @@ export function serializeStylesheet(ast, options) {
80
86
return cssStr
81
87
}
82
88
89
+ type SingleIterator < T > = ( item : T ) => boolean | void
90
+
83
91
/**
84
92
* Converts a walkStyleRules() iterator to mark nodes with `.$$remove=true` instead of actually removing them.
85
93
* This means they can be removed in a second pass, allowing the first pass to be nondestructive (eg: to preserve mirrored sheets).
86
94
* @private
87
- * @param {Function } iterator Invoked on each node in the tree. Return `false` to remove that node.
88
- * @returns {(rule) => void } nonDestructiveIterator
95
+ * @param predicate Invoked on each node in the tree. Return `false` to remove that node.
89
96
*/
90
- export function markOnly ( predicate ) {
97
+ export function markOnly ( predicate : SingleIterator < ChildNode | Root_ > ) : ( rule : Rule | ChildNode | Root_ ) => void {
91
98
return ( rule ) => {
92
- const sel = rule . selectors
99
+ const sel = 'selectors' in rule ? rule . selectors : undefined
93
100
if ( predicate ( rule ) === false ) {
94
101
rule . $$remove = true
95
102
}
96
- rule . $$markedSelectors = rule . selectors
103
+ if ( 'selectors' in rule ) {
104
+ rule . $$markedSelectors = rule . selectors
105
+ rule . selectors = sel !
106
+ }
97
107
if ( rule . _other ) {
98
108
rule . _other . $$markedSelectors = rule . _other . selectors
99
109
}
100
- rule . selectors = sel
101
110
}
102
111
}
103
112
104
113
/**
105
114
* Apply filtered selectors to a rule from a previous markOnly run.
106
115
* @private
107
- * @param { css.Rule } rule The Rule to apply marked selectors to (if they exist).
116
+ * @param rule The Rule to apply marked selectors to (if they exist).
108
117
*/
109
- export function applyMarkedSelectors ( rule ) {
118
+ export function applyMarkedSelectors ( rule : Rule ) {
110
119
if ( rule . $$markedSelectors ) {
111
120
rule . selectors = rule . $$markedSelectors
112
121
}
@@ -118,11 +127,14 @@ export function applyMarkedSelectors(rule) {
118
127
/**
119
128
* Recursively walk all rules in a stylesheet.
120
129
* @private
121
- * @param { css.Rule } node A Stylesheet or Rule to descend into.
122
- * @param { Function } iterator Invoked on each node in the tree. Return `false` to remove that node.
130
+ * @param node A Stylesheet or Rule to descend into.
131
+ * @param iterator Invoked on each node in the tree. Return `false` to remove that node.
123
132
*/
124
- export function walkStyleRules ( node , iterator ) {
125
- node . nodes = node . nodes . filter ( ( rule ) => {
133
+ export function walkStyleRules ( node : ChildNode | Root_ , iterator : SingleIterator < ChildNode | Root_ | Rule > ) {
134
+ if ( ! ( 'nodes' in node ) ) {
135
+ return
136
+ }
137
+ node . nodes = node . nodes ?. filter ( ( rule ) => {
126
138
if ( hasNestedRules ( rule ) ) {
127
139
walkStyleRules ( rule , iterator )
128
140
}
@@ -135,23 +147,23 @@ export function walkStyleRules(node, iterator) {
135
147
/**
136
148
* Recursively walk all rules in two identical stylesheets, filtering nodes into one or the other based on a predicate.
137
149
* @private
138
- * @param { css.Rule } node A Stylesheet or Rule to descend into.
139
- * @param { css.Rule } node2 A second tree identical to `node`
140
- * @param { Function } iterator Invoked on each node in the tree. Return `false` to remove that node from the first tree, true to remove it from the second.
150
+ * @param node A Stylesheet or Rule to descend into.
151
+ * @param node2 A second tree identical to `node`
152
+ * @param iterator Invoked on each node in the tree. Return `false` to remove that node from the first tree, true to remove it from the second.
141
153
*/
142
- export function walkStyleRulesWithReverseMirror ( node , node2 , iterator ) {
143
- if ( node2 === null )
154
+ export function walkStyleRulesWithReverseMirror ( node : Rule | Root_ , node2 : Rule | Root_ | undefined | null , iterator : SingleIterator < ChildNode | Root_ > ) {
155
+ if ( ! node2 )
144
156
return walkStyleRules ( node , iterator ) ;
145
157
146
158
[ node . nodes , node2 . nodes ] = splitFilter (
147
159
node . nodes ,
148
160
node2 . nodes ,
149
- ( rule , index , rules , rules2 ) => {
150
- const rule2 = rules2 [ index ]
161
+ ( rule , index , _rules , rules2 ) => {
162
+ const rule2 = rules2 ?. [ index ]
151
163
if ( hasNestedRules ( rule ) ) {
152
- walkStyleRulesWithReverseMirror ( rule , rule2 , iterator )
164
+ walkStyleRulesWithReverseMirror ( rule , rule2 as Rule , iterator )
153
165
}
154
- rule . _other = rule2
166
+ rule . _other = rule2 as Rule
155
167
rule . filterSelectors = filterSelectors
156
168
return iterator ( rule ) !== false
157
169
} ,
@@ -160,33 +172,35 @@ export function walkStyleRulesWithReverseMirror(node, node2, iterator) {
160
172
161
173
// Checks if a node has nested rules, like @media
162
174
// @keyframes are an exception since they are evaluated as a whole
163
- function hasNestedRules ( rule ) {
175
+ function hasNestedRules ( rule : ChildNode ) : rule is Rule {
164
176
return (
165
- rule . nodes ?. length
166
- && rule . name !== 'keyframes'
167
- && rule . name !== '-webkit-keyframes'
177
+ ' nodes' in rule
178
+ && ! ! rule . nodes ?. length
179
+ && ( ! ( 'name' in rule ) || ( rule . name !== 'keyframes' && rule . name !== ' -webkit-keyframes') )
168
180
&& rule . nodes . some ( n => n . type === 'rule' || n . type === 'atrule' )
169
181
)
170
182
}
171
183
172
184
// Like [].filter(), but applies the opposite filtering result to a second copy of the Array without a second pass.
173
185
// This is just a quicker version of generating the compliment of the set returned from a filter operation.
174
- function splitFilter ( a , b , predicate ) {
175
- const aOut = [ ]
176
- const bOut = [ ]
186
+ type SplitIterator < T > = ( item : T , index : number , a : T [ ] , b ?: T [ ] ) => boolean
187
+ function splitFilter < T > ( a : T [ ] , b : T [ ] , predicate : SplitIterator < T > ) {
188
+ const aOut : T [ ] = [ ]
189
+ const bOut : T [ ] = [ ]
177
190
for ( let index = 0 ; index < a . length ; index ++ ) {
178
- if ( predicate ( a [ index ] , index , a , b ) ) {
179
- aOut . push ( a [ index ] )
191
+ const item = a [ index ] !
192
+ if ( predicate ( item , index , a , b ) ) {
193
+ aOut . push ( item )
180
194
}
181
195
else {
182
- bOut . push ( a [ index ] )
196
+ bOut . push ( item )
183
197
}
184
198
}
185
- return [ aOut , bOut ]
199
+ return [ aOut , bOut ] as const
186
200
}
187
201
188
202
// can be invoked on a style rule to subset its selectors (with reverse mirroring)
189
- function filterSelectors ( predicate ) {
203
+ function filterSelectors ( this : Rule , predicate : SplitIterator < string > ) {
190
204
if ( this . _other ) {
191
205
const [ a , b ] = splitFilter (
192
206
this . selectors ,
@@ -218,7 +232,7 @@ const MEDIA_FEATURES = new Set(
218
232
] . flatMap ( feature => [ feature , `min-${ feature } ` , `max-${ feature } ` ] ) ,
219
233
)
220
234
221
- function validateMediaType ( node ) {
235
+ function validateMediaType ( node : Child | Root ) {
222
236
const { type : nodeType , value : nodeValue } = node
223
237
if ( nodeType === 'media-type' ) {
224
238
return MEDIA_TYPES . has ( nodeValue )
@@ -232,24 +246,23 @@ function validateMediaType(node) {
232
246
}
233
247
234
248
/**
235
- *
236
- * @param {string } Media query to validate
237
- * @returns {boolean }
238
249
*
239
250
* This function performs a basic media query validation
240
251
* to ensure the values passed as part of the 'media' config
241
252
* is HTML safe and does not cause any injection issue
253
+ *
254
+ * @param query Media query to validate
242
255
*/
243
- export function validateMediaQuery ( query ) {
256
+ export function validateMediaQuery ( query : string ) : boolean {
244
257
// The below is needed for consumption with webpack.
245
- const mediaParserFn = 'default' in mediaParser ? mediaParser . default : mediaParser
258
+ const mediaParserFn = 'default' in mediaParser ? mediaParser . default as unknown as typeof mediaParser : mediaParser
246
259
const mediaTree = mediaParserFn ( query )
247
260
const nodeTypes = new Set ( [ 'media-type' , 'keyword' , 'media-feature' ] )
248
261
249
- const stack = [ mediaTree ]
262
+ const stack : Array < Child | Root > = [ mediaTree ]
250
263
251
264
while ( stack . length > 0 ) {
252
- const node = stack . pop ( )
265
+ const node = stack . pop ( ) !
253
266
254
267
if ( nodeTypes . has ( node . type ) && ! validateMediaType ( node ) ) {
255
268
return false
@@ -262,3 +275,12 @@ export function validateMediaQuery(query) {
262
275
263
276
return true
264
277
}
278
+
279
+ declare module 'postcss' {
280
+ interface Node {
281
+ _other ?: Rule
282
+ $$remove ?: boolean
283
+ $$markedSelectors ?: string [ ]
284
+ filterSelectors ?: typeof filterSelectors
285
+ }
286
+ }
0 commit comments