@pathscale/rebuild-plugin-ui-css-purge 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1 @@
1
+ # bun-plugin-ui-css-purge
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Lib-side database generator.
3
+ *
4
+ * Reads all `*.classnames.ts` files from @pathscale/ui's component tree
5
+ * and produces a `purge-manifest.json` that the consumer-side plugin uses
6
+ * to build safelists.
7
+ *
8
+ * Usage: bun run src/generate-manifest.ts <path-to-ui-src/components>
9
+ * Output: purge-manifest.json in cwd (or pass --out <path>)
10
+ */
11
+ export {};
@@ -0,0 +1,2 @@
1
+ export { pluginCssPurge } from "./rsbuild-plugin";
2
+ export type { CssPurgeOptions } from "./rsbuild-plugin";
package/dist/index.js ADDED
@@ -0,0 +1,337 @@
1
+ // src/rsbuild-plugin.ts
2
+ import { PurgeCSS } from "purgecss";
3
+ import postcss from "postcss";
4
+ import swc from "@swc/core";
5
+ import path from "path";
6
+ import { readFile as nodeReadFile } from "node:fs/promises";
7
+ import { glob as fastGlob } from "fast-glob";
8
+ function walkAST(node, visitor) {
9
+ if (!node || typeof node !== "object")
10
+ return;
11
+ visitor(node);
12
+ for (const key of Object.keys(node)) {
13
+ if (key === "span")
14
+ continue;
15
+ const val = node[key];
16
+ if (Array.isArray(val)) {
17
+ for (const item of val)
18
+ walkAST(item, visitor);
19
+ } else if (val && typeof val === "object") {
20
+ walkAST(val, visitor);
21
+ }
22
+ }
23
+ }
24
+ function extractUIImports(ast) {
25
+ const imports = new Map;
26
+ for (const node of ast.body) {
27
+ if (node.type !== "ImportDeclaration")
28
+ continue;
29
+ const src = node.source?.value;
30
+ if (!src || !src.startsWith("@pathscale/ui"))
31
+ continue;
32
+ for (const spec of node.specifiers) {
33
+ if (spec.type === "ImportSpecifier") {
34
+ const imported = spec.imported?.value ?? spec.local?.value;
35
+ const local = spec.local?.value;
36
+ if (local && imported)
37
+ imports.set(local, imported);
38
+ } else if (spec.type === "ImportDefaultSpecifier") {
39
+ const local = spec.local?.value;
40
+ if (local)
41
+ imports.set(local, local);
42
+ }
43
+ }
44
+ }
45
+ return imports;
46
+ }
47
+ function extractJSXUsages(ast, uiComponents) {
48
+ const usages = [];
49
+ walkAST(ast, (node) => {
50
+ if (node.type !== "JSXOpeningElement")
51
+ return;
52
+ let elementName = null;
53
+ let rootName = null;
54
+ if (node.name?.type === "Identifier") {
55
+ elementName = node.name.value;
56
+ rootName = elementName;
57
+ } else if (node.name?.type === "JSXMemberExpression") {
58
+ const parts = [];
59
+ let cursor = node.name;
60
+ while (cursor?.type === "JSXMemberExpression") {
61
+ parts.unshift(cursor.property?.value);
62
+ cursor = cursor.object;
63
+ }
64
+ if (cursor?.type === "Identifier") {
65
+ parts.unshift(cursor.value);
66
+ rootName = cursor.value;
67
+ }
68
+ elementName = parts.join(".");
69
+ }
70
+ if (!rootName || !uiComponents.has(rootName))
71
+ return;
72
+ const usage = {
73
+ component: elementName,
74
+ props: new Map,
75
+ booleanProps: new Set,
76
+ hasSpread: false
77
+ };
78
+ for (const attr of node.attributes || []) {
79
+ if (attr.type === "SpreadElement" || attr.type === "JSXSpreadAttribute") {
80
+ usage.hasSpread = true;
81
+ continue;
82
+ }
83
+ if (attr.type !== "JSXAttribute")
84
+ continue;
85
+ const propName = attr.name?.value;
86
+ if (!propName)
87
+ continue;
88
+ if (!attr.value) {
89
+ usage.booleanProps.add(propName);
90
+ } else if (attr.value.type === "StringLiteral") {
91
+ usage.props.set(propName, attr.value.value);
92
+ } else {
93
+ usage.props.set(propName, "DYNAMIC");
94
+ }
95
+ }
96
+ usages.push(usage);
97
+ });
98
+ return usages;
99
+ }
100
+ function buildSafelists(allUsages, manifest) {
101
+ const classSafelist = new Set;
102
+ const attrSafelist = new Set;
103
+ const componentUsages = new Map;
104
+ for (const usage of allUsages) {
105
+ const existing = componentUsages.get(usage.component) ?? [];
106
+ existing.push(usage);
107
+ componentUsages.set(usage.component, existing);
108
+ }
109
+ for (const [entryName, entry] of Object.entries(manifest)) {
110
+ const matchingUsages = findMatchingUsages(entryName, componentUsages);
111
+ if (matchingUsages.length === 0)
112
+ continue;
113
+ for (const cls of entry.classes.always)
114
+ classSafelist.add(cls);
115
+ for (const [propOrSlot, value] of Object.entries(entry.classes.byProp)) {
116
+ if (Array.isArray(value)) {
117
+ if (isPropUsed(propOrSlot, matchingUsages)) {
118
+ for (const cls of value)
119
+ classSafelist.add(cls);
120
+ }
121
+ } else {
122
+ const usedValues = getUsedEnumValues(propOrSlot, matchingUsages);
123
+ if (usedValues === "ALL") {
124
+ for (const classes of Object.values(value)) {
125
+ for (const cls of classes)
126
+ classSafelist.add(cls);
127
+ }
128
+ } else {
129
+ for (const val of usedValues) {
130
+ if (value[val]) {
131
+ for (const cls of value[val])
132
+ classSafelist.add(cls);
133
+ }
134
+ }
135
+ }
136
+ }
137
+ }
138
+ if (entry.attrs) {
139
+ for (const [propName, attrMap] of Object.entries(entry.attrs)) {
140
+ if (isPropUsed(propName, matchingUsages)) {
141
+ for (const [attr, val] of Object.entries(attrMap)) {
142
+ attrSafelist.add(`[${attr}="${val}"]`);
143
+ }
144
+ }
145
+ }
146
+ }
147
+ }
148
+ return { classSafelist, attrSafelist };
149
+ }
150
+ function findMatchingUsages(entryName, usageMap) {
151
+ if (usageMap.has(entryName))
152
+ return usageMap.get(entryName);
153
+ const results = [];
154
+ for (const [usageName, usages] of usageMap) {
155
+ if (usageName === entryName) {
156
+ results.push(...usages);
157
+ }
158
+ if (entryName.includes(".")) {
159
+ const [family, part] = entryName.split(".");
160
+ if (part === family && usageName === family) {
161
+ results.push(...usages);
162
+ }
163
+ }
164
+ }
165
+ return results;
166
+ }
167
+ function isPropUsed(propName, usages) {
168
+ for (const usage of usages) {
169
+ if (usage.hasSpread)
170
+ return true;
171
+ if (usage.booleanProps.has(propName))
172
+ return true;
173
+ if (usage.props.has(propName))
174
+ return true;
175
+ }
176
+ return false;
177
+ }
178
+ function getUsedEnumValues(slotName, usages) {
179
+ const values = new Set;
180
+ for (const usage of usages) {
181
+ if (usage.hasSpread)
182
+ return "ALL";
183
+ const val = usage.props.get(slotName);
184
+ if (val === "DYNAMIC")
185
+ return "ALL";
186
+ if (val !== undefined)
187
+ values.add(val);
188
+ if (usage.booleanProps.has(slotName))
189
+ return "ALL";
190
+ }
191
+ return values;
192
+ }
193
+ async function scanConsumerSource(srcDir) {
194
+ const allUsages = [];
195
+ const files = await fastGlob("**/*.{tsx,ts,jsx,js}", { cwd: srcDir, ignore: ["**/node_modules/**"] });
196
+ for (const relPath of files) {
197
+ const fullPath = path.join(srcDir, relPath);
198
+ const code = await nodeReadFile(fullPath, "utf-8");
199
+ if (!code.includes("@pathscale/ui"))
200
+ continue;
201
+ const isTsx = /\.[tj]sx$/.test(relPath);
202
+ const ast = await swc.parse(code, { syntax: "typescript", tsx: isTsx });
203
+ const uiImports = extractUIImports(ast);
204
+ if (uiImports.size === 0)
205
+ continue;
206
+ allUsages.push(...extractJSXUsages(ast, uiImports));
207
+ }
208
+ return allUsages;
209
+ }
210
+ function purgeAttributes(css, attrSafelist) {
211
+ const root = postcss.parse(css);
212
+ root.walkRules((rule) => {
213
+ const selectors = rule.selectors;
214
+ const kept = [];
215
+ for (const sel of selectors) {
216
+ const attrMatches = sel.matchAll(/\[(data-[a-z-]+|aria-[a-z-]+)="([^"]+)"\]/g);
217
+ let shouldKeep = true;
218
+ for (const match of attrMatches) {
219
+ const attrSelector = `[${match[1]}="${match[2]}"]`;
220
+ if (match[1] === "data-slot")
221
+ continue;
222
+ if (!attrSafelist.has(attrSelector)) {
223
+ shouldKeep = false;
224
+ break;
225
+ }
226
+ }
227
+ if (shouldKeep)
228
+ kept.push(sel);
229
+ }
230
+ if (kept.length === 0) {
231
+ rule.remove();
232
+ } else if (kept.length < selectors.length) {
233
+ rule.selectors = kept;
234
+ }
235
+ });
236
+ return root.toString();
237
+ }
238
+ function cleanUnusedVars(css) {
239
+ let changed = true;
240
+ let result = css;
241
+ while (changed) {
242
+ changed = false;
243
+ const root = postcss.parse(result);
244
+ const declared = new Map;
245
+ root.walkDecls(/^--/, (decl) => {
246
+ const entries = declared.get(decl.prop) ?? [];
247
+ entries.push({ rule: decl.parent, prop: decl.prop, index: entries.length });
248
+ declared.set(decl.prop, entries);
249
+ });
250
+ const referenced = new Set;
251
+ root.walkDecls((decl) => {
252
+ const refs = decl.value.matchAll(/var\(\s*(--[a-zA-Z0-9_-]+)/g);
253
+ for (const ref of refs) {
254
+ referenced.add(ref[1]);
255
+ }
256
+ });
257
+ for (const [varName, entries] of declared) {
258
+ if (!referenced.has(varName)) {
259
+ for (const entry of entries) {
260
+ entry.rule.walkDecls(entry.prop, (decl) => {
261
+ decl.remove();
262
+ changed = true;
263
+ });
264
+ }
265
+ }
266
+ }
267
+ root.walkRules((rule) => {
268
+ if (rule.nodes && rule.nodes.length === 0)
269
+ rule.remove();
270
+ });
271
+ root.walkAtRules((atRule) => {
272
+ if (atRule.nodes && atRule.nodes.length === 0)
273
+ atRule.remove();
274
+ });
275
+ result = root.toString();
276
+ }
277
+ return result;
278
+ }
279
+ var pluginCssPurge = (options) => ({
280
+ name: "plugin-css-purge",
281
+ setup(api) {
282
+ const {
283
+ manifest: manifestPath,
284
+ srcDir = "src",
285
+ attrPurge = true,
286
+ cleanVars = true,
287
+ verbose = true
288
+ } = options;
289
+ api.processAssets({ stage: "optimize-size" }, async ({ assets, sources, compilation }) => {
290
+ const log = verbose ? console.log.bind(console) : () => {};
291
+ const manifest = JSON.parse(await nodeReadFile(path.resolve(manifestPath), "utf-8"));
292
+ log(`[css-purge] Manifest loaded: ${Object.keys(manifest).length} entries`);
293
+ const resolvedSrc = path.resolve(srcDir);
294
+ const usages = await scanConsumerSource(resolvedSrc);
295
+ log(`[css-purge] Scanned ${resolvedSrc}: ${usages.length} component usages`);
296
+ const { classSafelist, attrSafelist } = buildSafelists(usages, manifest);
297
+ log(`[css-purge] Safelist: ${classSafelist.size} classes, ${attrSafelist.size} attrs`);
298
+ for (const [name, asset] of Object.entries(assets)) {
299
+ if (!name.endsWith(".css"))
300
+ continue;
301
+ const originalCss = asset.source().toString();
302
+ const originalSize = Buffer.byteLength(originalCss, "utf-8");
303
+ log(`[css-purge] Processing ${name} (${(originalSize / 1024).toFixed(1)} KB)`);
304
+ const purgeResult = await new PurgeCSS().purge({
305
+ content: [],
306
+ css: [{ raw: originalCss }],
307
+ safelist: [...classSafelist],
308
+ keyframes: false,
309
+ fontFace: false
310
+ });
311
+ let purgedCss = purgeResult[0]?.css ?? originalCss;
312
+ const afterL1 = Buffer.byteLength(purgedCss, "utf-8");
313
+ log(`[css-purge] L1 class purge: ${(originalSize / 1024).toFixed(1)} → ${(afterL1 / 1024).toFixed(1)} KB`);
314
+ if (attrPurge && attrSafelist.size > 0) {
315
+ purgedCss = purgeAttributes(purgedCss, attrSafelist);
316
+ const afterL2 = Buffer.byteLength(purgedCss, "utf-8");
317
+ log(`[css-purge] L2 attr purge: ${(afterL1 / 1024).toFixed(1)} → ${(afterL2 / 1024).toFixed(1)} KB`);
318
+ }
319
+ if (cleanVars) {
320
+ purgedCss = cleanUnusedVars(purgedCss);
321
+ const afterL3 = Buffer.byteLength(purgedCss, "utf-8");
322
+ log(`[css-purge] L3 var cleanup: → ${(afterL3 / 1024).toFixed(1)} KB`);
323
+ }
324
+ const finalSize = Buffer.byteLength(purgedCss, "utf-8");
325
+ log(`[css-purge] Final: ${(originalSize / 1024).toFixed(1)} → ${(finalSize / 1024).toFixed(1)} KB (${((1 - finalSize / originalSize) * 100).toFixed(1)}% reduction)`);
326
+ const source = new sources.RawSource(purgedCss);
327
+ compilation.updateAsset(name, source);
328
+ }
329
+ });
330
+ }
331
+ });
332
+ export {
333
+ pluginCssPurge
334
+ };
335
+
336
+ //# debugId=CB5B2093063663E864756E2164756E21
337
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/rsbuild-plugin.ts"],
  "sourcesContent": [
    "/**\n * rsbuild-plugin-css-purge\n *\n * Two-level CSS purge for @pathscale/ui consumers.\n *\n * Level 1: class-level purge via purgecss — removes entire rules whose selectors\n *          don't match the safelist built from consumer JSX analysis.\n * Level 2: attribute-level purge — within kept rules, strips compound selectors\n *          containing data-attr / aria-attr attribute selectors not in the attr safelist.\n *\n * Usage in rsbuild.config.ts:\n *   import { pluginCssPurge } from \"@pathscale/rebuild-plugin-ui-css-purge\";\n *   export default defineConfig({ plugins: [pluginCssPurge({ manifest: \"...\" })] });\n */\n\nimport type { RsbuildPlugin } from \"@rsbuild/core\";\nimport { PurgeCSS } from \"purgecss\";\nimport postcss from \"postcss\";\nimport type { Rule, AtRule } from \"postcss\";\nimport swc from \"@swc/core\";\nimport path from \"path\";\nimport { readFile as nodeReadFile } from \"node:fs/promises\";\nimport { glob as fastGlob } from \"fast-glob\";\n\n// ── Types ─────────────────────────────────────────────────────────────────────\n\ninterface ComponentManifest {\n  classes: {\n    always: string[];\n    byProp: Record<string, string[] | Record<string, string[]>>;\n  };\n  attrs?: Record<string, Record<string, string>>;\n}\n\ntype PurgeManifest = Record<string, ComponentManifest>;\n\ninterface PropUsage {\n  component: string;\n  props: Map<string, string | \"DYNAMIC\">;\n  booleanProps: Set<string>;\n  hasSpread: boolean;\n}\n\n// ── Plugin options ─────────────────────────────────────────────────────────────\n\nexport interface CssPurgeOptions {\n  /** Path to purge-manifest.json (generated by generate-manifest.ts) */\n  manifest: string;\n  /** Consumer source directory to scan for JSX usage (default: \"src\") */\n  srcDir?: string;\n  /** Enable Level 2 attribute purge (default: true) */\n  attrPurge?: boolean;\n  /** Enable CSS variable cleanup (default: true) */\n  cleanVars?: boolean;\n  /** Log purge stats (default: true) */\n  verbose?: boolean;\n}\n\n// ── AST utilities ─────────────────────────────────────────────────────────────\n\nfunction walkAST(node: any, visitor: (node: any) => void) {\n  if (!node || typeof node !== \"object\") return;\n  visitor(node);\n  for (const key of Object.keys(node)) {\n    if (key === \"span\") continue;\n    const val = node[key];\n    if (Array.isArray(val)) {\n      for (const item of val) walkAST(item, visitor);\n    } else if (val && typeof val === \"object\") {\n      walkAST(val, visitor);\n    }\n  }\n}\n\nfunction extractUIImports(ast: any): Map<string, string> {\n  const imports = new Map<string, string>();\n  for (const node of ast.body) {\n    if (node.type !== \"ImportDeclaration\") continue;\n    const src = node.source?.value as string;\n    if (!src || !src.startsWith(\"@pathscale/ui\")) continue;\n    for (const spec of node.specifiers) {\n      if (spec.type === \"ImportSpecifier\") {\n        const imported = spec.imported?.value ?? spec.local?.value;\n        const local = spec.local?.value;\n        if (local && imported) imports.set(local, imported);\n      } else if (spec.type === \"ImportDefaultSpecifier\") {\n        const local = spec.local?.value;\n        if (local) imports.set(local, local);\n      }\n    }\n  }\n  return imports;\n}\n\nfunction extractJSXUsages(ast: any, uiComponents: Map<string, string>): PropUsage[] {\n  const usages: PropUsage[] = [];\n  walkAST(ast, (node) => {\n    if (node.type !== \"JSXOpeningElement\") return;\n    let elementName: string | null = null;\n    let rootName: string | null = null;\n    if (node.name?.type === \"Identifier\") {\n      elementName = node.name.value;\n      rootName = elementName;\n    } else if (node.name?.type === \"JSXMemberExpression\") {\n      const parts: string[] = [];\n      let cursor = node.name;\n      while (cursor?.type === \"JSXMemberExpression\") {\n        parts.unshift(cursor.property?.value);\n        cursor = cursor.object;\n      }\n      if (cursor?.type === \"Identifier\") {\n        parts.unshift(cursor.value);\n        rootName = cursor.value;\n      }\n      elementName = parts.join(\".\");\n    }\n    if (!rootName || !uiComponents.has(rootName)) return;\n    const usage: PropUsage = {\n      component: elementName!,\n      props: new Map(),\n      booleanProps: new Set(),\n      hasSpread: false,\n    };\n    for (const attr of node.attributes || []) {\n      if (attr.type === \"SpreadElement\" || attr.type === \"JSXSpreadAttribute\") {\n        usage.hasSpread = true;\n        continue;\n      }\n      if (attr.type !== \"JSXAttribute\") continue;\n      const propName = attr.name?.value;\n      if (!propName) continue;\n      if (!attr.value) {\n        usage.booleanProps.add(propName);\n      } else if (attr.value.type === \"StringLiteral\") {\n        usage.props.set(propName, attr.value.value);\n      } else {\n        usage.props.set(propName, \"DYNAMIC\");\n      }\n    }\n    usages.push(usage);\n  });\n  return usages;\n}\n\n// ── Safelist builder ───────────────────────────────────────────────────────────\n\nfunction buildSafelists(\n  allUsages: PropUsage[],\n  manifest: PurgeManifest,\n): { classSafelist: Set<string>; attrSafelist: Set<string> } {\n  const classSafelist = new Set<string>();\n  const attrSafelist = new Set<string>();\n\n  const componentUsages = new Map<string, PropUsage[]>();\n  for (const usage of allUsages) {\n    const existing = componentUsages.get(usage.component) ?? [];\n    existing.push(usage);\n    componentUsages.set(usage.component, existing);\n  }\n\n  for (const [entryName, entry] of Object.entries(manifest)) {\n    const matchingUsages = findMatchingUsages(entryName, componentUsages);\n    if (matchingUsages.length === 0) continue;\n\n    for (const cls of entry.classes.always) classSafelist.add(cls);\n\n    for (const [propOrSlot, value] of Object.entries(entry.classes.byProp)) {\n      if (Array.isArray(value)) {\n        if (isPropUsed(propOrSlot, matchingUsages)) {\n          for (const cls of value) classSafelist.add(cls);\n        }\n      } else {\n        const usedValues = getUsedEnumValues(propOrSlot, matchingUsages);\n        if (usedValues === \"ALL\") {\n          for (const classes of Object.values(value)) {\n            for (const cls of classes) classSafelist.add(cls);\n          }\n        } else {\n          for (const val of usedValues) {\n            if (value[val]) {\n              for (const cls of value[val]) classSafelist.add(cls);\n            }\n          }\n        }\n      }\n    }\n\n    if (entry.attrs) {\n      for (const [propName, attrMap] of Object.entries(entry.attrs)) {\n        if (isPropUsed(propName, matchingUsages)) {\n          for (const [attr, val] of Object.entries(attrMap)) {\n            attrSafelist.add(`[${attr}=\"${val}\"]`);\n          }\n        }\n      }\n    }\n  }\n\n  return { classSafelist, attrSafelist };\n}\n\nfunction findMatchingUsages(entryName: string, usageMap: Map<string, PropUsage[]>): PropUsage[] {\n  if (usageMap.has(entryName)) return usageMap.get(entryName)!;\n  const results: PropUsage[] = [];\n  for (const [usageName, usages] of usageMap) {\n    if (usageName === entryName) {\n      results.push(...usages);\n    }\n    if (entryName.includes(\".\")) {\n      const [family, part] = entryName.split(\".\");\n      if (part === family && usageName === family) {\n        results.push(...usages);\n      }\n    }\n  }\n  return results;\n}\n\nfunction isPropUsed(propName: string, usages: PropUsage[]): boolean {\n  for (const usage of usages) {\n    if (usage.hasSpread) return true;\n    if (usage.booleanProps.has(propName)) return true;\n    if (usage.props.has(propName)) return true;\n  }\n  return false;\n}\n\nfunction getUsedEnumValues(slotName: string, usages: PropUsage[]): Set<string> | \"ALL\" {\n  const values = new Set<string>();\n  for (const usage of usages) {\n    if (usage.hasSpread) return \"ALL\";\n    const val = usage.props.get(slotName);\n    if (val === \"DYNAMIC\") return \"ALL\";\n    if (val !== undefined) values.add(val);\n    if (usage.booleanProps.has(slotName)) return \"ALL\";\n  }\n  return values;\n}\n\n// ── Consumer source scanning ───────────────────────────────────────────────────\n\nasync function scanConsumerSource(srcDir: string): Promise<PropUsage[]> {\n  const allUsages: PropUsage[] = [];\n  const files = await fastGlob(\"**/*.{tsx,ts,jsx,js}\", { cwd: srcDir, ignore: [\"**/node_modules/**\"] });\n\n  for (const relPath of files) {\n    const fullPath = path.join(srcDir, relPath);\n    const code = await nodeReadFile(fullPath, \"utf-8\");\n    if (!code.includes(\"@pathscale/ui\")) continue;\n\n    const isTsx = /\\.[tj]sx$/.test(relPath);\n    const ast = await swc.parse(code, { syntax: \"typescript\", tsx: isTsx });\n    const uiImports = extractUIImports(ast);\n    if (uiImports.size === 0) continue;\n\n    allUsages.push(...extractJSXUsages(ast, uiImports));\n  }\n\n  return allUsages;\n}\n\n// ── Level 2: attribute purge ───────────────────────────────────────────────────\n\nfunction purgeAttributes(css: string, attrSafelist: Set<string>): string {\n  const root = postcss.parse(css);\n\n  root.walkRules((rule) => {\n    const selectors = rule.selectors;\n    const kept: string[] = [];\n\n    for (const sel of selectors) {\n      const attrMatches = sel.matchAll(/\\[(data-[a-z-]+|aria-[a-z-]+)=\"([^\"]+)\"\\]/g);\n      let shouldKeep = true;\n\n      for (const match of attrMatches) {\n        const attrSelector = `[${match[1]}=\"${match[2]}\"]`;\n        if (match[1] === \"data-slot\") continue;\n        if (!attrSafelist.has(attrSelector)) {\n          shouldKeep = false;\n          break;\n        }\n      }\n\n      if (shouldKeep) kept.push(sel);\n    }\n\n    if (kept.length === 0) {\n      rule.remove();\n    } else if (kept.length < selectors.length) {\n      rule.selectors = kept;\n    }\n  });\n\n  return root.toString();\n}\n\n// ── Level 3: unused CSS variable cleanup ───────────────────────────────────────\n\nfunction cleanUnusedVars(css: string): string {\n  let changed = true;\n  let result = css;\n\n  while (changed) {\n    changed = false;\n    const root = postcss.parse(result);\n\n    const declared = new Map<string, { rule: Rule | AtRule; prop: string; index: number }[]>();\n    root.walkDecls(/^--/, (decl) => {\n      const entries = declared.get(decl.prop) ?? [];\n      entries.push({ rule: decl.parent as Rule, prop: decl.prop, index: entries.length });\n      declared.set(decl.prop, entries);\n    });\n\n    const referenced = new Set<string>();\n    root.walkDecls((decl) => {\n      const refs = decl.value.matchAll(/var\\(\\s*(--[a-zA-Z0-9_-]+)/g);\n      for (const ref of refs) {\n        referenced.add(ref[1]);\n      }\n    });\n\n    for (const [varName, entries] of declared) {\n      if (!referenced.has(varName)) {\n        for (const entry of entries) {\n          entry.rule.walkDecls(entry.prop, (decl) => {\n            decl.remove();\n            changed = true;\n          });\n        }\n      }\n    }\n\n    root.walkRules((rule) => {\n      if (rule.nodes && rule.nodes.length === 0) rule.remove();\n    });\n    root.walkAtRules((atRule) => {\n      if (atRule.nodes && atRule.nodes.length === 0) atRule.remove();\n    });\n\n    result = root.toString();\n  }\n\n  return result;\n}\n\n// ── The rsbuild plugin ─────────────────────────────────────────────────────────\n\nexport const pluginCssPurge = (options: CssPurgeOptions): RsbuildPlugin => ({\n  name: \"plugin-css-purge\",\n\n  setup(api) {\n    const {\n      manifest: manifestPath,\n      srcDir = \"src\",\n      attrPurge = true,\n      cleanVars = true,\n      verbose = true,\n    } = options;\n\n    api.processAssets(\n      { stage: \"optimize-size\" },\n      async ({ assets, sources, compilation }) => {\n        const log = verbose ? console.log.bind(console) : () => {};\n\n        // 1. Load manifest\n        const manifest: PurgeManifest = JSON.parse(\n          await nodeReadFile(path.resolve(manifestPath), \"utf-8\"),\n        );\n        log(`[css-purge] Manifest loaded: ${Object.keys(manifest).length} entries`);\n\n        // 2. Scan consumer source\n        const resolvedSrc = path.resolve(srcDir);\n        const usages = await scanConsumerSource(resolvedSrc);\n        log(`[css-purge] Scanned ${resolvedSrc}: ${usages.length} component usages`);\n\n        // 3. Build safelists\n        const { classSafelist, attrSafelist } = buildSafelists(usages, manifest);\n        log(`[css-purge] Safelist: ${classSafelist.size} classes, ${attrSafelist.size} attrs`);\n\n        // 4. Process each CSS asset\n        for (const [name, asset] of Object.entries(assets)) {\n          if (!name.endsWith(\".css\")) continue;\n\n          const originalCss = asset.source().toString();\n          const originalSize = Buffer.byteLength(originalCss, \"utf-8\");\n          log(`[css-purge] Processing ${name} (${(originalSize / 1024).toFixed(1)} KB)`);\n\n          // Level 1: class-level purge\n          const purgeResult = await new PurgeCSS().purge({\n            content: [],\n            css: [{ raw: originalCss }],\n            safelist: [...classSafelist],\n            keyframes: false,\n            fontFace: false,\n          });\n\n          let purgedCss = purgeResult[0]?.css ?? originalCss;\n          const afterL1 = Buffer.byteLength(purgedCss, \"utf-8\");\n          log(`[css-purge]   L1 class purge: ${(originalSize / 1024).toFixed(1)} → ${(afterL1 / 1024).toFixed(1)} KB`);\n\n          // Level 2: attribute-level purge\n          if (attrPurge && attrSafelist.size > 0) {\n            purgedCss = purgeAttributes(purgedCss, attrSafelist);\n            const afterL2 = Buffer.byteLength(purgedCss, \"utf-8\");\n            log(`[css-purge]   L2 attr purge: ${(afterL1 / 1024).toFixed(1)} → ${(afterL2 / 1024).toFixed(1)} KB`);\n          }\n\n          // Level 3: unused CSS variable cleanup\n          if (cleanVars) {\n            purgedCss = cleanUnusedVars(purgedCss);\n            const afterL3 = Buffer.byteLength(purgedCss, \"utf-8\");\n            log(`[css-purge]   L3 var cleanup: → ${(afterL3 / 1024).toFixed(1)} KB`);\n          }\n\n          const finalSize = Buffer.byteLength(purgedCss, \"utf-8\");\n          log(`[css-purge]   Final: ${(originalSize / 1024).toFixed(1)} → ${(finalSize / 1024).toFixed(1)} KB (${((1 - finalSize / originalSize) * 100).toFixed(1)}% reduction)`);\n\n          // Write back\n          const source = new sources.RawSource(purgedCss);\n          compilation.updateAsset(name, source);\n        }\n      },\n    );\n  },\n});\n"
  ],
  "mappings": ";AAgBA;AACA;AAEA;AACA;AACA,qBAAS;AACT,iBAAS;AAsCT,SAAS,OAAO,CAAC,MAAW,SAA8B;AAAA,EACxD,IAAI,CAAC,QAAQ,OAAO,SAAS;AAAA,IAAU;AAAA,EACvC,QAAQ,IAAI;AAAA,EACZ,WAAW,OAAO,OAAO,KAAK,IAAI,GAAG;AAAA,IACnC,IAAI,QAAQ;AAAA,MAAQ;AAAA,IACpB,MAAM,MAAM,KAAK;AAAA,IACjB,IAAI,MAAM,QAAQ,GAAG,GAAG;AAAA,MACtB,WAAW,QAAQ;AAAA,QAAK,QAAQ,MAAM,OAAO;AAAA,IAC/C,EAAO,SAAI,OAAO,OAAO,QAAQ,UAAU;AAAA,MACzC,QAAQ,KAAK,OAAO;AAAA,IACtB;AAAA,EACF;AAAA;AAGF,SAAS,gBAAgB,CAAC,KAA+B;AAAA,EACvD,MAAM,UAAU,IAAI;AAAA,EACpB,WAAW,QAAQ,IAAI,MAAM;AAAA,IAC3B,IAAI,KAAK,SAAS;AAAA,MAAqB;AAAA,IACvC,MAAM,MAAM,KAAK,QAAQ;AAAA,IACzB,IAAI,CAAC,OAAO,CAAC,IAAI,WAAW,eAAe;AAAA,MAAG;AAAA,IAC9C,WAAW,QAAQ,KAAK,YAAY;AAAA,MAClC,IAAI,KAAK,SAAS,mBAAmB;AAAA,QACnC,MAAM,WAAW,KAAK,UAAU,SAAS,KAAK,OAAO;AAAA,QACrD,MAAM,QAAQ,KAAK,OAAO;AAAA,QAC1B,IAAI,SAAS;AAAA,UAAU,QAAQ,IAAI,OAAO,QAAQ;AAAA,MACpD,EAAO,SAAI,KAAK,SAAS,0BAA0B;AAAA,QACjD,MAAM,QAAQ,KAAK,OAAO;AAAA,QAC1B,IAAI;AAAA,UAAO,QAAQ,IAAI,OAAO,KAAK;AAAA,MACrC;AAAA,IACF;AAAA,EACF;AAAA,EACA,OAAO;AAAA;AAGT,SAAS,gBAAgB,CAAC,KAAU,cAAgD;AAAA,EAClF,MAAM,SAAsB,CAAC;AAAA,EAC7B,QAAQ,KAAK,CAAC,SAAS;AAAA,IACrB,IAAI,KAAK,SAAS;AAAA,MAAqB;AAAA,IACvC,IAAI,cAA6B;AAAA,IACjC,IAAI,WAA0B;AAAA,IAC9B,IAAI,KAAK,MAAM,SAAS,cAAc;AAAA,MACpC,cAAc,KAAK,KAAK;AAAA,MACxB,WAAW;AAAA,IACb,EAAO,SAAI,KAAK,MAAM,SAAS,uBAAuB;AAAA,MACpD,MAAM,QAAkB,CAAC;AAAA,MACzB,IAAI,SAAS,KAAK;AAAA,MAClB,OAAO,QAAQ,SAAS,uBAAuB;AAAA,QAC7C,MAAM,QAAQ,OAAO,UAAU,KAAK;AAAA,QACpC,SAAS,OAAO;AAAA,MAClB;AAAA,MACA,IAAI,QAAQ,SAAS,cAAc;AAAA,QACjC,MAAM,QAAQ,OAAO,KAAK;AAAA,QAC1B,WAAW,OAAO;AAAA,MACpB;AAAA,MACA,cAAc,MAAM,KAAK,GAAG;AAAA,IAC9B;AAAA,IACA,IAAI,CAAC,YAAY,CAAC,aAAa,IAAI,QAAQ;AAAA,MAAG;AAAA,IAC9C,MAAM,QAAmB;AAAA,MACvB,WAAW;AAAA,MACX,OAAO,IAAI;AAAA,MACX,cAAc,IAAI;AAAA,MAClB,WAAW;AAAA,IACb;AAAA,IACA,WAAW,QAAQ,KAAK,cAAc,CAAC,GAAG;AAAA,MACxC,IAAI,KAAK,SAAS,mBAAmB,KAAK,SAAS,sBAAsB;AAAA,QACvE,MAAM,YAAY;AAAA,QAClB;AAAA,MACF;AAAA,MACA,IAAI,KAAK,SAAS;AAAA,QAAgB;AAAA,MAClC,MAAM,WAAW,KAAK,MAAM;AAAA,MAC5B,IAAI,CAAC;AAAA,QAAU;AAAA,MACf,IAAI,CAAC,KAAK,OAAO;AAAA,QACf,MAAM,aAAa,IAAI,QAAQ;AAAA,MACjC,EAAO,SAAI,KAAK,MAAM,SAAS,iBAAiB;AAAA,QAC9C,MAAM,MAAM,IAAI,UAAU,KAAK,MAAM,KAAK;AAAA,MAC5C,EAAO;AAAA,QACL,MAAM,MAAM,IAAI,UAAU,SAAS;AAAA;AAAA,IAEvC;AAAA,IACA,OAAO,KAAK,KAAK;AAAA,GAClB;AAAA,EACD,OAAO;AAAA;AAKT,SAAS,cAAc,CACrB,WACA,UAC2D;AAAA,EAC3D,MAAM,gBAAgB,IAAI;AAAA,EAC1B,MAAM,eAAe,IAAI;AAAA,EAEzB,MAAM,kBAAkB,IAAI;AAAA,EAC5B,WAAW,SAAS,WAAW;AAAA,IAC7B,MAAM,WAAW,gBAAgB,IAAI,MAAM,SAAS,KAAK,CAAC;AAAA,IAC1D,SAAS,KAAK,KAAK;AAAA,IACnB,gBAAgB,IAAI,MAAM,WAAW,QAAQ;AAAA,EAC/C;AAAA,EAEA,YAAY,WAAW,UAAU,OAAO,QAAQ,QAAQ,GAAG;AAAA,IACzD,MAAM,iBAAiB,mBAAmB,WAAW,eAAe;AAAA,IACpE,IAAI,eAAe,WAAW;AAAA,MAAG;AAAA,IAEjC,WAAW,OAAO,MAAM,QAAQ;AAAA,MAAQ,cAAc,IAAI,GAAG;AAAA,IAE7D,YAAY,YAAY,UAAU,OAAO,QAAQ,MAAM,QAAQ,MAAM,GAAG;AAAA,MACtE,IAAI,MAAM,QAAQ,KAAK,GAAG;AAAA,QACxB,IAAI,WAAW,YAAY,cAAc,GAAG;AAAA,UAC1C,WAAW,OAAO;AAAA,YAAO,cAAc,IAAI,GAAG;AAAA,QAChD;AAAA,MACF,EAAO;AAAA,QACL,MAAM,aAAa,kBAAkB,YAAY,cAAc;AAAA,QAC/D,IAAI,eAAe,OAAO;AAAA,UACxB,WAAW,WAAW,OAAO,OAAO,KAAK,GAAG;AAAA,YAC1C,WAAW,OAAO;AAAA,cAAS,cAAc,IAAI,GAAG;AAAA,UAClD;AAAA,QACF,EAAO;AAAA,UACL,WAAW,OAAO,YAAY;AAAA,YAC5B,IAAI,MAAM,MAAM;AAAA,cACd,WAAW,OAAO,MAAM;AAAA,gBAAM,cAAc,IAAI,GAAG;AAAA,YACrD;AAAA,UACF;AAAA;AAAA;AAAA,IAGN;AAAA,IAEA,IAAI,MAAM,OAAO;AAAA,MACf,YAAY,UAAU,YAAY,OAAO,QAAQ,MAAM,KAAK,GAAG;AAAA,QAC7D,IAAI,WAAW,UAAU,cAAc,GAAG;AAAA,UACxC,YAAY,MAAM,QAAQ,OAAO,QAAQ,OAAO,GAAG;AAAA,YACjD,aAAa,IAAI,IAAI,SAAS,OAAO;AAAA,UACvC;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,OAAO,EAAE,eAAe,aAAa;AAAA;AAGvC,SAAS,kBAAkB,CAAC,WAAmB,UAAiD;AAAA,EAC9F,IAAI,SAAS,IAAI,SAAS;AAAA,IAAG,OAAO,SAAS,IAAI,SAAS;AAAA,EAC1D,MAAM,UAAuB,CAAC;AAAA,EAC9B,YAAY,WAAW,WAAW,UAAU;AAAA,IAC1C,IAAI,cAAc,WAAW;AAAA,MAC3B,QAAQ,KAAK,GAAG,MAAM;AAAA,IACxB;AAAA,IACA,IAAI,UAAU,SAAS,GAAG,GAAG;AAAA,MAC3B,OAAO,QAAQ,QAAQ,UAAU,MAAM,GAAG;AAAA,MAC1C,IAAI,SAAS,UAAU,cAAc,QAAQ;AAAA,QAC3C,QAAQ,KAAK,GAAG,MAAM;AAAA,MACxB;AAAA,IACF;AAAA,EACF;AAAA,EACA,OAAO;AAAA;AAGT,SAAS,UAAU,CAAC,UAAkB,QAA8B;AAAA,EAClE,WAAW,SAAS,QAAQ;AAAA,IAC1B,IAAI,MAAM;AAAA,MAAW,OAAO;AAAA,IAC5B,IAAI,MAAM,aAAa,IAAI,QAAQ;AAAA,MAAG,OAAO;AAAA,IAC7C,IAAI,MAAM,MAAM,IAAI,QAAQ;AAAA,MAAG,OAAO;AAAA,EACxC;AAAA,EACA,OAAO;AAAA;AAGT,SAAS,iBAAiB,CAAC,UAAkB,QAA0C;AAAA,EACrF,MAAM,SAAS,IAAI;AAAA,EACnB,WAAW,SAAS,QAAQ;AAAA,IAC1B,IAAI,MAAM;AAAA,MAAW,OAAO;AAAA,IAC5B,MAAM,MAAM,MAAM,MAAM,IAAI,QAAQ;AAAA,IACpC,IAAI,QAAQ;AAAA,MAAW,OAAO;AAAA,IAC9B,IAAI,QAAQ;AAAA,MAAW,OAAO,IAAI,GAAG;AAAA,IACrC,IAAI,MAAM,aAAa,IAAI,QAAQ;AAAA,MAAG,OAAO;AAAA,EAC/C;AAAA,EACA,OAAO;AAAA;AAKT,eAAe,kBAAkB,CAAC,QAAsC;AAAA,EACtE,MAAM,YAAyB,CAAC;AAAA,EAChC,MAAM,QAAQ,MAAM,SAAS,wBAAwB,EAAE,KAAK,QAAQ,QAAQ,CAAC,oBAAoB,EAAE,CAAC;AAAA,EAEpG,WAAW,WAAW,OAAO;AAAA,IAC3B,MAAM,WAAW,KAAK,KAAK,QAAQ,OAAO;AAAA,IAC1C,MAAM,OAAO,MAAM,aAAa,UAAU,OAAO;AAAA,IACjD,IAAI,CAAC,KAAK,SAAS,eAAe;AAAA,MAAG;AAAA,IAErC,MAAM,QAAQ,YAAY,KAAK,OAAO;AAAA,IACtC,MAAM,MAAM,MAAM,IAAI,MAAM,MAAM,EAAE,QAAQ,cAAc,KAAK,MAAM,CAAC;AAAA,IACtE,MAAM,YAAY,iBAAiB,GAAG;AAAA,IACtC,IAAI,UAAU,SAAS;AAAA,MAAG;AAAA,IAE1B,UAAU,KAAK,GAAG,iBAAiB,KAAK,SAAS,CAAC;AAAA,EACpD;AAAA,EAEA,OAAO;AAAA;AAKT,SAAS,eAAe,CAAC,KAAa,cAAmC;AAAA,EACvE,MAAM,OAAO,QAAQ,MAAM,GAAG;AAAA,EAE9B,KAAK,UAAU,CAAC,SAAS;AAAA,IACvB,MAAM,YAAY,KAAK;AAAA,IACvB,MAAM,OAAiB,CAAC;AAAA,IAExB,WAAW,OAAO,WAAW;AAAA,MAC3B,MAAM,cAAc,IAAI,SAAS,4CAA4C;AAAA,MAC7E,IAAI,aAAa;AAAA,MAEjB,WAAW,SAAS,aAAa;AAAA,QAC/B,MAAM,eAAe,IAAI,MAAM,OAAO,MAAM;AAAA,QAC5C,IAAI,MAAM,OAAO;AAAA,UAAa;AAAA,QAC9B,IAAI,CAAC,aAAa,IAAI,YAAY,GAAG;AAAA,UACnC,aAAa;AAAA,UACb;AAAA,QACF;AAAA,MACF;AAAA,MAEA,IAAI;AAAA,QAAY,KAAK,KAAK,GAAG;AAAA,IAC/B;AAAA,IAEA,IAAI,KAAK,WAAW,GAAG;AAAA,MACrB,KAAK,OAAO;AAAA,IACd,EAAO,SAAI,KAAK,SAAS,UAAU,QAAQ;AAAA,MACzC,KAAK,YAAY;AAAA,IACnB;AAAA,GACD;AAAA,EAED,OAAO,KAAK,SAAS;AAAA;AAKvB,SAAS,eAAe,CAAC,KAAqB;AAAA,EAC5C,IAAI,UAAU;AAAA,EACd,IAAI,SAAS;AAAA,EAEb,OAAO,SAAS;AAAA,IACd,UAAU;AAAA,IACV,MAAM,OAAO,QAAQ,MAAM,MAAM;AAAA,IAEjC,MAAM,WAAW,IAAI;AAAA,IACrB,KAAK,UAAU,OAAO,CAAC,SAAS;AAAA,MAC9B,MAAM,UAAU,SAAS,IAAI,KAAK,IAAI,KAAK,CAAC;AAAA,MAC5C,QAAQ,KAAK,EAAE,MAAM,KAAK,QAAgB,MAAM,KAAK,MAAM,OAAO,QAAQ,OAAO,CAAC;AAAA,MAClF,SAAS,IAAI,KAAK,MAAM,OAAO;AAAA,KAChC;AAAA,IAED,MAAM,aAAa,IAAI;AAAA,IACvB,KAAK,UAAU,CAAC,SAAS;AAAA,MACvB,MAAM,OAAO,KAAK,MAAM,SAAS,6BAA6B;AAAA,MAC9D,WAAW,OAAO,MAAM;AAAA,QACtB,WAAW,IAAI,IAAI,EAAE;AAAA,MACvB;AAAA,KACD;AAAA,IAED,YAAY,SAAS,YAAY,UAAU;AAAA,MACzC,IAAI,CAAC,WAAW,IAAI,OAAO,GAAG;AAAA,QAC5B,WAAW,SAAS,SAAS;AAAA,UAC3B,MAAM,KAAK,UAAU,MAAM,MAAM,CAAC,SAAS;AAAA,YACzC,KAAK,OAAO;AAAA,YACZ,UAAU;AAAA,WACX;AAAA,QACH;AAAA,MACF;AAAA,IACF;AAAA,IAEA,KAAK,UAAU,CAAC,SAAS;AAAA,MACvB,IAAI,KAAK,SAAS,KAAK,MAAM,WAAW;AAAA,QAAG,KAAK,OAAO;AAAA,KACxD;AAAA,IACD,KAAK,YAAY,CAAC,WAAW;AAAA,MAC3B,IAAI,OAAO,SAAS,OAAO,MAAM,WAAW;AAAA,QAAG,OAAO,OAAO;AAAA,KAC9D;AAAA,IAED,SAAS,KAAK,SAAS;AAAA,EACzB;AAAA,EAEA,OAAO;AAAA;AAKF,IAAM,iBAAiB,CAAC,aAA6C;AAAA,EAC1E,MAAM;AAAA,EAEN,KAAK,CAAC,KAAK;AAAA,IACT;AAAA,MACE,UAAU;AAAA,MACV,SAAS;AAAA,MACT,YAAY;AAAA,MACZ,YAAY;AAAA,MACZ,UAAU;AAAA,QACR;AAAA,IAEJ,IAAI,cACF,EAAE,OAAO,gBAAgB,GACzB,SAAS,QAAQ,SAAS,kBAAkB;AAAA,MAC1C,MAAM,MAAM,UAAU,QAAQ,IAAI,KAAK,OAAO,IAAI,MAAM;AAAA,MAGxD,MAAM,WAA0B,KAAK,MACnC,MAAM,aAAa,KAAK,QAAQ,YAAY,GAAG,OAAO,CACxD;AAAA,MACA,IAAI,gCAAgC,OAAO,KAAK,QAAQ,EAAE,gBAAgB;AAAA,MAG1E,MAAM,cAAc,KAAK,QAAQ,MAAM;AAAA,MACvC,MAAM,SAAS,MAAM,mBAAmB,WAAW;AAAA,MACnD,IAAI,uBAAuB,gBAAgB,OAAO,yBAAyB;AAAA,MAG3E,QAAQ,eAAe,iBAAiB,eAAe,QAAQ,QAAQ;AAAA,MACvE,IAAI,yBAAyB,cAAc,iBAAiB,aAAa,YAAY;AAAA,MAGrF,YAAY,MAAM,UAAU,OAAO,QAAQ,MAAM,GAAG;AAAA,QAClD,IAAI,CAAC,KAAK,SAAS,MAAM;AAAA,UAAG;AAAA,QAE5B,MAAM,cAAc,MAAM,OAAO,EAAE,SAAS;AAAA,QAC5C,MAAM,eAAe,OAAO,WAAW,aAAa,OAAO;AAAA,QAC3D,IAAI,0BAA0B,UAAU,eAAe,MAAM,QAAQ,CAAC,OAAO;AAAA,QAG7E,MAAM,cAAc,MAAM,IAAI,SAAS,EAAE,MAAM;AAAA,UAC7C,SAAS,CAAC;AAAA,UACV,KAAK,CAAC,EAAE,KAAK,YAAY,CAAC;AAAA,UAC1B,UAAU,CAAC,GAAG,aAAa;AAAA,UAC3B,WAAW;AAAA,UACX,UAAU;AAAA,QACZ,CAAC;AAAA,QAED,IAAI,YAAY,YAAY,IAAI,OAAO;AAAA,QACvC,MAAM,UAAU,OAAO,WAAW,WAAW,OAAO;AAAA,QACpD,IAAI,kCAAkC,eAAe,MAAM,QAAQ,CAAC,QAAO,UAAU,MAAM,QAAQ,CAAC,MAAM;AAAA,QAG1G,IAAI,aAAa,aAAa,OAAO,GAAG;AAAA,UACtC,YAAY,gBAAgB,WAAW,YAAY;AAAA,UACnD,MAAM,UAAU,OAAO,WAAW,WAAW,OAAO;AAAA,UACpD,IAAI,iCAAiC,UAAU,MAAM,QAAQ,CAAC,QAAO,UAAU,MAAM,QAAQ,CAAC,MAAM;AAAA,QACtG;AAAA,QAGA,IAAI,WAAW;AAAA,UACb,YAAY,gBAAgB,SAAS;AAAA,UACrC,MAAM,UAAU,OAAO,WAAW,WAAW,OAAO;AAAA,UACpD,IAAI,oCAAmC,UAAU,MAAM,QAAQ,CAAC,MAAM;AAAA,QACxE;AAAA,QAEA,MAAM,YAAY,OAAO,WAAW,WAAW,OAAO;AAAA,QACtD,IAAI,yBAAyB,eAAe,MAAM,QAAQ,CAAC,QAAO,YAAY,MAAM,QAAQ,CAAC,WAAW,IAAI,YAAY,gBAAgB,KAAK,QAAQ,CAAC,eAAe;AAAA,QAGrK,MAAM,SAAS,IAAI,QAAQ,UAAU,SAAS;AAAA,QAC9C,YAAY,YAAY,MAAM,MAAM;AAAA,MACtC;AAAA,KAEJ;AAAA;AAEJ;",
  "debugId": "CB5B2093063663E864756E2164756E21",
  "names": []
}
@@ -0,0 +1,28 @@
1
+ /**
2
+ * rsbuild-plugin-css-purge
3
+ *
4
+ * Two-level CSS purge for @pathscale/ui consumers.
5
+ *
6
+ * Level 1: class-level purge via purgecss — removes entire rules whose selectors
7
+ * don't match the safelist built from consumer JSX analysis.
8
+ * Level 2: attribute-level purge — within kept rules, strips compound selectors
9
+ * containing data-attr / aria-attr attribute selectors not in the attr safelist.
10
+ *
11
+ * Usage in rsbuild.config.ts:
12
+ * import { pluginCssPurge } from "@pathscale/rebuild-plugin-ui-css-purge";
13
+ * export default defineConfig({ plugins: [pluginCssPurge({ manifest: "..." })] });
14
+ */
15
+ import type { RsbuildPlugin } from "@rsbuild/core";
16
+ export interface CssPurgeOptions {
17
+ /** Path to purge-manifest.json (generated by generate-manifest.ts) */
18
+ manifest: string;
19
+ /** Consumer source directory to scan for JSX usage (default: "src") */
20
+ srcDir?: string;
21
+ /** Enable Level 2 attribute purge (default: true) */
22
+ attrPurge?: boolean;
23
+ /** Enable CSS variable cleanup (default: true) */
24
+ cleanVars?: boolean;
25
+ /** Log purge stats (default: true) */
26
+ verbose?: boolean;
27
+ }
28
+ export declare const pluginCssPurge: (options: CssPurgeOptions) => RsbuildPlugin;
@@ -0,0 +1,35 @@
1
+ /**
2
+ * Consumer-side JSX scanner.
3
+ *
4
+ * Walks a consumer's source tree, finds component imports from @pathscale/ui,
5
+ * collects prop values, and cross-references with the purge manifest to build
6
+ * Level 1 (class) and Level 2 (attribute) safelists.
7
+ *
8
+ * Usage: bun run src/scan-consumer.ts <consumer-src-dir> <purge-manifest.json>
9
+ */
10
+ interface ComponentManifest {
11
+ classes: {
12
+ always: string[];
13
+ byProp: Record<string, string[] | Record<string, string[]>>;
14
+ };
15
+ attrs?: Record<string, Record<string, string>>;
16
+ }
17
+ type PurgeManifest = Record<string, ComponentManifest>;
18
+ /** What we collect per component usage from JSX */
19
+ interface PropUsage {
20
+ component: string;
21
+ props: Map<string, string | "DYNAMIC">;
22
+ booleanProps: Set<string>;
23
+ hasSpread: boolean;
24
+ }
25
+ /** Extract @pathscale/ui imports from a parsed module */
26
+ declare function extractUIImports(ast: any): Map<string, string>;
27
+ /** Extract JSX usages of UI components */
28
+ declare function extractJSXUsages(ast: any, uiComponents: Map<string, string>): PropUsage[];
29
+ interface Safelists {
30
+ classSafelist: Set<string>;
31
+ attrSafelist: Set<string>;
32
+ }
33
+ declare function buildSafelists(allUsages: PropUsage[], manifest: PurgeManifest): Safelists;
34
+ export { extractUIImports, extractJSXUsages, buildSafelists };
35
+ export type { PropUsage, PurgeManifest, ComponentManifest, Safelists };
package/package.json ADDED
@@ -0,0 +1,35 @@
1
+ {
2
+ "name": "@pathscale/rebuild-plugin-ui-css-purge",
3
+ "version": "0.1.0",
4
+ "type": "module",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "exports": {
8
+ ".": {
9
+ "types": "./dist/index.d.ts",
10
+ "import": "./dist/index.js"
11
+ }
12
+ },
13
+ "bin": {
14
+ "generate-manifest": "src/generate-manifest.ts"
15
+ },
16
+ "files": ["dist", "src/generate-manifest.ts"],
17
+ "dependencies": {
18
+ "@swc/core": "^1.15.24",
19
+ "fast-glob": "^3.3.3",
20
+ "postcss": "^8.5.9",
21
+ "purgecss": "^8.0.0"
22
+ },
23
+ "peerDependencies": {
24
+ "@rsbuild/core": "^1.7.5"
25
+ },
26
+ "scripts": {
27
+ "build": "bun run build.ts",
28
+ "lint": "biome check .",
29
+ "format": "biome format . --write"
30
+ },
31
+ "devDependencies": {
32
+ "@rsbuild/core": "^1.7.5",
33
+ "bun-types": "^1.3.12"
34
+ }
35
+ }
@@ -0,0 +1,155 @@
1
+ /**
2
+ * Lib-side database generator.
3
+ *
4
+ * Reads all `*.classnames.ts` files from @pathscale/ui's component tree
5
+ * and produces a `purge-manifest.json` that the consumer-side plugin uses
6
+ * to build safelists.
7
+ *
8
+ * Usage: bun run src/generate-manifest.ts <path-to-ui-src/components>
9
+ * Output: purge-manifest.json in cwd (or pass --out <path>)
10
+ */
11
+
12
+ import { Glob } from "bun";
13
+ import path from "path";
14
+
15
+ // ── Types ──────────────────────────────────────────────────────────────────────
16
+
17
+ type ClassValue = string | readonly string[];
18
+
19
+ interface ComponentManifest {
20
+ classes: {
21
+ always: string[];
22
+ byProp: Record<string, string[] | Record<string, string[]>>;
23
+ };
24
+ attrs?: Record<string, Record<string, string>>;
25
+ }
26
+
27
+ type PurgeManifest = Record<string, ComponentManifest>;
28
+
29
+ // ── Helpers ────────────────────────────────────────────────────────────────────
30
+
31
+ /** Flatten a class value (string or string[]) into an array of individual class names. */
32
+ function flattenClasses(val: ClassValue): string[] {
33
+ if (typeof val === "string") {
34
+ return val.split(/\s+/).filter(Boolean);
35
+ }
36
+ return (val as readonly string[]).flatMap((s) => s.split(/\s+/).filter(Boolean));
37
+ }
38
+
39
+ /** Check if a value is a plain object (not array, not null). */
40
+ function isRecord(v: unknown): v is Record<string, unknown> {
41
+ return v !== null && typeof v === "object" && !Array.isArray(v);
42
+ }
43
+
44
+ /**
45
+ * Walk a single CLASSES object (one component or one part of a compound component)
46
+ * and produce the manifest entry.
47
+ */
48
+ function walkClassesObject(obj: Record<string, unknown>): ComponentManifest {
49
+ const always: string[] = [];
50
+ const byProp: Record<string, string[] | Record<string, string[]>> = {};
51
+ let attrs: Record<string, Record<string, string>> | undefined;
52
+
53
+ for (const [slot, value] of Object.entries(obj)) {
54
+ if (slot === "base") {
55
+ always.push(...flattenClasses(value as ClassValue));
56
+ } else if (slot === "attrs") {
57
+ if (isRecord(value)) {
58
+ attrs = {};
59
+ for (const [propName, attrMap] of Object.entries(value)) {
60
+ if (isRecord(attrMap)) {
61
+ attrs[propName] = attrMap as Record<string, string>;
62
+ }
63
+ }
64
+ }
65
+ } else if (slot === "flag") {
66
+ if (isRecord(value)) {
67
+ for (const [propName, classVal] of Object.entries(value)) {
68
+ byProp[propName] = flattenClasses(classVal as ClassValue);
69
+ }
70
+ }
71
+ } else if (isRecord(value)) {
72
+ const enumMap: Record<string, string[]> = {};
73
+ for (const [enumKey, classVal] of Object.entries(value)) {
74
+ enumMap[enumKey] = flattenClasses(classVal as ClassValue);
75
+ }
76
+ byProp[slot] = enumMap;
77
+ }
78
+ }
79
+
80
+ const manifest: ComponentManifest = { classes: { always, byProp } };
81
+ if (attrs && Object.keys(attrs).length > 0) {
82
+ manifest.attrs = attrs;
83
+ }
84
+ return manifest;
85
+ }
86
+
87
+ /**
88
+ * Detect whether CLASSES is compound (top-level keys are part names like Root, Item)
89
+ * or flat (top-level keys are slots like base, variant, flag).
90
+ */
91
+ const KNOWN_SLOTS = new Set(["base", "variant", "size", "flag", "color", "attrs"]);
92
+
93
+ function isCompound(obj: Record<string, unknown>): boolean {
94
+ for (const key of Object.keys(obj)) {
95
+ if (KNOWN_SLOTS.has(key)) return false;
96
+ }
97
+ return true;
98
+ }
99
+
100
+ // ── Main ───────────────────────────────────────────────────────────────────────
101
+
102
+ async function main() {
103
+ const args = process.argv.slice(2);
104
+ let componentsDir = args[0];
105
+ let outPath = "purge-manifest.json";
106
+
107
+ const outIdx = args.indexOf("--out");
108
+ if (outIdx !== -1 && args[outIdx + 1]) {
109
+ outPath = args[outIdx + 1];
110
+ }
111
+
112
+ if (!componentsDir) {
113
+ console.error("Usage: bun run src/generate-manifest.ts <path-to-components-dir> [--out <path>]");
114
+ process.exit(1);
115
+ }
116
+
117
+ componentsDir = path.resolve(componentsDir);
118
+ console.log(`Scanning ${componentsDir} for *.classnames.ts files…`);
119
+
120
+ const manifest: PurgeManifest = {};
121
+ const glob = new Glob("**/*.classnames.ts");
122
+
123
+ for await (const relPath of glob.scan({ cwd: componentsDir })) {
124
+ const fullPath = path.join(componentsDir, relPath);
125
+
126
+ const mod = await import(fullPath);
127
+ const CLASSES = mod.CLASSES;
128
+
129
+ if (!CLASSES || typeof CLASSES !== "object") {
130
+ console.warn(` SKIP ${relPath} — no CLASSES export found`);
131
+ continue;
132
+ }
133
+
134
+ const fileName = path.basename(relPath, ".classnames.ts");
135
+
136
+ if (isCompound(CLASSES as Record<string, unknown>)) {
137
+ for (const [partName, partObj] of Object.entries(CLASSES as Record<string, unknown>)) {
138
+ if (isRecord(partObj)) {
139
+ const entryName = `${fileName}.${partName}`;
140
+ manifest[entryName] = walkClassesObject(partObj as Record<string, unknown>);
141
+ console.log(` ✓ ${entryName}`);
142
+ }
143
+ }
144
+ } else {
145
+ manifest[fileName] = walkClassesObject(CLASSES as Record<string, unknown>);
146
+ console.log(` ✓ ${fileName}`);
147
+ }
148
+ }
149
+
150
+ const json = JSON.stringify(manifest, null, 2);
151
+ await Bun.write(outPath, json);
152
+ console.log(`\nWrote ${outPath} (${Object.keys(manifest).length} entries, ${json.length} bytes)`);
153
+ }
154
+
155
+ main();