playwriter 0.1.0 → 0.3.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/dist/bippy.js +5 -5
- package/dist/cdp-log.d.ts +4 -1
- package/dist/cdp-log.d.ts.map +1 -1
- package/dist/cdp-log.js +39 -2
- package/dist/cdp-log.js.map +1 -1
- package/dist/cdp-log.test.d.ts +2 -0
- package/dist/cdp-log.test.d.ts.map +1 -0
- package/dist/cdp-log.test.js +109 -0
- package/dist/cdp-log.test.js.map +1 -0
- package/dist/cdp-relay.d.ts.map +1 -1
- package/dist/cdp-relay.js +120 -11
- package/dist/cdp-relay.js.map +1 -1
- package/dist/cli-help.test.js +22 -0
- package/dist/cli-help.test.js.map +1 -1
- package/dist/cli.js +69 -25
- package/dist/cli.js.map +1 -1
- package/dist/executor.d.ts +4 -0
- package/dist/executor.d.ts.map +1 -1
- package/dist/executor.js +140 -33
- package/dist/executor.js.map +1 -1
- package/dist/extension/background.js +343 -62
- package/dist/extension/manifest.json +1 -1
- package/dist/mcp.d.ts.map +1 -1
- package/dist/mcp.js +6 -1
- package/dist/mcp.js.map +1 -1
- package/dist/performance-examples.d.ts +5 -0
- package/dist/performance-examples.d.ts.map +1 -0
- package/dist/performance-examples.js +112 -0
- package/dist/performance-examples.js.map +1 -0
- package/dist/performance-profiling.md +417 -0
- package/dist/prompt.md +51 -18
- package/dist/react-source.d.ts +44 -0
- package/dist/react-source.d.ts.map +1 -1
- package/dist/react-source.js +207 -20
- package/dist/react-source.js.map +1 -1
- package/dist/readability.js +1 -1
- package/dist/relay-client.d.ts +11 -0
- package/dist/relay-client.d.ts.map +1 -1
- package/dist/relay-client.js +46 -1
- package/dist/relay-client.js.map +1 -1
- package/dist/relay-core.test.js +10 -6
- package/dist/relay-core.test.js.map +1 -1
- package/dist/relay-session.test.js +43 -7
- package/dist/relay-session.test.js.map +1 -1
- package/dist/relay-state.test.js +57 -1
- package/dist/relay-state.test.js.map +1 -1
- package/dist/screen-recording.d.ts.map +1 -1
- package/dist/screen-recording.js +19 -4
- package/dist/screen-recording.js.map +1 -1
- package/dist/selector-generator.js +1 -1
- package/dist/start-relay-server.d.ts +1 -1
- package/dist/start-relay-server.d.ts.map +1 -1
- package/dist/start-relay-server.js +23 -1
- package/dist/start-relay-server.js.map +1 -1
- package/dist/utils.d.ts +2 -1
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +4 -1
- package/dist/utils.js.map +1 -1
- package/package.json +3 -3
- package/src/cdp-log.test.ts +131 -0
- package/src/cdp-log.ts +44 -2
- package/src/cdp-relay.ts +127 -10
- package/src/cli-help.test.ts +22 -0
- package/src/cli.ts +74 -24
- package/src/executor.ts +166 -39
- package/src/mcp.ts +6 -1
- package/src/performance-examples.ts +186 -0
- package/src/react-source.ts +310 -24
- package/src/relay-client.ts +62 -5
- package/src/relay-core.test.ts +10 -6
- package/src/relay-session.test.ts +45 -11
- package/src/relay-state.test.ts +67 -1
- package/src/screen-recording.ts +20 -4
- package/src/skill.md +62 -19
- package/src/start-relay-server.ts +22 -1
- package/src/utils.ts +5 -0
|
@@ -1,4 +1,173 @@
|
|
|
1
|
-
//#region ../node_modules/.pnpm/
|
|
1
|
+
//#region ../node_modules/.pnpm/string-dedent@3.0.2/node_modules/string-dedent/dist/dedent.mjs
|
|
2
|
+
var cache = /* @__PURE__ */ new WeakMap();
|
|
3
|
+
var newline = /(\n|\r\n?|\u2028|\u2029)/g;
|
|
4
|
+
var leadingWhitespace = /^\s*/;
|
|
5
|
+
var nonWhitespace = /\S/;
|
|
6
|
+
var slice = Array.prototype.slice;
|
|
7
|
+
var zero = "0".charCodeAt(0);
|
|
8
|
+
var nine = "9".charCodeAt(0);
|
|
9
|
+
var lowerA = "a".charCodeAt(0);
|
|
10
|
+
var lowerF = "f".charCodeAt(0);
|
|
11
|
+
var upperA = "A".charCodeAt(0);
|
|
12
|
+
var upperF = "F".charCodeAt(0);
|
|
13
|
+
function dedent(arg) {
|
|
14
|
+
if (typeof arg === "string") return process([arg])[0];
|
|
15
|
+
if (typeof arg === "function") return function() {
|
|
16
|
+
const args = slice.call(arguments);
|
|
17
|
+
args[0] = processTemplateStringsArray(args[0]);
|
|
18
|
+
return arg.apply(this, args);
|
|
19
|
+
};
|
|
20
|
+
const strings = processTemplateStringsArray(arg);
|
|
21
|
+
let s = getCooked(strings, 0);
|
|
22
|
+
for (let i = 1; i < strings.length; i++) s += arguments[i] + getCooked(strings, i);
|
|
23
|
+
return s;
|
|
24
|
+
}
|
|
25
|
+
function getCooked(strings, index) {
|
|
26
|
+
const str = strings[index];
|
|
27
|
+
if (str === void 0) throw new TypeError(`invalid cooked string at index ${index}`);
|
|
28
|
+
return str;
|
|
29
|
+
}
|
|
30
|
+
function processTemplateStringsArray(strings) {
|
|
31
|
+
const cached = cache.get(strings);
|
|
32
|
+
if (cached) return cached;
|
|
33
|
+
const raw = process(strings.raw);
|
|
34
|
+
const cooked = raw.map(cook);
|
|
35
|
+
Object.defineProperty(cooked, "raw", { value: Object.freeze(raw) });
|
|
36
|
+
Object.freeze(cooked);
|
|
37
|
+
cache.set(strings, cooked);
|
|
38
|
+
return cooked;
|
|
39
|
+
}
|
|
40
|
+
function process(strings) {
|
|
41
|
+
const splitQuasis = strings.map((quasi) => quasi.split(newline));
|
|
42
|
+
let common;
|
|
43
|
+
for (let i = 0; i < splitQuasis.length; i++) {
|
|
44
|
+
const lines = splitQuasis[i];
|
|
45
|
+
const firstSplit = i === 0;
|
|
46
|
+
const lastSplit = i + 1 === splitQuasis.length;
|
|
47
|
+
if (firstSplit) {
|
|
48
|
+
if (lines.length === 1 || lines[0].length > 0) throw new Error("invalid content on opening line");
|
|
49
|
+
lines[1] = "";
|
|
50
|
+
}
|
|
51
|
+
if (lastSplit) {
|
|
52
|
+
if (lines.length === 1 || nonWhitespace.test(lines[lines.length - 1])) throw new Error("invalid content on closing line");
|
|
53
|
+
lines[lines.length - 2] = "";
|
|
54
|
+
lines[lines.length - 1] = "";
|
|
55
|
+
}
|
|
56
|
+
for (let j = 2; j < lines.length; j += 2) {
|
|
57
|
+
const text = lines[j];
|
|
58
|
+
const lineContainsTemplateExpression = j + 1 === lines.length && !lastSplit;
|
|
59
|
+
const leading = leadingWhitespace.exec(text)[0];
|
|
60
|
+
if (!lineContainsTemplateExpression && leading.length === text.length) {
|
|
61
|
+
lines[j] = "";
|
|
62
|
+
continue;
|
|
63
|
+
}
|
|
64
|
+
common = commonStart(leading, common);
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
const min = common ? common.length : 0;
|
|
68
|
+
return splitQuasis.map((lines) => {
|
|
69
|
+
let quasi = lines[0];
|
|
70
|
+
for (let i = 1; i < lines.length; i += 2) {
|
|
71
|
+
const newline = lines[i];
|
|
72
|
+
const text = lines[i + 1];
|
|
73
|
+
quasi += newline + text.slice(min);
|
|
74
|
+
}
|
|
75
|
+
return quasi;
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
function commonStart(a, b) {
|
|
79
|
+
if (b === void 0 || a === b) return a;
|
|
80
|
+
let i = 0;
|
|
81
|
+
for (const len = Math.min(a.length, b.length); i < len; i++) if (a[i] !== b[i]) break;
|
|
82
|
+
return a.slice(0, i);
|
|
83
|
+
}
|
|
84
|
+
function cook(raw) {
|
|
85
|
+
let out = "";
|
|
86
|
+
let start = 0;
|
|
87
|
+
let i = 0;
|
|
88
|
+
while ((i = raw.indexOf("\\", i)) > -1) {
|
|
89
|
+
out += raw.slice(start, i);
|
|
90
|
+
if (++i === raw.length) return void 0;
|
|
91
|
+
const next = raw[i++];
|
|
92
|
+
switch (next) {
|
|
93
|
+
case "b":
|
|
94
|
+
out += "\b";
|
|
95
|
+
break;
|
|
96
|
+
case "t":
|
|
97
|
+
out += " ";
|
|
98
|
+
break;
|
|
99
|
+
case "n":
|
|
100
|
+
out += "\n";
|
|
101
|
+
break;
|
|
102
|
+
case "v":
|
|
103
|
+
out += "\v";
|
|
104
|
+
break;
|
|
105
|
+
case "f":
|
|
106
|
+
out += "\f";
|
|
107
|
+
break;
|
|
108
|
+
case "r":
|
|
109
|
+
out += "\r";
|
|
110
|
+
break;
|
|
111
|
+
case "\r": if (i < raw.length && raw[i] === "\n") ++i;
|
|
112
|
+
case "\n":
|
|
113
|
+
case "\u2028":
|
|
114
|
+
case "\u2029": break;
|
|
115
|
+
case "0":
|
|
116
|
+
if (isDigit(raw, i)) return void 0;
|
|
117
|
+
out += "\0";
|
|
118
|
+
break;
|
|
119
|
+
case "x": {
|
|
120
|
+
const n = parseHex(raw, i, i + 2);
|
|
121
|
+
if (n === -1) return void 0;
|
|
122
|
+
i += 2;
|
|
123
|
+
out += String.fromCharCode(n);
|
|
124
|
+
break;
|
|
125
|
+
}
|
|
126
|
+
case "u": {
|
|
127
|
+
let n;
|
|
128
|
+
if (i < raw.length && raw[i] === "{") {
|
|
129
|
+
const end = raw.indexOf("}", ++i);
|
|
130
|
+
if (end === -1) return void 0;
|
|
131
|
+
n = parseHex(raw, i, end);
|
|
132
|
+
i = end + 1;
|
|
133
|
+
} else {
|
|
134
|
+
n = parseHex(raw, i, i + 4);
|
|
135
|
+
i += 4;
|
|
136
|
+
}
|
|
137
|
+
if (n === -1 || n > 1114111) return void 0;
|
|
138
|
+
out += String.fromCodePoint(n);
|
|
139
|
+
break;
|
|
140
|
+
}
|
|
141
|
+
default:
|
|
142
|
+
if (isDigit(next, 0)) return void 0;
|
|
143
|
+
out += next;
|
|
144
|
+
}
|
|
145
|
+
start = i;
|
|
146
|
+
}
|
|
147
|
+
return out + raw.slice(start);
|
|
148
|
+
}
|
|
149
|
+
function isDigit(str, index) {
|
|
150
|
+
const c = str.charCodeAt(index);
|
|
151
|
+
return c >= zero && c <= nine;
|
|
152
|
+
}
|
|
153
|
+
function parseHex(str, index, end) {
|
|
154
|
+
if (end >= str.length) return -1;
|
|
155
|
+
let n = 0;
|
|
156
|
+
for (; index < end; index++) {
|
|
157
|
+
const c = hexToInt(str.charCodeAt(index));
|
|
158
|
+
if (c === -1) return -1;
|
|
159
|
+
n = n * 16 + c;
|
|
160
|
+
}
|
|
161
|
+
return n;
|
|
162
|
+
}
|
|
163
|
+
function hexToInt(c) {
|
|
164
|
+
if (c >= zero && c <= nine) return c - zero;
|
|
165
|
+
if (c >= lowerA && c <= lowerF) return c - lowerA + 10;
|
|
166
|
+
if (c >= upperA && c <= upperF) return c - upperA + 10;
|
|
167
|
+
return -1;
|
|
168
|
+
}
|
|
169
|
+
//#endregion
|
|
170
|
+
//#region ../node_modules/.pnpm/zustand@5.0.13_@types+react@19.2.7_react@19.2.6/node_modules/zustand/esm/vanilla.mjs
|
|
2
171
|
var createStoreImpl = (createState) => {
|
|
3
172
|
let state;
|
|
4
173
|
const listeners = /* @__PURE__ */ new Set();
|
|
@@ -250,37 +419,8 @@ function initPlaywriterToolbar() {
|
|
|
250
419
|
if (target) positionOverlay(target);
|
|
251
420
|
else hideOverlay();
|
|
252
421
|
}
|
|
253
|
-
function
|
|
254
|
-
|
|
255
|
-
const id = el.id || "";
|
|
256
|
-
const cls = typeof el.className === "string" ? el.className : "";
|
|
257
|
-
const role = el.getAttribute("role") || "";
|
|
258
|
-
const aria = el.getAttribute("aria-label") || "";
|
|
259
|
-
const nameAttr = el.getAttribute("name") || "";
|
|
260
|
-
const href = el.getAttribute("href") || "";
|
|
261
|
-
const typeAttr = el.getAttribute("type") || "";
|
|
262
|
-
const text = (el.textContent || "").replace(/\s+/g, " ").trim().slice(0, 300);
|
|
263
|
-
const r = el.getBoundingClientRect();
|
|
264
|
-
const rect = `x=${Math.round(r.x)} y=${Math.round(r.y)} w=${Math.round(r.width)} h=${Math.round(r.height)}`;
|
|
265
|
-
const visible = r.width > 0 && r.height > 0;
|
|
266
|
-
return [
|
|
267
|
-
`Pinned #${n} (globalThis.playwriterPinnedElem${n})`,
|
|
268
|
-
`URL: ${location.href}`,
|
|
269
|
-
`Tag: ${tag}`,
|
|
270
|
-
!!id && `ID: ${id}`,
|
|
271
|
-
!!cls && `Classes: ${cls.slice(0, 200)}`,
|
|
272
|
-
!!role && `Role: ${role}`,
|
|
273
|
-
!!aria && `Aria-label: ${aria}`,
|
|
274
|
-
!!nameAttr && `Name: ${nameAttr}`,
|
|
275
|
-
!!href && `Href: ${href.slice(0, 200)}`,
|
|
276
|
-
!!typeAttr && `Type: ${typeAttr}`,
|
|
277
|
-
!!text && `Text: ${text}`,
|
|
278
|
-
`Rect: ${rect}`,
|
|
279
|
-
`Visible: ${visible}`
|
|
280
|
-
].filter((line) => typeof line === "string").join("\n");
|
|
281
|
-
}
|
|
282
|
-
function buildInspectionCode(n, url, summary) {
|
|
283
|
-
return `state.page=context.pages().find(x=>x.url()===${JSON.stringify(url).replace(/'/g, "\\u0027")})||context.pages()[0]; console.log(${JSON.stringify(summary).replace(/'/g, "\\u0027")}+"\\n\\nouterHTML:\\n"+await state.page.evaluate(n=>globalThis["playwriterPinnedElem"+n]?.outerHTML,${n}))`;
|
|
422
|
+
function buildInspectionCode(n, url) {
|
|
423
|
+
return `inspectPinnedElement(${JSON.stringify(url).replace(/'/g, "\\u0027")},"globalThis.playwriterPinnedElem${n}")`;
|
|
284
424
|
}
|
|
285
425
|
function onClick(e) {
|
|
286
426
|
if (isOverToolbar(e)) return;
|
|
@@ -293,8 +433,8 @@ function initPlaywriterToolbar() {
|
|
|
293
433
|
window[name] = target;
|
|
294
434
|
flashElement(target);
|
|
295
435
|
const url = location.href;
|
|
296
|
-
copyText("
|
|
297
|
-
showToast(
|
|
436
|
+
copyText("playwriter -e '" + buildInspectionCode(n, url) + "'");
|
|
437
|
+
showToast("Copied playwriter element reference, use it in your agent prompt", target.getBoundingClientRect());
|
|
298
438
|
setPinMode(false);
|
|
299
439
|
}
|
|
300
440
|
function onKeyDown(e) {
|
|
@@ -406,6 +546,9 @@ async function handleGhostBrowserCommand(params, chromeApi) {
|
|
|
406
546
|
//#region ../playwriter/dist/ghost-cursor-client.js?raw
|
|
407
547
|
var ghost_cursor_client_default = "(() => {\n var __defProp = Object.defineProperty;\n var __getOwnPropNames = Object.getOwnPropertyNames;\n var __getOwnPropDesc = Object.getOwnPropertyDescriptor;\n var __hasOwnProp = Object.prototype.hasOwnProperty;\n function __accessProp(key) {\n return this[key];\n }\n var __toCommonJS = (from) => {\n var entry = (__moduleCache ??= new WeakMap).get(from), desc;\n if (entry)\n return entry;\n entry = __defProp({}, \"__esModule\", { value: true });\n if (from && typeof from === \"object\" || typeof from === \"function\") {\n for (var key of __getOwnPropNames(from))\n if (!__hasOwnProp.call(entry, key))\n __defProp(entry, key, {\n get: __accessProp.bind(from, key),\n enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable\n });\n }\n __moduleCache.set(from, entry);\n return entry;\n };\n var __moduleCache;\n\n // src/ghost-cursor-client.ts\n var exports_ghost_cursor_client = {};\n\n // src/assets/cursors/screen-studio/pointer-macos-tahoe-data-url.ts\n var SCREENSTUDIO_POINTER_MACOS_TAHOE_DATA_URL = \"data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNjE4IiBoZWlnaHQ9Ijk1OCIgdmlld0JveD0iMCAwIDYxOCA5NTgiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxnIGZpbHRlcj0idXJsKCNmaWx0ZXIwX2RfMzg0XzI3KSI+CjxwYXRoIGZpbGwtcnVsZT0iZXZlbm9kZCIgY2xpcC1ydWxlPSJldmVub2RkIiBkPSJNMTI3LjA2MiAzNy4wMzMxTDU0MC42OTYgNDUxLjU1NUM1OTIuNjUzIDUwMy42NiA1NTUuNzk0IDU5Mi41NzQgNDgyLjIyNiA1OTIuNTc0TDQyMS44MzEgNTkyLjU2OUw0ODEuODIxIDczNS4wNTRDNDkyLjMzMSA3NjAuMDIxIDQ5Mi40NzkgNzg3LjY1MiA0ODIuMjY1IDgxMi43NjdDNDcyLjAwMiA4MzcuOTMyIDQ1Mi41NjEgODU3LjU3IDQyNy40OTYgODY4LjA4QzQxNC44NjQgODczLjM1OSA0MDEuNjQgODc2LjAyNCAzODguMTIxIDg3Ni4wMjRDMzQ3LjExNyA4NzYuMDI0IDMxMC4zNTggODUxLjYgMjk0LjQ3IDgxMy44MDRMMjMxLjQyIDY2My45MThMMTkwLjM2OCA3MDAuMzM3QzEzNy4wMjkgNzQ3LjUwOCA1MyA3MDkuNjYzIDUzIDYzOC40MTNWNjcuNjc0NEM1MyAyOC45OTAzIDk5LjcyNjggOS42NDgyOCAxMjcuMDYyIDM3LjAzMzFaIiBmaWxsPSJ3aGl0ZSIvPgo8cGF0aCBmaWxsLXJ1bGU9ImV2ZW5vZGQiIGNsaXAtcnVsZT0iZXZlbm9kZCIgZD0iTTEwMi4zMTYgOTkuNjUyQzEwMi4zMTYgOTMuMTg4MiAxMTAuMTYyIDg5LjkzMTYgMTE0LjcwMSA5NC41MjA0TDUwNC44OTcgNDg1LjU1NUM1MjYuMTY0IDUwNi44NzEgNTExLjA2NSA1NDMuMjM2IDQ4MC45NjcgNTQzLjIzNkwzNDcuNTQ2IDU0My4xNjFMNDM2LjM0MiA3NTQuMTQzQzQ0Ny41NDIgNzgwLjc4OCA0MzUuMDA5IDgxMS40MjkgNDA4LjQxNCA4MjIuNTgxQzM4MS43MiA4MzMuNzgxIDM1MS4xMjggODIxLjI5OCAzMzkuOTc3IDc5NC43MDJMMjUwLjI5MyA1ODEuMzUyTDE1OC41MTcgNjYyLjY0NEMxMzcuOTkxIDY4MC44MDEgMTA2LjMxOSA2NjguMTQ1IDEwMi42NjQgNjQyLjMyM0wxMDIuMzE2IDYzNy4zMzFWOTkuNjUyWiIgZmlsbD0iYmxhY2siLz4KPC9nPgo8ZGVmcz4KPGZpbHRlciBpZD0iZmlsdGVyMF9kXzM4NF8yNyIgeD0iMC4zNCIgeT0iMC43OTkyMTkiIHdpZHRoPSI2MTcuMzIiIGhlaWdodD0iOTU3LjE0NCIgZmlsdGVyVW5pdHM9InVzZXJTcGFjZU9uVXNlIiBjb2xvci1pbnRlcnBvbGF0aW9uLWZpbHRlcnM9InNSR0IiPgo8ZmVGbG9vZCBmbG9vZC1vcGFjaXR5PSIwIiByZXN1bHQ9IkJhY2tncm91bmRJbWFnZUZpeCIvPgo8ZmVDb2xvck1hdHJpeCBpbj0iU291cmNlQWxwaGEiIHR5cGU9Im1hdHJpeCIgdmFsdWVzPSIwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAxMjcgMCIgcmVzdWx0PSJoYXJkQWxwaGEiLz4KPGZlT2Zmc2V0IGR5PSIyOS4yNiIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSIyNi4zMyIvPgo8ZmVDb21wb3NpdGUgaW4yPSJoYXJkQWxwaGEiIG9wZXJhdG9yPSJvdXQiLz4KPGZlQ29sb3JNYXRyaXggdHlwZT0ibWF0cml4IiB2YWx1ZXM9IjAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAuNjUgMCIvPgo8ZmVCbGVuZCBtb2RlPSJub3JtYWwiIGluMj0iQmFja2dyb3VuZEltYWdlRml4IiByZXN1bHQ9ImVmZmVjdDFfZHJvcFNoYWRvd18zODRfMjciLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMzg0XzI3IiByZXN1bHQ9InNoYXBlIi8+CjwvZmlsdGVyPgo8L2RlZnM+Cjwvc3ZnPg==\";\n\n // src/ghost-cursor-client.ts\n var isTopFrame = (() => {\n try {\n return window === window.top;\n } catch {\n return false;\n }\n })();\n var CURSOR_ID = \"__playwriter_ghost_cursor__\";\n var SCREENSTUDIO_POINTER_ASPECT_RATIO = 618 / 958;\n var SCREENSTUDIO_HOTSPOT_X_RATIO = 0.14;\n var SCREENSTUDIO_HOTSPOT_Y_RATIO = 0.06;\n var MINIMAL_TRIANGLE_HOTSPOT_X_RATIO = 0.07;\n var MINIMAL_TRIANGLE_HOTSPOT_Y_RATIO = 0.06;\n var MOVE_EASING = \"cubic-bezier(0.65, 0, 0.35, 1)\";\n var PRESS_EASING = \"cubic-bezier(0.23, 1, 0.32, 1)\";\n var PRESS_DURATION_MS = 140;\n var IDLE_HIDE_DELAY_MS = 5000;\n var IDLE_FADE_OUT_MS = 600;\n var DEFAULT_OPTIONS = {\n style: \"minimal\",\n color: \"#111827\",\n size: 22,\n zIndex: 2147483647,\n easing: MOVE_EASING,\n minDurationMs: 220,\n maxDurationMs: 1500,\n speedPxPerMs: 1.2\n };\n var runtime = {\n outerElement: null,\n innerElement: null,\n options: DEFAULT_OPTIONS,\n x: 0,\n y: 0,\n scale: 1,\n hasPosition: false,\n enabled: false,\n idleHidden: false\n };\n var idleHideTimer = null;\n function clamp(options) {\n const { value, min, max } = options;\n return Math.min(max, Math.max(min, value));\n }\n function mergeOptions(options) {\n if (!options) {\n return DEFAULT_OPTIONS;\n }\n return {\n style: options.style ?? DEFAULT_OPTIONS.style,\n color: options.color ?? DEFAULT_OPTIONS.color,\n size: options.size ?? DEFAULT_OPTIONS.size,\n zIndex: options.zIndex ?? DEFAULT_OPTIONS.zIndex,\n easing: options.easing ?? DEFAULT_OPTIONS.easing,\n minDurationMs: options.minDurationMs ?? DEFAULT_OPTIONS.minDurationMs,\n maxDurationMs: options.maxDurationMs ?? DEFAULT_OPTIONS.maxDurationMs,\n speedPxPerMs: options.speedPxPerMs ?? DEFAULT_OPTIONS.speedPxPerMs\n };\n }\n function getCursorDimensions() {\n if (runtime.options.style === \"screenstudio\") {\n const height = runtime.options.size;\n const width = Math.max(10, Math.round(height * SCREENSTUDIO_POINTER_ASPECT_RATIO));\n return { width, height };\n }\n if (runtime.options.style === \"minimal\") {\n const size = Math.max(12, runtime.options.size);\n return { width: size, height: size };\n }\n return { width: runtime.options.size, height: runtime.options.size };\n }\n function getHotspotOffsetPx() {\n const dimensions = getCursorDimensions();\n if (runtime.options.style === \"screenstudio\") {\n return {\n x: Math.round(dimensions.width * SCREENSTUDIO_HOTSPOT_X_RATIO),\n y: Math.round(dimensions.height * SCREENSTUDIO_HOTSPOT_Y_RATIO)\n };\n }\n if (runtime.options.style === \"minimal\") {\n return {\n x: Math.round(dimensions.width * MINIMAL_TRIANGLE_HOTSPOT_X_RATIO),\n y: Math.round(dimensions.height * MINIMAL_TRIANGLE_HOTSPOT_Y_RATIO)\n };\n }\n return {\n x: Math.round(dimensions.width / 2),\n y: Math.round(dimensions.height / 2)\n };\n }\n function getBaseOpacity() {\n if (runtime.options.style === \"screenstudio\") {\n return \"0.95\";\n }\n if (runtime.options.style === \"minimal\") {\n return \"1\";\n }\n return \"0.72\";\n }\n function applyTranslate() {\n if (!runtime.outerElement) {\n return;\n }\n const hotspot = getHotspotOffsetPx();\n runtime.outerElement.style.transform = `translate3d(${runtime.x - hotspot.x}px, ${runtime.y - hotspot.y}px, 0)`;\n }\n function applyScale() {\n if (!runtime.innerElement) {\n return;\n }\n runtime.innerElement.style.transform = `scale(${runtime.scale})`;\n }\n function computeDurationMs(options) {\n if (!runtime.hasPosition) {\n return 0;\n }\n const dx = options.targetX - runtime.x;\n const dy = options.targetY - runtime.y;\n const distance = Math.hypot(dx, dy);\n const rawDurationMs = distance / runtime.options.speedPxPerMs;\n return clamp({\n value: rawDurationMs,\n min: runtime.options.minDurationMs,\n max: runtime.options.maxDurationMs\n });\n }\n function createCursorElement() {\n const outer = document.createElement(\"div\");\n outer.id = CURSOR_ID;\n outer.setAttribute(\"aria-hidden\", \"true\");\n outer.style.position = \"fixed\";\n outer.style.left = \"0\";\n outer.style.top = \"0\";\n outer.style.pointerEvents = \"none\";\n outer.style.zIndex = `${runtime.options.zIndex}`;\n outer.style.transitionProperty = \"transform\";\n outer.style.transitionTimingFunction = runtime.options.easing;\n outer.style.transitionDuration = \"0ms\";\n outer.style.willChange = \"transform\";\n const inner = document.createElement(\"div\");\n inner.style.transitionProperty = \"transform, opacity\";\n inner.style.transitionTimingFunction = PRESS_EASING;\n inner.style.transitionDuration = `${PRESS_DURATION_MS}ms`;\n inner.style.opacity = getBaseOpacity();\n outer.appendChild(inner);\n runtime.outerElement = outer;\n runtime.innerElement = inner;\n applyRuntimeVisualOptions();\n return outer;\n }\n function ensureCursorElement() {\n const existing = document.getElementById(CURSOR_ID);\n if (existing) {\n runtime.outerElement = existing;\n runtime.innerElement = existing.firstElementChild || null;\n return existing;\n }\n const outer = createCursorElement();\n const root = document.documentElement || document.body;\n root.appendChild(outer);\n return outer;\n }\n function applyRuntimeVisualOptions() {\n if (!runtime.innerElement) {\n return;\n }\n const dimensions = getCursorDimensions();\n runtime.innerElement.style.width = `${dimensions.width}px`;\n runtime.innerElement.style.height = `${dimensions.height}px`;\n if (runtime.outerElement) {\n runtime.outerElement.style.zIndex = `${runtime.options.zIndex}`;\n runtime.outerElement.style.transitionTimingFunction = runtime.options.easing;\n }\n const hotspot = getHotspotOffsetPx();\n runtime.innerElement.style.transformOrigin = `${hotspot.x}px ${hotspot.y}px`;\n if (runtime.options.style === \"screenstudio\") {\n runtime.innerElement.style.borderRadius = \"0\";\n runtime.innerElement.style.border = \"none\";\n runtime.innerElement.style.backgroundColor = \"transparent\";\n runtime.innerElement.style.backgroundImage = `url(\"${SCREENSTUDIO_POINTER_MACOS_TAHOE_DATA_URL}\")`;\n runtime.innerElement.style.backgroundRepeat = \"no-repeat\";\n runtime.innerElement.style.backgroundPosition = \"left top\";\n runtime.innerElement.style.backgroundSize = \"contain\";\n runtime.innerElement.style.backdropFilter = \"none\";\n runtime.innerElement.style.filter = \"none\";\n runtime.innerElement.style.boxShadow = \"none\";\n runtime.innerElement.style.opacity = getBaseOpacity();\n return;\n }\n if (runtime.options.style === \"minimal\") {\n const triangleSvg = `<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"32\" height=\"32\" viewBox=\"-1 -1 26 26\"><path fill=\"white\" stroke=\"${runtime.options.color}\" stroke-width=\"1.5\" stroke-linejoin=\"round\" d=\"m23.284 19.124l-6.866-6.895a.4.4 0 0 1-.118-.296a.43.43 0 0 1 .163-.282l4.439-3.077a1.48 1.48 0 0 0 .621-1.48a1.48 1.48 0 0 0-1.036-1.198L1.623.302a1.14 1.14 0 0 0-1.11.282A1.13 1.13 0 0 0 .29 1.649L5.928 20.44a1.48 1.48 0 0 0 1.183 1.035a1.48 1.48 0 0 0 1.48-.621l3.078-4.44a.37.37 0 0 1 .31-.118a.43.43 0 0 1 .296.104l6.91 6.91a1.48 1.48 0 0 0 2.087 0l2.086-2.086a1.48 1.48 0 0 0-.074-2.101\"/></svg>`;\n const triangleDataUrl = `url(\"data:image/svg+xml,${encodeURIComponent(triangleSvg)}\")`;\n runtime.innerElement.style.borderRadius = \"0\";\n runtime.innerElement.style.border = \"none\";\n runtime.innerElement.style.backgroundColor = \"transparent\";\n runtime.innerElement.style.backgroundImage = triangleDataUrl;\n runtime.innerElement.style.backgroundRepeat = \"no-repeat\";\n runtime.innerElement.style.backgroundSize = \"contain\";\n runtime.innerElement.style.backgroundPosition = \"left top\";\n runtime.innerElement.style.backdropFilter = \"none\";\n runtime.innerElement.style.boxShadow = \"none\";\n runtime.innerElement.style.filter = \"drop-shadow(0 1px 2px rgba(0, 0, 0, 0.4))\";\n runtime.innerElement.style.opacity = getBaseOpacity();\n return;\n }\n runtime.innerElement.style.borderRadius = \"999px\";\n runtime.innerElement.style.border = \"none\";\n runtime.innerElement.style.backgroundColor = runtime.options.color;\n runtime.innerElement.style.backgroundImage = \"none\";\n runtime.innerElement.style.backdropFilter = \"none\";\n runtime.innerElement.style.filter = \"none\";\n runtime.innerElement.style.boxShadow = \"0 2px 10px rgba(0, 0, 0, 0.18), inset 0 0 0 2px rgba(255, 255, 255, 0.55)\";\n runtime.innerElement.style.opacity = getBaseOpacity();\n }\n function clearIdleHideTimer() {\n if (idleHideTimer !== null) {\n clearTimeout(idleHideTimer);\n idleHideTimer = null;\n }\n }\n function scheduleIdleHide() {\n clearIdleHideTimer();\n idleHideTimer = setTimeout(() => {\n idleHideTimer = null;\n if (!runtime.enabled || !runtime.innerElement) {\n return;\n }\n runtime.idleHidden = true;\n runtime.innerElement.style.transitionDuration = `${IDLE_FADE_OUT_MS}ms`;\n runtime.innerElement.style.transitionTimingFunction = PRESS_EASING;\n runtime.innerElement.style.opacity = \"0\";\n }, IDLE_HIDE_DELAY_MS);\n }\n function wakeFromIdle(options) {\n runtime.x = options.x;\n runtime.y = options.y;\n runtime.hasPosition = true;\n if (runtime.innerElement) {\n runtime.innerElement.style.transitionDuration = `${PRESS_DURATION_MS}ms`;\n runtime.innerElement.style.transitionTimingFunction = PRESS_EASING;\n runtime.innerElement.style.opacity = getBaseOpacity();\n }\n }\n function moveCursor(options) {\n if (!runtime.enabled) {\n return;\n }\n ensureCursorElement();\n const durationMs = computeDurationMs({ targetX: options.x, targetY: options.y });\n if (runtime.outerElement) {\n runtime.outerElement.style.transitionDuration = `${Math.round(durationMs)}ms`;\n runtime.outerElement.style.transitionTimingFunction = runtime.options.easing;\n }\n runtime.x = options.x;\n runtime.y = options.y;\n runtime.hasPosition = true;\n applyTranslate();\n }\n function setPressed(options) {\n if (!runtime.enabled || !runtime.innerElement) {\n return;\n }\n runtime.scale = options.pressed ? runtime.options.style === \"dot\" ? 0.92 : 0.95 : 1;\n runtime.innerElement.style.transitionDuration = `${PRESS_DURATION_MS}ms`;\n runtime.innerElement.style.transitionTimingFunction = PRESS_EASING;\n runtime.innerElement.style.opacity = options.pressed ? \"1\" : getBaseOpacity();\n applyScale();\n }\n function enable(options) {\n runtime.options = mergeOptions(options);\n runtime.enabled = true;\n ensureCursorElement();\n applyRuntimeVisualOptions();\n if (!runtime.hasPosition) {\n runtime.x = Math.round(window.innerWidth / 2);\n runtime.y = Math.round(window.innerHeight / 2);\n runtime.scale = 1;\n runtime.hasPosition = true;\n }\n runtime.idleHidden = false;\n if (runtime.innerElement) {\n runtime.innerElement.style.opacity = getBaseOpacity();\n }\n applyTranslate();\n applyScale();\n scheduleIdleHide();\n }\n function disable() {\n runtime.enabled = false;\n runtime.scale = 1;\n runtime.hasPosition = false;\n runtime.idleHidden = false;\n clearIdleHideTimer();\n if (runtime.outerElement) {\n runtime.outerElement.remove();\n runtime.outerElement = null;\n runtime.innerElement = null;\n }\n }\n function applyMouseAction(action) {\n if (!runtime.enabled) {\n return;\n }\n if (runtime.idleHidden) {\n runtime.idleHidden = false;\n wakeFromIdle({ x: action.x, y: action.y });\n }\n if (action.type === \"move\" || action.type === \"wheel\") {\n moveCursor({ x: action.x, y: action.y });\n } else if (action.type === \"down\") {\n moveCursor({ x: action.x, y: action.y });\n setPressed({ pressed: true });\n } else if (action.type === \"up\") {\n moveCursor({ x: action.x, y: action.y });\n setPressed({ pressed: false });\n }\n scheduleIdleHide();\n }\n var api = {\n enable,\n disable,\n applyMouseAction,\n isEnabled: () => {\n return runtime.enabled;\n }\n };\n if (isTopFrame) {\n globalThis.__playwriterGhostCursor = api;\n try {\n if (document.readyState === \"loading\") {\n document.addEventListener(\"DOMContentLoaded\", () => {\n try {\n api.enable();\n } catch {}\n }, { once: true });\n } else {\n api.enable();\n }\n } catch {}\n }\n})();\n";
|
|
408
548
|
//#endregion
|
|
549
|
+
//#region ../playwriter/dist/bippy.js?raw
|
|
550
|
+
var bippy_default = "(() => {\n // ../node_modules/.pnpm/bippy@0.5.27_@types+react@19.2.7_react@19.2.6/node_modules/bippy/dist/rdt-hook-BZMdLD7S.js\n var e = `0.5.27`;\n var t = `bippy-${e}`;\n var n = Object.defineProperty;\n var r = Object.prototype.hasOwnProperty;\n var i = () => {};\n var a = (e2) => {\n try {\n let t2 = Function.prototype.toString.call(e2);\n t2.indexOf(`^_^`) > -1 && setTimeout(() => {\n throw Error(`React is running in production mode, but dead code elimination has not been applied. Read how to correctly configure React for production: https://reactjs.org/link/perf-use-production-build`);\n });\n } catch {}\n };\n var o = (e2 = h()) => (`getFiberRoots` in e2);\n var s = false;\n var c;\n var l = (e2 = h()) => s ? true : (typeof e2.inject == `function` && (c = e2.inject.toString()), !!c?.includes(`(injected)`));\n var u = new Set;\n var d = new Set;\n var f = (e2) => {\n let r2 = new Map, o2 = 0, s2 = { _instrumentationIsActive: false, _instrumentationSource: t, checkDCE: a, hasUnsupportedRendererAttached: false, inject(e3) {\n let t2 = ++o2;\n return r2.set(t2, e3), d.add(e3), s2._instrumentationIsActive || (s2._instrumentationIsActive = true, u.forEach((e4) => e4())), t2;\n }, on: i, onCommitFiberRoot: i, onCommitFiberUnmount: i, onPostCommitFiberRoot: i, renderers: r2, supportsFiber: true, supportsFlight: true };\n try {\n n(globalThis, `__REACT_DEVTOOLS_GLOBAL_HOOK__`, { configurable: true, enumerable: true, get() {\n return s2;\n }, set(t3) {\n if (t3 && typeof t3 == `object`) {\n let n2 = s2.renderers;\n s2 = t3, n2.size > 0 && (n2.forEach((e3, n3) => {\n d.add(e3), t3.renderers.set(n3, e3);\n }), p(e2));\n }\n } });\n let t2 = window.hasOwnProperty, r3 = false;\n n(window, `hasOwnProperty`, { configurable: true, value: function(...e3) {\n try {\n if (!r3 && e3[0] === `__REACT_DEVTOOLS_GLOBAL_HOOK__`)\n return globalThis.__REACT_DEVTOOLS_GLOBAL_HOOK__ = undefined, r3 = true, -0;\n } catch {}\n return t2.apply(this, e3);\n }, writable: true });\n } catch {\n p(e2);\n }\n return s2;\n };\n var p = (e2) => {\n e2 && u.add(e2);\n try {\n let n2 = globalThis.__REACT_DEVTOOLS_GLOBAL_HOOK__;\n if (!n2)\n return;\n if (!n2._instrumentationSource) {\n n2.checkDCE = a, n2.supportsFiber = true, n2.supportsFlight = true, n2.hasUnsupportedRendererAttached = false, n2._instrumentationSource = t, n2._instrumentationIsActive = false;\n let e3 = o(n2);\n if (e3 || (n2.on = i), n2.renderers.size) {\n n2._instrumentationIsActive = true, u.forEach((e4) => e4());\n return;\n }\n let r2 = n2.inject, c2 = l(n2);\n if (c2 && !e3) {\n s = true;\n let e4 = n2.inject({ scheduleRefresh() {} });\n e4 && (n2._instrumentationIsActive = true);\n }\n n2.inject = (e4) => {\n let t2 = r2(e4);\n return d.add(e4), c2 && n2.renderers.set(t2, e4), n2._instrumentationIsActive = true, u.forEach((e5) => e5()), t2;\n };\n }\n (n2.renderers.size || n2._instrumentationIsActive || l()) && e2?.();\n } catch {}\n };\n var m = () => r.call(globalThis, `__REACT_DEVTOOLS_GLOBAL_HOOK__`);\n var h = (e2) => m() ? (p(e2), globalThis.__REACT_DEVTOOLS_GLOBAL_HOOK__) : f(e2);\n var g = () => !!(typeof window < `u` && (window.document?.createElement || window.navigator?.product === `ReactNative`));\n var _ = () => {\n try {\n g() && h();\n } catch {}\n };\n\n // ../node_modules/.pnpm/bippy@0.5.27_@types+react@19.2.7_react@19.2.6/node_modules/bippy/dist/install-hook-only-BOBPiBkc.js\n _();\n\n // ../node_modules/.pnpm/bippy@0.5.27_@types+react@19.2.7_react@19.2.6/node_modules/bippy/dist/core-coQbWNwP.js\n var a2 = 0;\n var o2 = 1;\n var c2 = 5;\n var f2 = 11;\n var p2 = 13;\n var m2 = 14;\n var h2 = 15;\n var ee = 16;\n var te = 19;\n var y = 26;\n var b = 27;\n var ne = 28;\n var re = 30;\n var ie = 2;\n var ae = 4096;\n var oe = 4;\n var se = 16;\n var ce = 32;\n var le = 1024;\n var ue = 8192;\n var O = ie | oe | se | ce | ae | ue | le;\n var k = (e2) => {\n switch (e2.tag) {\n case c2:\n case y:\n case b:\n return true;\n default:\n return typeof e2.type == `string`;\n }\n };\n var pe = (e2) => {\n switch (e2.tag) {\n case o2:\n case f2:\n case a2:\n case m2:\n case h2:\n return true;\n default:\n return false;\n }\n };\n function N(e2, t2, n2 = false) {\n if (!e2)\n return null;\n let r2 = t2(e2);\n if (r2 instanceof Promise)\n return (async () => {\n if (await r2 === true)\n return e2;\n let i3 = n2 ? e2.return : e2.child;\n for (;i3; ) {\n let e3 = await F(i3, t2, n2);\n if (e3)\n return e3;\n i3 = n2 ? null : i3.sibling;\n }\n return null;\n })();\n if (r2 === true)\n return e2;\n let i2 = n2 ? e2.return : e2.child;\n for (;i2; ) {\n let e3 = P(i2, t2, n2);\n if (e3)\n return e3;\n i2 = n2 ? null : i2.sibling;\n }\n return null;\n }\n var P = (e2, t2, n2 = false) => {\n if (!e2)\n return null;\n if (t2(e2) === true)\n return e2;\n let r2 = n2 ? e2.return : e2.child;\n for (;r2; ) {\n let e3 = P(r2, t2, n2);\n if (e3)\n return e3;\n r2 = n2 ? null : r2.sibling;\n }\n return null;\n };\n var F = async (e2, t2, n2 = false) => {\n if (!e2)\n return null;\n if (await t2(e2) === true)\n return e2;\n let r2 = n2 ? e2.return : e2.child;\n for (;r2; ) {\n let e3 = await F(r2, t2, n2);\n if (e3)\n return e3;\n r2 = n2 ? null : r2.sibling;\n }\n return null;\n };\n var I = (e2) => {\n let t2 = e2;\n return typeof t2 == `function` ? t2 : typeof t2 == `object` && t2 ? I(t2.type || t2.render) : null;\n };\n var Te = (e2) => {\n let t2 = e2;\n if (typeof t2 == `string`)\n return t2;\n if (typeof t2 != `function` && !(typeof t2 == `object` && t2))\n return null;\n let n2 = t2.displayName || t2.name || null;\n if (n2)\n return n2;\n let r2 = I(t2);\n return r2 && (r2.displayName || r2.name) || null;\n };\n var z = new WeakMap;\n var K = new WeakMap;\n var Pe = (e2) => {\n let n2 = h();\n for (let t2 of n2.renderers.values())\n try {\n let n3 = t2.findFiberByHostInstance?.(e2);\n if (n3)\n return n3;\n } catch {}\n if (typeof e2 == `object` && e2) {\n if (`_reactRootContainer` in e2)\n return e2._reactRootContainer?._internalRoot?.current?.child;\n for (let t2 in e2)\n if (t2.startsWith(`__reactContainer$`) || t2.startsWith(`__reactInternalInstance$`) || t2.startsWith(`__reactFiber`))\n return e2[t2] || null;\n }\n return null;\n };\n var Fe = Error();\n var $ = new Set;\n\n // ../node_modules/.pnpm/bippy@0.5.27_@types+react@19.2.7_react@19.2.6/node_modules/bippy/dist/source.js\n var g3 = Object.create;\n var _3 = Object.defineProperty;\n var v2 = Object.getOwnPropertyDescriptor;\n var y2 = Object.getOwnPropertyNames;\n var b2 = Object.getPrototypeOf;\n var x2 = Object.prototype.hasOwnProperty;\n var S2 = (e2, t2) => () => (t2 || e2((t2 = { exports: {} }).exports, t2), t2.exports);\n var ee2 = (e2, t2, n2, r2) => {\n if (t2 && typeof t2 == `object` || typeof t2 == `function`)\n for (var i2 = y2(t2), a3 = 0, o3 = i2.length, s3;a3 < o3; a3++)\n s3 = i2[a3], !x2.call(e2, s3) && s3 !== n2 && _3(e2, s3, { get: ((e3) => t2[e3]).bind(null, s3), enumerable: !(r2 = v2(t2, s3)) || r2.enumerable });\n return e2;\n };\n var C2 = (e2, t2, n2) => (n2 = e2 == null ? {} : g3(b2(e2)), ee2(t2 || !e2 || !e2.__esModule ? _3(n2, `default`, { value: e2, enumerable: true }) : n2, e2));\n var w2 = /^[a-zA-Z][a-zA-Z\\d+\\-.]*:/;\n var te2 = [`rsc://`, `file:///`, `webpack://`, `webpack-internal://`, `node:`, `turbopack://`, `metro://`, `/app-pages-browser/`];\n var T2 = `about://React/`;\n var ne2 = [`<anonymous>`, `eval`, ``];\n var re2 = /\\.(jsx|tsx|ts|js)$/;\n var ie2 = /(\\.min|bundle|chunk|vendor|vendors|runtime|polyfill|polyfills)\\.(js|mjs|cjs)$|(chunk|bundle|vendor|vendors|runtime|polyfill|polyfills|framework|app|main|index)[-_.][A-Za-z0-9_-]{4,}\\.(js|mjs|cjs)$|[\\da-f]{8,}\\.(js|mjs|cjs)$|[-_.][\\da-f]{20,}\\.(js|mjs|cjs)$|\\/dist\\/|\\/build\\/|\\/.next\\/|\\/out\\/|\\/node_modules\\/|\\.webpack\\.|\\.vite\\.|\\.turbopack\\./i;\n var ae2 = /^\\?[\\w~.\\-]+(?:=[^&#]*)?(?:&[\\w~.\\-]+(?:=[^&#]*)?)*$/;\n var E = `(at Server)`;\n var oe2 = /(^|@)\\S+:\\d+/;\n var se2 = /^\\s*at .*(\\S+:\\d+|\\(native\\))/m;\n var ce2 = /^(eval@)?(\\[native code\\])?$/;\n var D = (e2, t2) => {\n if (t2?.includeInElement !== false) {\n let n2 = e2.split(`\n`), r2 = [];\n for (let e3 of n2)\n if (/^\\s*at\\s+/.test(e3)) {\n let t3 = A2(e3, undefined)[0];\n t3 && r2.push(t3);\n } else if (/^\\s*in\\s+/.test(e3)) {\n let t3 = e3.replace(/^\\s*in\\s+/, ``).replace(/\\s*\\(at .*\\)$/, ``);\n r2.push({ functionName: t3, source: e3 });\n } else if (e3.match(oe2)) {\n let t3 = j2(e3, undefined)[0];\n t3 && r2.push(t3);\n }\n return k2(r2, t2);\n }\n return e2.match(se2) ? A2(e2, t2) : j2(e2, t2);\n };\n var O2 = (e2) => {\n if (!e2.includes(`:`))\n return [e2, undefined, undefined];\n let t2 = /(.+?)(?::(\\d+))?(?::(\\d+))?$/, n2 = e2.startsWith(`(`) && e2.endsWith(`)`) ? e2.slice(1, -1) : e2, r2 = t2.exec(n2);\n return [r2[1], r2[2] || undefined, r2[3] || undefined];\n };\n var k2 = (e2, t2) => t2 && t2.slice != null ? Array.isArray(t2.slice) ? e2.slice(t2.slice[0], t2.slice[1]) : e2.slice(0, t2.slice) : e2;\n var A2 = (e2, t2) => {\n let n2 = k2(e2.split(`\n`).filter((e3) => !!e3.match(se2)), t2);\n return n2.map((e3) => {\n let t3 = e3;\n t3.includes(`(eval `) && (t3 = t3.replace(/eval code/g, `eval`).replace(/(\\(eval at [^()]*)|(,.*$)/g, ``));\n let n3 = t3.replace(/^\\s+/, ``).replace(/\\(eval code/g, `(`).replace(/^.*?\\s+/, ``), r2 = n3.match(/ (\\(.+\\)$)/);\n n3 = r2 ? n3.replace(r2[0], ``) : n3;\n let i2 = O2(r2 ? r2[1] : n3), a3 = r2 && n3 || undefined, o3 = [`eval`, `<anonymous>`].includes(i2[0]) ? undefined : i2[0];\n return { functionName: a3, fileName: o3, lineNumber: i2[1] ? +i2[1] : undefined, columnNumber: i2[2] ? +i2[2] : undefined, source: t3 };\n });\n };\n var j2 = (e2, t2) => {\n let n2 = k2(e2.split(`\n`).filter((e3) => !e3.match(ce2)), t2);\n return n2.map((e3) => {\n let t3 = e3;\n if (t3.includes(` > eval`) && (t3 = t3.replace(/ line (\\d+)(?: > eval line \\d+)* > eval:\\d+:\\d+/g, `:$1`)), !t3.includes(`@`) && !t3.includes(`:`))\n return { functionName: t3 };\n {\n let e4 = /(([^\\n\\r\"\\u2028\\u2029]*\".[^\\n\\r\"\\u2028\\u2029]*\"[^\\n\\r@\\u2028\\u2029]*(?:@[^\\n\\r\"\\u2028\\u2029]*\"[^\\n\\r@\\u2028\\u2029]*)*(?:[\\n\\r\\u2028\\u2029][^@]*)?)?[^@]*)@/, n3 = t3.match(e4), r2 = n3 && n3[1] ? n3[1] : undefined, i2 = O2(t3.replace(e4, ``));\n return { functionName: r2, fileName: i2[0], lineNumber: i2[1] ? +i2[1] : undefined, columnNumber: i2[2] ? +i2[2] : undefined, source: t3 };\n }\n });\n };\n var pe2 = S2((exports, t2) => {\n (function(n2, r2) {\n typeof exports == `object` && t2 !== undefined ? r2(exports) : typeof define == `function` && define.amd ? define([`exports`], r2) : (n2 = typeof globalThis < `u` ? globalThis : n2 || self, r2(n2.sourcemapCodec = {}));\n })(undefined, function(e2) {\n let t3 = 44, n2 = 59, r2 = `ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/`, i2 = new Uint8Array(64), a3 = new Uint8Array(128);\n for (let e3 = 0;e3 < r2.length; e3++) {\n let t4 = r2.charCodeAt(e3);\n i2[e3] = t4, a3[t4] = e3;\n }\n function o3(e3, t4) {\n let n3 = 0, r3 = 0, i3 = 0;\n do {\n let t5 = e3.next();\n i3 = a3[t5], n3 |= (i3 & 31) << r3, r3 += 5;\n } while (i3 & 32);\n let o4 = n3 & 1;\n return n3 >>>= 1, o4 && (n3 = -2147483648 | -n3), t4 + n3;\n }\n function s3(e3, t4, n3) {\n let r3 = t4 - n3;\n r3 = r3 < 0 ? -r3 << 1 | 1 : r3 << 1;\n do {\n let t5 = r3 & 31;\n r3 >>>= 5, r3 > 0 && (t5 |= 32), e3.write(i2[t5]);\n } while (r3 > 0);\n return t4;\n }\n function c3(e3, n3) {\n return e3.pos >= n3 ? false : e3.peek() !== t3;\n }\n let l3 = 1024 * 16, u3 = typeof TextDecoder < `u` ? new TextDecoder : typeof Buffer < `u` ? { decode(e3) {\n let t4 = Buffer.from(e3.buffer, e3.byteOffset, e3.byteLength);\n return t4.toString();\n } } : { decode(e3) {\n let t4 = ``;\n for (let n3 = 0;n3 < e3.length; n3++)\n t4 += String.fromCharCode(e3[n3]);\n return t4;\n } };\n\n class d3 {\n constructor() {\n this.pos = 0, this.out = ``, this.buffer = new Uint8Array(l3);\n }\n write(e3) {\n let { buffer: t4 } = this;\n t4[this.pos++] = e3, this.pos === l3 && (this.out += u3.decode(t4), this.pos = 0);\n }\n flush() {\n let { buffer: e3, out: t4, pos: n3 } = this;\n return n3 > 0 ? t4 + u3.decode(e3.subarray(0, n3)) : t4;\n }\n }\n\n class f3 {\n constructor(e3) {\n this.pos = 0, this.buffer = e3;\n }\n next() {\n return this.buffer.charCodeAt(this.pos++);\n }\n peek() {\n return this.buffer.charCodeAt(this.pos);\n }\n indexOf(e3) {\n let { buffer: t4, pos: n3 } = this, r3 = t4.indexOf(e3, n3);\n return r3 === -1 ? t4.length : r3;\n }\n }\n let p3 = [];\n function m3(e3) {\n let { length: t4 } = e3, n3 = new f3(e3), r3 = [], i3 = [], a4 = 0;\n for (;n3.pos < t4; n3.pos++) {\n a4 = o3(n3, a4);\n let e4 = o3(n3, 0);\n if (!c3(n3, t4)) {\n let t5 = i3.pop();\n t5[2] = a4, t5[3] = e4;\n continue;\n }\n let s4 = o3(n3, 0), l4 = o3(n3, 0), u4 = l4 & 1, d4 = u4 ? [a4, e4, 0, 0, s4, o3(n3, 0)] : [a4, e4, 0, 0, s4], f4 = p3;\n if (c3(n3, t4)) {\n f4 = [];\n do {\n let e5 = o3(n3, 0);\n f4.push(e5);\n } while (c3(n3, t4));\n }\n d4.vars = f4, r3.push(d4), i3.push(d4);\n }\n return r3;\n }\n function h3(e3) {\n let t4 = new d3;\n for (let n3 = 0;n3 < e3.length; )\n n3 = g4(e3, n3, t4, [0]);\n return t4.flush();\n }\n function g4(e3, n3, r3, i3) {\n let a4 = e3[n3], { 0: o4, 1: c4, 2: l4, 3: u4, 4: d4, vars: f4 } = a4;\n n3 > 0 && r3.write(t3), i3[0] = s3(r3, o4, i3[0]), s3(r3, c4, 0), s3(r3, d4, 0);\n let p4 = a4.length === 6 ? 1 : 0;\n s3(r3, p4, 0), a4.length === 6 && s3(r3, a4[5], 0);\n for (let e4 of f4)\n s3(r3, e4, 0);\n for (n3++;n3 < e3.length; ) {\n let t4 = e3[n3], { 0: a5, 1: o5 } = t4;\n if (a5 > l4 || a5 === l4 && o5 >= u4)\n break;\n n3 = g4(e3, n3, r3, i3);\n }\n return r3.write(t3), i3[0] = s3(r3, l4, i3[0]), s3(r3, u4, 0), n3;\n }\n function _4(e3) {\n let { length: t4 } = e3, n3 = new f3(e3), r3 = [], i3 = [], a4 = 0, s4 = 0, l4 = 0, u4 = 0, d4 = 0, m4 = 0, h4 = 0, g5 = 0;\n do {\n let e4 = n3.indexOf(`;`), t5 = 0;\n for (;n3.pos < e4; n3.pos++) {\n if (t5 = o3(n3, t5), !c3(n3, e4)) {\n let e5 = i3.pop();\n e5[2] = a4, e5[3] = t5;\n continue;\n }\n let f4 = o3(n3, 0), _5 = f4 & 1, v4 = f4 & 2, y4 = f4 & 4, b4 = null, x4 = p3, S4;\n if (_5) {\n let e5 = o3(n3, s4);\n l4 = o3(n3, s4 === e5 ? l4 : 0), s4 = e5, S4 = [a4, t5, 0, 0, e5, l4];\n } else\n S4 = [a4, t5, 0, 0];\n if (S4.isScope = !!y4, v4) {\n let e5 = u4, t6 = d4;\n u4 = o3(n3, u4);\n let r4 = e5 === u4;\n d4 = o3(n3, r4 ? d4 : 0), m4 = o3(n3, r4 && t6 === d4 ? m4 : 0), b4 = [u4, d4, m4];\n }\n if (S4.callsite = b4, c3(n3, e4)) {\n x4 = [];\n do {\n h4 = a4, g5 = t5;\n let e5 = o3(n3, 0), r4;\n if (e5 < -1) {\n r4 = [[o3(n3, 0)]];\n for (let t6 = -1;t6 > e5; t6--) {\n let e6 = h4;\n h4 = o3(n3, h4), g5 = o3(n3, h4 === e6 ? g5 : 0);\n let t7 = o3(n3, 0);\n r4.push([t7, h4, g5]);\n }\n } else\n r4 = [[e5]];\n x4.push(r4);\n } while (c3(n3, e4));\n }\n S4.bindings = x4, r3.push(S4), i3.push(S4);\n }\n a4++, n3.pos = e4 + 1;\n } while (n3.pos < t4);\n return r3;\n }\n function v3(e3) {\n if (e3.length === 0)\n return ``;\n let t4 = new d3;\n for (let n3 = 0;n3 < e3.length; )\n n3 = y3(e3, n3, t4, [0, 0, 0, 0, 0, 0, 0]);\n return t4.flush();\n }\n function y3(e3, n3, r3, i3) {\n let a4 = e3[n3], { 0: o4, 1: c4, 2: l4, 3: u4, isScope: d4, callsite: f4, bindings: p4 } = a4;\n i3[0] < o4 ? (b3(r3, i3[0], o4), i3[0] = o4, i3[1] = 0) : n3 > 0 && r3.write(t3), i3[1] = s3(r3, a4[1], i3[1]);\n let m4 = (a4.length === 6 ? 1 : 0) | (f4 ? 2 : 0) | (d4 ? 4 : 0);\n if (s3(r3, m4, 0), a4.length === 6) {\n let { 4: e4, 5: t4 } = a4;\n e4 !== i3[2] && (i3[3] = 0), i3[2] = s3(r3, e4, i3[2]), i3[3] = s3(r3, t4, i3[3]);\n }\n if (f4) {\n let { 0: e4, 1: t4, 2: n4 } = a4.callsite;\n e4 === i3[4] ? t4 !== i3[5] && (i3[6] = 0) : (i3[5] = 0, i3[6] = 0), i3[4] = s3(r3, e4, i3[4]), i3[5] = s3(r3, t4, i3[5]), i3[6] = s3(r3, n4, i3[6]);\n }\n if (p4)\n for (let e4 of p4) {\n e4.length > 1 && s3(r3, -e4.length, 0);\n let t4 = e4[0][0];\n s3(r3, t4, 0);\n let n4 = o4, i4 = c4;\n for (let t5 = 1;t5 < e4.length; t5++) {\n let a5 = e4[t5];\n n4 = s3(r3, a5[1], n4), i4 = s3(r3, a5[2], i4), s3(r3, a5[0], 0);\n }\n }\n for (n3++;n3 < e3.length; ) {\n let t4 = e3[n3], { 0: a5, 1: o5 } = t4;\n if (a5 > l4 || a5 === l4 && o5 >= u4)\n break;\n n3 = y3(e3, n3, r3, i3);\n }\n return i3[0] < l4 ? (b3(r3, i3[0], l4), i3[0] = l4, i3[1] = 0) : r3.write(t3), i3[1] = s3(r3, u4, i3[1]), n3;\n }\n function b3(e3, t4, r3) {\n do\n e3.write(n2);\n while (++t4 < r3);\n }\n function x3(e3) {\n let { length: t4 } = e3, n3 = new f3(e3), r3 = [], i3 = 0, a4 = 0, s4 = 0, l4 = 0, u4 = 0;\n do {\n let e4 = n3.indexOf(`;`), t5 = [], d4 = true, f4 = 0;\n for (i3 = 0;n3.pos < e4; ) {\n let r4;\n i3 = o3(n3, i3), i3 < f4 && (d4 = false), f4 = i3, c3(n3, e4) ? (a4 = o3(n3, a4), s4 = o3(n3, s4), l4 = o3(n3, l4), c3(n3, e4) ? (u4 = o3(n3, u4), r4 = [i3, a4, s4, l4, u4]) : r4 = [i3, a4, s4, l4]) : r4 = [i3], t5.push(r4), n3.pos++;\n }\n d4 || S3(t5), r3.push(t5), n3.pos = e4 + 1;\n } while (n3.pos <= t4);\n return r3;\n }\n function S3(e3) {\n e3.sort(ee3);\n }\n function ee3(e3, t4) {\n return e3[0] - t4[0];\n }\n function C3(e3) {\n let r3 = new d3, i3 = 0, a4 = 0, o4 = 0, c4 = 0;\n for (let l4 = 0;l4 < e3.length; l4++) {\n let u4 = e3[l4];\n if (l4 > 0 && r3.write(n2), u4.length === 0)\n continue;\n let d4 = 0;\n for (let e4 = 0;e4 < u4.length; e4++) {\n let n3 = u4[e4];\n e4 > 0 && r3.write(t3), d4 = s3(r3, n3[0], d4), n3.length !== 1 && (i3 = s3(r3, n3[1], i3), a4 = s3(r3, n3[2], a4), o4 = s3(r3, n3[3], o4), n3.length !== 4 && (c4 = s3(r3, n3[4], c4)));\n }\n }\n return r3.flush();\n }\n e2.decode = x3, e2.decodeGeneratedRanges = _4, e2.decodeOriginalScopes = m3, e2.encode = C3, e2.encodeGeneratedRanges = v3, e2.encodeOriginalScopes = h3, Object.defineProperty(e2, `__esModule`, { value: true });\n });\n });\n var F2 = C2(pe2(), 1);\n var I2 = /^[a-zA-Z][a-zA-Z\\d+\\-.]*:/;\n var me2 = /^data:application\\/json[^,]+base64,/;\n var he2 = /(?:\\/\\/[@#][ \\t]+sourceMappingURL=([^\\s'\"]+?)[ \\t]*$)|(?:\\/\\*[@#][ \\t]+sourceMappingURL=([^*]+?)[ \\t]*(?:\\*\\/)[ \\t]*$)/;\n var L2 = typeof WeakRef < `u`;\n var R = new Map;\n var z2 = new Map;\n var ge2 = (e2) => L2 && e2 instanceof WeakRef;\n var B2 = (e2, t2, n2, r2) => {\n if (n2 < 0 || n2 >= e2.length)\n return null;\n let i2 = e2[n2];\n if (!i2 || i2.length === 0)\n return null;\n let a3 = null;\n for (let e3 of i2)\n if (e3[0] <= r2)\n a3 = e3;\n else\n break;\n if (!a3 || a3.length < 4)\n return null;\n let [, o3, s3, c3] = a3;\n if (o3 === undefined || s3 === undefined || c3 === undefined)\n return null;\n let l3 = t2[o3];\n return l3 ? { columnNumber: c3, fileName: l3, lineNumber: s3 + 1 } : null;\n };\n var V2 = (e2, t2, n2) => {\n if (e2.sections) {\n let r2 = null;\n for (let i3 of e2.sections)\n if (t2 > i3.offset.line || t2 === i3.offset.line && n2 >= i3.offset.column)\n r2 = i3;\n else\n break;\n if (!r2)\n return null;\n let i2 = t2 - r2.offset.line, a3 = t2 === r2.offset.line ? n2 - r2.offset.column : n2;\n return B2(r2.map.mappings, r2.map.sources, i2, a3);\n }\n return B2(e2.mappings, e2.sources, t2 - 1, n2);\n };\n var _e2 = (e2, t2) => {\n let n2 = t2.split(`\n`), r2;\n for (let e3 = n2.length - 1;e3 >= 0 && !r2; e3--) {\n let t3 = n2[e3].match(he2);\n t3 && (r2 = t3[1] || t3[2]);\n }\n if (!r2)\n return null;\n let i2 = I2.test(r2);\n if (!(me2.test(r2) || i2 || r2.startsWith(`/`))) {\n let t3 = e2.split(`/`);\n t3[t3.length - 1] = r2, r2 = t3.join(`/`);\n }\n return r2;\n };\n var ve2 = (e2) => ({ file: e2.file, mappings: (0, F2.decode)(e2.mappings), names: e2.names, sourceRoot: e2.sourceRoot, sources: e2.sources, sourcesContent: e2.sourcesContent, version: 3 });\n var ye2 = (e2) => {\n let t2 = e2.sections.map(({ map: e3, offset: t3 }) => ({ map: { ...e3, mappings: (0, F2.decode)(e3.mappings) }, offset: t3 })), n2 = new Set;\n for (let e3 of t2)\n for (let t3 of e3.map.sources)\n n2.add(t3);\n return { file: e2.file, mappings: [], names: [], sections: t2, sourceRoot: undefined, sources: Array.from(n2), sourcesContent: undefined, version: 3 };\n };\n var H2 = (e2) => {\n if (!e2)\n return false;\n let t2 = e2.trim();\n if (!t2)\n return false;\n let n2 = t2.match(I2);\n if (!n2)\n return true;\n let r2 = n2[0].toLowerCase();\n return r2 === `http:` || r2 === `https:`;\n };\n var U2 = async (e2, t2 = fetch) => {\n if (!H2(e2))\n return null;\n let n2;\n try {\n let r3 = await t2(e2);\n n2 = await r3.text();\n } catch {\n return null;\n }\n if (!n2)\n return null;\n let r2 = _e2(e2, n2);\n if (!r2 || !H2(r2))\n return null;\n try {\n let e3 = await t2(r2), n3 = await e3.json();\n return `sections` in n3 ? ye2(n3) : ve2(n3);\n } catch {\n return null;\n }\n };\n var W2 = async (e2, t2 = true, n2) => {\n if (t2 && R.has(e2)) {\n let t3 = R.get(e2);\n if (t3 == null)\n return null;\n if (ge2(t3)) {\n let n3 = t3.deref();\n if (n3)\n return n3;\n R.delete(e2);\n } else\n return t3;\n }\n if (t2 && z2.has(e2))\n return z2.get(e2);\n let r2 = U2(e2, n2);\n t2 && z2.set(e2, r2);\n let i2 = await r2;\n return t2 && z2.delete(e2), t2 && (i2 === null ? R.set(e2, null) : R.set(e2, L2 ? new WeakRef(i2) : i2)), i2;\n };\n var G2 = async (e2, t2 = true, n2) => await Promise.all(e2.map(async (e3) => {\n if (!e3.fileName)\n return e3;\n let r2 = await W2(e3.fileName, t2, n2);\n if (!r2 || typeof e3.lineNumber != `number` || typeof e3.columnNumber != `number`)\n return e3;\n let i2 = V2(r2, e3.lineNumber, e3.columnNumber);\n return i2 ? { ...e3, source: i2.fileName && e3.source ? e3.source.replace(e3.fileName, i2.fileName) : e3.source, fileName: i2.fileName, lineNumber: i2.lineNumber, columnNumber: i2.columnNumber, isSymbolicated: true } : e3;\n }));\n var K2 = (e2) => e2._debugStack instanceof Error && typeof e2._debugStack?.stack == `string`;\n var be2 = () => {\n let n2 = h();\n for (let t2 of [...Array.from(d), ...Array.from(n2.renderers.values())]) {\n let e2 = t2.currentDispatcherRef;\n if (e2 && typeof e2 == `object`)\n return `H` in e2 ? e2.H : e2.current;\n }\n return null;\n };\n var q = (t2) => {\n for (let n2 of d) {\n let e2 = n2.currentDispatcherRef;\n e2 && typeof e2 == `object` && (`H` in e2 ? e2.H = t2 : e2.current = t2);\n }\n };\n var J = (e2) => `\n in ${e2}`;\n var Y = (e2, t2) => {\n let n2 = J(e2);\n return t2 && (n2 += ` (at ${t2})`), n2;\n };\n var X2 = false;\n var Z = (e2, t2) => {\n if (!e2 || X2)\n return ``;\n let n2 = Error.prepareStackTrace;\n Error.prepareStackTrace = undefined, X2 = true;\n let r2 = be2();\n q(null);\n let { error: i2, warn: a3 } = console;\n console.error = () => {}, console.warn = () => {};\n try {\n let n3 = { DetermineComponentFrameRoot() {\n let n4;\n try {\n if (t2) {\n let t3 = function() {\n throw Error();\n };\n if (Object.defineProperty(t3.prototype, `props`, { set: function() {\n throw Error();\n } }), typeof Reflect == `object` && Reflect.construct) {\n try {\n Reflect.construct(t3, []);\n } catch (e3) {\n n4 = e3;\n }\n Reflect.construct(e2, [], t3);\n } else {\n try {\n t3.call();\n } catch (e3) {\n n4 = e3;\n }\n e2.call(t3.prototype);\n }\n } else {\n try {\n throw Error();\n } catch (e3) {\n n4 = e3;\n }\n let t3 = e2();\n t3 && typeof t3.catch == `function` && t3.catch(() => {});\n }\n } catch (e3) {\n if (e3 instanceof Error && n4 instanceof Error && typeof e3.stack == `string`)\n return [e3.stack, n4.stack];\n }\n return [null, null];\n } };\n n3.DetermineComponentFrameRoot.displayName = `DetermineComponentFrameRoot`;\n let r3 = Object.getOwnPropertyDescriptor(n3.DetermineComponentFrameRoot, `name`);\n r3?.configurable && Object.defineProperty(n3.DetermineComponentFrameRoot, `name`, { value: `DetermineComponentFrameRoot` });\n let [i3, a4] = n3.DetermineComponentFrameRoot();\n if (i3 && a4) {\n let t3 = i3.split(`\n`), n4 = a4.split(`\n`), r4 = 0, o4 = 0;\n for (;r4 < t3.length && !t3[r4].includes(`DetermineComponentFrameRoot`); )\n r4++;\n for (;o4 < n4.length && !n4[o4].includes(`DetermineComponentFrameRoot`); )\n o4++;\n if (r4 === t3.length || o4 === n4.length)\n for (r4 = t3.length - 1, o4 = n4.length - 1;r4 >= 1 && o4 >= 0 && t3[r4] !== n4[o4]; )\n o4--;\n for (;r4 >= 1 && o4 >= 0; r4--, o4--)\n if (t3[r4] !== n4[o4]) {\n if (r4 !== 1 || o4 !== 1)\n do\n if (r4--, o4--, o4 < 0 || t3[r4] !== n4[o4]) {\n let n5 = `\n${t3[r4].replace(` at new `, ` at `)}`, i4 = Te(e2);\n return i4 && n5.includes(`<anonymous>`) && (n5 = n5.replace(`<anonymous>`, i4)), n5;\n }\n while (r4 >= 1 && o4 >= 0);\n break;\n }\n }\n } finally {\n X2 = false, Error.prepareStackTrace = n2, q(r2), console.error = i2, console.warn = a3;\n }\n let o3 = e2 ? Te(e2) : ``, s3 = o3 ? J(o3) : ``;\n return s3;\n };\n var xe2 = (e2, t2) => {\n let m3 = e2.tag, h3 = ``;\n switch (m3) {\n case ne:\n h3 = J(`Activity`);\n break;\n case o2:\n h3 = Z(e2.type, true);\n break;\n case f2:\n h3 = Z(e2.type.render, false);\n break;\n case a2:\n case h2:\n h3 = Z(e2.type, false);\n break;\n case c2:\n case y:\n case b:\n h3 = J(e2.type);\n break;\n case ee:\n h3 = J(`Lazy`);\n break;\n case p2:\n h3 = e2.child !== t2 && t2 !== null ? J(`Suspense Fallback`) : J(`Suspense`);\n break;\n case te:\n h3 = J(`SuspenseList`);\n break;\n case re:\n h3 = J(`ViewTransition`);\n break;\n default:\n return ``;\n }\n return h3;\n };\n var Se2 = (e2) => {\n try {\n let t2 = ``, n2 = e2, r2 = null;\n do {\n t2 += xe2(n2, r2);\n let e3 = n2._debugInfo;\n if (e3 && Array.isArray(e3))\n for (let n3 = e3.length - 1;n3 >= 0; n3--) {\n let r3 = e3[n3];\n typeof r3.name == `string` && (t2 += Y(r3.name, r3.env));\n }\n r2 = n2, n2 = n2.return;\n } while (n2);\n return t2;\n } catch (e3) {\n return e3 instanceof Error ? `\nError generating stack: ${e3.message}\n${e3.stack}` : ``;\n }\n };\n var Ce2 = (e2) => {\n let t2 = Error.prepareStackTrace;\n Error.prepareStackTrace = undefined;\n let n2 = e2;\n if (!n2)\n return ``;\n Error.prepareStackTrace = t2, n2.startsWith(`Error: react-stack-top-frame\n`) && (n2 = n2.slice(29));\n let r2 = n2.indexOf(`\n`);\n if (r2 !== -1 && (n2 = n2.slice(r2 + 1)), r2 = Math.max(n2.indexOf(`react_stack_bottom_frame`), n2.indexOf(`react-stack-bottom-frame`)), r2 !== -1 && (r2 = n2.lastIndexOf(`\n`, r2)), r2 !== -1)\n n2 = n2.slice(0, r2);\n else\n return ``;\n return n2;\n };\n var we2 = (e2) => !!(e2.fileName?.startsWith(`rsc://`) && e2.functionName);\n var Te2 = (e2, t2) => e2.fileName === t2.fileName && e2.lineNumber === t2.lineNumber && e2.columnNumber === t2.columnNumber;\n var Ee2 = (e2) => {\n let t2 = new Map;\n for (let n2 of e2)\n for (let e3 of n2.stackFrames) {\n if (!we2(e3))\n continue;\n let n3 = e3.functionName, r2 = t2.get(n3) ?? [], i2 = r2.some((t3) => Te2(t3, e3));\n i2 || (r2.push(e3), t2.set(n3, r2));\n }\n return t2;\n };\n var De2 = (e2, t2, n2) => {\n if (!e2.functionName)\n return { ...e2, isServer: true };\n let r2 = t2.get(e2.functionName);\n if (!r2 || r2.length === 0)\n return { ...e2, isServer: true };\n let i2 = n2.get(e2.functionName) ?? 0, a3 = r2[i2 % r2.length];\n return n2.set(e2.functionName, i2 + 1), { ...e2, isServer: true, fileName: a3.fileName, lineNumber: a3.lineNumber, columnNumber: a3.columnNumber, source: e2.source?.replace(E, `(${a3.fileName}:${a3.lineNumber}:${a3.columnNumber})`) };\n };\n var Oe = (e2) => {\n let t2 = [];\n return N(e2, (e3) => {\n if (!K2(e3))\n return;\n let n2 = typeof e3.type == `string` ? e3.type : Te(e3.type) || `<anonymous>`;\n t2.push({ componentName: n2, stackFrames: D(Ce2(e3._debugStack?.stack)) });\n }, true), t2;\n };\n var Q = async (e2, t2 = true, n2) => {\n let r2 = Oe(e2), i2 = D(Se2(e2)), a3 = Ee2(r2), o3 = new Map, s3 = i2.map((e3) => {\n let t3 = e3.source?.includes(E) ?? false;\n return t3 ? De2(e3, a3, o3) : e3;\n }), c3 = s3.filter((e3, t3, n3) => {\n if (t3 === 0)\n return true;\n let r3 = n3[t3 - 1];\n return e3.functionName !== r3.functionName;\n });\n return G2(c3, t2, n2);\n };\n var ke2 = (e2) => {\n let t2 = e2._debugSource;\n return t2 ? typeof t2 == `object` && !!t2 && `fileName` in t2 && typeof t2.fileName == `string` && `lineNumber` in t2 && typeof t2.lineNumber == `number` : false;\n };\n var Ae2 = async (e2, t2 = true, n2) => {\n if (ke2(e2)) {\n let t3 = e2._debugSource;\n return t3 || null;\n }\n let r2 = await Q(e2, t2, n2);\n for (let e3 of r2)\n if (e3.fileName)\n return { fileName: e3.fileName, lineNumber: e3.lineNumber, columnNumber: e3.columnNumber, functionName: e3.functionName };\n return null;\n };\n var $2 = (e2) => {\n if (!e2 || ne2.some((t3) => t3 === e2))\n return ``;\n let t2 = e2;\n if (t2.startsWith(`http://`) || t2.startsWith(`https://`))\n try {\n let e3 = new URL(t2);\n t2 = e3.pathname;\n } catch {}\n if (t2.startsWith(T2)) {\n let e3 = t2.slice(T2.length), n3 = e3.indexOf(`/`), r3 = e3.indexOf(`:`);\n t2 = n3 !== -1 && (r3 === -1 || n3 < r3) ? e3.slice(n3 + 1) : e3;\n }\n let n2 = true;\n for (;n2; ) {\n n2 = false;\n for (let e3 of te2)\n if (t2.startsWith(e3)) {\n t2 = t2.slice(e3.length), e3 === `file:///` && (t2 = `/${t2.replace(/^\\/+/, ``)}`), n2 = true;\n break;\n }\n }\n if (w2.test(t2)) {\n let e3 = t2.match(w2);\n e3 && (t2 = t2.slice(e3[0].length));\n }\n if (t2.startsWith(`//`)) {\n let e3 = t2.indexOf(`/`, 2);\n t2 = e3 === -1 ? `` : t2.slice(e3);\n }\n let r2 = t2.indexOf(`?`);\n if (r2 !== -1) {\n let e3 = t2.slice(r2);\n ae2.test(e3) && (t2 = t2.slice(0, r2));\n }\n return t2;\n };\n var je2 = (e2) => {\n let t2 = $2(e2);\n return !(!t2 || !re2.test(t2) || ie2.test(t2));\n };\n\n // dist/_bippy-entry-28588-1781364082573.js\n globalThis.__bippy = {\n getFiberFromHostInstance: Pe,\n getDisplayName: Te,\n traverseFiber: N,\n isCompositeFiber: pe,\n isHostFiber: k,\n getSource: Ae2,\n getOwnerStack: Q,\n normalizeFileName: $2,\n isSourceFile: je2\n };\n})();\n";
|
|
551
|
+
//#endregion
|
|
409
552
|
//#region src/recording.ts
|
|
410
553
|
var activeRecordings = /* @__PURE__ */ new Map();
|
|
411
554
|
var offscreenDocumentCreating = null;
|
|
@@ -624,6 +767,7 @@ async function cleanupRecordingForTab(tabId) {
|
|
|
624
767
|
}
|
|
625
768
|
//#endregion
|
|
626
769
|
//#region src/background.ts
|
|
770
|
+
var js = dedent;
|
|
627
771
|
var RELAY_HOST = "127.0.0.1";
|
|
628
772
|
var RELAY_PORT = 19988;
|
|
629
773
|
function sleep(ms) {
|
|
@@ -636,19 +780,30 @@ function createInstallId() {
|
|
|
636
780
|
return value.toString(36);
|
|
637
781
|
}).join("");
|
|
638
782
|
}
|
|
783
|
+
function browserNameFromBrands(brands) {
|
|
784
|
+
const brandNames = brands.map((brand) => {
|
|
785
|
+
return brand.brand.trim().toLowerCase();
|
|
786
|
+
});
|
|
787
|
+
if (brandNames.some((brand) => brand === "brave")) return "Brave";
|
|
788
|
+
if (brandNames.some((brand) => brand === "microsoft edge")) return "Edge";
|
|
789
|
+
if (brandNames.some((brand) => brand === "opera")) return "Opera";
|
|
790
|
+
if (brandNames.some((brand) => brand === "vivaldi")) return "Vivaldi";
|
|
791
|
+
if (brandNames.some((brand) => brand === "google chrome canary")) return "Chrome Canary";
|
|
792
|
+
if (brandNames.some((brand) => brand === "google chrome")) return "Chrome";
|
|
793
|
+
if (brandNames.some((brand) => brand === "chromium")) return "Chromium";
|
|
794
|
+
return null;
|
|
795
|
+
}
|
|
639
796
|
async function detectBrowserName() {
|
|
640
797
|
if (chrome.ghostPublicAPI) return "Ghost";
|
|
641
|
-
const
|
|
798
|
+
const navigatorWithUaData = navigator;
|
|
799
|
+
const brands = navigatorWithUaData.userAgentData?.brands;
|
|
800
|
+
const highEntropyName = browserNameFromBrands((await navigatorWithUaData.userAgentData?.getHighEntropyValues?.(["fullVersionList"]).catch(() => {
|
|
801
|
+
return null;
|
|
802
|
+
}))?.fullVersionList || []);
|
|
803
|
+
if (highEntropyName) return highEntropyName;
|
|
642
804
|
if (brands && brands.length > 0) {
|
|
643
|
-
const
|
|
644
|
-
|
|
645
|
-
});
|
|
646
|
-
if (brandNames.some((brand) => brand === "brave")) return "Brave";
|
|
647
|
-
if (brandNames.some((brand) => brand === "microsoft edge")) return "Edge";
|
|
648
|
-
if (brandNames.some((brand) => brand === "opera")) return "Opera";
|
|
649
|
-
if (brandNames.some((brand) => brand === "vivaldi")) return "Vivaldi";
|
|
650
|
-
if (brandNames.some((brand) => brand === "google chrome")) return "Chrome";
|
|
651
|
-
if (brandNames.some((brand) => brand === "chromium")) return "Chromium";
|
|
805
|
+
const lowEntropyName = browserNameFromBrands(brands);
|
|
806
|
+
if (lowEntropyName) return lowEntropyName;
|
|
652
807
|
}
|
|
653
808
|
const ua = navigator.userAgent.toLowerCase();
|
|
654
809
|
if (ua.includes("edg/")) return "Edge";
|
|
@@ -778,7 +933,7 @@ var ConnectionManager = class {
|
|
|
778
933
|
if (identity.email) relayUrl.searchParams.set("email", identity.email);
|
|
779
934
|
if (identity.id) relayUrl.searchParams.set("id", identity.id);
|
|
780
935
|
if (identity.installId) relayUrl.searchParams.set("installId", identity.installId);
|
|
781
|
-
relayUrl.searchParams.set("v", "0.
|
|
936
|
+
relayUrl.searchParams.set("v", "0.3.0");
|
|
782
937
|
logger.debug("Creating WebSocket connection to:", relayUrl);
|
|
783
938
|
const socket = new WebSocket(relayUrl.toString());
|
|
784
939
|
await new Promise((resolve, reject) => {
|
|
@@ -1514,7 +1669,7 @@ async function attachTab(tabId, { skipAttachedEvent = false } = {}) {
|
|
|
1514
1669
|
} catch (error) {
|
|
1515
1670
|
logger.debug("Failed to apply auto-attach for tab:", tabId, error);
|
|
1516
1671
|
}
|
|
1517
|
-
const contextMenuScript = `
|
|
1672
|
+
const contextMenuScript = js`
|
|
1518
1673
|
document.addEventListener('contextmenu', (e) => {
|
|
1519
1674
|
window.__playwriter_lastRightClicked = e.target;
|
|
1520
1675
|
}, true);
|
|
@@ -1939,10 +2094,22 @@ chrome.contextMenus.remove("playwriter-pin-element").catch(() => {}).finally(()
|
|
|
1939
2094
|
visible: false
|
|
1940
2095
|
});
|
|
1941
2096
|
});
|
|
2097
|
+
chrome.contextMenus.remove("playwriter-copy-react-source").catch(() => {}).finally(() => {
|
|
2098
|
+
chrome.contextMenus?.create({
|
|
2099
|
+
id: "playwriter-copy-react-source",
|
|
2100
|
+
title: "Copy React Component Source Path",
|
|
2101
|
+
contexts: ["all"],
|
|
2102
|
+
visible: false
|
|
2103
|
+
});
|
|
2104
|
+
});
|
|
1942
2105
|
function updateContextMenuVisibility() {
|
|
1943
2106
|
const { currentTabId, tabs } = store.getState();
|
|
1944
2107
|
const isConnected = currentTabId !== void 0 && tabs.get(currentTabId)?.state === "connected";
|
|
1945
2108
|
chrome.contextMenus?.update("playwriter-pin-element", { visible: isConnected });
|
|
2109
|
+
chrome.contextMenus?.update("playwriter-copy-react-source", { visible: isConnected });
|
|
2110
|
+
}
|
|
2111
|
+
function buildPinnedElementInspectionCode(options) {
|
|
2112
|
+
return `inspectPinnedElement(${JSON.stringify(options.url).replace(/'/g, "\\u0027")},"globalThis.${options.pinName}")`;
|
|
1946
2113
|
}
|
|
1947
2114
|
chrome.runtime.onInstalled.addListener((details) => {});
|
|
1948
2115
|
function serializeTabs(tabs) {
|
|
@@ -2079,44 +2246,45 @@ chrome.windows.onCreated.addListener(async (popupWindow) => {
|
|
|
2079
2246
|
}
|
|
2080
2247
|
});
|
|
2081
2248
|
chrome.contextMenus?.onClicked.addListener(async (info, tab) => {
|
|
2082
|
-
if (
|
|
2249
|
+
if (!tab?.id) return;
|
|
2083
2250
|
const tabInfo = store.getState().tabs.get(tab.id);
|
|
2084
2251
|
if (!tabInfo || tabInfo.state !== "connected") {
|
|
2085
2252
|
logger.debug("Tab not connected, ignoring");
|
|
2086
2253
|
return;
|
|
2087
2254
|
}
|
|
2088
2255
|
const debuggee = { tabId: tab.id };
|
|
2089
|
-
|
|
2090
|
-
|
|
2091
|
-
const hasMultiplePages = connectedTabs.length > 1;
|
|
2092
|
-
try {
|
|
2093
|
-
const name = `playwriterPinnedElem${(await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", {
|
|
2094
|
-
expression: `
|
|
2256
|
+
if (info.menuItemId === "playwriter-pin-element") try {
|
|
2257
|
+
const jsAllocatePin = js`
|
|
2095
2258
|
(function() {
|
|
2096
2259
|
window.__playwriterPinCount = (window.__playwriterPinCount || 0) + 1;
|
|
2097
2260
|
return window.__playwriterPinCount;
|
|
2098
2261
|
})()
|
|
2099
|
-
|
|
2262
|
+
`;
|
|
2263
|
+
const name = `playwriterPinnedElem${(await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", {
|
|
2264
|
+
expression: jsAllocatePin,
|
|
2100
2265
|
returnByValue: true
|
|
2101
2266
|
})).result?.value ?? 1}`;
|
|
2102
|
-
const
|
|
2103
|
-
expression: `
|
|
2267
|
+
const jsAssignPin = js`
|
|
2104
2268
|
if (window.__playwriter_lastRightClicked) {
|
|
2105
2269
|
window.${name} = window.__playwriter_lastRightClicked;
|
|
2106
2270
|
'${name}';
|
|
2107
2271
|
} else {
|
|
2108
2272
|
throw new Error('No element was right-clicked');
|
|
2109
2273
|
}
|
|
2110
|
-
|
|
2274
|
+
`;
|
|
2275
|
+
const result = await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", {
|
|
2276
|
+
expression: jsAssignPin,
|
|
2111
2277
|
returnByValue: true
|
|
2112
2278
|
});
|
|
2113
2279
|
if (result.exceptionDetails) {
|
|
2114
2280
|
logger.error("Failed to pin element:", result.exceptionDetails.text);
|
|
2115
2281
|
return;
|
|
2116
2282
|
}
|
|
2117
|
-
const clipboardText =
|
|
2118
|
-
|
|
2119
|
-
|
|
2283
|
+
const clipboardText = "playwriter -e '" + buildPinnedElementInspectionCode({
|
|
2284
|
+
pinName: name,
|
|
2285
|
+
url: tab.url || ""
|
|
2286
|
+
}) + "'";
|
|
2287
|
+
const jsPinFlashAndCopy = js`
|
|
2120
2288
|
(() => {
|
|
2121
2289
|
const el = window.${name};
|
|
2122
2290
|
if (!el) return;
|
|
@@ -2125,13 +2293,126 @@ chrome.contextMenus?.onClicked.addListener(async (info, tab) => {
|
|
|
2125
2293
|
setTimeout(() => el.setAttribute('style', orig), 300);
|
|
2126
2294
|
return navigator.clipboard.writeText(${JSON.stringify(clipboardText)});
|
|
2127
2295
|
})()
|
|
2128
|
-
|
|
2129
|
-
|
|
2296
|
+
`;
|
|
2297
|
+
await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", {
|
|
2298
|
+
expression: jsPinFlashAndCopy,
|
|
2299
|
+
awaitPromise: true,
|
|
2300
|
+
userGesture: true
|
|
2130
2301
|
});
|
|
2131
2302
|
logger.debug("Pinned element as:", name);
|
|
2132
2303
|
} catch (error) {
|
|
2133
2304
|
logger.error("Failed to pin element:", error.message);
|
|
2134
2305
|
}
|
|
2306
|
+
if (info.menuItemId === "playwriter-copy-react-source") try {
|
|
2307
|
+
if (!(await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", {
|
|
2308
|
+
expression: "!!globalThis.__bippy",
|
|
2309
|
+
returnByValue: true
|
|
2310
|
+
})).result?.value) await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", { expression: bippy_default });
|
|
2311
|
+
const jsResolveSource = js`
|
|
2312
|
+
(async () => {
|
|
2313
|
+
const el = window.__playwriter_lastRightClicked;
|
|
2314
|
+
if (!el) return JSON.stringify({ error: 'No element was right-clicked' });
|
|
2315
|
+
|
|
2316
|
+
const bippy = globalThis.__bippy;
|
|
2317
|
+
if (!bippy) return JSON.stringify({ error: 'bippy not loaded' });
|
|
2318
|
+
|
|
2319
|
+
// bippy.normalizeFileName strips "/app-pages-browser/" but not the parenthesized
|
|
2320
|
+
// form "/(app-pages-browser)/" that Next.js webpack actually uses. This regex
|
|
2321
|
+
// strips all Next.js webpack layer prefixes: (app-pages-browser), (ssr), (rsc),
|
|
2322
|
+
// (action-browser), (pages-dir-browser), (pages-dir-edge), (pages-dir-node).
|
|
2323
|
+
// Also strips leading "./" that often follows the layer prefix.
|
|
2324
|
+
const cleanFileName = (name) => {
|
|
2325
|
+
let f = bippy.normalizeFileName(name);
|
|
2326
|
+
f = f.replace(/^\/?\\([-\\w]+\\)\\//, '');
|
|
2327
|
+
f = f.replace(/^\\.[\\/]/, '');
|
|
2328
|
+
return f;
|
|
2329
|
+
};
|
|
2330
|
+
|
|
2331
|
+
let fiber;
|
|
2332
|
+
try { fiber = bippy.getFiberFromHostInstance(el); } catch {}
|
|
2333
|
+
if (!fiber) return JSON.stringify({ error: 'No React fiber found. Is this a React app?' });
|
|
2334
|
+
|
|
2335
|
+
// Walk up to find nearest composite fiber with source info
|
|
2336
|
+
let current = fiber;
|
|
2337
|
+
for (let i = 0; i < 50 && current; i++) {
|
|
2338
|
+
try {
|
|
2339
|
+
if (bippy.isCompositeFiber(current)) {
|
|
2340
|
+
const source = await bippy.getSource(current);
|
|
2341
|
+
if (source && source.fileName && bippy.isSourceFile(source.fileName)) {
|
|
2342
|
+
return JSON.stringify({
|
|
2343
|
+
fileName: cleanFileName(source.fileName),
|
|
2344
|
+
lineNumber: source.lineNumber || null,
|
|
2345
|
+
columnNumber: source.columnNumber || null,
|
|
2346
|
+
componentName: source.functionName || bippy.getDisplayName(current.type) || null,
|
|
2347
|
+
});
|
|
2348
|
+
}
|
|
2349
|
+
// Try owner stack as fallback for this fiber
|
|
2350
|
+
const ownerStack = await bippy.getOwnerStack(current);
|
|
2351
|
+
for (const frame of ownerStack) {
|
|
2352
|
+
if (frame.fileName && bippy.isSourceFile(frame.fileName)) {
|
|
2353
|
+
return JSON.stringify({
|
|
2354
|
+
fileName: cleanFileName(frame.fileName),
|
|
2355
|
+
lineNumber: frame.lineNumber || null,
|
|
2356
|
+
columnNumber: frame.columnNumber || null,
|
|
2357
|
+
componentName: frame.functionName || bippy.getDisplayName(current.type) || null,
|
|
2358
|
+
});
|
|
2359
|
+
}
|
|
2360
|
+
}
|
|
2361
|
+
}
|
|
2362
|
+
} catch {}
|
|
2363
|
+
current = current.return;
|
|
2364
|
+
}
|
|
2365
|
+
return JSON.stringify({ error: 'No React source location found. Is this a dev build with source maps?' });
|
|
2366
|
+
})()
|
|
2367
|
+
`;
|
|
2368
|
+
const sourceResult = await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", {
|
|
2369
|
+
expression: jsResolveSource,
|
|
2370
|
+
returnByValue: true,
|
|
2371
|
+
awaitPromise: true
|
|
2372
|
+
});
|
|
2373
|
+
if (sourceResult.exceptionDetails) {
|
|
2374
|
+
logger.error("Failed to get React source:", sourceResult.exceptionDetails.text);
|
|
2375
|
+
return;
|
|
2376
|
+
}
|
|
2377
|
+
const parsed = JSON.parse(sourceResult.result?.value || "{}");
|
|
2378
|
+
if (!parsed.fileName && !parsed.error) parsed.error = "React source result missing fileName";
|
|
2379
|
+
if (parsed.error) {
|
|
2380
|
+
const jsFlashRed = js`
|
|
2381
|
+
(() => {
|
|
2382
|
+
const el = window.__playwriter_lastRightClicked;
|
|
2383
|
+
if (!el) return;
|
|
2384
|
+
const orig = el.getAttribute('style') || '';
|
|
2385
|
+
el.setAttribute('style', orig + '; outline: 3px solid #ef4444 !important; outline-offset: 2px !important;');
|
|
2386
|
+
setTimeout(() => el.setAttribute('style', orig), 600);
|
|
2387
|
+
})()
|
|
2388
|
+
`;
|
|
2389
|
+
await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", { expression: jsFlashRed });
|
|
2390
|
+
logger.debug("React source not found:", parsed.error);
|
|
2391
|
+
return;
|
|
2392
|
+
}
|
|
2393
|
+
const clipboardText = (() => {
|
|
2394
|
+
if (parsed.lineNumber) return `${parsed.fileName}:${parsed.lineNumber}`;
|
|
2395
|
+
return parsed.fileName;
|
|
2396
|
+
})();
|
|
2397
|
+
const jsFlashGreenAndCopy = js`
|
|
2398
|
+
(() => {
|
|
2399
|
+
const el = window.__playwriter_lastRightClicked;
|
|
2400
|
+
if (!el) return;
|
|
2401
|
+
const orig = el.getAttribute('style') || '';
|
|
2402
|
+
el.setAttribute('style', orig + '; outline: 3px solid #22c55e !important; outline-offset: 2px !important; box-shadow: 0 0 0 3px #22c55e !important;');
|
|
2403
|
+
setTimeout(() => el.setAttribute('style', orig), 300);
|
|
2404
|
+
return navigator.clipboard.writeText(${JSON.stringify(clipboardText)});
|
|
2405
|
+
})()
|
|
2406
|
+
`;
|
|
2407
|
+
await chrome.debugger.sendCommand(debuggee, "Runtime.evaluate", {
|
|
2408
|
+
expression: jsFlashGreenAndCopy,
|
|
2409
|
+
awaitPromise: true,
|
|
2410
|
+
userGesture: true
|
|
2411
|
+
});
|
|
2412
|
+
logger.debug("Copied React source path:", clipboardText, "component:", parsed.componentName);
|
|
2413
|
+
} catch (error) {
|
|
2414
|
+
logger.error("Failed to copy React source:", error.message);
|
|
2415
|
+
}
|
|
2135
2416
|
});
|
|
2136
2417
|
updateIcons();
|
|
2137
2418
|
chrome.runtime.onMessage.addListener((message, _sender, _sendResponse) => {
|
package/dist/mcp.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"mcp.d.ts","sourceRoot":"","sources":["../src/mcp.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"mcp.d.ts","sourceRoot":"","sources":["../src/mcp.ts"],"names":[],"mappings":"AAgWA,wBAAsB,QAAQ,CAAC,OAAO,GAAE;IAAE,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAA;CAAO,iBAuB7E"}
|
package/dist/mcp.js
CHANGED
|
@@ -43,9 +43,14 @@ function getLogServerUrl() {
|
|
|
43
43
|
}
|
|
44
44
|
async function sendLogToRelayServer(level, ...args) {
|
|
45
45
|
try {
|
|
46
|
+
const headers = { 'Content-Type': 'application/json' };
|
|
47
|
+
const token = process.env.PLAYWRITER_TOKEN;
|
|
48
|
+
if (token) {
|
|
49
|
+
headers['Authorization'] = `Bearer ${token}`;
|
|
50
|
+
}
|
|
46
51
|
await fetch(getLogServerUrl(), {
|
|
47
52
|
method: 'POST',
|
|
48
|
-
headers
|
|
53
|
+
headers,
|
|
49
54
|
body: JSON.stringify({ level, args }),
|
|
50
55
|
signal: AbortSignal.timeout(1000),
|
|
51
56
|
});
|