forked from CopilotKit/CopilotKit
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathscope-preflight.mjs
More file actions
100 lines (82 loc) · 2.75 KB
/
Copy pathscope-preflight.mjs
File metadata and controls
100 lines (82 loc) · 2.75 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
/**
* Post-processes compiled Tailwind CSS to scope all @layer base rules
* under [data-copilotkit], preventing CopilotKit styles from leaking
* into the host application.
*
* Run after `tailwindcss` CLI: node scripts/scope-preflight.mjs <file>
*/
import { readFileSync, writeFileSync } from "fs";
import postcss from "postcss";
const SCOPE = "[data-copilotkit]";
const file = process.argv[2];
if (!file) {
console.error("Usage: node scripts/scope-preflight.mjs <css-file>");
process.exit(1);
}
/** Selectors that are already scoped and should be left alone. */
function isAlreadyScoped(selector) {
return (
selector.includes("[data-copilot") || selector.includes("[data-sidebar")
);
}
/** Rewrite a single selector to be scoped under [data-copilotkit]. */
function scopeSelector(sel) {
sel = sel.trim();
// Already scoped — keep as-is
if (isAlreadyScoped(sel)) return sel;
// html, :host → [data-copilotkit]
if (sel === "html" || sel === ":host" || sel === "html,:host") {
return SCOPE;
}
// body → null (remove)
if (sel === "body") return null;
// ::backdrop → null (cannot be scoped to a container)
if (sel === "::backdrop") return null;
// Bare universal selector → scope to container + descendants
if (sel === "*") return `${SCOPE}, ${SCOPE} *`;
// Pseudo-elements (::) and vendor pseudo-classes (:-) → descendant only
// These can't be combined with an attribute selector suffix
if (sel.startsWith(":")) {
return `${SCOPE} ${sel}`;
}
// Element / attribute selectors → descendant AND self-matching
// e.g. button → [data-copilotkit] button, button[data-copilotkit]
// This ensures <button data-copilotkit> also receives the preflight reset
return `${SCOPE} ${sel}, ${sel}${SCOPE}`;
}
function scopeRule(rule) {
const newSelectors = [];
for (const sel of rule.selectors) {
const scoped = scopeSelector(sel);
if (scoped !== null) {
// scopeSelector may return comma-separated selectors (for *)
if (typeof scoped === "string" && scoped.includes(", ")) {
newSelectors.push(...scoped.split(", "));
} else if (scoped) {
newSelectors.push(scoped);
}
}
}
if (newSelectors.length === 0) {
rule.remove();
} else {
rule.selectors = newSelectors;
}
}
/** Recursively scope all rules within a node (handles @supports, @media, etc.) */
function scopeChildren(node) {
node.walk((child) => {
if (child.type === "rule") {
scopeRule(child);
}
// @supports / @media blocks are walked automatically
});
}
// --- Main ---
const css = readFileSync(file, "utf8");
const root = postcss.parse(css);
root.walkAtRules("layer", (layer) => {
if (layer.params !== "base") return;
scopeChildren(layer);
});
writeFileSync(file, root.toString());