@playwright-repl/browser-extension 0.24.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/LICENSE +21 -0
- package/README.md +176 -0
- package/dist/background.js +162567 -0
- package/dist/background.js.map +1 -0
- package/dist/content/recorder.js +479 -0
- package/dist/content/trace-loader.js +12 -0
- package/dist/devtools/console.html +17 -0
- package/dist/devtools/console.js +44 -0
- package/dist/devtools/console.js.map +1 -0
- package/dist/devtools/devtools.html +8 -0
- package/dist/devtools/devtools.js +7 -0
- package/dist/devtools/devtools.js.map +1 -0
- package/dist/icons/dramaturg_icon_128.png +0 -0
- package/dist/icons/dramaturg_icon_16.png +0 -0
- package/dist/icons/dramaturg_icon_32.png +0 -0
- package/dist/icons/dramaturg_icon_48.png +0 -0
- package/dist/index.css +1353 -0
- package/dist/index.js +12462 -0
- package/dist/index.js.map +1 -0
- package/dist/index2.js +27328 -0
- package/dist/index2.js.map +1 -0
- package/dist/manifest.json +49 -0
- package/dist/modulepreload-polyfill.js +30 -0
- package/dist/modulepreload-polyfill.js.map +1 -0
- package/dist/offscreen/offscreen.html +6 -0
- package/dist/offscreen/offscreen.js +151 -0
- package/dist/offscreen/offscreen.js.map +1 -0
- package/dist/panel/panel.html +16 -0
- package/dist/panel/panel.js +2258 -0
- package/dist/panel/panel.js.map +1 -0
- package/dist/preferences/preferences.html +14 -0
- package/dist/preferences/preferences.js +102 -0
- package/dist/preferences/preferences.js.map +1 -0
- package/dist/settings.js +13 -0
- package/dist/settings.js.map +1 -0
- package/dist/sw-debugger-core.js +1139 -0
- package/dist/sw-debugger-core.js.map +1 -0
- package/package.json +80 -0
|
@@ -0,0 +1,479 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
(() => {
|
|
3
|
+
// src/content/locator.ts
|
|
4
|
+
var IMPLICIT_ROLES = {
|
|
5
|
+
A: (el) => el.hasAttribute("href") ? "link" : null,
|
|
6
|
+
BUTTON: "button",
|
|
7
|
+
H1: "heading",
|
|
8
|
+
H2: "heading",
|
|
9
|
+
H3: "heading",
|
|
10
|
+
H4: "heading",
|
|
11
|
+
H5: "heading",
|
|
12
|
+
H6: "heading",
|
|
13
|
+
INPUT: (el) => {
|
|
14
|
+
const type = el.type.toLowerCase();
|
|
15
|
+
if (type === "checkbox") return "checkbox";
|
|
16
|
+
if (type === "radio") return "radio";
|
|
17
|
+
if (type === "submit" || type === "reset" || type === "button") return "button";
|
|
18
|
+
if (type === "hidden") return null;
|
|
19
|
+
return "textbox";
|
|
20
|
+
},
|
|
21
|
+
TEXTAREA: "textbox",
|
|
22
|
+
SELECT: "combobox",
|
|
23
|
+
IMG: "img",
|
|
24
|
+
NAV: "navigation",
|
|
25
|
+
MAIN: "main",
|
|
26
|
+
HEADER: "banner",
|
|
27
|
+
FOOTER: "contentinfo",
|
|
28
|
+
P: "paragraph",
|
|
29
|
+
UL: "list",
|
|
30
|
+
OL: "list",
|
|
31
|
+
LI: "listitem",
|
|
32
|
+
TABLE: "table",
|
|
33
|
+
TR: "row",
|
|
34
|
+
TH: "columnheader",
|
|
35
|
+
TD: "cell",
|
|
36
|
+
FORM: "form",
|
|
37
|
+
DIALOG: "dialog",
|
|
38
|
+
ARTICLE: "article"
|
|
39
|
+
};
|
|
40
|
+
function getImplicitRole(el) {
|
|
41
|
+
const explicit = el.getAttribute("role");
|
|
42
|
+
if (explicit && explicit !== "none" && explicit !== "presentation") return explicit;
|
|
43
|
+
const entry = IMPLICIT_ROLES[el.tagName];
|
|
44
|
+
if (!entry) return null;
|
|
45
|
+
return typeof entry === "function" ? entry(el) : entry;
|
|
46
|
+
}
|
|
47
|
+
function getAccessibleName(el) {
|
|
48
|
+
const ariaLabel = el.getAttribute("aria-label");
|
|
49
|
+
if (ariaLabel) return ariaLabel.trim();
|
|
50
|
+
const labelledBy = el.getAttribute("aria-labelledby");
|
|
51
|
+
if (labelledBy) {
|
|
52
|
+
const parts = labelledBy.split(/\s+/).map((id) => document.getElementById(id)?.textContent?.trim()).filter(Boolean);
|
|
53
|
+
if (parts.length) return parts.join(" ");
|
|
54
|
+
}
|
|
55
|
+
if (el instanceof HTMLInputElement || el instanceof HTMLTextAreaElement || el instanceof HTMLSelectElement) {
|
|
56
|
+
const label = getLabel(el);
|
|
57
|
+
if (label) return label;
|
|
58
|
+
}
|
|
59
|
+
const role = getImplicitRole(el);
|
|
60
|
+
const NAME_FROM_CONTENT = /* @__PURE__ */ new Set([
|
|
61
|
+
"button",
|
|
62
|
+
"link",
|
|
63
|
+
"heading",
|
|
64
|
+
"tab",
|
|
65
|
+
"menuitem",
|
|
66
|
+
"menuitemcheckbox",
|
|
67
|
+
"menuitemradio",
|
|
68
|
+
"option",
|
|
69
|
+
"radio",
|
|
70
|
+
"checkbox",
|
|
71
|
+
"switch",
|
|
72
|
+
"cell",
|
|
73
|
+
"columnheader",
|
|
74
|
+
"rowheader",
|
|
75
|
+
"tooltip",
|
|
76
|
+
"treeitem"
|
|
77
|
+
]);
|
|
78
|
+
if (role && NAME_FROM_CONTENT.has(role)) {
|
|
79
|
+
const text = (el.textContent || "").trim();
|
|
80
|
+
if (text && text.length <= 80) return text;
|
|
81
|
+
}
|
|
82
|
+
if (el.tagName === "IMG") {
|
|
83
|
+
const alt = el.getAttribute("alt");
|
|
84
|
+
if (alt) return alt.trim();
|
|
85
|
+
}
|
|
86
|
+
return "";
|
|
87
|
+
}
|
|
88
|
+
function getLabel(el) {
|
|
89
|
+
if (el.id) {
|
|
90
|
+
const label = document.querySelector(`label[for="${CSS.escape(el.id)}"]`);
|
|
91
|
+
if (label) return (label.textContent || "").trim();
|
|
92
|
+
}
|
|
93
|
+
const parentLabel = el.closest("label");
|
|
94
|
+
if (parentLabel) {
|
|
95
|
+
const clone = parentLabel.cloneNode(true);
|
|
96
|
+
clone.querySelectorAll("input,textarea,select").forEach((c) => c.remove());
|
|
97
|
+
const text = (clone.textContent || "").trim();
|
|
98
|
+
if (text) return text;
|
|
99
|
+
}
|
|
100
|
+
return "";
|
|
101
|
+
}
|
|
102
|
+
var CONTAINER_ROLES = /* @__PURE__ */ new Set(["listitem", "row", "article", "group"]);
|
|
103
|
+
function getContextText(ancestor, exclude) {
|
|
104
|
+
function findText(node) {
|
|
105
|
+
for (const child of node.childNodes) {
|
|
106
|
+
if (child === exclude) continue;
|
|
107
|
+
if (child.nodeType === Node.ELEMENT_NODE && child.contains(exclude)) {
|
|
108
|
+
const inner = findText(child);
|
|
109
|
+
if (inner) return inner;
|
|
110
|
+
continue;
|
|
111
|
+
}
|
|
112
|
+
const text = (child.textContent || "").trim();
|
|
113
|
+
if (text && text.length <= 50) return text;
|
|
114
|
+
}
|
|
115
|
+
return "";
|
|
116
|
+
}
|
|
117
|
+
return findText(ancestor);
|
|
118
|
+
}
|
|
119
|
+
function findContainerAncestor(el) {
|
|
120
|
+
let current = el.parentElement;
|
|
121
|
+
while (current && current !== document.body && current !== document.documentElement) {
|
|
122
|
+
const role = getImplicitRole(current);
|
|
123
|
+
if (role && CONTAINER_ROLES.has(role)) return { ancestor: current, role };
|
|
124
|
+
current = current.parentElement;
|
|
125
|
+
}
|
|
126
|
+
return null;
|
|
127
|
+
}
|
|
128
|
+
function tryAncestorContext(el, role, name, matches) {
|
|
129
|
+
const container = findContainerAncestor(el);
|
|
130
|
+
if (!container) return null;
|
|
131
|
+
const contextText = getContextText(container.ancestor, el);
|
|
132
|
+
if (!contextText || contextText.length > 50) return null;
|
|
133
|
+
let count = 0;
|
|
134
|
+
for (const match of matches) {
|
|
135
|
+
const mc = findContainerAncestor(match);
|
|
136
|
+
if (!mc || mc.role !== container.role) continue;
|
|
137
|
+
if ((mc.ancestor.textContent || "").includes(contextText)) {
|
|
138
|
+
count++;
|
|
139
|
+
if (count > 1) return null;
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
if (count !== 1) return null;
|
|
143
|
+
return `getByRole(${escapeString(container.role)}).filter({ hasText: ${escapeString(contextText)} }).getByRole(${escapeString(role)}, { name: ${escapeString(name)} })`;
|
|
144
|
+
}
|
|
145
|
+
function findByRoleAndName(role, name) {
|
|
146
|
+
const matches = [];
|
|
147
|
+
for (const el of document.querySelectorAll("*")) {
|
|
148
|
+
if (getImplicitRole(el) === role && getAccessibleName(el) === name && el.checkVisibility?.() !== false)
|
|
149
|
+
matches.push(el);
|
|
150
|
+
}
|
|
151
|
+
return matches;
|
|
152
|
+
}
|
|
153
|
+
function findHoverAncestor(el) {
|
|
154
|
+
let ancestor = el.parentElement;
|
|
155
|
+
while (ancestor && ancestor !== document.body && ancestor !== document.documentElement) {
|
|
156
|
+
if (ancestor.matches(":hover")) return ancestor;
|
|
157
|
+
ancestor = ancestor.parentElement;
|
|
158
|
+
}
|
|
159
|
+
return null;
|
|
160
|
+
}
|
|
161
|
+
function isHoverRevealed(el) {
|
|
162
|
+
for (const sheet of document.styleSheets) {
|
|
163
|
+
try {
|
|
164
|
+
for (const rule of sheet.cssRules) {
|
|
165
|
+
if (!(rule instanceof CSSStyleRule)) continue;
|
|
166
|
+
if (!rule.selectorText.includes(":hover")) continue;
|
|
167
|
+
const s = rule.style;
|
|
168
|
+
const reveals = s.display && s.display !== "none" || s.visibility === "visible" || s.opacity && s.opacity !== "0";
|
|
169
|
+
if (!reveals) continue;
|
|
170
|
+
try {
|
|
171
|
+
if (el.matches(rule.selectorText)) return true;
|
|
172
|
+
} catch {
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
} catch {
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
return false;
|
|
179
|
+
}
|
|
180
|
+
function locatorToPwArgs(locator, role) {
|
|
181
|
+
const q = (s) => `"${s}"`;
|
|
182
|
+
let nth = "";
|
|
183
|
+
if (/\.first\(\)/.test(locator)) nth = " --nth 0";
|
|
184
|
+
else if (/\.last\(\)/.test(locator)) nth = " --nth -1";
|
|
185
|
+
else {
|
|
186
|
+
const nthMatch = locator.match(/\.nth\((\d+)\)/);
|
|
187
|
+
if (nthMatch) nth = ` --nth ${nthMatch[1]}`;
|
|
188
|
+
}
|
|
189
|
+
const roleNameMatch = locator.match(/getByRole\(['"](.+?)['"],\s*\{[^}]*name:\s*['"](.+?)['"]/);
|
|
190
|
+
if (roleNameMatch) return `${roleNameMatch[1]} ${q(roleNameMatch[2])}${nth}`;
|
|
191
|
+
const roleMatch = locator.match(/getByRole\(['"](.+?)['"]\)/);
|
|
192
|
+
if (roleMatch) return `${roleMatch[1]}${nth}`;
|
|
193
|
+
const testIdMatch = locator.match(/getByTestId\(['"](.+?)['"]\)/);
|
|
194
|
+
if (testIdMatch) return `${q(testIdMatch[1])}${nth}`;
|
|
195
|
+
const getByMatch = locator.match(/getBy\w+\(['"](.+?)['"]\)/);
|
|
196
|
+
if (getByMatch) {
|
|
197
|
+
const prefix = role ? `${role} ` : "";
|
|
198
|
+
return `${prefix}${q(getByMatch[1])}${nth}`;
|
|
199
|
+
}
|
|
200
|
+
const locatorMatch = locator.match(/locator\(['"](.+?)['"]\)/);
|
|
201
|
+
if (locatorMatch) return `${q(locatorMatch[1])}${nth}`;
|
|
202
|
+
return q(locator);
|
|
203
|
+
}
|
|
204
|
+
function escapeString(s) {
|
|
205
|
+
if (!s.includes("'")) return `'${s}'`;
|
|
206
|
+
if (!s.includes('"')) return `"${s}"`;
|
|
207
|
+
return `'${s.replace(/'/g, "\\'")}'`;
|
|
208
|
+
}
|
|
209
|
+
function generateLocator(el) {
|
|
210
|
+
const testId = el.getAttribute("data-testid") || el.getAttribute("data-test-id");
|
|
211
|
+
if (testId) return `getByTestId(${escapeString(testId)})`;
|
|
212
|
+
const role = getImplicitRole(el);
|
|
213
|
+
const name = getAccessibleName(el);
|
|
214
|
+
if (role && name) {
|
|
215
|
+
const matches = findByRoleAndName(role, name);
|
|
216
|
+
if (matches.length > 1) {
|
|
217
|
+
const ancestorLocator = tryAncestorContext(el, role, name, matches);
|
|
218
|
+
if (ancestorLocator) return ancestorLocator;
|
|
219
|
+
const base = `getByRole(${escapeString(role)}, { name: ${escapeString(name)}, exact: true })`;
|
|
220
|
+
const idx = matches.indexOf(el);
|
|
221
|
+
return idx === 0 ? base + ".first()" : base + `.nth(${idx})`;
|
|
222
|
+
}
|
|
223
|
+
return `getByRole(${escapeString(role)}, { name: ${escapeString(name)} })`;
|
|
224
|
+
}
|
|
225
|
+
if (el instanceof HTMLInputElement || el instanceof HTMLTextAreaElement || el instanceof HTMLSelectElement) {
|
|
226
|
+
const label = getLabel(el);
|
|
227
|
+
if (label) return `getByLabel(${escapeString(label)})`;
|
|
228
|
+
}
|
|
229
|
+
const placeholder = el.getAttribute("placeholder");
|
|
230
|
+
if (placeholder) return `getByPlaceholder(${escapeString(placeholder)})`;
|
|
231
|
+
const alt = el.getAttribute("alt");
|
|
232
|
+
if (alt && ["IMG", "APPLET", "AREA", "INPUT"].includes(el.tagName))
|
|
233
|
+
return `getByAltText(${escapeString(alt)})`;
|
|
234
|
+
const title = el.getAttribute("title");
|
|
235
|
+
if (title) return `getByTitle(${escapeString(title)})`;
|
|
236
|
+
const text = (el.textContent || "").trim();
|
|
237
|
+
if (text) {
|
|
238
|
+
const snippet = text.length <= 80 ? text : text.slice(0, 50).replace(/\s+\S*$/, "");
|
|
239
|
+
if (snippet) return `getByText(${escapeString(snippet)})`;
|
|
240
|
+
}
|
|
241
|
+
if (role) return `getByRole(${escapeString(role)})`;
|
|
242
|
+
return `locator(${escapeString(buildCssSelector(el))})`;
|
|
243
|
+
}
|
|
244
|
+
var ROLE_SHORTHANDS = { listitem: "list" };
|
|
245
|
+
function generateLocatorPair(el) {
|
|
246
|
+
const jsLocator = generateLocator(el);
|
|
247
|
+
if (!jsLocator.includes(".filter(")) return { js: jsLocator, pw: jsLocator };
|
|
248
|
+
const role = getImplicitRole(el);
|
|
249
|
+
const name = getAccessibleName(el);
|
|
250
|
+
const container = findContainerAncestor(el);
|
|
251
|
+
const contextText = container ? getContextText(container.ancestor, el) : "";
|
|
252
|
+
if (container && contextText) {
|
|
253
|
+
const pwLocator2 = `getByRole(${escapeString(role)}, { name: ${escapeString(name)} })`;
|
|
254
|
+
const shortRole = ROLE_SHORTHANDS[container.role] ?? container.role;
|
|
255
|
+
return { js: jsLocator, pw: pwLocator2, ancestor: { role: shortRole, text: contextText } };
|
|
256
|
+
}
|
|
257
|
+
const matches = findByRoleAndName(role, name);
|
|
258
|
+
const base = `getByRole(${escapeString(role)}, { name: ${escapeString(name)}, exact: true })`;
|
|
259
|
+
const idx = matches.indexOf(el);
|
|
260
|
+
const pwLocator = idx === 0 ? base + ".first()" : base + `.nth(${idx})`;
|
|
261
|
+
return { js: jsLocator, pw: pwLocator };
|
|
262
|
+
}
|
|
263
|
+
function isTextField(el) {
|
|
264
|
+
if (el instanceof HTMLTextAreaElement) return true;
|
|
265
|
+
if (el instanceof HTMLInputElement) {
|
|
266
|
+
const type = el.type.toLowerCase();
|
|
267
|
+
return !["checkbox", "radio", "submit", "reset", "button", "hidden", "file", "image", "range", "color"].includes(type);
|
|
268
|
+
}
|
|
269
|
+
if (el.getAttribute("contenteditable") === "true") return true;
|
|
270
|
+
return false;
|
|
271
|
+
}
|
|
272
|
+
function isCheckable(el) {
|
|
273
|
+
return el instanceof HTMLInputElement && (el.type === "checkbox" || el.type === "radio");
|
|
274
|
+
}
|
|
275
|
+
function buildCommands(action, el, opts) {
|
|
276
|
+
const { js: jsLocator, pw: pwLocator, ancestor } = generateLocatorPair(el);
|
|
277
|
+
const jsLoc = `page.${jsLocator}`;
|
|
278
|
+
const role = getImplicitRole(el);
|
|
279
|
+
const pwArgs = locatorToPwArgs(pwLocator, role);
|
|
280
|
+
const q = (s) => `"${s}"`;
|
|
281
|
+
const inFlag = ancestor ? ` --in ${ancestor.role} ${q(ancestor.text)}` : "";
|
|
282
|
+
switch (action) {
|
|
283
|
+
case "hover":
|
|
284
|
+
return {
|
|
285
|
+
pw: `hover ${pwArgs}${inFlag}`,
|
|
286
|
+
js: `await ${jsLoc}.hover();`
|
|
287
|
+
};
|
|
288
|
+
case "click":
|
|
289
|
+
return {
|
|
290
|
+
pw: `click ${pwArgs}${inFlag}`,
|
|
291
|
+
js: `await ${jsLoc}.click();`
|
|
292
|
+
};
|
|
293
|
+
case "fill": {
|
|
294
|
+
const val = opts?.value ?? "";
|
|
295
|
+
return {
|
|
296
|
+
pw: `fill ${pwArgs} ${q(val)}${inFlag}`,
|
|
297
|
+
js: `await ${jsLoc}.fill(${escapeString(val)});`
|
|
298
|
+
};
|
|
299
|
+
}
|
|
300
|
+
case "check":
|
|
301
|
+
return {
|
|
302
|
+
pw: `check ${pwArgs}${inFlag}`,
|
|
303
|
+
js: `await ${jsLoc}.check();`
|
|
304
|
+
};
|
|
305
|
+
case "uncheck":
|
|
306
|
+
return {
|
|
307
|
+
pw: `uncheck ${pwArgs}${inFlag}`,
|
|
308
|
+
js: `await ${jsLoc}.uncheck();`
|
|
309
|
+
};
|
|
310
|
+
case "select": {
|
|
311
|
+
const optVal = opts?.option ?? "";
|
|
312
|
+
return {
|
|
313
|
+
pw: `select ${pwArgs} ${q(optVal)}${inFlag}`,
|
|
314
|
+
js: `await ${jsLoc}.selectOption(${escapeString(optVal)});`
|
|
315
|
+
};
|
|
316
|
+
}
|
|
317
|
+
case "press": {
|
|
318
|
+
const key = opts?.key ?? "";
|
|
319
|
+
if (pwArgs) {
|
|
320
|
+
return {
|
|
321
|
+
pw: `press ${pwArgs} ${key}${inFlag}`,
|
|
322
|
+
js: `await ${jsLoc}.press(${escapeString(key)});`
|
|
323
|
+
};
|
|
324
|
+
}
|
|
325
|
+
return {
|
|
326
|
+
pw: `press ${key}`,
|
|
327
|
+
js: `await page.keyboard.press(${escapeString(key)});`
|
|
328
|
+
};
|
|
329
|
+
}
|
|
330
|
+
default:
|
|
331
|
+
return null;
|
|
332
|
+
}
|
|
333
|
+
}
|
|
334
|
+
function buildCssSelector(el) {
|
|
335
|
+
const tag = el.tagName.toLowerCase();
|
|
336
|
+
if (el.id) return `${tag}#${CSS.escape(el.id)}`;
|
|
337
|
+
const classes = [...el.classList].slice(0, 2).map((c) => "." + CSS.escape(c)).join("");
|
|
338
|
+
if (classes) return `${tag}${classes}`;
|
|
339
|
+
return tag;
|
|
340
|
+
}
|
|
341
|
+
|
|
342
|
+
// src/content/recorder.ts
|
|
343
|
+
var SPECIAL_KEYS = /* @__PURE__ */ new Set([
|
|
344
|
+
"Enter",
|
|
345
|
+
"Tab",
|
|
346
|
+
"Escape",
|
|
347
|
+
"Backspace",
|
|
348
|
+
"Delete",
|
|
349
|
+
"ArrowUp",
|
|
350
|
+
"ArrowDown",
|
|
351
|
+
"ArrowLeft",
|
|
352
|
+
"ArrowRight",
|
|
353
|
+
"Home",
|
|
354
|
+
"End",
|
|
355
|
+
"PageUp",
|
|
356
|
+
"PageDown",
|
|
357
|
+
"F1",
|
|
358
|
+
"F2",
|
|
359
|
+
"F3",
|
|
360
|
+
"F4",
|
|
361
|
+
"F5",
|
|
362
|
+
"F6",
|
|
363
|
+
"F7",
|
|
364
|
+
"F8",
|
|
365
|
+
"F9",
|
|
366
|
+
"F10",
|
|
367
|
+
"F11",
|
|
368
|
+
"F12"
|
|
369
|
+
]);
|
|
370
|
+
var pendingFill = null;
|
|
371
|
+
function flushPendingFill() {
|
|
372
|
+
pendingFill = null;
|
|
373
|
+
}
|
|
374
|
+
function onClickCapture(e) {
|
|
375
|
+
const target = e.target;
|
|
376
|
+
if (!target) return;
|
|
377
|
+
if (isTextField(target)) return;
|
|
378
|
+
if (isCheckable(target)) return;
|
|
379
|
+
flushPendingFill();
|
|
380
|
+
if (isHoverRevealed(target)) {
|
|
381
|
+
const hoverTarget = findHoverAncestor(target);
|
|
382
|
+
if (hoverTarget) {
|
|
383
|
+
const hoverCmds = buildCommands("hover", hoverTarget);
|
|
384
|
+
if (hoverCmds) {
|
|
385
|
+
chrome.runtime.sendMessage({ type: "recorded-action", action: hoverCmds });
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
const cmds = buildCommands("click", target);
|
|
390
|
+
if (cmds) {
|
|
391
|
+
chrome.runtime.sendMessage({ type: "recorded-action", action: cmds });
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
function onInputCapture(e) {
|
|
395
|
+
const target = e.target;
|
|
396
|
+
if (!target || !isTextField(target)) return;
|
|
397
|
+
const value = target.value ?? "";
|
|
398
|
+
if (pendingFill && pendingFill.el === target) {
|
|
399
|
+
pendingFill.value = value;
|
|
400
|
+
const cmds = buildCommands("fill", target, { value });
|
|
401
|
+
if (cmds) {
|
|
402
|
+
chrome.runtime.sendMessage({ type: "recorded-fill-update", action: cmds });
|
|
403
|
+
}
|
|
404
|
+
} else {
|
|
405
|
+
flushPendingFill();
|
|
406
|
+
pendingFill = { el: target, value };
|
|
407
|
+
const cmds = buildCommands("fill", target, { value });
|
|
408
|
+
if (cmds) {
|
|
409
|
+
chrome.runtime.sendMessage({ type: "recorded-action", action: cmds });
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
function onChangeCapture(e) {
|
|
414
|
+
const target = e.target;
|
|
415
|
+
if (!target) return;
|
|
416
|
+
if (isCheckable(target)) {
|
|
417
|
+
flushPendingFill();
|
|
418
|
+
const checked = target.checked;
|
|
419
|
+
const cmds = buildCommands(checked ? "check" : "uncheck", target);
|
|
420
|
+
if (cmds) {
|
|
421
|
+
chrome.runtime.sendMessage({ type: "recorded-action", action: cmds });
|
|
422
|
+
}
|
|
423
|
+
return;
|
|
424
|
+
}
|
|
425
|
+
if (target instanceof HTMLSelectElement) {
|
|
426
|
+
flushPendingFill();
|
|
427
|
+
const option = target.value;
|
|
428
|
+
const cmds = buildCommands("select", target, { option });
|
|
429
|
+
if (cmds) {
|
|
430
|
+
chrome.runtime.sendMessage({ type: "recorded-action", action: cmds });
|
|
431
|
+
}
|
|
432
|
+
return;
|
|
433
|
+
}
|
|
434
|
+
}
|
|
435
|
+
function onKeyDownCapture(e) {
|
|
436
|
+
if (!SPECIAL_KEYS.has(e.key)) return;
|
|
437
|
+
const target = e.target;
|
|
438
|
+
if (e.key === "Tab") {
|
|
439
|
+
flushPendingFill();
|
|
440
|
+
return;
|
|
441
|
+
}
|
|
442
|
+
if (e.key !== "Enter" && target && isTextField(target)) return;
|
|
443
|
+
flushPendingFill();
|
|
444
|
+
const cmds = target && target !== document.body && target !== document.documentElement ? buildCommands("press", target, { key: e.key }) : { pw: `press ${e.key}`, js: `await page.keyboard.press(${escapeString(e.key)});` };
|
|
445
|
+
if (cmds) {
|
|
446
|
+
chrome.runtime.sendMessage({ type: "recorded-action", action: cmds });
|
|
447
|
+
}
|
|
448
|
+
}
|
|
449
|
+
function onFocusOutCapture(e) {
|
|
450
|
+
if (pendingFill && e.target === pendingFill.el) {
|
|
451
|
+
flushPendingFill();
|
|
452
|
+
}
|
|
453
|
+
}
|
|
454
|
+
function cleanup() {
|
|
455
|
+
flushPendingFill();
|
|
456
|
+
window.__pw_recorder_active = false;
|
|
457
|
+
document.removeEventListener("click", onClickCapture, true);
|
|
458
|
+
document.removeEventListener("input", onInputCapture, true);
|
|
459
|
+
document.removeEventListener("change", onChangeCapture, true);
|
|
460
|
+
document.removeEventListener("keydown", onKeyDownCapture, true);
|
|
461
|
+
document.removeEventListener("focusout", onFocusOutCapture, true);
|
|
462
|
+
chrome.runtime.onMessage.removeListener(onMessage);
|
|
463
|
+
}
|
|
464
|
+
function onMessage(msg) {
|
|
465
|
+
if (msg.type === "record-stop") cleanup();
|
|
466
|
+
}
|
|
467
|
+
function init() {
|
|
468
|
+
if (window.__pw_recorder_active) return;
|
|
469
|
+
window.__pw_recorder_active = true;
|
|
470
|
+
chrome.runtime.onMessage.addListener(onMessage);
|
|
471
|
+
document.addEventListener("click", onClickCapture, true);
|
|
472
|
+
document.addEventListener("input", onInputCapture, true);
|
|
473
|
+
document.addEventListener("change", onChangeCapture, true);
|
|
474
|
+
document.addEventListener("keydown", onKeyDownCapture, true);
|
|
475
|
+
document.addEventListener("focusout", onFocusOutCapture, true);
|
|
476
|
+
}
|
|
477
|
+
init();
|
|
478
|
+
})();
|
|
479
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/content/locator.ts", "src/content/recorder.ts"],
  "sourcesContent": ["/**\r\n * Shared locator generation utilities.\r\n * Used by both picker.ts and recorder.ts content scripts.\r\n * Vite inlines this into each content script bundle \u2014 no runtime module loading.\r\n */\r\n\r\n// \u2500\u2500\u2500 Implicit ARIA roles \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport const IMPLICIT_ROLES: Record<string, string | ((el: Element) => string | null)> = {\r\n    A: (el) => el.hasAttribute('href') ? 'link' : null,\r\n    BUTTON: 'button',\r\n    H1: 'heading', H2: 'heading', H3: 'heading', H4: 'heading', H5: 'heading', H6: 'heading',\r\n    INPUT: (el) => {\r\n        const type = (el as HTMLInputElement).type.toLowerCase();\r\n        if (type === 'checkbox') return 'checkbox';\r\n        if (type === 'radio') return 'radio';\r\n        if (type === 'submit' || type === 'reset' || type === 'button') return 'button';\r\n        if (type === 'hidden') return null;\r\n        return 'textbox';\r\n    },\r\n    TEXTAREA: 'textbox',\r\n    SELECT: 'combobox',\r\n    IMG: 'img',\r\n    NAV: 'navigation',\r\n    MAIN: 'main',\r\n    HEADER: 'banner',\r\n    FOOTER: 'contentinfo',\r\n    P: 'paragraph',\r\n    UL: 'list', OL: 'list',\r\n    LI: 'listitem',\r\n    TABLE: 'table',\r\n    TR: 'row',\r\n    TH: 'columnheader',\r\n    TD: 'cell',\r\n    FORM: 'form',\r\n    DIALOG: 'dialog',\r\n    ARTICLE: 'article',\r\n};\r\n\r\nexport function getImplicitRole(el: Element): string | null {\r\n    const explicit = el.getAttribute('role');\r\n    if (explicit && explicit !== 'none' && explicit !== 'presentation') return explicit;\r\n    const entry = IMPLICIT_ROLES[el.tagName];\r\n    if (!entry) return null;\r\n    return typeof entry === 'function' ? entry(el) : entry;\r\n}\r\n\r\n// \u2500\u2500\u2500 Accessible name \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport function getAccessibleName(el: Element): string {\r\n    // aria-label\r\n    const ariaLabel = el.getAttribute('aria-label');\r\n    if (ariaLabel) return ariaLabel.trim();\r\n\r\n    // aria-labelledby\r\n    const labelledBy = el.getAttribute('aria-labelledby');\r\n    if (labelledBy) {\r\n        const parts = labelledBy.split(/\\s+/).map(id => document.getElementById(id)?.textContent?.trim()).filter(Boolean);\r\n        if (parts.length) return parts.join(' ');\r\n    }\r\n\r\n    // For inputs: associated <label>\r\n    if (el instanceof HTMLInputElement || el instanceof HTMLTextAreaElement || el instanceof HTMLSelectElement) {\r\n        const label = getLabel(el);\r\n        if (label) return label;\r\n    }\r\n\r\n    // For roles that get name from content (ARIA \"name from content\" roles)\r\n    const role = getImplicitRole(el);\r\n    const NAME_FROM_CONTENT = new Set([\r\n        'button', 'link', 'heading', 'tab', 'menuitem', 'menuitemcheckbox',\r\n        'menuitemradio', 'option', 'radio', 'checkbox', 'switch', 'cell',\r\n        'columnheader', 'rowheader', 'tooltip', 'treeitem',\r\n    ]);\r\n    if (role && NAME_FROM_CONTENT.has(role)) {\r\n        const text = (el.textContent || '').trim();\r\n        if (text && text.length <= 80) return text;\r\n    }\r\n\r\n    // alt for images\r\n    if (el.tagName === 'IMG') {\r\n        const alt = el.getAttribute('alt');\r\n        if (alt) return alt.trim();\r\n    }\r\n\r\n    return '';\r\n}\r\n\r\nexport function getLabel(el: HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement): string {\r\n    // Explicit label via for attribute\r\n    if (el.id) {\r\n        const label = document.querySelector(`label[for=\"${CSS.escape(el.id)}\"]`);\r\n        if (label) return (label.textContent || '').trim();\r\n    }\r\n    // Implicit label (ancestor)\r\n    const parentLabel = el.closest('label');\r\n    if (parentLabel) {\r\n        // Get label text excluding the input's own text\r\n        const clone = parentLabel.cloneNode(true) as HTMLElement;\r\n        clone.querySelectorAll('input,textarea,select').forEach(c => c.remove());\r\n        const text = (clone.textContent || '').trim();\r\n        if (text) return text;\r\n    }\r\n    return '';\r\n}\r\n\r\n// \u2500\u2500\u2500 Ancestor context disambiguation \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\n/** Container roles suitable for ancestor-context disambiguation. */\r\nconst CONTAINER_ROLES = new Set(['listitem', 'row', 'article', 'group']);\r\n\r\n/**\r\n * Find distinctive text in an ancestor that doesn't come from the excluded element's subtree.\r\n * Returns text from the first child subtree that doesn't contain the excluded element,\r\n * recursing into wrappers that do contain it. This ensures the returned text is a\r\n * contiguous substring of the ancestor's full textContent (needed for hasText matching).\r\n */\r\nfunction getContextText(ancestor: Element, exclude: Element): string {\r\n    function findText(node: Node): string {\r\n        for (const child of node.childNodes) {\r\n            if (child === exclude) continue;\r\n            if (child.nodeType === Node.ELEMENT_NODE && (child as Element).contains(exclude)) {\r\n                // This subtree contains the target \u2014 recurse to find non-target siblings\r\n                const inner = findText(child);\r\n                if (inner) return inner;\r\n                continue;\r\n            }\r\n            const text = (child.textContent || '').trim();\r\n            if (text && text.length <= 50) return text;\r\n        }\r\n        return '';\r\n    }\r\n    return findText(ancestor);\r\n}\r\n\r\n/** Walk up from el to find nearest ancestor with a container role. */\r\nfunction findContainerAncestor(el: Element): { ancestor: Element; role: string } | null {\r\n    let current = el.parentElement;\r\n    while (current && current !== document.body && current !== document.documentElement) {\r\n        const role = getImplicitRole(current);\r\n        if (role && CONTAINER_ROLES.has(role)) return { ancestor: current, role };\r\n        current = current.parentElement;\r\n    }\r\n    return null;\r\n}\r\n\r\n/**\r\n * Try to disambiguate using ancestor context.\r\n * Returns a chained locator like:\r\n *   getByRole('listitem').filter({ hasText: 'reading' }).getByRole('button', { name: 'Delete' })\r\n * or null if ancestor context doesn't produce a unique result.\r\n */\r\nfunction tryAncestorContext(el: Element, role: string, name: string, matches: Element[]): string | null {\r\n    const container = findContainerAncestor(el);\r\n    if (!container) return null;\r\n\r\n    const contextText = getContextText(container.ancestor, el);\r\n    if (!contextText || contextText.length > 50) return null;\r\n\r\n    // Verify uniqueness: only one match's container ancestor should contain this text\r\n    let count = 0;\r\n    for (const match of matches) {\r\n        const mc = findContainerAncestor(match);\r\n        if (!mc || mc.role !== container.role) continue;\r\n        if ((mc.ancestor.textContent || '').includes(contextText)) {\r\n            count++;\r\n            if (count > 1) return null;\r\n        }\r\n    }\r\n    if (count !== 1) return null;\r\n\r\n    return `getByRole(${escapeString(container.role)}).filter({ hasText: ${escapeString(contextText)} }).getByRole(${escapeString(role)}, { name: ${escapeString(name)} })`;\r\n}\r\n\r\n// \u2500\u2500\u2500 Locator disambiguation \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport function findByRoleAndName(role: string, name: string): Element[] {\r\n    const matches: Element[] = [];\r\n    for (const el of document.querySelectorAll('*')) {\r\n        if (getImplicitRole(el) === role && getAccessibleName(el) === name\r\n            && (el as HTMLElement).checkVisibility?.() !== false)\r\n            matches.push(el);\r\n    }\r\n    return matches;\r\n}\r\n\r\n/** Like findByRoleAndName but includes hidden elements (for hover detection). */\r\nexport function findAllByRoleAndName(role: string, name: string): Element[] {\r\n    const matches: Element[] = [];\r\n    for (const el of document.querySelectorAll('*')) {\r\n        if (getImplicitRole(el) === role && getAccessibleName(el) === name)\r\n            matches.push(el);\r\n    }\r\n    return matches;\r\n}\r\n\r\n/** Find the nearest :hover ancestor (for recording hover before click). */\r\nexport function findHoverAncestor(el: Element): Element | null {\r\n    let ancestor = el.parentElement;\r\n    while (ancestor && ancestor !== document.body && ancestor !== document.documentElement) {\r\n        if (ancestor.matches(':hover')) return ancestor;\r\n        ancestor = ancestor.parentElement;\r\n    }\r\n    return null;\r\n}\r\n\r\n/**\r\n * Check if element is revealed by a :hover CSS rule.\r\n * Scans stylesheets for rules like `.parent:hover .child { display: inline-block }`.\r\n * At click time the ancestor IS hovered, so el.matches() works against :hover selectors.\r\n */\r\nexport function isHoverRevealed(el: Element): boolean {\r\n    for (const sheet of document.styleSheets) {\r\n        try {\r\n            for (const rule of sheet.cssRules) {\r\n                if (!(rule instanceof CSSStyleRule)) continue;\r\n                if (!rule.selectorText.includes(':hover')) continue;\r\n                const s = rule.style;\r\n                const reveals = (s.display && s.display !== 'none') ||\r\n                    s.visibility === 'visible' ||\r\n                    (s.opacity && s.opacity !== '0');\r\n                if (!reveals) continue;\r\n                try { if (el.matches(rule.selectorText)) return true; } catch { /* invalid selector */ }\r\n            }\r\n        } catch { /* cross-origin stylesheet */ }\r\n    }\r\n    return false;\r\n}\r\n\r\n// \u2500\u2500\u2500 Locator string conversion \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\n/**\r\n * Parse a JS locator string into PW keyword args.\r\n * e.g. `getByRole('tab', { name: 'npm', exact: true }).nth(1)` \u2192 `tab \"npm\" --nth 1`\r\n */\r\nexport function locatorToPwArgs(locator: string, role?: string | null): string {\r\n    const q = (s: string) => `\"${s}\"`;\r\n\r\n    // Extract nth modifier\r\n    let nth = '';\r\n    if (/\\.first\\(\\)/.test(locator)) nth = ' --nth 0';\r\n    else if (/\\.last\\(\\)/.test(locator)) nth = ' --nth -1';\r\n    else {\r\n        const nthMatch = locator.match(/\\.nth\\((\\d+)\\)/);\r\n        if (nthMatch) nth = ` --nth ${nthMatch[1]}`;\r\n    }\r\n\r\n    // getByRole with name \u2014 already has role\r\n    const roleNameMatch = locator.match(/getByRole\\(['\"](.+?)['\"],\\s*\\{[^}]*name:\\s*['\"](.+?)['\"]/);\r\n    if (roleNameMatch) return `${roleNameMatch[1]} ${q(roleNameMatch[2])}${nth}`;\r\n\r\n    // getByRole without name \u2014 already has role\r\n    const roleMatch = locator.match(/getByRole\\(['\"](.+?)['\"]\\)/);\r\n    if (roleMatch) return `${roleMatch[1]}${nth}`;\r\n\r\n    // getByTestId \u2014 test ID is not an accessible name, don't add role\r\n    const testIdMatch = locator.match(/getByTestId\\(['\"](.+?)['\"]\\)/);\r\n    if (testIdMatch) return `${q(testIdMatch[1])}${nth}`;\r\n\r\n    // getByLabel / getByText / getByPlaceholder / getByTitle / getByAltText \u2014 prepend role if available\r\n    const getByMatch = locator.match(/getBy\\w+\\(['\"](.+?)['\"]\\)/);\r\n    if (getByMatch) {\r\n        const prefix = role ? `${role} ` : '';\r\n        return `${prefix}${q(getByMatch[1])}${nth}`;\r\n    }\r\n\r\n    // locator('css') fallback\r\n    const locatorMatch = locator.match(/locator\\(['\"](.+?)['\"]\\)/);\r\n    if (locatorMatch) return `${q(locatorMatch[1])}${nth}`;\r\n\r\n    return q(locator);\r\n}\r\n\r\n// \u2500\u2500\u2500 Locator generation \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport function escapeString(s: string): string {\r\n    if (!s.includes(\"'\")) return `'${s}'`;\r\n    if (!s.includes('\"')) return `\"${s}\"`;\r\n    return `'${s.replace(/'/g, \"\\\\'\")}'`;\r\n}\r\n\r\nexport function generateLocator(el: Element): string {\r\n    // 1. Test ID\r\n    const testId = el.getAttribute('data-testid') || el.getAttribute('data-test-id');\r\n    if (testId) return `getByTestId(${escapeString(testId)})`;\r\n\r\n    // 2. Role + accessible name\r\n    const role = getImplicitRole(el);\r\n    const name = getAccessibleName(el);\r\n    if (role && name) {\r\n        // Disambiguate when multiple elements share same role + name\r\n        const matches = findByRoleAndName(role, name);\r\n        if (matches.length > 1) {\r\n            // Try ancestor context first (readable chained locators)\r\n            const ancestorLocator = tryAncestorContext(el, role, name, matches);\r\n            if (ancestorLocator) return ancestorLocator;\r\n            // Fallback to nth-based disambiguation\r\n            const base = `getByRole(${escapeString(role)}, { name: ${escapeString(name)}, exact: true })`;\r\n            const idx = matches.indexOf(el);\r\n            return idx === 0 ? base + '.first()' : base + `.nth(${idx})`;\r\n        }\r\n        return `getByRole(${escapeString(role)}, { name: ${escapeString(name)} })`;\r\n    }\r\n\r\n    // 3. Label (for form elements)\r\n    if (el instanceof HTMLInputElement || el instanceof HTMLTextAreaElement || el instanceof HTMLSelectElement) {\r\n        const label = getLabel(el);\r\n        if (label) return `getByLabel(${escapeString(label)})`;\r\n    }\r\n\r\n    // 4. Placeholder\r\n    const placeholder = el.getAttribute('placeholder');\r\n    if (placeholder) return `getByPlaceholder(${escapeString(placeholder)})`;\r\n\r\n    // 5. Alt text\r\n    const alt = el.getAttribute('alt');\r\n    if (alt && ['IMG', 'APPLET', 'AREA', 'INPUT'].includes(el.tagName))\r\n        return `getByAltText(${escapeString(alt)})`;\r\n\r\n    // 6. Title\r\n    const title = el.getAttribute('title');\r\n    if (title) return `getByTitle(${escapeString(title)})`;\r\n\r\n    // 7. Text content (use substring for long text \u2014 getByText does partial matching)\r\n    const text = (el.textContent || '').trim();\r\n    if (text) {\r\n        const snippet = text.length <= 80 ? text : text.slice(0, 50).replace(/\\s+\\S*$/, '');\r\n        if (snippet) return `getByText(${escapeString(snippet)})`;\r\n    }\r\n\r\n    // 8. Role without name\r\n    if (role) return `getByRole(${escapeString(role)})`;\r\n\r\n    // 9. CSS fallback\r\n    return `locator(${escapeString(buildCssSelector(el))})`;\r\n}\r\n\r\n/** Shorthand mapping for PW --in flag (listitem \u2192 list for readability). */\r\nconst ROLE_SHORTHANDS: Record<string, string> = { listitem: 'list' };\r\n\r\n/**\r\n * Generate separate JS and PW locators.\r\n * JS uses ancestor context (.filter chains); PW uses --in flag.\r\n * Falls back to .nth() when ancestor context isn't available.\r\n */\r\nexport function generateLocatorPair(el: Element): { js: string; pw: string; ancestor?: { role: string; text: string } } {\r\n    const jsLocator = generateLocator(el);\r\n    if (!jsLocator.includes('.filter(')) return { js: jsLocator, pw: jsLocator };\r\n\r\n    // JS used ancestor context \u2014 extract ancestor info for PW --in flag\r\n    const role = getImplicitRole(el)!;\r\n    const name = getAccessibleName(el);\r\n    const container = findContainerAncestor(el);\r\n    const contextText = container ? getContextText(container.ancestor, el) : '';\r\n\r\n    if (container && contextText) {\r\n        const pwLocator = `getByRole(${escapeString(role)}, { name: ${escapeString(name)} })`;\r\n        const shortRole = ROLE_SHORTHANDS[container.role] ?? container.role;\r\n        return { js: jsLocator, pw: pwLocator, ancestor: { role: shortRole, text: contextText } };\r\n    }\r\n\r\n    // No ancestor context \u2014 fall back to .nth()\r\n    const matches = findByRoleAndName(role, name);\r\n    const base = `getByRole(${escapeString(role)}, { name: ${escapeString(name)}, exact: true })`;\r\n    const idx = matches.indexOf(el);\r\n    const pwLocator = idx === 0 ? base + '.first()' : base + `.nth(${idx})`;\r\n    return { js: jsLocator, pw: pwLocator };\r\n}\r\n\r\n// \u2500\u2500\u2500 Element classification \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\n/** Check if element is a text-entry field */\r\nexport function isTextField(el: Element): boolean {\r\n    if (el instanceof HTMLTextAreaElement) return true;\r\n    if (el instanceof HTMLInputElement) {\r\n        const type = el.type.toLowerCase();\r\n        return !['checkbox', 'radio', 'submit', 'reset', 'button', 'hidden', 'file', 'image', 'range', 'color'].includes(type);\r\n    }\r\n    // contenteditable\r\n    if (el.getAttribute('contenteditable') === 'true') return true;\r\n    return false;\r\n}\r\n\r\n/** Check if element is a checkbox or radio */\r\nexport function isCheckable(el: Element): boolean {\r\n    return el instanceof HTMLInputElement && (el.type === 'checkbox' || el.type === 'radio');\r\n}\r\n\r\n// \u2500\u2500\u2500 Command building \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\n/** Build both PW and JS command strings for a recorder action */\r\nexport function buildCommands(action: string, el: Element, opts?: {\r\n    value?: string;\r\n    key?: string;\r\n    checked?: boolean;\r\n    option?: string;\r\n}): { pw: string; js: string } | null {\r\n    const { js: jsLocator, pw: pwLocator, ancestor } = generateLocatorPair(el);\r\n    const jsLoc = `page.${jsLocator}`;\r\n    const role = getImplicitRole(el);\r\n    const pwArgs = locatorToPwArgs(pwLocator, role);\r\n    const q = (s: string) => `\"${s}\"`;\r\n    const inFlag = ancestor ? ` --in ${ancestor.role} ${q(ancestor.text)}` : '';\r\n\r\n    switch (action) {\r\n        case 'hover':\r\n            return {\r\n                pw: `hover ${pwArgs}${inFlag}`,\r\n                js: `await ${jsLoc}.hover();`,\r\n            };\r\n\r\n        case 'click':\r\n            return {\r\n                pw: `click ${pwArgs}${inFlag}`,\r\n                js: `await ${jsLoc}.click();`,\r\n            };\r\n\r\n        case 'fill': {\r\n            const val = opts?.value ?? '';\r\n            return {\r\n                pw: `fill ${pwArgs} ${q(val)}${inFlag}`,\r\n                js: `await ${jsLoc}.fill(${escapeString(val)});`,\r\n            };\r\n        }\r\n\r\n        case 'check':\r\n            return {\r\n                pw: `check ${pwArgs}${inFlag}`,\r\n                js: `await ${jsLoc}.check();`,\r\n            };\r\n\r\n        case 'uncheck':\r\n            return {\r\n                pw: `uncheck ${pwArgs}${inFlag}`,\r\n                js: `await ${jsLoc}.uncheck();`,\r\n            };\r\n\r\n        case 'select': {\r\n            const optVal = opts?.option ?? '';\r\n            return {\r\n                pw: `select ${pwArgs} ${q(optVal)}${inFlag}`,\r\n                js: `await ${jsLoc}.selectOption(${escapeString(optVal)});`,\r\n            };\r\n        }\r\n\r\n        case 'press': {\r\n            const key = opts?.key ?? '';\r\n            if (pwArgs) {\r\n                return {\r\n                    pw: `press ${pwArgs} ${key}${inFlag}`,\r\n                    js: `await ${jsLoc}.press(${escapeString(key)});`,\r\n                };\r\n            }\r\n            // Global key press (no locator context)\r\n            return {\r\n                pw: `press ${key}`,\r\n                js: `await page.keyboard.press(${escapeString(key)});`,\r\n            };\r\n        }\r\n\r\n        default:\r\n            return null;\r\n    }\r\n}\r\n\r\n// \u2500\u2500\u2500 CSS selector \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport function buildCssSelector(el: Element): string {\r\n    const tag = el.tagName.toLowerCase();\r\n    if (el.id) return `${tag}#${CSS.escape(el.id)}`;\r\n    const classes = [...el.classList].slice(0, 2).map(c => '.' + CSS.escape(c)).join('');\r\n    if (classes) return `${tag}${classes}`;\r\n    return tag;\r\n}\r\n", "/**\r\n * Recorder content script.\r\n * Injected into the active tab via chrome.scripting.executeScript.\r\n * Captures DOM events, generates locator + PW/JS commands, sends to panel.\r\n *\r\n * Transparent: never calls preventDefault/stopPropagation \u2014 user actions flow normally.\r\n */\r\nimport { escapeString, isTextField, isCheckable, buildCommands, findHoverAncestor, isHoverRevealed } from './locator';\r\n\r\n// \u2500\u2500\u2500 Special key detection \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport const SPECIAL_KEYS = new Set([\r\n    'Enter', 'Tab', 'Escape', 'Backspace', 'Delete',\r\n    'ArrowUp', 'ArrowDown', 'ArrowLeft', 'ArrowRight',\r\n    'Home', 'End', 'PageUp', 'PageDown',\r\n    'F1', 'F2', 'F3', 'F4', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12',\r\n]);\r\n\r\n// \u2500\u2500\u2500 Fill buffering state machine \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport let pendingFill: { el: Element; value: string } | null = null;\r\n\r\nexport function flushPendingFill() {\r\n    pendingFill = null;\r\n}\r\n\r\n// \u2500\u2500\u2500 Event handlers (capture phase, transparent) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport function onClickCapture(e: MouseEvent) {\r\n    const target = e.target as Element;\r\n    if (!target) return;\r\n\r\n    // Skip clicks on text fields (focus-click noise before fill)\r\n    if (isTextField(target)) return;\r\n\r\n    // Skip clicks on checkable elements (handled by change event)\r\n    if (isCheckable(target)) return;\r\n\r\n    // Flush any pending fill\r\n    flushPendingFill();\r\n\r\n    // Detect hover-revealed elements: if a :hover CSS rule reveals this element,\r\n    // emit a hover command on the :hover ancestor so replay works.\r\n    if (isHoverRevealed(target)) {\r\n        const hoverTarget = findHoverAncestor(target);\r\n        if (hoverTarget) {\r\n            const hoverCmds = buildCommands('hover', hoverTarget);\r\n            if (hoverCmds) {\r\n                chrome.runtime.sendMessage({ type: 'recorded-action', action: hoverCmds });\r\n            }\r\n        }\r\n    }\r\n\r\n    const cmds = buildCommands('click', target);\r\n    if (cmds) {\r\n        chrome.runtime.sendMessage({ type: 'recorded-action', action: cmds });\r\n    }\r\n}\r\n\r\nexport function onInputCapture(e: Event) {\r\n    const target = e.target as Element;\r\n    if (!target || !isTextField(target)) return;\r\n\r\n    const value = (target as HTMLInputElement | HTMLTextAreaElement).value ?? '';\r\n\r\n    if (pendingFill && pendingFill.el === target) {\r\n        // Same element \u2014 update\r\n        pendingFill.value = value;\r\n        const cmds = buildCommands('fill', target, { value });\r\n        if (cmds) {\r\n            chrome.runtime.sendMessage({ type: 'recorded-fill-update', action: cmds });\r\n        }\r\n    } else {\r\n        // Different element or first input \u2014 flush old, start new\r\n        flushPendingFill();\r\n        pendingFill = { el: target, value };\r\n        const cmds = buildCommands('fill', target, { value });\r\n        if (cmds) {\r\n            chrome.runtime.sendMessage({ type: 'recorded-action', action: cmds });\r\n        }\r\n    }\r\n}\r\n\r\nexport function onChangeCapture(e: Event) {\r\n    const target = e.target as Element;\r\n    if (!target) return;\r\n\r\n    // Checkbox / radio\r\n    if (isCheckable(target)) {\r\n        flushPendingFill();\r\n        const checked = (target as HTMLInputElement).checked;\r\n        const cmds = buildCommands(checked ? 'check' : 'uncheck', target);\r\n        if (cmds) {\r\n            chrome.runtime.sendMessage({ type: 'recorded-action', action: cmds });\r\n        }\r\n        return;\r\n    }\r\n\r\n    // Select\r\n    if (target instanceof HTMLSelectElement) {\r\n        flushPendingFill();\r\n        const option = target.value;\r\n        const cmds = buildCommands('select', target, { option });\r\n        if (cmds) {\r\n            chrome.runtime.sendMessage({ type: 'recorded-action', action: cmds });\r\n        }\r\n        return;\r\n    }\r\n}\r\n\r\nexport function onKeyDownCapture(e: KeyboardEvent) {\r\n    if (!SPECIAL_KEYS.has(e.key)) return;\r\n\r\n    const target = e.target as Element;\r\n\r\n    // Tab changes focus but is navigation noise \u2014 flush fill, don't emit\r\n    if (e.key === 'Tab') { flushPendingFill(); return; }\r\n\r\n    // Inside a text field, only Enter is a meaningful action \u2014\r\n    // everything else (Backspace, arrows, etc.) is editing noise\r\n    if (e.key !== 'Enter' && target && isTextField(target)) return;\r\n\r\n    // Any special key during fill \u2192 flush fill, then fall through to emit press\r\n    flushPendingFill();\r\n\r\n    const cmds = target && target !== document.body && target !== document.documentElement\r\n        ? buildCommands('press', target, { key: e.key })\r\n        : { pw: `press ${e.key}`, js: `await page.keyboard.press(${escapeString(e.key)});` };\r\n    if (cmds) {\r\n        chrome.runtime.sendMessage({ type: 'recorded-action', action: cmds });\r\n    }\r\n}\r\n\r\nexport function onFocusOutCapture(e: FocusEvent) {\r\n    if (pendingFill && e.target === pendingFill.el) {\r\n        flushPendingFill();\r\n    }\r\n}\r\n\r\n// \u2500\u2500\u2500 Lifecycle \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport function cleanup() {\r\n    flushPendingFill();\r\n    (window as any).__pw_recorder_active = false;\r\n    document.removeEventListener('click', onClickCapture, true);\r\n    document.removeEventListener('input', onInputCapture, true);\r\n    document.removeEventListener('change', onChangeCapture, true);\r\n    document.removeEventListener('keydown', onKeyDownCapture, true);\r\n    document.removeEventListener('focusout', onFocusOutCapture, true);\r\n    chrome.runtime.onMessage.removeListener(onMessage);\r\n}\r\n\r\nfunction onMessage(msg: any) {\r\n    if (msg.type === 'record-stop') cleanup();\r\n}\r\n\r\n// \u2500\u2500\u2500 Init \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\r\n\r\nexport function init() {\r\n    // Guard against double-injection\r\n    if ((window as any).__pw_recorder_active) return;\r\n    (window as any).__pw_recorder_active = true;\r\n\r\n    chrome.runtime.onMessage.addListener(onMessage);\r\n    document.addEventListener('click', onClickCapture, true);\r\n    document.addEventListener('input', onInputCapture, true);\r\n    document.addEventListener('change', onChangeCapture, true);\r\n    document.addEventListener('keydown', onKeyDownCapture, true);\r\n    document.addEventListener('focusout', onFocusOutCapture, true);\r\n}\r\n\r\ninit();\r\n"],
  "mappings": "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",
  "names": ["pwLocator"]
}

|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
(() => {
|
|
3
|
+
// src/content/trace-loader.ts
|
|
4
|
+
chrome.runtime.onMessage.addListener((msg) => {
|
|
5
|
+
if (msg.type !== "load-trace" || !msg.data) return;
|
|
6
|
+
const bytes = new Uint8Array(msg.data);
|
|
7
|
+
const blob = new Blob([bytes], { type: "application/zip" });
|
|
8
|
+
window.postMessage({ method: "load", params: { trace: blob } }, "*");
|
|
9
|
+
});
|
|
10
|
+
chrome.runtime.sendMessage({ type: "trace-loader-ready" });
|
|
11
|
+
})();
|
|
12
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsic3JjL2NvbnRlbnQvdHJhY2UtbG9hZGVyLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvLyBDb250ZW50IHNjcmlwdCBpbmplY3RlZCBpbnRvIHRyYWNlLnBsYXl3cmlnaHQuZGV2XHJcbi8vIFJlY2VpdmVzIHRyYWNlIGRhdGEgZnJvbSB0aGUgYmFja2dyb3VuZCBTVyBhbmQgbG9hZHMgaXQgdmlhIHBvc3RNZXNzYWdlLlxyXG5cclxuY2hyb21lLnJ1bnRpbWUub25NZXNzYWdlLmFkZExpc3RlbmVyKChtc2cpID0+IHtcclxuICBpZiAobXNnLnR5cGUgIT09ICdsb2FkLXRyYWNlJyB8fCAhbXNnLmRhdGEpIHJldHVybjtcclxuICBjb25zdCBieXRlcyA9IG5ldyBVaW50OEFycmF5KG1zZy5kYXRhKTtcclxuICBjb25zdCBibG9iID0gbmV3IEJsb2IoW2J5dGVzXSwgeyB0eXBlOiAnYXBwbGljYXRpb24vemlwJyB9KTtcclxuICB3aW5kb3cucG9zdE1lc3NhZ2UoeyBtZXRob2Q6ICdsb2FkJywgcGFyYW1zOiB7IHRyYWNlOiBibG9iIH0gfSwgJyonKTtcclxufSk7XHJcblxyXG4vLyBTaWduYWwgcmVhZHlcclxuY2hyb21lLnJ1bnRpbWUuc2VuZE1lc3NhZ2UoeyB0eXBlOiAndHJhY2UtbG9hZGVyLXJlYWR5JyB9KTtcclxuIl0sCiAgIm1hcHBpbmdzIjogIjs7O0FBR0EsU0FBTyxRQUFRLFVBQVUsWUFBWSxDQUFDLFFBQVE7QUFDNUMsUUFBSSxJQUFJLFNBQVMsZ0JBQWdCLENBQUMsSUFBSSxLQUFNO0FBQzVDLFVBQU0sUUFBUSxJQUFJLFdBQVcsSUFBSSxJQUFJO0FBQ3JDLFVBQU0sT0FBTyxJQUFJLEtBQUssQ0FBQyxLQUFLLEdBQUcsRUFBRSxNQUFNLGtCQUFrQixDQUFDO0FBQzFELFdBQU8sWUFBWSxFQUFFLFFBQVEsUUFBUSxRQUFRLEVBQUUsT0FBTyxLQUFLLEVBQUUsR0FBRyxHQUFHO0FBQUEsRUFDckUsQ0FBQztBQUdELFNBQU8sUUFBUSxZQUFZLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQzsiLAogICJuYW1lcyI6IFtdCn0K
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8">
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
6
|
+
<title>REPL</title>
|
|
7
|
+
<script type="module" crossorigin src="../devtools/console.js"></script>
|
|
8
|
+
<link rel="modulepreload" crossorigin href="../modulepreload-polyfill.js">
|
|
9
|
+
<link rel="modulepreload" crossorigin href="../index.js">
|
|
10
|
+
<link rel="modulepreload" crossorigin href="../sw-debugger-core.js">
|
|
11
|
+
<link rel="modulepreload" crossorigin href="../index2.js">
|
|
12
|
+
<link rel="stylesheet" crossorigin href="../index.css">
|
|
13
|
+
</head>
|
|
14
|
+
<body>
|
|
15
|
+
<div id="root"></div>
|
|
16
|
+
</body>
|
|
17
|
+
</html>
|