Skip to content

Commit 7a43836

Browse files
authored
refactor(beasties): rewrite in typescript (#16)
1 parent edc522e commit 7a43836

12 files changed

+2072
-709
lines changed

package.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
"build": "pnpm -r build",
2828
"build:main": "pnpm --filter beasties run build",
2929
"build:webpack": "pnpm --filter beasties-webpack-plugin run build",
30+
"postinstall": "pnpm -r build:stub",
3031
"docs": "pnpm -r docs",
3132
"lint": "eslint .",
3233
"release": "bumpp && pnpm publish",
@@ -36,8 +37,9 @@
3637
"devDependencies": {
3738
"@antfu/eslint-config": "3.8.0",
3839
"@codspeed/vitest-plugin": "3.1.1",
40+
"@types/node": "18.13.0",
3941
"@vitest/coverage-v8": "2.1.3",
40-
"bumpp": "^9.7.1",
42+
"bumpp": "9.7.1",
4143
"cheerio": "1.0.0",
4244
"css-what": "6.1.0",
4345
"eslint": "9.13.0",

packages/beasties/package.json

+10-10
Original file line numberDiff line numberDiff line change
@@ -30,26 +30,26 @@
3030
"exports": {
3131
".": {
3232
"types": "./dist/index.d.ts",
33-
"import": "./dist/beasties.mjs",
34-
"require": "./dist/beasties.js",
35-
"default": "./dist/beasties.mjs"
33+
"import": "./dist/index.mjs",
34+
"require": "./dist/index.cjs",
35+
"default": "./dist/index.mjs"
3636
}
3737
},
38-
"main": "dist/beasties.js",
39-
"module": "dist/beasties.mjs",
40-
"source": "src/index.js",
38+
"main": "dist/index.cjs",
39+
"module": "dist/index.mjs",
4140
"types": "./dist/index.d.ts",
4241
"files": [
43-
"dist",
44-
"src"
42+
"dist"
4543
],
4644
"scripts": {
47-
"build": "microbundle --target node --no-sourcemap -f cjs,esm && cp src/index.d.ts dist/index.d.ts",
45+
"build": "unbuild && cp src/index.d.ts dist/index.d.ts",
46+
"build:stub": "unbuild --stub && cp src/index.d.ts dist/index.d.ts",
4847
"docs": "documentation readme src -q --no-markdown-toc -a public -s Usage --sort-order alpha",
4948
"prepack": "npm run -s build"
5049
},
5150
"dependencies": {
5251
"css-select": "^5.1.0",
52+
"css-what": "^6.1.0",
5353
"dom-serializer": "^2.0.0",
5454
"domhandler": "^5.0.3",
5555
"htmlparser2": "^9.0.0",
@@ -60,6 +60,6 @@
6060
"devDependencies": {
6161
"@types/postcss-media-query-parser": "0.2.4",
6262
"documentation": "14.0.3",
63-
"microbundle": "0.15.1"
63+
"unbuild": "^2.0.0"
6464
}
6565
}

packages/beasties/src/css.js packages/beasties/src/css.ts

+72-50
Original file line numberDiff line numberDiff line change
@@ -14,29 +14,35 @@
1414
* the License.
1515
*/
1616

17+
import type { AnyNode, ChildNode, Rule } from 'postcss'
18+
import type Root_ from 'postcss/lib/root'
1719
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'
1921

2022
/**
2123
* Parse a textual CSS Stylesheet into a Stylesheet instance.
2224
* Stylesheet is a mutable postcss AST with format similar to CSSOM.
2325
* @see https://github.com/postcss/postcss/
2426
* @private
25-
* @param {string} stylesheet
26-
* @returns {css.Stylesheet} ast
2727
*/
28-
export function parseStylesheet(stylesheet) {
28+
export function parseStylesheet(stylesheet: string) {
2929
return parse(stylesheet)
3030
}
3131

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+
3240
/**
3341
* Serialize a postcss Stylesheet to a String of CSS.
3442
* @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()`
3844
*/
39-
export function serializeStylesheet(ast, options) {
45+
export function serializeStylesheet(ast: AnyNode, options: SerializeStylesheetOptions) {
4046
let cssStr = ''
4147

4248
stringify(ast, (result, node, type) => {
@@ -61,7 +67,7 @@ export function serializeStylesheet(ast, options) {
6167
}
6268

6369
if (type === 'start') {
64-
if (node.type === 'rule' && node.selectors) {
70+
if (node?.type === 'rule' && node.selectors) {
6571
cssStr += `${node.selectors.join(',')}{`
6672
}
6773
else {
@@ -80,33 +86,36 @@ export function serializeStylesheet(ast, options) {
8086
return cssStr
8187
}
8288

89+
type SingleIterator<T> = (item: T) => boolean | void
90+
8391
/**
8492
* Converts a walkStyleRules() iterator to mark nodes with `.$$remove=true` instead of actually removing them.
8593
* This means they can be removed in a second pass, allowing the first pass to be nondestructive (eg: to preserve mirrored sheets).
8694
* @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.
8996
*/
90-
export function markOnly(predicate) {
97+
export function markOnly(predicate: SingleIterator<ChildNode | Root_>): (rule: Rule | ChildNode | Root_) => void {
9198
return (rule) => {
92-
const sel = rule.selectors
99+
const sel = 'selectors' in rule ? rule.selectors : undefined
93100
if (predicate(rule) === false) {
94101
rule.$$remove = true
95102
}
96-
rule.$$markedSelectors = rule.selectors
103+
if ('selectors' in rule) {
104+
rule.$$markedSelectors = rule.selectors
105+
rule.selectors = sel!
106+
}
97107
if (rule._other) {
98108
rule._other.$$markedSelectors = rule._other.selectors
99109
}
100-
rule.selectors = sel
101110
}
102111
}
103112

104113
/**
105114
* Apply filtered selectors to a rule from a previous markOnly run.
106115
* @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).
108117
*/
109-
export function applyMarkedSelectors(rule) {
118+
export function applyMarkedSelectors(rule: Rule) {
110119
if (rule.$$markedSelectors) {
111120
rule.selectors = rule.$$markedSelectors
112121
}
@@ -118,11 +127,14 @@ export function applyMarkedSelectors(rule) {
118127
/**
119128
* Recursively walk all rules in a stylesheet.
120129
* @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.
123132
*/
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) => {
126138
if (hasNestedRules(rule)) {
127139
walkStyleRules(rule, iterator)
128140
}
@@ -135,23 +147,23 @@ export function walkStyleRules(node, iterator) {
135147
/**
136148
* Recursively walk all rules in two identical stylesheets, filtering nodes into one or the other based on a predicate.
137149
* @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.
141153
*/
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)
144156
return walkStyleRules(node, iterator);
145157

146158
[node.nodes, node2.nodes] = splitFilter(
147159
node.nodes,
148160
node2.nodes,
149-
(rule, index, rules, rules2) => {
150-
const rule2 = rules2[index]
161+
(rule, index, _rules, rules2) => {
162+
const rule2 = rules2?.[index]
151163
if (hasNestedRules(rule)) {
152-
walkStyleRulesWithReverseMirror(rule, rule2, iterator)
164+
walkStyleRulesWithReverseMirror(rule, rule2 as Rule, iterator)
153165
}
154-
rule._other = rule2
166+
rule._other = rule2 as Rule
155167
rule.filterSelectors = filterSelectors
156168
return iterator(rule) !== false
157169
},
@@ -160,33 +172,35 @@ export function walkStyleRulesWithReverseMirror(node, node2, iterator) {
160172

161173
// Checks if a node has nested rules, like @media
162174
// @keyframes are an exception since they are evaluated as a whole
163-
function hasNestedRules(rule) {
175+
function hasNestedRules(rule: ChildNode): rule is Rule {
164176
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'))
168180
&& rule.nodes.some(n => n.type === 'rule' || n.type === 'atrule')
169181
)
170182
}
171183

172184
// Like [].filter(), but applies the opposite filtering result to a second copy of the Array without a second pass.
173185
// 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[] = []
177190
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)
180194
}
181195
else {
182-
bOut.push(a[index])
196+
bOut.push(item)
183197
}
184198
}
185-
return [aOut, bOut]
199+
return [aOut, bOut] as const
186200
}
187201

188202
// 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>) {
190204
if (this._other) {
191205
const [a, b] = splitFilter(
192206
this.selectors,
@@ -218,7 +232,7 @@ const MEDIA_FEATURES = new Set(
218232
].flatMap(feature => [feature, `min-${feature}`, `max-${feature}`]),
219233
)
220234

221-
function validateMediaType(node) {
235+
function validateMediaType(node: Child | Root) {
222236
const { type: nodeType, value: nodeValue } = node
223237
if (nodeType === 'media-type') {
224238
return MEDIA_TYPES.has(nodeValue)
@@ -232,24 +246,23 @@ function validateMediaType(node) {
232246
}
233247

234248
/**
235-
*
236-
* @param {string} Media query to validate
237-
* @returns {boolean}
238249
*
239250
* This function performs a basic media query validation
240251
* to ensure the values passed as part of the 'media' config
241252
* is HTML safe and does not cause any injection issue
253+
*
254+
* @param query Media query to validate
242255
*/
243-
export function validateMediaQuery(query) {
256+
export function validateMediaQuery(query: string): boolean {
244257
// 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
246259
const mediaTree = mediaParserFn(query)
247260
const nodeTypes = new Set(['media-type', 'keyword', 'media-feature'])
248261

249-
const stack = [mediaTree]
262+
const stack: Array<Child | Root> = [mediaTree]
250263

251264
while (stack.length > 0) {
252-
const node = stack.pop()
265+
const node = stack.pop()!
253266

254267
if (nodeTypes.has(node.type) && !validateMediaType(node)) {
255268
return false
@@ -262,3 +275,12 @@ export function validateMediaQuery(query) {
262275

263276
return true
264277
}
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

Comments
 (0)