@sleep2agi/agent-network-dashboard 0.5.1-preview.4 → 0.5.1-preview.40

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.
Files changed (190) hide show
  1. package/.next/BUILD_ID +1 -1
  2. package/.next/build-manifest.json +3 -3
  3. package/.next/diagnostics/route-bundle-stats.json +6 -6
  4. package/.next/fallback-build-manifest.json +3 -3
  5. package/.next/server/app/_global-error.html +1 -1
  6. package/.next/server/app/_global-error.rsc +1 -1
  7. package/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
  8. package/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  9. package/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  10. package/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  11. package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  12. package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  13. package/.next/server/app/_not-found.html +2 -2
  14. package/.next/server/app/_not-found.rsc +2 -2
  15. package/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
  16. package/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  17. package/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
  18. package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  19. package/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  20. package/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
  21. package/.next/server/app/admin/page_client-reference-manifest.js +1 -1
  22. package/.next/server/app/admin.html +2 -2
  23. package/.next/server/app/admin.rsc +2 -2
  24. package/.next/server/app/admin.segments/_full.segment.rsc +2 -2
  25. package/.next/server/app/admin.segments/_head.segment.rsc +1 -1
  26. package/.next/server/app/admin.segments/_index.segment.rsc +2 -2
  27. package/.next/server/app/admin.segments/_tree.segment.rsc +2 -2
  28. package/.next/server/app/admin.segments/admin/__PAGE__.segment.rsc +1 -1
  29. package/.next/server/app/admin.segments/admin.segment.rsc +1 -1
  30. package/.next/server/app/index.html +2 -2
  31. package/.next/server/app/index.rsc +3 -3
  32. package/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  33. package/.next/server/app/index.segments/_full.segment.rsc +3 -3
  34. package/.next/server/app/index.segments/_head.segment.rsc +1 -1
  35. package/.next/server/app/index.segments/_index.segment.rsc +2 -2
  36. package/.next/server/app/index.segments/_tree.segment.rsc +2 -2
  37. package/.next/server/app/login/page_client-reference-manifest.js +1 -1
  38. package/.next/server/app/login.html +2 -2
  39. package/.next/server/app/login.rsc +3 -3
  40. package/.next/server/app/login.segments/_full.segment.rsc +3 -3
  41. package/.next/server/app/login.segments/_head.segment.rsc +1 -1
  42. package/.next/server/app/login.segments/_index.segment.rsc +2 -2
  43. package/.next/server/app/login.segments/_tree.segment.rsc +2 -2
  44. package/.next/server/app/login.segments/login/__PAGE__.segment.rsc +2 -2
  45. package/.next/server/app/login.segments/login.segment.rsc +1 -1
  46. package/.next/server/app/logs/page_client-reference-manifest.js +1 -1
  47. package/.next/server/app/logs.html +2 -2
  48. package/.next/server/app/logs.rsc +2 -2
  49. package/.next/server/app/logs.segments/_full.segment.rsc +2 -2
  50. package/.next/server/app/logs.segments/_head.segment.rsc +1 -1
  51. package/.next/server/app/logs.segments/_index.segment.rsc +2 -2
  52. package/.next/server/app/logs.segments/_tree.segment.rsc +2 -2
  53. package/.next/server/app/logs.segments/logs/__PAGE__.segment.rsc +1 -1
  54. package/.next/server/app/logs.segments/logs.segment.rsc +1 -1
  55. package/.next/server/app/messages/page_client-reference-manifest.js +1 -1
  56. package/.next/server/app/messages.html +2 -2
  57. package/.next/server/app/messages.rsc +2 -2
  58. package/.next/server/app/messages.segments/_full.segment.rsc +2 -2
  59. package/.next/server/app/messages.segments/_head.segment.rsc +1 -1
  60. package/.next/server/app/messages.segments/_index.segment.rsc +2 -2
  61. package/.next/server/app/messages.segments/_tree.segment.rsc +2 -2
  62. package/.next/server/app/messages.segments/messages/__PAGE__.segment.rsc +1 -1
  63. package/.next/server/app/messages.segments/messages.segment.rsc +1 -1
  64. package/.next/server/app/node/page_client-reference-manifest.js +1 -1
  65. package/.next/server/app/node.html +2 -2
  66. package/.next/server/app/node.rsc +2 -2
  67. package/.next/server/app/node.segments/_full.segment.rsc +2 -2
  68. package/.next/server/app/node.segments/_head.segment.rsc +1 -1
  69. package/.next/server/app/node.segments/_index.segment.rsc +2 -2
  70. package/.next/server/app/node.segments/_tree.segment.rsc +2 -2
  71. package/.next/server/app/node.segments/node/__PAGE__.segment.rsc +1 -1
  72. package/.next/server/app/node.segments/node.segment.rsc +1 -1
  73. package/.next/server/app/nodes/page_client-reference-manifest.js +1 -1
  74. package/.next/server/app/nodes.html +2 -2
  75. package/.next/server/app/nodes.rsc +2 -2
  76. package/.next/server/app/nodes.segments/_full.segment.rsc +2 -2
  77. package/.next/server/app/nodes.segments/_head.segment.rsc +1 -1
  78. package/.next/server/app/nodes.segments/_index.segment.rsc +2 -2
  79. package/.next/server/app/nodes.segments/_tree.segment.rsc +2 -2
  80. package/.next/server/app/nodes.segments/nodes/__PAGE__.segment.rsc +1 -1
  81. package/.next/server/app/nodes.segments/nodes.segment.rsc +1 -1
  82. package/.next/server/app/page_client-reference-manifest.js +1 -1
  83. package/.next/server/app/server-logs/page_client-reference-manifest.js +1 -1
  84. package/.next/server/app/server-logs.html +2 -2
  85. package/.next/server/app/server-logs.rsc +2 -2
  86. package/.next/server/app/server-logs.segments/_full.segment.rsc +2 -2
  87. package/.next/server/app/server-logs.segments/_head.segment.rsc +1 -1
  88. package/.next/server/app/server-logs.segments/_index.segment.rsc +2 -2
  89. package/.next/server/app/server-logs.segments/_tree.segment.rsc +2 -2
  90. package/.next/server/app/server-logs.segments/server-logs/__PAGE__.segment.rsc +1 -1
  91. package/.next/server/app/server-logs.segments/server-logs.segment.rsc +1 -1
  92. package/.next/server/app/settings/networks/page_client-reference-manifest.js +1 -1
  93. package/.next/server/app/settings/networks.html +2 -2
  94. package/.next/server/app/settings/networks.rsc +2 -2
  95. package/.next/server/app/settings/networks.segments/_full.segment.rsc +2 -2
  96. package/.next/server/app/settings/networks.segments/_head.segment.rsc +1 -1
  97. package/.next/server/app/settings/networks.segments/_index.segment.rsc +2 -2
  98. package/.next/server/app/settings/networks.segments/_tree.segment.rsc +2 -2
  99. package/.next/server/app/settings/networks.segments/settings/networks/__PAGE__.segment.rsc +1 -1
  100. package/.next/server/app/settings/networks.segments/settings/networks.segment.rsc +1 -1
  101. package/.next/server/app/settings/networks.segments/settings.segment.rsc +1 -1
  102. package/.next/server/app/settings/page_client-reference-manifest.js +1 -1
  103. package/.next/server/app/settings/tokens/page_client-reference-manifest.js +1 -1
  104. package/.next/server/app/settings/tokens.html +2 -2
  105. package/.next/server/app/settings/tokens.rsc +2 -2
  106. package/.next/server/app/settings/tokens.segments/_full.segment.rsc +2 -2
  107. package/.next/server/app/settings/tokens.segments/_head.segment.rsc +1 -1
  108. package/.next/server/app/settings/tokens.segments/_index.segment.rsc +2 -2
  109. package/.next/server/app/settings/tokens.segments/_tree.segment.rsc +2 -2
  110. package/.next/server/app/settings/tokens.segments/settings/tokens/__PAGE__.segment.rsc +1 -1
  111. package/.next/server/app/settings/tokens.segments/settings/tokens.segment.rsc +1 -1
  112. package/.next/server/app/settings/tokens.segments/settings.segment.rsc +1 -1
  113. package/.next/server/app/settings.html +2 -2
  114. package/.next/server/app/settings.rsc +3 -3
  115. package/.next/server/app/settings.segments/_full.segment.rsc +3 -3
  116. package/.next/server/app/settings.segments/_head.segment.rsc +1 -1
  117. package/.next/server/app/settings.segments/_index.segment.rsc +2 -2
  118. package/.next/server/app/settings.segments/_tree.segment.rsc +2 -2
  119. package/.next/server/app/settings.segments/settings/__PAGE__.segment.rsc +2 -2
  120. package/.next/server/app/settings.segments/settings.segment.rsc +1 -1
  121. package/.next/server/app/tasks/[id]/page_client-reference-manifest.js +1 -1
  122. package/.next/server/app/tasks/page_client-reference-manifest.js +1 -1
  123. package/.next/server/app/tasks.html +2 -2
  124. package/.next/server/app/tasks.rsc +2 -2
  125. package/.next/server/app/tasks.segments/_full.segment.rsc +2 -2
  126. package/.next/server/app/tasks.segments/_head.segment.rsc +1 -1
  127. package/.next/server/app/tasks.segments/_index.segment.rsc +2 -2
  128. package/.next/server/app/tasks.segments/_tree.segment.rsc +2 -2
  129. package/.next/server/app/tasks.segments/tasks/__PAGE__.segment.rsc +1 -1
  130. package/.next/server/app/tasks.segments/tasks.segment.rsc +1 -1
  131. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js +3 -3
  132. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js.map +1 -1
  133. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js +1 -1
  134. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js.map +1 -1
  135. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js +1 -1
  136. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js.map +1 -1
  137. package/.next/server/middleware-build-manifest.js +3 -3
  138. package/.next/server/pages/404.html +2 -2
  139. package/.next/server/pages/500.html +1 -1
  140. package/.next/static/chunks/05.asnoxq2t39.js +1 -0
  141. package/.next/static/chunks/06_k1dgr3isa2.js +1 -0
  142. package/.next/static/chunks/0fxhdt5-198se.js +4 -0
  143. package/.next/static/chunks/0hwb_h953qm4d.css +2 -0
  144. package/.next/trace +2 -2
  145. package/.next/trace-build +1 -1
  146. package/app/components/TopoGraph.tsx +760 -58
  147. package/package.json +1 -1
  148. package/scripts/topo-chip-digit-fontweight-test.mjs +105 -0
  149. package/scripts/topo-chrome-segmented-radius-test.mjs +100 -0
  150. package/scripts/topo-edge-badge-fontsize-test.mjs +90 -0
  151. package/scripts/topo-edge-badge-opacity-test.mjs +80 -0
  152. package/scripts/topo-edge-badge-stroke-test.mjs +92 -0
  153. package/scripts/topo-filter-pill-hover-opacity-test.mjs +110 -0
  154. package/scripts/topo-filter-pill-x-hover-scale-test.mjs +99 -0
  155. package/scripts/topo-flow-rail-linecap-test.mjs +79 -0
  156. package/scripts/topo-freshness-chip-tabular-test.mjs +41 -0
  157. package/scripts/topo-freshness-floor-lift-test.mjs +92 -0
  158. package/scripts/topo-freshness-suffix-tabular-test.mjs +88 -0
  159. package/scripts/topo-fullscreen-icon-hover-scale-test.mjs +91 -0
  160. package/scripts/topo-group-label-count-fontweight-test.mjs +108 -0
  161. package/scripts/topo-hub-digit-fontsize-test.mjs +86 -0
  162. package/scripts/topo-hub-highlight-radius-test.mjs +90 -0
  163. package/scripts/topo-hub-hover-ring-opacity-test.mjs +96 -0
  164. package/scripts/topo-layout-toggle-hover-tracking-test.mjs +109 -0
  165. package/scripts/topo-layout-toggle-radius-test.mjs +87 -0
  166. package/scripts/topo-legend-label-fontweight-test.mjs +94 -0
  167. package/scripts/topo-minimap-offline-opacity-test.mjs +90 -0
  168. package/scripts/topo-minimap-viewport-hover-test.mjs +109 -0
  169. package/scripts/topo-minimap-viewport-linejoin-test.mjs +75 -0
  170. package/scripts/topo-more-flows-fontweight-test.mjs +103 -0
  171. package/scripts/topo-more-flows-hover-letterspacing-test.mjs +98 -0
  172. package/scripts/topo-panel-count-letterspacing-test.mjs +89 -0
  173. package/scripts/topo-panel-rect-opacity-hover-test.mjs +109 -0
  174. package/scripts/topo-panel-title-hover-letterspacing-test.mjs +97 -0
  175. package/scripts/topo-pressure-bar-height-test.mjs +92 -0
  176. package/scripts/topo-pressure-kicker-fontweight-test.mjs +76 -0
  177. package/scripts/topo-recent-pip-radius-test.mjs +76 -0
  178. package/scripts/topo-recent-row-text-fontweight-test.mjs +90 -0
  179. package/scripts/topo-reset-hover-rotate-test.mjs +102 -0
  180. package/scripts/topo-vendor-glyph-fontweight-test.mjs +102 -0
  181. package/scripts/topo-vendor-letter-hover-scale-test.mjs +129 -0
  182. package/scripts/topo-zoom-icon-hover-scale-test.mjs +114 -0
  183. package/scripts/topo-zoom-level-hover-letterspacing-test.mjs +91 -0
  184. package/.next/static/chunks/04sya_8oheege.js +0 -4
  185. package/.next/static/chunks/08c-dt44c.swe.js +0 -1
  186. package/.next/static/chunks/0aauz~36q5n2a.css +0 -2
  187. package/.next/static/chunks/12e0z26h2p0xl.js +0 -1
  188. /package/.next/static/{CkuUcpLXVJ-OsmjZqi3xX → PtUnxSzwOvGhpWPUawz9k}/_buildManifest.js +0 -0
  189. /package/.next/static/{CkuUcpLXVJ-OsmjZqi3xX → PtUnxSzwOvGhpWPUawz9k}/_clientMiddlewareManifest.js +0 -0
  190. /package/.next/static/{CkuUcpLXVJ-OsmjZqi3xX → PtUnxSzwOvGhpWPUawz9k}/_ssgManifest.js +0 -0
