134 lines
4.9 KiB
JavaScript
134 lines
4.9 KiB
JavaScript
const { chromium } = require('playwright');
|
||
|
||
(async () => {
|
||
// 启动浏览器并打开页面
|
||
const browser = await chromium.launch({ headless: true });
|
||
const page = await browser.newPage();
|
||
// await page.goto('https://www.baidu.com');
|
||
// await page.goto('https://play.grafana.org/a/grafana-synthetic-monitoring-app/probes', { waitUntil: 'domcontentloaded' });
|
||
await page.goto('https://play.grafana.org/d/be9htelw63ke8b/metrics-rename-example?orgId=1&from=now-6h&to=now&timezone=utc', { waitUntil: 'domcontentloaded' });
|
||
|
||
await page.waitForTimeout(10000);
|
||
|
||
console.log('page loaded');
|
||
|
||
// 提取完整的 AXTree(设置 interestingOnly: false 可获取全部节点)
|
||
const axTree = await page.accessibility.snapshot({ interestingOnly: false });
|
||
|
||
// 全局计数器和编号到"伪选择器"映射的对象
|
||
let idCounter = 1;
|
||
const idToSelector = {};
|
||
|
||
// 在文件开头添加一个全局变量来存储父子关系
|
||
const nodeParents = new Map();
|
||
|
||
function traverse(node, depth = 0, parent = null) {
|
||
// 记录父节点关系
|
||
nodeParents.set(node, parent);
|
||
|
||
// 增加 InlineTextBox 到过滤条件中
|
||
if (((node.role === 'none' || node.role === 'generic' || node.role === 'InlineTextBox') &&
|
||
!node.name &&
|
||
!node.focusable &&
|
||
!node.focused &&
|
||
node.expanded === undefined) ||
|
||
node.role === 'InlineTextBox' // 无论如何都跳过 InlineTextBox
|
||
) {
|
||
// 直接处理子节点
|
||
if (node.children && node.children.length > 0) {
|
||
for (const child of node.children) {
|
||
traverse(child, depth, node);
|
||
}
|
||
}
|
||
return;
|
||
}
|
||
|
||
const currentId = idCounter++;
|
||
|
||
// 构建更详细的 selector
|
||
let selectorParts = [`role=${node.role}`];
|
||
|
||
if (node.name) {
|
||
selectorParts.push(`[name="${node.name}"]`);
|
||
}
|
||
|
||
// 添加其他可能的属性
|
||
if (node.selected) selectorParts.push('[selected=true]');
|
||
if (node.checked !== undefined) selectorParts.push(`[checked=${node.checked}]`);
|
||
if (node.pressed !== undefined) selectorParts.push(`[pressed=${node.pressed}]`);
|
||
|
||
// 如果有父节点,添加父节点信息
|
||
if (parent && parent.role !== 'WebArea') {
|
||
let parentSelector = `role=${parent.role}`;
|
||
if (parent.name) {
|
||
parentSelector += `[name="${parent.name}"]`;
|
||
}
|
||
selectorParts.unshift(`${parentSelector} >>`);
|
||
}
|
||
|
||
// 如果是列表项,添加位置信息
|
||
if (parent && parent.children) {
|
||
const siblingIndex = parent.children.findIndex(child => child === node);
|
||
if (siblingIndex !== -1) {
|
||
selectorParts.push(`:nth-match(${siblingIndex + 1})`);
|
||
}
|
||
}
|
||
|
||
idToSelector[currentId] = selectorParts.join(' ');
|
||
|
||
// 收集所有可能的属性
|
||
let props = [];
|
||
if (node.focusable) props.push('focusable');
|
||
if (node.focused) props.push('focused');
|
||
if (node.expanded !== undefined) props.push(`expanded=${node.expanded}`);
|
||
if (node.selected) props.push('selected');
|
||
if (node.checked !== undefined) props.push(`checked=${node.checked}`);
|
||
if (node.disabled) props.push('disabled');
|
||
if (node.required) props.push('required');
|
||
if (node.pressed !== undefined) props.push(`pressed=${node.pressed}`);
|
||
|
||
// 判断元素是否可点击
|
||
const clickableRoles = ['button', 'link', 'menuitem', 'tab', 'checkbox', 'radio', 'switch', 'option'];
|
||
const isClickable = clickableRoles.includes(node.role) ||
|
||
node.focusable ||
|
||
node.role === 'generic' && node.name && node.focusable;
|
||
if (isClickable) props.push('clickable');
|
||
|
||
const indent = ' '.repeat(depth * 4);
|
||
console.log(`${indent}[${currentId}] ${node.role} '${node.name || ''}'${props.length > 0 ? ' (' + props.join(', ') + ')' : ''}`);
|
||
|
||
if (node.children && node.children.length > 0) {
|
||
for (const child of node.children) {
|
||
traverse(child, depth + 1, node);
|
||
}
|
||
}
|
||
}
|
||
|
||
// 输出 AXTree 的整体结构
|
||
console.log('## AXTree:');
|
||
// 打印根节点信息(这里用 Root+role 来模拟输出)
|
||
let rootProps = [];
|
||
if (axTree.focusable) rootProps.push('focusable=True');
|
||
if (axTree.focused) rootProps.push('focused');
|
||
console.log(`Root${axTree.role ? axTree.role : 'WebArea'} '${axTree.name || ''}'${rootProps.length > 0 ? ', ' + rootProps.join(', ') : ''}`);
|
||
|
||
if (axTree.children && axTree.children.length > 0) {
|
||
for (const child of axTree.children) {
|
||
traverse(child, 1, axTree);
|
||
}
|
||
}
|
||
|
||
// 输出编号与伪选择器的映射
|
||
console.log('\n编号与 Selector 映射:');
|
||
console.log(idToSelector);
|
||
|
||
// 示例:后续可根据编号获取对应的 selector 进行操作
|
||
// 比如要点击编号为 25 的节点:
|
||
// const selectorForId25 = idToSelector[25];
|
||
// await page.click(selectorForId25);
|
||
//
|
||
// 注意:上面的 selector 为伪生成示例,实际操作中需要根据页面结构生成能唯一定位该元素的 selector。
|
||
|
||
await browser.close();
|
||
})();
|