@sleep2agi/agent-network-dashboard 0.5.1-preview.7 → 0.5.1-preview.71
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/.next/BUILD_ID +1 -1
- package/.next/build-manifest.json +3 -3
- package/.next/diagnostics/route-bundle-stats.json +6 -6
- package/.next/fallback-build-manifest.json +3 -3
- package/.next/server/app/_global-error.html +1 -1
- package/.next/server/app/_global-error.rsc +1 -1
- package/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/.next/server/app/_not-found.html +2 -2
- package/.next/server/app/_not-found.rsc +2 -2
- package/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
- package/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
- package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/admin/page_client-reference-manifest.js +1 -1
- package/.next/server/app/admin.html +2 -2
- package/.next/server/app/admin.rsc +2 -2
- package/.next/server/app/admin.segments/_full.segment.rsc +2 -2
- package/.next/server/app/admin.segments/_head.segment.rsc +1 -1
- package/.next/server/app/admin.segments/_index.segment.rsc +2 -2
- package/.next/server/app/admin.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/admin.segments/admin/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/admin.segments/admin.segment.rsc +1 -1
- package/.next/server/app/index.html +2 -2
- package/.next/server/app/index.rsc +3 -3
- package/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/.next/server/app/index.segments/_full.segment.rsc +3 -3
- package/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/.next/server/app/index.segments/_index.segment.rsc +2 -2
- package/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/login/page_client-reference-manifest.js +1 -1
- package/.next/server/app/login.html +2 -2
- package/.next/server/app/login.rsc +3 -3
- package/.next/server/app/login.segments/_full.segment.rsc +3 -3
- package/.next/server/app/login.segments/_head.segment.rsc +1 -1
- package/.next/server/app/login.segments/_index.segment.rsc +2 -2
- package/.next/server/app/login.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/login.segments/login/__PAGE__.segment.rsc +2 -2
- package/.next/server/app/login.segments/login.segment.rsc +1 -1
- package/.next/server/app/logs/page_client-reference-manifest.js +1 -1
- package/.next/server/app/logs.html +2 -2
- package/.next/server/app/logs.rsc +2 -2
- package/.next/server/app/logs.segments/_full.segment.rsc +2 -2
- package/.next/server/app/logs.segments/_head.segment.rsc +1 -1
- package/.next/server/app/logs.segments/_index.segment.rsc +2 -2
- package/.next/server/app/logs.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/logs.segments/logs/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/logs.segments/logs.segment.rsc +1 -1
- package/.next/server/app/messages/page_client-reference-manifest.js +1 -1
- package/.next/server/app/messages.html +2 -2
- package/.next/server/app/messages.rsc +2 -2
- package/.next/server/app/messages.segments/_full.segment.rsc +2 -2
- package/.next/server/app/messages.segments/_head.segment.rsc +1 -1
- package/.next/server/app/messages.segments/_index.segment.rsc +2 -2
- package/.next/server/app/messages.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/messages.segments/messages/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/messages.segments/messages.segment.rsc +1 -1
- package/.next/server/app/node/page_client-reference-manifest.js +1 -1
- package/.next/server/app/node.html +2 -2
- package/.next/server/app/node.rsc +2 -2
- package/.next/server/app/node.segments/_full.segment.rsc +2 -2
- package/.next/server/app/node.segments/_head.segment.rsc +1 -1
- package/.next/server/app/node.segments/_index.segment.rsc +2 -2
- package/.next/server/app/node.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/node.segments/node/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/node.segments/node.segment.rsc +1 -1
- package/.next/server/app/nodes/page_client-reference-manifest.js +1 -1
- package/.next/server/app/nodes.html +2 -2
- package/.next/server/app/nodes.rsc +2 -2
- package/.next/server/app/nodes.segments/_full.segment.rsc +2 -2
- package/.next/server/app/nodes.segments/_head.segment.rsc +1 -1
- package/.next/server/app/nodes.segments/_index.segment.rsc +2 -2
- package/.next/server/app/nodes.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/nodes.segments/nodes/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/nodes.segments/nodes.segment.rsc +1 -1
- package/.next/server/app/page_client-reference-manifest.js +1 -1
- package/.next/server/app/server-logs/page_client-reference-manifest.js +1 -1
- package/.next/server/app/server-logs.html +2 -2
- package/.next/server/app/server-logs.rsc +2 -2
- package/.next/server/app/server-logs.segments/_full.segment.rsc +2 -2
- package/.next/server/app/server-logs.segments/_head.segment.rsc +1 -1
- package/.next/server/app/server-logs.segments/_index.segment.rsc +2 -2
- package/.next/server/app/server-logs.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/server-logs.segments/server-logs/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/server-logs.segments/server-logs.segment.rsc +1 -1
- package/.next/server/app/settings/networks/page_client-reference-manifest.js +1 -1
- package/.next/server/app/settings/networks.html +2 -2
- package/.next/server/app/settings/networks.rsc +2 -2
- package/.next/server/app/settings/networks.segments/_full.segment.rsc +2 -2
- package/.next/server/app/settings/networks.segments/_head.segment.rsc +1 -1
- package/.next/server/app/settings/networks.segments/_index.segment.rsc +2 -2
- package/.next/server/app/settings/networks.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/settings/networks.segments/settings/networks/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/settings/networks.segments/settings/networks.segment.rsc +1 -1
- package/.next/server/app/settings/networks.segments/settings.segment.rsc +1 -1
- package/.next/server/app/settings/page_client-reference-manifest.js +1 -1
- package/.next/server/app/settings/tokens/page_client-reference-manifest.js +1 -1
- package/.next/server/app/settings/tokens.html +2 -2
- package/.next/server/app/settings/tokens.rsc +2 -2
- package/.next/server/app/settings/tokens.segments/_full.segment.rsc +2 -2
- package/.next/server/app/settings/tokens.segments/_head.segment.rsc +1 -1
- package/.next/server/app/settings/tokens.segments/_index.segment.rsc +2 -2
- package/.next/server/app/settings/tokens.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/settings/tokens.segments/settings/tokens/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/settings/tokens.segments/settings/tokens.segment.rsc +1 -1
- package/.next/server/app/settings/tokens.segments/settings.segment.rsc +1 -1
- package/.next/server/app/settings.html +2 -2
- package/.next/server/app/settings.rsc +3 -3
- package/.next/server/app/settings.segments/_full.segment.rsc +3 -3
- package/.next/server/app/settings.segments/_head.segment.rsc +1 -1
- package/.next/server/app/settings.segments/_index.segment.rsc +2 -2
- package/.next/server/app/settings.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/settings.segments/settings/__PAGE__.segment.rsc +2 -2
- package/.next/server/app/settings.segments/settings.segment.rsc +1 -1
- package/.next/server/app/tasks/[id]/page_client-reference-manifest.js +1 -1
- package/.next/server/app/tasks/page_client-reference-manifest.js +1 -1
- package/.next/server/app/tasks.html +2 -2
- package/.next/server/app/tasks.rsc +2 -2
- package/.next/server/app/tasks.segments/_full.segment.rsc +2 -2
- package/.next/server/app/tasks.segments/_head.segment.rsc +1 -1
- package/.next/server/app/tasks.segments/_index.segment.rsc +2 -2
- package/.next/server/app/tasks.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/tasks.segments/tasks/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/tasks.segments/tasks.segment.rsc +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js +3 -3
- package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js.map +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js.map +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js.map +1 -1
- package/.next/server/middleware-build-manifest.js +3 -3
- package/.next/server/pages/404.html +2 -2
- package/.next/server/pages/500.html +1 -1
- package/.next/static/chunks/04.j~y8e~sbs4.js +1 -0
- package/.next/static/chunks/06llamqb4jsu..js +4 -0
- package/.next/static/chunks/0_p8jkzdw5x2_.css +2 -0
- package/.next/static/chunks/12heglqfrp1bm.js +1 -0
- package/.next/trace +2 -2
- package/.next/trace-build +1 -1
- package/app/components/TopoGraph.tsx +1426 -84
- package/package.json +1 -1
- package/scripts/topo-active-links-chip-hover-lift-test.mjs +93 -0
- package/scripts/topo-chip-digit-fontweight-test.mjs +105 -0
- package/scripts/topo-chip-row-hover-lift-test.mjs +95 -0
- package/scripts/topo-chrome-button-hover-lift-test.mjs +94 -0
- package/scripts/topo-chrome-segmented-radius-test.mjs +100 -0
- package/scripts/topo-click-ripple-opacity-test.mjs +99 -0
- package/scripts/topo-edge-badge-fontsize-test.mjs +90 -0
- package/scripts/topo-edge-badge-hover-opacity-test.mjs +94 -0
- package/scripts/topo-edge-badge-hover-stroke-test.mjs +92 -0
- package/scripts/topo-edge-badge-opacity-test.mjs +80 -0
- package/scripts/topo-edge-badge-pin-opacity-test.mjs +86 -0
- package/scripts/topo-edge-badge-stroke-test.mjs +92 -0
- package/scripts/topo-edge-freshness-floor-test.mjs +99 -0
- package/scripts/topo-edge-visible-linecap-test.mjs +89 -0
- package/scripts/topo-filter-pill-hover-lift-test.mjs +101 -0
- package/scripts/topo-filter-pill-hover-opacity-test.mjs +110 -0
- package/scripts/topo-filter-pill-x-hover-scale-test.mjs +99 -0
- package/scripts/topo-flow-rail-linecap-test.mjs +79 -0
- package/scripts/topo-freshness-chip-hierarchy-test.mjs +93 -0
- package/scripts/topo-freshness-chip-tabular-test.mjs +41 -0
- package/scripts/topo-freshness-floor-lift-test.mjs +92 -0
- package/scripts/topo-freshness-suffix-tabular-test.mjs +88 -0
- package/scripts/topo-fullscreen-icon-hover-scale-test.mjs +91 -0
- package/scripts/topo-group-box-stroke-test.mjs +105 -0
- package/scripts/topo-group-label-count-fontweight-test.mjs +108 -0
- package/scripts/topo-hover-detail-body-fw-test.mjs +101 -0
- package/scripts/topo-hover-detail-model-fw-test.mjs +98 -0
- package/scripts/topo-hover-detail-opacity-test.mjs +98 -0
- package/scripts/topo-hover-detail-rx-test.mjs +81 -0
- package/scripts/topo-hub-digit-fontsize-test.mjs +86 -0
- package/scripts/topo-hub-halo-light-trough-test.mjs +88 -0
- package/scripts/topo-hub-halo-radius-test.mjs +86 -0
- package/scripts/topo-hub-halo-trough-test.mjs +83 -0
- package/scripts/topo-hub-highlight-opacity-test.mjs +88 -0
- package/scripts/topo-hub-highlight-radius-test.mjs +90 -0
- package/scripts/topo-hub-hover-ring-opacity-test.mjs +96 -0
- package/scripts/topo-hub-hover-ring-stroke-test.mjs +86 -0
- package/scripts/topo-hub-spoke-linecap-test.mjs +80 -0
- package/scripts/topo-layout-toggle-hover-tracking-test.mjs +109 -0
- package/scripts/topo-layout-toggle-radius-test.mjs +87 -0
- package/scripts/topo-legend-label-fontweight-test.mjs +94 -0
- package/scripts/topo-legend-pin-ring-stroke-test.mjs +101 -0
- package/scripts/topo-minimap-offline-opacity-test.mjs +90 -0
- package/scripts/topo-minimap-online-opacity-test.mjs +93 -0
- package/scripts/topo-minimap-online-radius-test.mjs +85 -0
- package/scripts/topo-minimap-viewport-linejoin-test.mjs +75 -0
- package/scripts/topo-minimap-viewport-rx-test.mjs +85 -0
- package/scripts/topo-more-flows-fontweight-test.mjs +103 -0
- package/scripts/topo-node-halo-offline-opacity-test.mjs +87 -0
- package/scripts/topo-node-pulse-peak-test.mjs +89 -0
- package/scripts/topo-panel-count-letterspacing-test.mjs +89 -0
- package/scripts/topo-panel-rect-opacity-hover-test.mjs +109 -0
- package/scripts/topo-pressure-bar-height-test.mjs +92 -0
- package/scripts/topo-pressure-kicker-fontweight-test.mjs +76 -0
- package/scripts/topo-recent-pip-radius-2-test.mjs +72 -0
- package/scripts/topo-recent-pip-radius-test.mjs +76 -0
- package/scripts/topo-recent-row-text-fontweight-test.mjs +90 -0
- package/scripts/topo-reset-hover-rotate-test.mjs +102 -0
- package/scripts/topo-spoke-active-opacity-test.mjs +104 -0
- package/scripts/topo-vendor-chip-hover-lift-test.mjs +87 -0
- package/scripts/topo-vendor-glyph-fontweight-test.mjs +102 -0
- package/scripts/topo-vendor-letter-hover-scale-test.mjs +129 -0
- package/scripts/topo-zoom-icon-hover-scale-test.mjs +114 -0
- package/scripts/topo-zoom-level-hover-letterspacing-test.mjs +91 -0
- package/.next/static/chunks/0aauz~36q5n2a.css +0 -2
- package/.next/static/chunks/0bja1amnrg3li.js +0 -1
- package/.next/static/chunks/0k~uc0~~19hyy.js +0 -4
- package/.next/static/chunks/0wtq_6dnzems6.js +0 -1
- /package/.next/static/{x9zCCrMkHsIYlXNY791KF → gaK6yNvVjshUCmKR9qrPn}/_buildManifest.js +0 -0
- /package/.next/static/{x9zCCrMkHsIYlXNY791KF → gaK6yNvVjshUCmKR9qrPn}/_clientMiddlewareManifest.js +0 -0
- /package/.next/static/{x9zCCrMkHsIYlXNY791KF → gaK6yNvVjshUCmKR9qrPn}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
/* Round 364 verification: legend-row label fontWeight 400 → 500.
|
|
2
|
+
* Sibling typography lift to R363 recent-row text fw 400 → 500.
|
|
3
|
+
* Both surfaces render small monospace text against panel chrome
|
|
4
|
+
* at fontSize 9-11 where SVG-default fw 400 sits at the legibility
|
|
5
|
+
* floor.
|
|
6
|
+
*
|
|
7
|
+
* Family snapshot post-R364:
|
|
8
|
+
* legend label fw 500 (R364, this round)
|
|
9
|
+
* legend count fw 600 (R309)
|
|
10
|
+
* recent alias fw 500 (R363)
|
|
11
|
+
* recent count fw 600/700 (R320)
|
|
12
|
+
*
|
|
13
|
+
* Contract (cyber, fixture with active rows):
|
|
14
|
+
* - Each rendered [data-legend-row-label] has font-weight === '500'.
|
|
15
|
+
* - Each has data-legend-row-label-font-weight === '500'.
|
|
16
|
+
* - Pre-R364 invariants:
|
|
17
|
+
* * fontSize=11 + fontFamily=monospace
|
|
18
|
+
* * R219 letter-spacing 0 → 0.5px on pin (style.transition still
|
|
19
|
+
* lists letter-spacing)
|
|
20
|
+
* * R55 fill transition retained
|
|
21
|
+
* * data-legend-row-label key surfaces (working / idle / offline)
|
|
22
|
+
*/
|
|
23
|
+
import { chromium } from 'playwright';
|
|
24
|
+
import { readFileSync } from 'node:fs';
|
|
25
|
+
|
|
26
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
27
|
+
const browser = await chromium.launch({ headless: true });
|
|
28
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
29
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
30
|
+
await ctx.addInitScript(() => {
|
|
31
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
32
|
+
});
|
|
33
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
34
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
35
|
+
const r = await route.fetch();
|
|
36
|
+
const b = await r.json();
|
|
37
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
38
|
+
const mk = (alias) => ({
|
|
39
|
+
alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
40
|
+
network_id: nid, project_dir: null,
|
|
41
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
42
|
+
});
|
|
43
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('a'), mk('b'), mk('c') ] } });
|
|
44
|
+
});
|
|
45
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
46
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
47
|
+
|
|
48
|
+
const page = await ctx.newPage();
|
|
49
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
50
|
+
await page.waitForSelector('[data-legend-row-label]', { timeout: 15000 });
|
|
51
|
+
await page.waitForTimeout(300);
|
|
52
|
+
|
|
53
|
+
const probe = await page.evaluate(() => {
|
|
54
|
+
const labels = Array.from(document.querySelectorAll('[data-legend-row-label]'));
|
|
55
|
+
return labels.map(el => {
|
|
56
|
+
const cs = getComputedStyle(el);
|
|
57
|
+
return {
|
|
58
|
+
key: el.getAttribute('data-legend-row-label'),
|
|
59
|
+
fontWeightAttr: el.getAttribute('font-weight'),
|
|
60
|
+
fontWeightData: el.getAttribute('data-legend-row-label-font-weight'),
|
|
61
|
+
fontSizeAttr: el.getAttribute('font-size'),
|
|
62
|
+
fontFamily: el.getAttribute('font-family'),
|
|
63
|
+
transition: cs.transition,
|
|
64
|
+
};
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
await browser.close();
|
|
69
|
+
|
|
70
|
+
const hasTrans = (s, prop) =>
|
|
71
|
+
new RegExp(`${prop}\\s+\\d*\\.?\\d*s|${prop}\\s+\\d+ms`, 'i').test(s || '');
|
|
72
|
+
|
|
73
|
+
const allRowsOk = probe.length > 0 && probe.every(r =>
|
|
74
|
+
r.fontWeightAttr === '500'
|
|
75
|
+
&& r.fontWeightData === '500'
|
|
76
|
+
&& r.fontSizeAttr === '11'
|
|
77
|
+
&& /monospace/i.test(r.fontFamily || '')
|
|
78
|
+
&& hasTrans(r.transition, 'letter-spacing')
|
|
79
|
+
&& hasTrans(r.transition, 'fill')
|
|
80
|
+
);
|
|
81
|
+
|
|
82
|
+
const results = {
|
|
83
|
+
rows_rendered: probe.length >= 1,
|
|
84
|
+
all_rows_fw_500: probe.every(r => r.fontWeightAttr === '500'),
|
|
85
|
+
all_rows_data_500: probe.every(r => r.fontWeightData === '500'),
|
|
86
|
+
all_rows_fontsize_11: probe.every(r => r.fontSizeAttr === '11'),
|
|
87
|
+
all_rows_monospace: probe.every(r => /monospace/i.test(r.fontFamily || '')),
|
|
88
|
+
all_rows_have_ls_trans: probe.every(r => hasTrans(r.transition, 'letter-spacing')),
|
|
89
|
+
all_rows_have_fill_trans: probe.every(r => hasTrans(r.transition, 'fill')),
|
|
90
|
+
};
|
|
91
|
+
const ok = allRowsOk && Object.values(results).every(Boolean);
|
|
92
|
+
console.log(`${ok ? '✅' : '❌'} legend-row label fw 400 → 500:`, JSON.stringify(results),
|
|
93
|
+
'\n rows: ', probe.map(r => ({ key: r.key, fw: r.fontWeightAttr })));
|
|
94
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
/* Round 402 verification: legend pin-ring strokeWidth 1.5 → 1.75.
|
|
2
|
+
* Sibling visual-weight bump (12th anchor) to R385 hub hover-ring
|
|
3
|
+
* — both are stroke-only pin/hover indicators painted as r=8 / r=14
|
|
4
|
+
* circles outside their target. R51 sentinel concern: 1.5 reserved
|
|
5
|
+
* for offline-node detection, but selector gated to g[data-node]
|
|
6
|
+
* ancestors so this legend-internal circle is invisible to probe.
|
|
7
|
+
*
|
|
8
|
+
* Contract:
|
|
9
|
+
* - data-legend-pin-ring-stroke-width === '1.75' (idle state too,
|
|
10
|
+
* since the attr surfaces the resolved value regardless of pin)
|
|
11
|
+
* - <circle> stroke-width attr === '1.75'
|
|
12
|
+
* - Pre-R402 invariants preserved:
|
|
13
|
+
* * r='8', fill='none', cx='16'
|
|
14
|
+
* * opacity reflects isPinned state (0 idle, 1 pinned)
|
|
15
|
+
* * data-legend-pin-ring-pinned attr present
|
|
16
|
+
* - Idle ring: opacity='0' (R181 always-mount + transition)
|
|
17
|
+
* - Pinned ring (after click): opacity='1'
|
|
18
|
+
*/
|
|
19
|
+
import { chromium } from 'playwright';
|
|
20
|
+
import { readFileSync } from 'node:fs';
|
|
21
|
+
|
|
22
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
23
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
24
|
+
|
|
25
|
+
const browser = await chromium.launch({ headless: true });
|
|
26
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
27
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
28
|
+
await ctx.addInitScript(() => {
|
|
29
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
30
|
+
});
|
|
31
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
32
|
+
const r = await route.fetch();
|
|
33
|
+
const b = await r.json();
|
|
34
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
35
|
+
const mk = (alias, status) => ({
|
|
36
|
+
alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
37
|
+
network_id: nid, project_dir: null,
|
|
38
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
39
|
+
});
|
|
40
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [
|
|
41
|
+
mk('alpha', 'working'),
|
|
42
|
+
mk('beta', 'idle'),
|
|
43
|
+
] } });
|
|
44
|
+
});
|
|
45
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
46
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
47
|
+
|
|
48
|
+
const page = await ctx.newPage();
|
|
49
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
50
|
+
await page.waitForSelector('[data-legend-pin-ring]', { timeout: 15000 });
|
|
51
|
+
await page.waitForTimeout(400);
|
|
52
|
+
|
|
53
|
+
const restProbe = await page.evaluate(() => {
|
|
54
|
+
const ring = document.querySelector('[data-legend-pin-ring]');
|
|
55
|
+
return ring ? {
|
|
56
|
+
strokeWidthAttr: ring.getAttribute('stroke-width'),
|
|
57
|
+
strokeWidthData: ring.getAttribute('data-legend-pin-ring-stroke-width'),
|
|
58
|
+
rAttr: ring.getAttribute('r'),
|
|
59
|
+
fillAttr: ring.getAttribute('fill'),
|
|
60
|
+
cxAttr: ring.getAttribute('cx'),
|
|
61
|
+
opacityAttr: ring.getAttribute('opacity'),
|
|
62
|
+
pinned: ring.getAttribute('data-legend-pin-ring-pinned'),
|
|
63
|
+
} : null;
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
// Click a legend row to pin it
|
|
67
|
+
await page.click('[data-legend-status="working"]');
|
|
68
|
+
await page.waitForTimeout(300);
|
|
69
|
+
|
|
70
|
+
const pinnedProbe = await page.evaluate(() => {
|
|
71
|
+
const rings = Array.from(document.querySelectorAll('[data-legend-pin-ring]'));
|
|
72
|
+
const pinned = rings.find((r) => r.getAttribute('data-legend-pin-ring-pinned') === 'true');
|
|
73
|
+
return pinned ? {
|
|
74
|
+
strokeWidthAttr: pinned.getAttribute('stroke-width'),
|
|
75
|
+
opacityAttr: pinned.getAttribute('opacity'),
|
|
76
|
+
rAttr: pinned.getAttribute('r'),
|
|
77
|
+
} : null;
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
await browser.close();
|
|
81
|
+
|
|
82
|
+
const results = {
|
|
83
|
+
// R402 wire: stroke-width attr + data both 1.75
|
|
84
|
+
rest_strokeWidth_1_75: restProbe?.strokeWidthAttr === '1.75',
|
|
85
|
+
rest_data_1_75: restProbe?.strokeWidthData === '1.75',
|
|
86
|
+
// Pre-R402 invariants (R181 + base geometry)
|
|
87
|
+
rest_r_8: restProbe?.rAttr === '8',
|
|
88
|
+
rest_fill_none: restProbe?.fillAttr === 'none',
|
|
89
|
+
rest_cx_16: restProbe?.cxAttr === '16',
|
|
90
|
+
rest_opacity_0: restProbe?.opacityAttr === '0', // R181 idle invariant
|
|
91
|
+
rest_pinned_false: restProbe?.pinned === 'false',
|
|
92
|
+
// Pinned state preserves the bumped stroke
|
|
93
|
+
pinned_strokeWidth_1_75: pinnedProbe?.strokeWidthAttr === '1.75',
|
|
94
|
+
pinned_opacity_1: pinnedProbe?.opacityAttr === '1', // R181 pinned alpha
|
|
95
|
+
pinned_r_8: pinnedProbe?.rAttr === '8', // geometry preserved
|
|
96
|
+
};
|
|
97
|
+
const ok = Object.values(results).every(Boolean);
|
|
98
|
+
console.log(`${ok ? '✅' : '❌'} legend pin-ring strokeWidth 1.5 → 1.75:`, JSON.stringify(results),
|
|
99
|
+
'\n rest: ', restProbe,
|
|
100
|
+
'\n pinned: ', pinnedProbe);
|
|
101
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
/* Round 372 verification: minimap offline-dot opacity 0.5 → 0.6.
|
|
2
|
+
* Sibling stale-state legibility lift to R358 (freshness ramp floor
|
|
3
|
+
* 0.25 → 0.30). Pre-R372 R198 drew offline dots at α=0.5; R372 lifts
|
|
4
|
+
* to 0.6 (+20 % relative) for better readability while keeping a
|
|
5
|
+
* clear two-tier distinction vs online α=0.9.
|
|
6
|
+
*
|
|
7
|
+
* The minimap only renders when view is non-default (zoom !== 1 ||
|
|
8
|
+
* pan). Test triggers zoom-in to mount it, then probes the offline
|
|
9
|
+
* dot's opacity attr.
|
|
10
|
+
*
|
|
11
|
+
* Contract:
|
|
12
|
+
* - data-topo-minimap-dot-online="false" element opacity === '0.6'.
|
|
13
|
+
* - data-topo-minimap-dot-opacity === '0.6' for offline dot.
|
|
14
|
+
* - Online dot still at opacity '0.9' (R198 invariant).
|
|
15
|
+
* - Pre-R372 invariants:
|
|
16
|
+
* * R198 offline r=1.2, online r=1.7 preserved
|
|
17
|
+
* * R198 transition list (opacity + fill + r) preserved
|
|
18
|
+
*/
|
|
19
|
+
import { chromium } from 'playwright';
|
|
20
|
+
import { readFileSync } from 'node:fs';
|
|
21
|
+
|
|
22
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
23
|
+
const browser = await chromium.launch({ headless: true });
|
|
24
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
25
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
26
|
+
await ctx.addInitScript(() => {
|
|
27
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
28
|
+
});
|
|
29
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
30
|
+
const stale = new Date(Date.now() - 600 * 1000).toISOString();
|
|
31
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
32
|
+
const r = await route.fetch();
|
|
33
|
+
const b = await r.json();
|
|
34
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
35
|
+
// Mix of online (working) + offline sessions.
|
|
36
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [
|
|
37
|
+
{ alias: 'on-a', status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
38
|
+
network_id: nid, project_dir: null, created_at: fresh, updated_at: fresh, last_seen_at: fresh },
|
|
39
|
+
{ alias: 'on-b', status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
40
|
+
network_id: nid, project_dir: null, created_at: fresh, updated_at: fresh, last_seen_at: fresh },
|
|
41
|
+
{ alias: 'off-c', status: 'offline', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
42
|
+
network_id: nid, project_dir: null, created_at: stale, updated_at: stale, last_seen_at: stale },
|
|
43
|
+
] } });
|
|
44
|
+
});
|
|
45
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
46
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
47
|
+
|
|
48
|
+
const page = await ctx.newPage();
|
|
49
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
50
|
+
// Minimap only mounts when view is non-default — zoom in twice.
|
|
51
|
+
await page.waitForSelector('[data-topo-chrome-zoom-in]', { timeout: 15000 });
|
|
52
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
53
|
+
await page.waitForTimeout(200);
|
|
54
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
55
|
+
await page.waitForSelector('[data-topo-minimap-dot]', { timeout: 5000 });
|
|
56
|
+
await page.waitForTimeout(400);
|
|
57
|
+
|
|
58
|
+
const probe = await page.evaluate(() => {
|
|
59
|
+
const offline = document.querySelector('[data-topo-minimap-dot-online="false"]');
|
|
60
|
+
const online = document.querySelector('[data-topo-minimap-dot-online="true"]');
|
|
61
|
+
const cs = offline ? getComputedStyle(offline) : null;
|
|
62
|
+
return {
|
|
63
|
+
offlineOpacity: offline?.getAttribute('opacity') ?? null,
|
|
64
|
+
offlineData: offline?.getAttribute('data-topo-minimap-dot-opacity') ?? null,
|
|
65
|
+
offlineR: offline?.getAttribute('r') ?? null,
|
|
66
|
+
offlineTrans: cs?.transition ?? null,
|
|
67
|
+
onlineOpacity: online?.getAttribute('opacity') ?? null,
|
|
68
|
+
onlineR: online?.getAttribute('r') ?? null,
|
|
69
|
+
};
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
await browser.close();
|
|
73
|
+
|
|
74
|
+
const hasTrans = (s, prop) =>
|
|
75
|
+
new RegExp(`${prop}\\s+\\d*\\.?\\d*s|${prop}\\s+\\d+ms`, 'i').test(s || '');
|
|
76
|
+
|
|
77
|
+
const results = {
|
|
78
|
+
offline_opacity_0_6: probe.offlineOpacity === '0.6',
|
|
79
|
+
offline_data_0_6: probe.offlineData === '0.6',
|
|
80
|
+
offline_r_1_2: probe.offlineR === '1.2', // R198 invariant
|
|
81
|
+
online_opacity_0_9: probe.onlineOpacity === '0.9', // R198 invariant
|
|
82
|
+
online_r_1_7: probe.onlineR === '1.7', // R198 invariant
|
|
83
|
+
trans_has_opacity: hasTrans(probe.offlineTrans, 'opacity'), // R198
|
|
84
|
+
trans_has_fill: hasTrans(probe.offlineTrans, 'fill'), // R198
|
|
85
|
+
trans_has_r: hasTrans(probe.offlineTrans, 'r'), // R198
|
|
86
|
+
};
|
|
87
|
+
const ok = Object.values(results).every(Boolean);
|
|
88
|
+
console.log(`${ok ? '✅' : '❌'} minimap offline opacity 0.5 → 0.6:`, JSON.stringify(results),
|
|
89
|
+
'\n probe:', probe);
|
|
90
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
/* Round 392 verification: minimap online dot opacity 0.9 → 0.95.
|
|
2
|
+
* Theme-consistency / canvas-presence polish family (7th anchor)
|
|
3
|
+
* after R370 hub-ring, R371 edge-badge, R372 minimap offline,
|
|
4
|
+
* R386 hub-highlight idle, R387 hover-detail panel, R391 hub-spoke
|
|
5
|
+
* active. Halves the online dot's idle alpha gap (0.10 → 0.05).
|
|
6
|
+
* Mirrors R386's hub-highlight idle 0.9 → 0.95 on the minimap.
|
|
7
|
+
*
|
|
8
|
+
* Contract:
|
|
9
|
+
* - Minimap only mounts when view is non-default — test clicks
|
|
10
|
+
* zoom-in twice to mount it before probing.
|
|
11
|
+
* - Online dots: opacity attr === '0.95', data-opacity === '0.95',
|
|
12
|
+
* R384 r=1.9 invariant, data-online='true'
|
|
13
|
+
* - Offline dot (one ghost session): opacity attr === '0.6'
|
|
14
|
+
* (R372 invariant), r=1.2 invariant, data-online='false'
|
|
15
|
+
*/
|
|
16
|
+
import { chromium } from 'playwright';
|
|
17
|
+
import { readFileSync } from 'node:fs';
|
|
18
|
+
|
|
19
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
20
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
21
|
+
const stale = new Date(Date.now() - 30 * 60 * 1000).toISOString(); // 30 min ago → offline
|
|
22
|
+
|
|
23
|
+
const browser = await chromium.launch({ headless: true });
|
|
24
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
25
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
26
|
+
await ctx.addInitScript(() => {
|
|
27
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
28
|
+
});
|
|
29
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
30
|
+
const r = await route.fetch();
|
|
31
|
+
const b = await r.json();
|
|
32
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
33
|
+
await route.fulfill({ response: r, json: {
|
|
34
|
+
...b,
|
|
35
|
+
sessions: [
|
|
36
|
+
{ alias: 'alpha', status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli', network_id: nid, project_dir: null, created_at: fresh, updated_at: fresh, last_seen_at: fresh },
|
|
37
|
+
{ alias: 'beta', status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli', network_id: nid, project_dir: null, created_at: fresh, updated_at: fresh, last_seen_at: fresh },
|
|
38
|
+
{ alias: 'ghost', status: 'offline', model: 'claude-opus-4', runtime: 'claude-code-cli', network_id: nid, project_dir: null, created_at: stale, updated_at: stale, last_seen_at: stale },
|
|
39
|
+
],
|
|
40
|
+
} });
|
|
41
|
+
});
|
|
42
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
43
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
44
|
+
|
|
45
|
+
const page = await ctx.newPage();
|
|
46
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
47
|
+
await page.waitForSelector('g[data-node]', { timeout: 15000 });
|
|
48
|
+
await page.waitForTimeout(300);
|
|
49
|
+
|
|
50
|
+
// Minimap mounts only on non-default view — click zoom-in twice
|
|
51
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
52
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
53
|
+
await page.waitForSelector('[data-topo-minimap-dot]', { timeout: 5000 });
|
|
54
|
+
await page.waitForTimeout(400);
|
|
55
|
+
|
|
56
|
+
const probe = await page.evaluate(() => {
|
|
57
|
+
const dots = Array.from(document.querySelectorAll('[data-topo-minimap-dot]'));
|
|
58
|
+
const online = dots.filter((d) => d.getAttribute('data-topo-minimap-dot-online') === 'true');
|
|
59
|
+
const offline = dots.filter((d) => d.getAttribute('data-topo-minimap-dot-online') === 'false');
|
|
60
|
+
const read = (el) => el ? ({
|
|
61
|
+
opacityAttr: el.getAttribute('opacity'),
|
|
62
|
+
opacityData: el.getAttribute('data-topo-minimap-dot-opacity'),
|
|
63
|
+
radiusAttr: el.getAttribute('r'),
|
|
64
|
+
}) : null;
|
|
65
|
+
return {
|
|
66
|
+
total: dots.length,
|
|
67
|
+
onlineCount: online.length,
|
|
68
|
+
offlineCount: offline.length,
|
|
69
|
+
onlineSample: read(online[0]),
|
|
70
|
+
offlineSample: read(offline[0]),
|
|
71
|
+
};
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
await browser.close();
|
|
75
|
+
|
|
76
|
+
const results = {
|
|
77
|
+
// Setup: 2 online + 1 offline
|
|
78
|
+
total_3_dots: probe.total === 3,
|
|
79
|
+
online_2_dots: probe.onlineCount === 2,
|
|
80
|
+
offline_1_dot: probe.offlineCount === 1,
|
|
81
|
+
// R392: online opacity 0.9 → 0.95
|
|
82
|
+
online_opacity_0_95: probe.onlineSample?.opacityAttr === '0.95',
|
|
83
|
+
online_data_0_95: probe.onlineSample?.opacityData === '0.95',
|
|
84
|
+
online_r_1_9: probe.onlineSample?.radiusAttr === '1.9', // R384 invariant
|
|
85
|
+
// R372 invariant: offline opacity 0.6
|
|
86
|
+
offline_opacity_0_6: probe.offlineSample?.opacityAttr === '0.6',
|
|
87
|
+
offline_data_0_6: probe.offlineSample?.opacityData === '0.6',
|
|
88
|
+
offline_r_1_2: probe.offlineSample?.radiusAttr === '1.2',
|
|
89
|
+
};
|
|
90
|
+
const ok = Object.values(results).every(Boolean);
|
|
91
|
+
console.log(`${ok ? '✅' : '❌'} minimap online dot opacity 0.9 → 0.95:`, JSON.stringify(results),
|
|
92
|
+
'\n probe:', JSON.stringify(probe, null, 2));
|
|
93
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
/* Round 384 verification: minimap online dot radius 1.7 → 1.9.
|
|
2
|
+
* Sibling visual-weight bump (10th anchor) to R383 recent-row pip
|
|
3
|
+
* 1.8 → 2.0. Widens online vs offline tier contrast (1.42× → 1.58×).
|
|
4
|
+
* R372 already lifted offline opacity 0.5 → 0.6; R384 lifts online
|
|
5
|
+
* radius — pair completes minimap-dot legibility polish across both
|
|
6
|
+
* states.
|
|
7
|
+
*
|
|
8
|
+
* The minimap only renders when view is non-default (zoom !== 1 ||
|
|
9
|
+
* pan), so test triggers zoom-in twice to mount it.
|
|
10
|
+
*
|
|
11
|
+
* Contract:
|
|
12
|
+
* - Online dot computed r === '1.9'.
|
|
13
|
+
* - Online dot data-topo-minimap-dot-radius === '1.9'.
|
|
14
|
+
* - Offline dot r === '1.2' (R198 invariant, untouched).
|
|
15
|
+
* - Offline dot data-radius === '1.2'.
|
|
16
|
+
* - Pre-R384 invariants:
|
|
17
|
+
* * R198 opacity (online 0.9 / offline 0.6 from R372)
|
|
18
|
+
* * R198 transition list (opacity + fill + r 200ms) preserved
|
|
19
|
+
*/
|
|
20
|
+
import { chromium } from 'playwright';
|
|
21
|
+
import { readFileSync } from 'node:fs';
|
|
22
|
+
|
|
23
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
24
|
+
const browser = await chromium.launch({ headless: true });
|
|
25
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
26
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
27
|
+
await ctx.addInitScript(() => {
|
|
28
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
29
|
+
});
|
|
30
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
31
|
+
const stale = new Date(Date.now() - 600 * 1000).toISOString();
|
|
32
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
33
|
+
const r = await route.fetch();
|
|
34
|
+
const b = await r.json();
|
|
35
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
36
|
+
// Mix of online (working) + offline.
|
|
37
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [
|
|
38
|
+
{ alias: 'on-a', status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
39
|
+
network_id: nid, project_dir: null, created_at: fresh, updated_at: fresh, last_seen_at: fresh },
|
|
40
|
+
{ alias: 'on-b', status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
41
|
+
network_id: nid, project_dir: null, created_at: fresh, updated_at: fresh, last_seen_at: fresh },
|
|
42
|
+
{ alias: 'off-c', status: 'offline', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
43
|
+
network_id: nid, project_dir: null, created_at: stale, updated_at: stale, last_seen_at: stale },
|
|
44
|
+
] } });
|
|
45
|
+
});
|
|
46
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
47
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
48
|
+
|
|
49
|
+
const page = await ctx.newPage();
|
|
50
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
51
|
+
// Trigger zoom-in to mount minimap (R30 gate).
|
|
52
|
+
await page.waitForSelector('[data-topo-chrome-zoom-in]', { timeout: 15000 });
|
|
53
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
54
|
+
await page.waitForTimeout(200);
|
|
55
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
56
|
+
await page.waitForSelector('[data-topo-minimap-dot]', { timeout: 5000 });
|
|
57
|
+
await page.waitForTimeout(400);
|
|
58
|
+
|
|
59
|
+
const probe = await page.evaluate(() => {
|
|
60
|
+
const online = document.querySelector('[data-topo-minimap-dot-online="true"]');
|
|
61
|
+
const offline = document.querySelector('[data-topo-minimap-dot-online="false"]');
|
|
62
|
+
return {
|
|
63
|
+
onlineR: online?.getAttribute('r') ?? null,
|
|
64
|
+
onlineRData: online?.getAttribute('data-topo-minimap-dot-radius') ?? null,
|
|
65
|
+
onlineOpacity: online?.getAttribute('opacity') ?? null,
|
|
66
|
+
offlineR: offline?.getAttribute('r') ?? null,
|
|
67
|
+
offlineRData: offline?.getAttribute('data-topo-minimap-dot-radius') ?? null,
|
|
68
|
+
offlineOpacity: offline?.getAttribute('opacity') ?? null,
|
|
69
|
+
};
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
await browser.close();
|
|
73
|
+
|
|
74
|
+
const results = {
|
|
75
|
+
online_r_1_9: probe.onlineR === '1.9',
|
|
76
|
+
online_r_data_1_9: probe.onlineRData === '1.9',
|
|
77
|
+
online_opacity_0_9: probe.onlineOpacity === '0.9', // R198 invariant
|
|
78
|
+
offline_r_1_2: probe.offlineR === '1.2', // R198 invariant
|
|
79
|
+
offline_r_data_1_2: probe.offlineRData === '1.2',
|
|
80
|
+
offline_opacity_0_6: probe.offlineOpacity === '0.6', // R372 lift
|
|
81
|
+
};
|
|
82
|
+
const ok = Object.values(results).every(Boolean);
|
|
83
|
+
console.log(`${ok ? '✅' : '❌'} minimap online dot radius 1.7 → 1.9:`, JSON.stringify(results),
|
|
84
|
+
'\n probe:', probe);
|
|
85
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
/* Round 379 verification: minimap viewport rect picks up
|
|
2
|
+
* strokeLinejoin='round'. Sibling SVG stroke-softening polish to
|
|
3
|
+
* R378 flow-rail linecap + R288 chrome icon linecap family.
|
|
4
|
+
*
|
|
5
|
+
* The minimap only mounts when view is non-default (zoom !== 1 ||
|
|
6
|
+
* pan), so test triggers zoom-in to mount it.
|
|
7
|
+
*
|
|
8
|
+
* Contract:
|
|
9
|
+
* - [data-topo-minimap-viewport] element stroke-linejoin attr === 'round'.
|
|
10
|
+
* - data-topo-minimap-viewport-linejoin === 'round'.
|
|
11
|
+
* - Pre-R379 invariants:
|
|
12
|
+
* * R287 strokeWidth=1.5 (cold rest)
|
|
13
|
+
* * R287 stroke=pal.legendAccent
|
|
14
|
+
* * R346 opacity=0.9 (cold rest)
|
|
15
|
+
* * R199 smoothView + R346 transition list still tween correctly
|
|
16
|
+
*/
|
|
17
|
+
import { chromium } from 'playwright';
|
|
18
|
+
import { readFileSync } from 'node:fs';
|
|
19
|
+
|
|
20
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
21
|
+
const browser = await chromium.launch({ headless: true });
|
|
22
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
23
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
24
|
+
await ctx.addInitScript(() => {
|
|
25
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
26
|
+
});
|
|
27
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
28
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
29
|
+
const r = await route.fetch();
|
|
30
|
+
const b = await r.json();
|
|
31
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
32
|
+
const mk = (alias) => ({
|
|
33
|
+
alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
34
|
+
network_id: nid, project_dir: null,
|
|
35
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
36
|
+
});
|
|
37
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('a'), mk('b'), mk('c') ] } });
|
|
38
|
+
});
|
|
39
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
40
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
41
|
+
|
|
42
|
+
const page = await ctx.newPage();
|
|
43
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
44
|
+
// Trigger zoom-in to mount minimap (R30 gate).
|
|
45
|
+
await page.waitForSelector('[data-topo-chrome-zoom-in]', { timeout: 15000 });
|
|
46
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
47
|
+
await page.waitForTimeout(200);
|
|
48
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
49
|
+
await page.waitForSelector('[data-topo-minimap-viewport]', { timeout: 5000 });
|
|
50
|
+
await page.waitForTimeout(400);
|
|
51
|
+
|
|
52
|
+
const probe = await page.evaluate(() => {
|
|
53
|
+
const r = document.querySelector('[data-topo-minimap-viewport]');
|
|
54
|
+
return {
|
|
55
|
+
linejoinAttr: r?.getAttribute('stroke-linejoin') ?? null,
|
|
56
|
+
linejoinData: r?.getAttribute('data-topo-minimap-viewport-linejoin') ?? null,
|
|
57
|
+
strokeWidthAttr: r?.getAttribute('stroke-width') ?? null,
|
|
58
|
+
opacityAttr: r?.getAttribute('opacity') ?? null,
|
|
59
|
+
fillAttr: r?.getAttribute('fill') ?? null,
|
|
60
|
+
};
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
await browser.close();
|
|
64
|
+
|
|
65
|
+
const results = {
|
|
66
|
+
linejoin_round: probe.linejoinAttr === 'round',
|
|
67
|
+
data_linejoin_round: probe.linejoinData === 'round',
|
|
68
|
+
stroke_width_1_5_rest: probe.strokeWidthAttr === '1.5', // R287 cold rest
|
|
69
|
+
opacity_0_9_rest: probe.opacityAttr === '0.9', // R346 cold rest
|
|
70
|
+
fill_none: probe.fillAttr === 'none', // viewport rect is stroke-only
|
|
71
|
+
};
|
|
72
|
+
const ok = Object.values(results).every(Boolean);
|
|
73
|
+
console.log(`${ok ? '✅' : '❌'} minimap viewport linejoin='round':`, JSON.stringify(results),
|
|
74
|
+
'\n probe:', probe);
|
|
75
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
/* Round 393 verification: minimap viewport rect rx 0 → 2.
|
|
2
|
+
* Corner-radius cascade family adds a sub-element tier (7th
|
|
3
|
+
* anchor) by softening the cyan-stroked viewport frame inside
|
|
4
|
+
* the R332 rounded minimap container. Small but visible polish
|
|
5
|
+
* — 2-px radius reads as "not sharp" without feeling pillowy on
|
|
6
|
+
* the 30-50px wide viewport rect.
|
|
7
|
+
*
|
|
8
|
+
* Contract:
|
|
9
|
+
* - Minimap only mounts on non-default view → zoom-in twice
|
|
10
|
+
* - viewport rect rx attr === '2'
|
|
11
|
+
* - data-topo-minimap-viewport-rx === '2'
|
|
12
|
+
* - Pre-R393 invariants preserved:
|
|
13
|
+
* * R346 hover-state strokeWidth tween (rest '1.5' / hover '1.75')
|
|
14
|
+
* * R346 opacity tween (rest '0.9' / hover '1')
|
|
15
|
+
* * R379 strokeLinejoin='round'
|
|
16
|
+
* * fill='none' + stroke=pal.legendAccent
|
|
17
|
+
*/
|
|
18
|
+
import { chromium } from 'playwright';
|
|
19
|
+
import { readFileSync } from 'node:fs';
|
|
20
|
+
|
|
21
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
22
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
23
|
+
|
|
24
|
+
const browser = await chromium.launch({ headless: true });
|
|
25
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
26
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
27
|
+
await ctx.addInitScript(() => {
|
|
28
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
29
|
+
});
|
|
30
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
31
|
+
const r = await route.fetch();
|
|
32
|
+
const b = await r.json();
|
|
33
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
34
|
+
const mk = (alias) => ({
|
|
35
|
+
alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
36
|
+
network_id: nid, project_dir: null,
|
|
37
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
38
|
+
});
|
|
39
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('alpha'), mk('beta'), mk('gamma') ] } });
|
|
40
|
+
});
|
|
41
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
42
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
43
|
+
|
|
44
|
+
const page = await ctx.newPage();
|
|
45
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
46
|
+
await page.waitForSelector('g[data-node]', { timeout: 15000 });
|
|
47
|
+
await page.waitForTimeout(300);
|
|
48
|
+
|
|
49
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
50
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
51
|
+
await page.waitForSelector('[data-topo-minimap-viewport]', { timeout: 5000 });
|
|
52
|
+
await page.waitForTimeout(400);
|
|
53
|
+
|
|
54
|
+
const probe = await page.evaluate(() => {
|
|
55
|
+
const v = document.querySelector('[data-topo-minimap-viewport]');
|
|
56
|
+
if (!v) return null;
|
|
57
|
+
return {
|
|
58
|
+
rxAttr: v.getAttribute('rx'),
|
|
59
|
+
rxData: v.getAttribute('data-topo-minimap-viewport-rx'),
|
|
60
|
+
strokeWidth: v.getAttribute('stroke-width'),
|
|
61
|
+
opacity: v.getAttribute('opacity'),
|
|
62
|
+
linejoin: v.getAttribute('stroke-linejoin'),
|
|
63
|
+
fill: v.getAttribute('fill'),
|
|
64
|
+
strokePresent: !!v.getAttribute('stroke') && v.getAttribute('stroke') !== 'none',
|
|
65
|
+
};
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
await browser.close();
|
|
69
|
+
|
|
70
|
+
const results = {
|
|
71
|
+
// R393: rx 0 → 2
|
|
72
|
+
rx_attr_2: probe?.rxAttr === '2',
|
|
73
|
+
rx_data_2: probe?.rxData === '2',
|
|
74
|
+
// R346 hover-state rest invariants (no hover yet → rest values)
|
|
75
|
+
rest_strokeWidth_15: probe?.strokeWidth === '1.5',
|
|
76
|
+
rest_opacity_09: probe?.opacity === '0.9',
|
|
77
|
+
// R379 + base invariants
|
|
78
|
+
linejoin_round: probe?.linejoin === 'round',
|
|
79
|
+
fill_none: probe?.fill === 'none',
|
|
80
|
+
stroke_present: probe?.strokePresent === true,
|
|
81
|
+
};
|
|
82
|
+
const ok = Object.values(results).every(Boolean);
|
|
83
|
+
console.log(`${ok ? '✅' : '❌'} minimap viewport rect rx 0 → 2:`, JSON.stringify(results),
|
|
84
|
+
'\n probe:', JSON.stringify(probe, null, 2));
|
|
85
|
+
process.exit(ok ? 0 : 1);
|