@@ -0,0 +1,129 @@
1
+ /* Round 354 verification: vendor letter glyph scales 1.0 → 1.1 on
2
+ * hover. R88 already dims OTHER vendors via canvas-wide opacity
3
+ * masking; R202 added a chip-level bg tint (color-mix 12 % alpha) so
4
+ * the chip itself responds. R354 closes the trio with a glyph-level
5
+ * lift: the focused vendor LETTER actively rises (transform scale)
6
+ * rather than the chip merely changing colour.
7
+ *
8
+ * Three layers of positive feedback on the hovered vendor + canvas-
9
+ * wide negative feedback on the others — a clean figure/ground
10
+ * separation.
11
+ *
12
+ * Contract (cyber, fixture with 3 distinct vendors so R281's > 2
13
+ * threshold mounts the chip):
14
+ * - Rest: vendor letter glyph transform === matrix(1, 0, 0, 1, 0, 0)
15
+ * (identity).
16
+ * - Hover one vendor chip: that glyph transform === matrix(1.1, 0,
17
+ * 0, 1.1, 0, 0).
18
+ * - data-vendor-letter-glyph-hover toggles false → true on
19
+ * hovered chip only; sibling glyphs stay 'false'.
20
+ * - Inline transition contains 'transform'.
21
+ * - Pre-R354 invariants: R202 bg color-mix on hover still applies,
22
+ * data-vendor-letter-count-suffix span (R333) still present.
23
+ */
24
+ import { chromium } from 'playwright';
25
+ import { readFileSync } from 'node:fs';
26
+
27
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
28
+ const browser = await chromium.launch({ headless: true });
29
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
30
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
31
+ await ctx.addInitScript(() => {
32
+ try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
33
+ });
34
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
35
+ await ctx.route('**/api/hub/status*', async (route) => {
36
+ const r = await route.fetch();
37
+ const b = await r.json();
38
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
39
+ // R281 vendor-dist threshold is > 2 distinct vendors — provide 3.
40
+ const mk = (alias, model) => ({
41
+ alias, status: 'working', model, runtime: 'claude-code-cli',
42
+ network_id: nid, project_dir: null,
43
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
44
+ });
45
+ await route.fulfill({ response: r, json: { ...b, sessions: [
46
+ mk('a1', 'claude-opus-4'),
47
+ mk('a2', 'claude-opus-4'),
48
+ mk('b1', 'gpt-4o'),
49
+ mk('b2', 'gpt-4o'),
50
+ mk('c1', 'internlm/internlm2'),
51
+ ] } });
52
+ });
53
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
54
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
55
+
56
+ const page = await ctx.newPage();
57
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
58
+ await page.waitForSelector('[data-vendor-letter-glyph]', { timeout: 15000 });
59
+ await page.waitForTimeout(300);
60
+
61
+ // Read all vendor letters present.
62
+ const inventory = await page.evaluate(() => {
63
+ const glyphs = Array.from(document.querySelectorAll('[data-vendor-letter-glyph]'));
64
+ return glyphs.map(el => el.getAttribute('data-vendor-letter-glyph'));
65
+ });
66
+
67
+ // Pick the first vendor letter for the hover test.
68
+ const target = inventory[0];
69
+
70
+ const restProbe = await page.evaluate((tgt) => {
71
+ const glyph = document.querySelector(`[data-vendor-letter-glyph="${tgt}"]`);
72
+ const cs = glyph ? getComputedStyle(glyph) : null;
73
+ const chip = glyph ? glyph.closest('[data-vendor-letter]') : null;
74
+ const suffix = chip ? chip.querySelector('[data-vendor-letter-count-suffix]') : null;
75
+ return {
76
+ transform: cs?.transform ?? null,
77
+ transition: cs?.transition ?? null,
78
+ hoverAttr: glyph?.getAttribute('data-vendor-letter-glyph-hover') ?? null,
79
+ display: cs?.display ?? null,
80
+ suffixPresent: !!suffix,
81
+ };
82
+ }, target);
83
+
84
+ // Hover the target vendor letter's chip.
85
+ await page.hover(`[data-vendor-letter="${target}"]`);
86
+ await page.waitForTimeout(300);
87
+ const hoverProbe = await page.evaluate((tgt) => {
88
+ const glyph = document.querySelector(`[data-vendor-letter-glyph="${tgt}"]`);
89
+ const cs = glyph ? getComputedStyle(glyph) : null;
90
+ // Read all sibling hover attrs to confirm only one is active.
91
+ const all = Array.from(document.querySelectorAll('[data-vendor-letter-glyph]'))
92
+ .map(el => ({ key: el.getAttribute('data-vendor-letter-glyph'),
93
+ hover: el.getAttribute('data-vendor-letter-glyph-hover') }));
94
+ return {
95
+ transform: cs?.transform ?? null,
96
+ hoverAttr: glyph?.getAttribute('data-vendor-letter-glyph-hover') ?? null,
97
+ siblings: all,
98
+ };
99
+ }, target);
100
+
101
+ await browser.close();
102
+
103
+ const isIdentity = (t) => t === 'none' || /matrix\(1, 0, 0, 1, 0, 0\)/.test(t || '');
104
+ const isScale110 = (t) => /matrix\(1\.1, 0, 0, 1\.1, 0, 0\)/.test(t || '');
105
+ const hasTrans = (s, prop) =>
106
+ new RegExp(`${prop}\\s+\\d*\\.?\\d*s|${prop}\\s+\\d+ms`, 'i').test(s || '');
107
+
108
+ const siblingsOnlyOneTrue = hoverProbe.siblings.filter(s => s.hover === 'true').length === 1
109
+ && hoverProbe.siblings.find(s => s.key === target)?.hover === 'true';
110
+
111
+ const results = {
112
+ inventory_has_glyphs: inventory.length >= 3,
113
+ rest_identity: isIdentity(restProbe.transform),
114
+ rest_hover_false: restProbe.hoverAttr === 'false',
115
+ // Flex children get display: block computed (spec blockification),
116
+ // even when declared inline-block — accept either for safety.
117
+ rest_display_block: restProbe.display === 'inline-block' || restProbe.display === 'block',
118
+ trans_has_transform: hasTrans(restProbe.transition, 'transform'),
119
+ suffix_present: restProbe.suffixPresent, // R333 preserved
120
+ hover_scale_110: isScale110(hoverProbe.transform),
121
+ hover_attr_true: hoverProbe.hoverAttr === 'true',
122
+ hover_isolates_target: siblingsOnlyOneTrue,
123
+ };
124
+ const ok = Object.values(results).every(Boolean);
125
+ console.log(`${ok ? '✅' : '❌'} vendor letter glyph hover-scale:`, JSON.stringify(results),
126
+ '\n inventory:', inventory,
127
+ '\n rest: ', restProbe,
128
+ '\n hover: ', hoverProbe);
129
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,114 @@
1
+ /* Round 352 verification: zoom-out + zoom-in icons gain group-hover:
2
+ * scale-110 — sibling to R350 reset hover-rotate. Pre-R352 hovering
3
+ * the zoom buttons only changed the bg (white/5); the icons stayed
4
+ * still. R352 lifts each icon 10 % on hover for a tactile "this
5
+ * button does something" cue. Pure-CSS via Tailwind `group` parent +
6
+ * `group-hover:scale-110` on the svg + `transition-transform
7
+ * duration-200 ease-out`. CSS-animation precedence keeps R186
8
+ * anet-chrome-pop owning transform during the 220 ms click pop;
9
+ * group-hover scale-110 picks up smoothly afterward.
10
+ *
11
+ * Contract:
12
+ * - Rest: both zoom icons computed transform 'matrix(1, 0, 0, 1, 0, 0)'
13
+ * (or 'none').
14
+ * - Hover zoom-out button: zoom-out icon transform matrix(1.1, 0, 0,
15
+ * 1.1, 0, 0).
16
+ * - Hover zoom-in button: zoom-in icon transform matrix(1.1, 0, 0,
17
+ * 1.1, 0, 0).
18
+ * - Inline transition on both icons contains 'transform'.
19
+ * - Pre-R352 invariants: data-topo-chrome-zoom-{out,in}-icon attrs
20
+ * intact, R288 strokeWidth=2.5 preserved, anet-chrome-pop class
21
+ * absent at rest.
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-topo-chrome-zoom-out-icon]', { timeout: 15000 });
51
+ await page.waitForSelector('[data-topo-chrome-zoom-in-icon]', { timeout: 5000 });
52
+ await page.waitForTimeout(300);
53
+
54
+ // Tailwind 4 compiles scale-110 to the modern CSS `scale` property
55
+ // (not legacy `transform: scale()`), so we probe both — `transform`
56
+ // stays at identity matrix and `scale` carries the 1.1 value.
57
+ const restProbe = await page.evaluate(() => {
58
+ const oico = document.querySelector('[data-topo-chrome-zoom-out-icon]');
59
+ const iico = document.querySelector('[data-topo-chrome-zoom-in-icon]');
60
+ const ocs = oico ? getComputedStyle(oico) : null;
61
+ const ics = iico ? getComputedStyle(iico) : null;
62
+ return {
63
+ outScale: ocs?.scale ?? null,
64
+ outTransition: ocs?.transition ?? null,
65
+ outStrokeW: oico?.getAttribute('stroke-width') ?? null,
66
+ outClass: oico?.getAttribute('class') ?? null,
67
+ inScale: ics?.scale ?? null,
68
+ inTransition: ics?.transition ?? null,
69
+ inStrokeW: iico?.getAttribute('stroke-width') ?? null,
70
+ inClass: iico?.getAttribute('class') ?? null,
71
+ };
72
+ });
73
+
74
+ // Hover zoom-out button.
75
+ await page.hover('[data-topo-chrome-zoom-out]');
76
+ await page.waitForTimeout(300);
77
+ const zoomOutHover = await page.evaluate(() => {
78
+ const ico = document.querySelector('[data-topo-chrome-zoom-out-icon]');
79
+ return { scale: ico ? getComputedStyle(ico).scale : null };
80
+ });
81
+
82
+ // Hover zoom-in button.
83
+ await page.hover('[data-topo-chrome-zoom-in]');
84
+ await page.waitForTimeout(300);
85
+ const zoomInHover = await page.evaluate(() => {
86
+ const ico = document.querySelector('[data-topo-chrome-zoom-in-icon]');
87
+ return { scale: ico ? getComputedStyle(ico).scale : null };
88
+ });
89
+
90
+ await browser.close();
91
+
92
+ const isRestScale = (s) => s === 'none' || s === '1' || s === '1 1' || s === null;
93
+ const isHoverScale110 = (s) => s === '1.1' || s === '1.1 1.1';
94
+ const hasTrans = (s, prop) =>
95
+ new RegExp(`${prop}\\s+\\d*\\.?\\d*s|${prop}\\s+\\d+ms`, 'i').test(s || '');
96
+
97
+ const results = {
98
+ rest_out_scale_1: isRestScale(restProbe.outScale),
99
+ rest_in_scale_1: isRestScale(restProbe.inScale),
100
+ out_trans_has_scale: hasTrans(restProbe.outTransition, 'scale') || hasTrans(restProbe.outTransition, 'transform'),
101
+ in_trans_has_scale: hasTrans(restProbe.inTransition, 'scale') || hasTrans(restProbe.inTransition, 'transform'),
102
+ out_strokew_2_5: restProbe.outStrokeW === '2.5',
103
+ in_strokew_2_5: restProbe.inStrokeW === '2.5',
104
+ out_no_pop: !/anet-chrome-pop/.test(restProbe.outClass || ''),
105
+ in_no_pop: !/anet-chrome-pop/.test(restProbe.inClass || ''),
106
+ hover_out_scale_110: isHoverScale110(zoomOutHover.scale),
107
+ hover_in_scale_110: isHoverScale110(zoomInHover.scale),
108
+ };
109
+ const ok = Object.values(results).every(Boolean);
110
+ console.log(`${ok ? '✅' : '❌'} zoom icons hover-scale:`, JSON.stringify(results),
111
+ '\n rest: ', restProbe,
112
+ '\n hover zoom-out: ', zoomOutHover,
113
+ '\n hover zoom-in: ', zoomInHover);
114
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,91 @@
1
+ /* Round 347 verification: zoom-level readout in the chrome strip
2
+ * gains a hover-state letter-spacing tween 0 → 0.5 px. Extends the
3
+ * R344/R345 hover-letter-spacing family from panel/footer surfaces
4
+ * into the HTML chrome strip. The readout is sandwiched between
5
+ * zoom-out / zoom-in buttons; hovering it spreads digits 0.5 px,
6
+ * signalling "this is alive". tabular-nums + minWidth: 46 from R225
7
+ * lock the column so the tween doesn't shove neighbouring controls.
8
+ *
9
+ * Contract:
10
+ * - Rest: zoom-level letter-spacing 0 (or 'normal' / '0px').
11
+ * - Hover: letter-spacing 0.5px.
12
+ * - data-topo-chrome-zoom-level-hover toggles false → true.
13
+ * - transition list contains 'letter-spacing'.
14
+ * - Pre-R347 invariants intact: R264 color + border-color transitions
15
+ * preserved; tabular-nums + minWidth=46 still applied.
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
+ await page.waitForSelector('[data-topo-chrome-zoom-level]', { timeout: 15000 });
45
+ await page.waitForTimeout(400);
46
+
47
+ const restProbe = await page.evaluate(() => {
48
+ const el = document.querySelector('[data-topo-chrome-zoom-level]');
49
+ const cs = el ? getComputedStyle(el) : null;
50
+ return {
51
+ ls: cs?.letterSpacing ?? null,
52
+ transition: cs?.transition ?? null,
53
+ hoverAttr: el?.getAttribute('data-topo-chrome-zoom-level-hover') ?? null,
54
+ minWidth: cs?.minWidth ?? null,
55
+ fontVariant: cs?.fontVariantNumeric ?? null,
56
+ };
57
+ });
58
+
59
+ await page.hover('[data-topo-chrome-zoom-level]');
60
+ await page.waitForTimeout(250);
61
+ const hoverProbe = await page.evaluate(() => {
62
+ const el = document.querySelector('[data-topo-chrome-zoom-level]');
63
+ const cs = el ? getComputedStyle(el) : null;
64
+ return {
65
+ ls: cs?.letterSpacing ?? null,
66
+ hoverAttr: el?.getAttribute('data-topo-chrome-zoom-level-hover') ?? null,
67
+ };
68
+ });
69
+
70
+ await browser.close();
71
+
72
+ const hasTrans = (s, prop) =>
73
+ new RegExp(`${prop}\\s+\\d*\\.?\\d*s|${prop}\\s+\\d+ms`, 'i').test(s || '');
74
+ const isRestLs = (v) => v === 'normal' || v === '0px' || v === '0' || v === null;
75
+
76
+ const results = {
77
+ rest_ls_zero: isRestLs(restProbe.ls),
78
+ rest_hover_attr_false: restProbe.hoverAttr === 'false',
79
+ trans_has_letterspacing: hasTrans(restProbe.transition, 'letter-spacing'),
80
+ trans_has_color: hasTrans(restProbe.transition, 'color'), // R264 preserved
81
+ trans_has_border: hasTrans(restProbe.transition, 'border-color'),
82
+ min_width_46: restProbe.minWidth === '46px',
83
+ tabular_nums: /tabular-nums/.test(restProbe.fontVariant || ''),
84
+ hover_ls_0_5: hoverProbe.ls === '0.5px',
85
+ hover_attr_true: hoverProbe.hoverAttr === 'true',
86
+ };
87
+ const ok = Object.values(results).every(Boolean);
88
+ console.log(`${ok ? '✅' : '❌'} zoom level hover letter-spacing:`, JSON.stringify(results),
89
+ '\n rest: ', restProbe,
90
+ '\n hover:', hoverProbe);
91
+ process.exit(ok ? 0 : 1);