@sleep2agi/agent-network-dashboard 0.5.3-preview.18 → 0.5.3-preview.181

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 (322) 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 +7 -7
  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/[root-of-the-server]__0sv~g.o._.js +1 -1
  132. package/.next/server/chunks/ssr/[root-of-the-server]__0sv~g.o._.js.map +1 -1
  133. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js +4 -4
  134. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js.map +1 -1
  135. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js +1 -1
  136. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js.map +1 -1
  137. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js +1 -1
  138. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js.map +1 -1
  139. package/.next/server/middleware-build-manifest.js +3 -3
  140. package/.next/server/pages/404.html +2 -2
  141. package/.next/server/pages/500.html +1 -1
  142. package/.next/static/chunks/0-8bxlg9uqt.w.css +2 -0
  143. package/.next/static/chunks/05hk317799_zi.js +1 -0
  144. package/.next/static/chunks/0chpuxik8zz52.js +1 -0
  145. package/.next/static/chunks/{0ha~3ah2r1rb1.js → 0jb6h4syeiiaj.js} +1 -1
  146. package/.next/static/chunks/0k.pw37g2bgyt.js +4 -0
  147. package/.next/trace +2 -2
  148. package/.next/trace-build +1 -1
  149. package/app/components/TopoGraph.tsx +4600 -174
  150. package/app/globals.css +58 -7
  151. package/package.json +4 -4
  152. package/scripts/topo-active-chrome-hover-text-test.mjs +107 -0
  153. package/scripts/topo-alias-chat-brightness-test.mjs +79 -0
  154. package/scripts/topo-alias-text-halo-layers-test.mjs +98 -0
  155. package/scripts/topo-avatar-chat-gate-test.mjs +77 -0
  156. package/scripts/topo-avatar-drop-shadow-test.mjs +86 -0
  157. package/scripts/topo-avatar-fallback-hover-test.mjs +104 -0
  158. package/scripts/topo-avatar-fallback-rotate-test.mjs +92 -0
  159. package/scripts/topo-avatar-rotate-test.mjs +85 -0
  160. package/scripts/topo-avatar-scale-test.mjs +89 -0
  161. package/scripts/topo-badge-chat-gate-test.mjs +74 -0
  162. package/scripts/topo-brand-drop-shadow-test.mjs +71 -0
  163. package/scripts/topo-brand-logo-breath-test.mjs +102 -0
  164. package/scripts/topo-brand-logo-hover-brightness-test.mjs +105 -0
  165. package/scripts/topo-brand-logo-hover-rotate-test.mjs +93 -0
  166. package/scripts/topo-brand-logo-hover-test.mjs +85 -0
  167. package/scripts/topo-card-chat-brightness-test.mjs +86 -0
  168. package/scripts/topo-chat-ring-breath-curve-test.mjs +114 -0
  169. package/scripts/topo-chat-ring-brightness-test.mjs +80 -0
  170. package/scripts/topo-chat-ring-halo-layers-test.mjs +100 -0
  171. package/scripts/topo-chat-ring-r-breath-test.mjs +121 -0
  172. package/scripts/topo-chat-ring-status-halo-test.mjs +106 -0
  173. package/scripts/topo-chat-ring-sw-breath-test.mjs +123 -0
  174. package/scripts/topo-chip-row-digit-ls-test.mjs +135 -0
  175. package/scripts/topo-chip-row-member-alias-lit-test.mjs +154 -0
  176. package/scripts/topo-chip-row-tier-glow-brightness-test.mjs +99 -0
  177. package/scripts/topo-chip-row-unit-hover-tracking-test.mjs +124 -0
  178. package/scripts/topo-chrome-control-halo-layers-test.mjs +22 -0
  179. package/scripts/topo-click-ripple-glow-test.mjs +86 -0
  180. package/scripts/topo-click-ripple-halo-layers-test.mjs +79 -0
  181. package/scripts/topo-click-ripple-sw-test.mjs +110 -0
  182. package/scripts/topo-cluster-count-attr-test.mjs +80 -0
  183. package/scripts/topo-crescent-breath-test.mjs +104 -0
  184. package/scripts/topo-crescent-recede-test.mjs +111 -0
  185. package/scripts/topo-edge-badge-circle-brightness-test.mjs +82 -0
  186. package/scripts/topo-edge-badge-circle-hot-pulse-test.mjs +100 -0
  187. package/scripts/topo-edge-badge-endpoint-gate-test.mjs +94 -0
  188. package/scripts/topo-edge-badge-halo-layers-test.mjs +85 -0
  189. package/scripts/topo-edge-badge-hot-pulse-test.mjs +92 -0
  190. package/scripts/topo-edge-badge-hover-glow-test.mjs +90 -0
  191. package/scripts/topo-edge-badge-text-brightness-test.mjs +83 -0
  192. package/scripts/topo-edge-chat-gate-test.mjs +71 -0
  193. package/scripts/topo-edge-particle-brightness-test.mjs +82 -0
  194. package/scripts/topo-edge-pill-glow-test.mjs +67 -0
  195. package/scripts/topo-edge-visible-brightness-test.mjs +84 -0
  196. package/scripts/topo-endpoint-ring-brightness-test.mjs +83 -0
  197. package/scripts/topo-endpoint-ring-flow-halo-test.mjs +107 -0
  198. package/scripts/topo-endpoint-ring-halo-layers-test.mjs +100 -0
  199. package/scripts/topo-filter-pill-glow-test.mjs +90 -0
  200. package/scripts/topo-filter-pill-halo-layers-test.mjs +27 -0
  201. package/scripts/topo-flow-arrow-brightness-test.mjs +82 -0
  202. package/scripts/topo-flow-rail-brightness-test.mjs +80 -0
  203. package/scripts/topo-freshness-chip-fade-test.mjs +105 -0
  204. package/scripts/topo-fullscreen-attr-test.mjs +73 -0
  205. package/scripts/topo-fullscreen-brightness-test.mjs +84 -0
  206. package/scripts/topo-fullscreen-icon-rotate-test.mjs +93 -0
  207. package/scripts/topo-grid-content-bottom-attr-test.mjs +72 -0
  208. package/scripts/topo-group-box-brightness-test.mjs +84 -0
  209. package/scripts/topo-group-chat-gate-test.mjs +77 -0
  210. package/scripts/topo-group-label-brightness-test.mjs +84 -0
  211. package/scripts/topo-group-label-halo-layers-test.mjs +78 -0
  212. package/scripts/topo-group-label-hover-glow-test.mjs +86 -0
  213. package/scripts/topo-group-label-member-alias-hover-test.mjs +125 -0
  214. package/scripts/topo-group-pill-glow-test.mjs +76 -0
  215. package/scripts/topo-group-tint-brightness-test.mjs +82 -0
  216. package/scripts/topo-halo-chat-gate-test.mjs +72 -0
  217. package/scripts/topo-hub-core-brightness-test.mjs +82 -0
  218. package/scripts/topo-hub-digit-brightness-test.mjs +79 -0
  219. package/scripts/topo-hub-digit-halo-layers-test.mjs +76 -0
  220. package/scripts/topo-hub-digit-ls-test.mjs +119 -0
  221. package/scripts/topo-hub-halo-brightness-test.mjs +80 -0
  222. package/scripts/topo-hub-halo-glow-test.mjs +96 -0
  223. package/scripts/topo-hub-halo-halo-layers-test.mjs +76 -0
  224. package/scripts/topo-hub-highlight-amplify-test.mjs +139 -0
  225. package/scripts/topo-hub-highlight-brightness-test.mjs +84 -0
  226. package/scripts/topo-hub-highlight-fill-transition-test.mjs +84 -0
  227. package/scripts/topo-hub-highlight-glow-test.mjs +99 -0
  228. package/scripts/topo-hub-highlight-halo-layers-test.mjs +78 -0
  229. package/scripts/topo-hub-highlight-r-test.mjs +112 -0
  230. package/scripts/topo-hub-highlight-recede-test.mjs +144 -0
  231. package/scripts/topo-hub-highlight-theme-fill-test.mjs +83 -0
  232. package/scripts/topo-hub-hover-ring-brightness-test.mjs +79 -0
  233. package/scripts/topo-hub-hover-ring-glow-test.mjs +97 -0
  234. package/scripts/topo-hub-hover-ring-halo-layers-test.mjs +71 -0
  235. package/scripts/topo-hub-idle-breath-test.mjs +7 -2
  236. package/scripts/topo-hub-recede-test.mjs +124 -0
  237. package/scripts/topo-hub-spoke-brightness-test.mjs +77 -0
  238. package/scripts/topo-hub-spoke-glow-test.mjs +112 -0
  239. package/scripts/topo-hub-spoke-halo-layers-test.mjs +97 -0
  240. package/scripts/topo-hub-spoke-self-filter-test.mjs +119 -0
  241. package/scripts/topo-label-card-brightness-test.mjs +81 -0
  242. package/scripts/topo-layout-hover-fw-test.mjs +98 -0
  243. package/scripts/topo-layout-toggle-brightness-test.mjs +94 -0
  244. package/scripts/topo-legend-count-brightness-test.mjs +80 -0
  245. package/scripts/topo-legend-count-letter-spacing-test.mjs +108 -0
  246. package/scripts/topo-legend-label-fw-test.mjs +107 -0
  247. package/scripts/topo-legend-pin-ring-brightness-test.mjs +82 -0
  248. package/scripts/topo-legend-pin-ring-halo-layers-test.mjs +71 -0
  249. package/scripts/topo-legend-row-count-brightness-test.mjs +85 -0
  250. package/scripts/topo-legend-row-label-glow-test.mjs +102 -0
  251. package/scripts/topo-legend-swatch-glow-test.mjs +109 -0
  252. package/scripts/topo-legend-swatch-member-alias-match-test.mjs +139 -0
  253. package/scripts/topo-legend-tint-brightness-test.mjs +83 -0
  254. package/scripts/topo-legend-trio-halo-layers-test.mjs +22 -0
  255. package/scripts/topo-minimap-dot-chat-gate-test.mjs +81 -0
  256. package/scripts/topo-minimap-hover-glow-test.mjs +109 -0
  257. package/scripts/topo-minimap-viewport-brightness-test.mjs +84 -0
  258. package/scripts/topo-minimap-viewport-halo-layers-test.mjs +24 -0
  259. package/scripts/topo-more-footer-brightness-test.mjs +94 -0
  260. package/scripts/topo-node-alias-brightness-test.mjs +84 -0
  261. package/scripts/topo-node-avatar-halo-layers-test.mjs +25 -0
  262. package/scripts/topo-node-hover-ring-halo-layers-test.mjs +70 -0
  263. package/scripts/topo-node-sub-text-brightness-test.mjs +88 -0
  264. package/scripts/topo-nodesize-brightness-test.mjs +82 -0
  265. package/scripts/topo-nodesize-hover-fw-test.mjs +99 -0
  266. package/scripts/topo-orphan-fill-opacity-test.mjs +91 -0
  267. package/scripts/topo-orphan-label-opacity-test.mjs +98 -0
  268. package/scripts/topo-panel-count-hover-ls-test.mjs +87 -0
  269. package/scripts/topo-panel-row-brightness-test.mjs +116 -0
  270. package/scripts/topo-panel-title-brightness-test.mjs +98 -0
  271. package/scripts/topo-panel-title-glow-test.mjs +111 -0
  272. package/scripts/topo-panel-titles-halo-layers-test.mjs +23 -0
  273. package/scripts/topo-pill-x-rotate-test.mjs +96 -0
  274. package/scripts/topo-pip-brightness-test.mjs +85 -0
  275. package/scripts/topo-pressure-bar-halo-layers-test.mjs +19 -0
  276. package/scripts/topo-pressure-seg-glow-test.mjs +92 -0
  277. package/scripts/topo-pressure-seg-member-alias-match-test.mjs +133 -0
  278. package/scripts/topo-pressure-seg-motion-test.mjs +101 -0
  279. package/scripts/topo-recent-count-brightness-test.mjs +84 -0
  280. package/scripts/topo-recent-more-fw-test.mjs +126 -0
  281. package/scripts/topo-recent-panel-hot-pulse-test.mjs +105 -0
  282. package/scripts/topo-recent-pip-halo-layers-test.mjs +82 -0
  283. package/scripts/topo-recent-row-chat-gate-test.mjs +75 -0
  284. package/scripts/topo-recent-row-content-lift-test.mjs +140 -0
  285. package/scripts/topo-recent-row-fw-test.mjs +115 -0
  286. package/scripts/topo-recent-row-text-glow-test.mjs +86 -0
  287. package/scripts/topo-recent-row-text-halo-layers-test.mjs +67 -0
  288. package/scripts/topo-recent-tint-brightness-test.mjs +80 -0
  289. package/scripts/topo-recent-ts-brightness-test.mjs +86 -0
  290. package/scripts/topo-reduced-motion-attr-test.mjs +69 -0
  291. package/scripts/topo-reset-brightness-test.mjs +83 -0
  292. package/scripts/topo-reset-icon-hover-scale-test.mjs +102 -0
  293. package/scripts/topo-runtime-badge-brightness-test.mjs +78 -0
  294. package/scripts/topo-runtime-badge-glow-test.mjs +108 -0
  295. package/scripts/topo-runtime-badge-halo-layers-test.mjs +87 -0
  296. package/scripts/topo-runtime-badge-rotate-test.mjs +85 -0
  297. package/scripts/topo-spoke-chat-gate-test.mjs +72 -0
  298. package/scripts/topo-starfield-hue-test.mjs +109 -0
  299. package/scripts/topo-status-pin-pill-halo-layers-test.mjs +17 -0
  300. package/scripts/topo-status-ring-brightness-test.mjs +84 -0
  301. package/scripts/topo-status-ring-chat-gate-test.mjs +72 -0
  302. package/scripts/topo-status-ring-halo-layers-test.mjs +105 -0
  303. package/scripts/topo-status-ring-status-halo-test.mjs +110 -0
  304. package/scripts/topo-sub-text-chat-brightness-test.mjs +81 -0
  305. package/scripts/topo-titleblock-h2-hover-fw-test.mjs +109 -0
  306. package/scripts/topo-titleblock-h2-hover-tracking-test.mjs +128 -0
  307. package/scripts/topo-titleblock-kicker-hover-test.mjs +134 -0
  308. package/scripts/topo-vendor-chip-glow-test.mjs +97 -0
  309. package/scripts/topo-vendor-chip-halo-layers-test.mjs +18 -0
  310. package/scripts/topo-vendor-pill-glow-test.mjs +98 -0
  311. package/scripts/topo-watermark-breath-test.mjs +100 -0
  312. package/scripts/topo-watermark-recede-test.mjs +114 -0
  313. package/scripts/topo-zoom-buttons-brightness-test.mjs +94 -0
  314. package/scripts/topo-zoom-level-brightness-test.mjs +83 -0
  315. package/scripts/topo-zoom-level-color-test.mjs +105 -0
  316. package/.next/static/chunks/0m.1mvl~t.avc.css +0 -2
  317. package/.next/static/chunks/0r19pt1ihol7g.js +0 -1
  318. package/.next/static/chunks/0u4p3l2hq_1kj.js +0 -4
  319. package/.next/static/chunks/0z6a3b81h8btk.js +0 -1
  320. /package/.next/static/{dLzRHfUo59WgRQX9Mbfuk → yS4qco8meQiV57CZRx0Uc}/_buildManifest.js +0 -0
  321. /package/.next/static/{dLzRHfUo59WgRQX9Mbfuk → yS4qco8meQiV57CZRx0Uc}/_clientMiddlewareManifest.js +0 -0
  322. /package/.next/static/{dLzRHfUo59WgRQX9Mbfuk → yS4qco8meQiV57CZRx0Uc}/_ssgManifest.js +0 -0
@@ -0,0 +1,84 @@
1
+ /* Round 574 verification: hub-highlight disc stacks brightness(1.15)
2
+ * onto R532 drop-shadow on hub-hover. 13th anchor in per-element
3
+ * brightness family.
4
+ *
5
+ * Mock: 0 working sessions → hub-highlight is visible (idle state).
6
+ *
7
+ * Test phases:
8
+ * 1. rest: filter='none' (no hover), brightness-attr='1'
9
+ * 2. source-side regex confirms stacked filter with brightness +
10
+ * data-attr conditional
11
+ */
12
+ import { chromium } from 'playwright';
13
+ import { readFileSync } from 'node:fs';
14
+
15
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
16
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
17
+
18
+ const browser = await chromium.launch({ headless: true });
19
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
20
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
21
+ await ctx.addInitScript(() => {
22
+ try {
23
+ localStorage.setItem('anet-theme', 'cyber');
24
+ localStorage.setItem('anet-topo-layout', 'ring');
25
+ sessionStorage.setItem('anet_v3_auth', '1');
26
+ } catch {}
27
+ });
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: 'idle', 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
+ // All idle → workingCount=0 → hub-highlight visible
38
+ await route.fulfill({ response: r, json: { ...b, sessions: [mk('a·1'), mk('a·2')] } });
39
+ });
40
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
41
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
42
+ const page = await ctx.newPage();
43
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
44
+ await page.waitForSelector('[data-topo-hub-highlight]', { timeout: 15000 });
45
+ await page.waitForTimeout(500);
46
+
47
+ const rest = await page.evaluate(() => {
48
+ const el = document.querySelector('[data-topo-hub-highlight]');
49
+ if (!el) return null;
50
+ const cs = getComputedStyle(el);
51
+ return {
52
+ filter: cs.filter,
53
+ transitionProperty: cs.transitionProperty,
54
+ brightnessAttr: el.getAttribute('data-topo-hub-highlight-brightness'),
55
+ glowAttr: el.getAttribute('data-topo-hub-highlight-glow'),
56
+ hoveredAttr: el.getAttribute('data-topo-hub-highlight-hovered'),
57
+ visibleAttr: el.getAttribute('data-topo-hub-highlight-visible'),
58
+ };
59
+ });
60
+
61
+ await browser.close();
62
+
63
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
64
+ // Both theme branches now include brightness(1.15)
65
+ const sourceLightFilter = /'drop-shadow\(0 0 2px rgba\(16, 185, 129, 0\.6\)\) brightness\(1\.15\)'/.test(src);
66
+ const sourceCyberFilter = /'drop-shadow\(0 0 3px rgba\(52, 211, 153, 0\.6\)\) brightness\(1\.15\)'/.test(src);
67
+ const sourceAttr = /data-topo-hub-highlight-brightness=\{!reducedMotion && hoveredHub \? '1\.15' : '1'\}/.test(src);
68
+
69
+ const results = {
70
+ visible_true: rest?.visibleAttr === 'true',
71
+ rest_filter_none: rest?.filter === 'none',
72
+ rest_brightness_1: rest?.brightnessAttr === '1',
73
+ rest_glow_false: rest?.glowAttr === 'false',
74
+ rest_hovered_false: rest?.hoveredAttr === 'false',
75
+ transition_has_filter: /filter/.test(rest?.transitionProperty || ''),
76
+ source_light_filter: sourceLightFilter,
77
+ source_cyber_filter: sourceCyberFilter,
78
+ source_attr: sourceAttr,
79
+ };
80
+ const ok = Object.values(results).every(Boolean);
81
+ console.log(`${ok ? '✅' : '❌'} R574 hub-highlight stacked brightness (13th anchor):`,
82
+ JSON.stringify(results, null, 2),
83
+ `\n rest: ${JSON.stringify(rest)}`);
84
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,84 @@
1
+ /* Round 510 verification: hub-highlight transition list extends to
2
+ * include `fill 200ms ease-out` alongside existing `opacity 300ms`.
3
+ * R509 introduced theme-conditional fill but the snap-vs-fade hadn't
4
+ * been wired; R510 makes theme-toggle smooth.
5
+ *
6
+ * Verifies:
7
+ * 1. computed transition-property includes both 'fill' and 'opacity'
8
+ * 2. computed transition-duration shows 200ms and 300ms
9
+ * 3. source-side regex confirms the new spec
10
+ * 4. R509 theme-conditional fill still works (cross-theme differential)
11
+ */
12
+ import { chromium } from 'playwright';
13
+ import { readFileSync } from 'node:fs';
14
+
15
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
16
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
17
+
18
+ async function probe(theme) {
19
+ const browser = await chromium.launch({ headless: true });
20
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
21
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
22
+ await ctx.addInitScript((t) => {
23
+ try {
24
+ localStorage.setItem('anet-theme', t);
25
+ localStorage.setItem('anet-topo-layout', 'ring');
26
+ sessionStorage.setItem('anet_v3_auth', '1');
27
+ } catch {}
28
+ }, theme);
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
+ const mk = (alias, status) => ({
34
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
35
+ network_id: nid, project_dir: null,
36
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
37
+ });
38
+ await route.fulfill({ response: r, json: { ...b, sessions: [mk('a·1', 'idle')] } });
39
+ });
40
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
41
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
42
+ const page = await ctx.newPage();
43
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
44
+ await page.waitForSelector('[data-topo-hub-highlight]', { timeout: 15000 });
45
+ await page.waitForTimeout(1500);
46
+ const result = await page.evaluate(() => {
47
+ const circle = document.querySelector('[data-topo-hub-highlight]');
48
+ if (!circle) return null;
49
+ const cs = window.getComputedStyle(circle);
50
+ return {
51
+ fill_attr: circle.getAttribute('fill'),
52
+ transition_property: cs.transitionProperty,
53
+ transition_duration: cs.transitionDuration,
54
+ };
55
+ });
56
+ await browser.close();
57
+ return result;
58
+ }
59
+
60
+ const cyber = await probe('cyber');
61
+ const light = await probe('light');
62
+
63
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
64
+ const sourceWired = /transition: 'opacity 300ms ease-out, fill 200ms ease-out'/.test(src);
65
+
66
+ const tpHas = (s, prop) => new RegExp(`\\b${prop}\\b`, 'i').test(s || '');
67
+ const durHas = (s, ms) => (s || '').split(',').map((x) => x.trim()).includes(ms);
68
+
69
+ const results = {
70
+ cyber_tp_has_fill: cyber && tpHas(cyber.transition_property, 'fill'),
71
+ cyber_tp_has_opacity: cyber && tpHas(cyber.transition_property, 'opacity'),
72
+ cyber_dur_200ms: cyber && durHas(cyber.transition_duration, '0.2s'),
73
+ cyber_dur_300ms: cyber && durHas(cyber.transition_duration, '0.3s'),
74
+ light_tp_has_fill: light && tpHas(light.transition_property, 'fill'),
75
+ light_tp_has_opacity: light && tpHas(light.transition_property, 'opacity'),
76
+ cyber_fill_d1fae5: cyber && cyber.fill_attr === '#d1fae5',
77
+ light_fill_10b981: light && light.fill_attr === '#10b981',
78
+ source_wired: sourceWired,
79
+ };
80
+ const ok = Object.values(results).every(Boolean);
81
+ console.log(`${ok ? '✅' : '❌'} R510 hub-highlight fill transition:`, JSON.stringify(results),
82
+ '\n cyber:', JSON.stringify(cyber),
83
+ '\n light:', JSON.stringify(light));
84
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,99 @@
1
+ /* Round 532 verification: hub-highlight disc gains drop-shadow glow on
2
+ * hub-hover — drop-shadow visual-polish family 8th anchor.
3
+ *
4
+ * Test phases (workingCount === 0 so highlight is visible):
5
+ * 1. rest: data-topo-hub-highlight-glow='false', computed filter='none'
6
+ * 2. hover (page.mouse.move to highlight bbox per R527 banked path):
7
+ * glow='true', computed filter matches 'drop-shadow(...)' with
8
+ * cyber emerald-400 hue
9
+ * 3. transition list includes 'filter 200ms ease-out'
10
+ * 4. source-side regex confirms filter ternary + transition list wiring
11
+ */
12
+ import { chromium } from 'playwright';
13
+ import { readFileSync } from 'node:fs';
14
+
15
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
16
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
17
+
18
+ const browser = await chromium.launch({ headless: true });
19
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
20
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
21
+ await ctx.addInitScript(() => {
22
+ try {
23
+ localStorage.setItem('anet-theme', 'cyber');
24
+ localStorage.setItem('anet-topo-layout', 'ring');
25
+ sessionStorage.setItem('anet_v3_auth', '1');
26
+ } catch {}
27
+ });
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: 'idle', 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: [
38
+ mk('a·1'), mk('a·2'), mk('a·3'),
39
+ ] } });
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
+ const page = await ctx.newPage();
44
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
45
+ await page.waitForSelector('[data-topo-hub-highlight]', { timeout: 15000 });
46
+ await page.waitForTimeout(800);
47
+
48
+ const restRead = async () => page.evaluate(() => {
49
+ const el = document.querySelector('[data-topo-hub-highlight]');
50
+ if (!el) return null;
51
+ const cs = getComputedStyle(el);
52
+ return {
53
+ attrGlow: el.getAttribute('data-topo-hub-highlight-glow'),
54
+ attrHovered: el.getAttribute('data-topo-hub-highlight-hovered'),
55
+ filter: cs.filter,
56
+ transition: cs.transition,
57
+ };
58
+ });
59
+
60
+ // Phase 1: rest
61
+ const rest = await restRead();
62
+
63
+ // Phase 2: hover hub via mouse.move to highlight bbox center
64
+ const bbox = await page.locator('[data-topo-hub-highlight]').first().boundingBox();
65
+ if (bbox) {
66
+ await page.mouse.move(bbox.x + bbox.width / 2, bbox.y + bbox.height / 2);
67
+ }
68
+ await page.waitForTimeout(400);
69
+ const hover = await restRead();
70
+
71
+ await browser.close();
72
+
73
+ // Source regex
74
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
75
+ const sourceFilterTernary =
76
+ /filter: !reducedMotion && hoveredHub\s+\? \(isLight\s+\? 'drop-shadow\(0 0 2px rgba\(16, 185, 129, 0\.6\)\)'\s+: 'drop-shadow\(0 0 3px rgba\(52, 211, 153, 0\.6\)\)'\)\s+: undefined,/.test(src);
77
+ const sourceAttrWired =
78
+ /data-topo-hub-highlight-glow=\{!reducedMotion && hoveredHub \? 'true' : 'false'\}/.test(src);
79
+ const sourceTransitionExt =
80
+ /transition: 'opacity 300ms ease-out, fill 200ms ease-out, r 200ms ease-out, filter 200ms ease-out'/.test(src);
81
+
82
+ const results = {
83
+ rest_glow_false: rest?.attrGlow === 'false',
84
+ rest_hovered_false: rest?.attrHovered === 'false',
85
+ rest_filter_none: rest?.filter === 'none' || rest?.filter === '',
86
+ rest_transition_has_filter: /\bfilter\b/.test(rest?.transition || ''),
87
+ hover_glow_true: hover?.attrGlow === 'true',
88
+ hover_hovered_true: hover?.attrHovered === 'true',
89
+ hover_filter_drop_shadow: /drop-shadow\(.+52,?\s*211,?\s*153/.test(hover?.filter || ''), // cyber emerald-400
90
+ source_filter_ternary: sourceFilterTernary,
91
+ source_attr_wired: sourceAttrWired,
92
+ source_transition_ext: sourceTransitionExt,
93
+ };
94
+ const ok = Object.values(results).every(Boolean);
95
+ console.log(`${ok ? '✅' : '❌'} R532 hub-highlight hover-glow:`,
96
+ JSON.stringify(results, null, 2),
97
+ '\n rest:', JSON.stringify(rest),
98
+ '\n hover:', JSON.stringify(hover));
99
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,78 @@
1
+ /* Round 652 — hub-highlight disc drop-shadow gains a SECOND outer
2
+ * layer. 11th anchor in multi-layer halo family (3rd hub-cluster
3
+ * anchor after R650 digit + R651 halo). The MIDDLE disc of the
4
+ * hub focal cluster now matches the outermost halo + innermost
5
+ * digit halo grammar.
6
+ *
7
+ * Test phases:
8
+ * 1. mock 2 nodes → hub-highlight renders
9
+ * 2. rest (no hub hover): halo-layers='0', no filter
10
+ * 3. source: both light + cyber filter branches stack 2 drop-
11
+ * shadows with emerald tint at descending alphas (0.6 → 0.3)
12
+ */
13
+ import { chromium } from 'playwright';
14
+ import { readFileSync } from 'node:fs';
15
+
16
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
17
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
18
+
19
+ const browser = await chromium.launch({ headless: true });
20
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
21
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
22
+ await ctx.addInitScript(() => {
23
+ try {
24
+ localStorage.setItem('anet-theme', 'cyber');
25
+ localStorage.setItem('anet-topo-layout', 'ring');
26
+ sessionStorage.setItem('anet_v3_auth', '1');
27
+ } 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
+ const mk = (alias) => ({
34
+ alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
35
+ network_id: nid, project_dir: null,
36
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
37
+ });
38
+ await route.fulfill({ response: r, json: { ...b, sessions: [mk('a·1'), mk('a·2')] } });
39
+ });
40
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
41
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
42
+ const page = await ctx.newPage();
43
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
44
+ await page.waitForSelector('[data-topo-hub-highlight-halo-layers]', { timeout: 15000, state: 'attached' });
45
+ await page.waitForTimeout(500);
46
+
47
+ const rest = await page.evaluate(() => {
48
+ const el = document.querySelector('[data-topo-hub-highlight-halo-layers]');
49
+ if (!el) return null;
50
+ const cs = getComputedStyle(el);
51
+ return {
52
+ layers: el.getAttribute('data-topo-hub-highlight-halo-layers'),
53
+ glow: el.getAttribute('data-topo-hub-highlight-glow'),
54
+ filter: cs.filter,
55
+ };
56
+ });
57
+
58
+ await browser.close();
59
+
60
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
61
+ const sourceLight = /'drop-shadow\(0 0 2px rgba\(16, 185, 129, 0\.6\)\) drop-shadow\(0 0 4px rgba\(16, 185, 129, 0\.3\)\) brightness\(1\.15\)'/.test(src);
62
+ const sourceCyber = /'drop-shadow\(0 0 3px rgba\(52, 211, 153, 0\.6\)\) drop-shadow\(0 0 6px rgba\(52, 211, 153, 0\.3\)\) brightness\(1\.15\)'/.test(src);
63
+ const sourceLayersAttr = /data-topo-hub-highlight-halo-layers=\{!reducedMotion && hoveredHub \? '2' : '0'\}/.test(src);
64
+
65
+ const results = {
66
+ rest_present: !!rest,
67
+ rest_layers_0: rest?.layers === '0',
68
+ rest_glow_false: rest?.glow === 'false',
69
+ rest_filter_no_drop_shadow: !/drop-shadow/.test(rest?.filter || ''),
70
+ source_light_filter: sourceLight,
71
+ source_cyber_filter: sourceCyber,
72
+ source_layers_attr: sourceLayersAttr,
73
+ };
74
+ const ok = Object.values(results).every(Boolean);
75
+ console.log(`${ok ? '✅' : '❌'} R652 hub-highlight disc multi-layer halo (3rd hub-cluster anchor):`,
76
+ JSON.stringify(results, null, 2),
77
+ `\n rest: ${JSON.stringify(rest)}`);
78
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,112 @@
1
+ /* Round 529 verification: hub-highlight gains geometric amplify
2
+ * r 5.5 → 6 on hub-hover — focal-amplify family 3rd anchor.
3
+ *
4
+ * Test phases (workingCount === 0 so highlight renders visibly):
5
+ * 1. rest: data-topo-hub-highlight-radius = 5.5,
6
+ * computed r ≈ 5.5px (CSS property),
7
+ * hovered attr = 'false'
8
+ * 2. hover hub (via page.mouse.move to highlight bbox):
9
+ * data-topo-hub-highlight-radius = 6,
10
+ * computed r ≈ 6px,
11
+ * hovered attr = 'true'
12
+ * 3. transition list includes `r 200ms ease-out`
13
+ * 4. source-side regex confirms highlightR ternary + style.r wiring
14
+ */
15
+ import { chromium } from 'playwright';
16
+ import { readFileSync } from 'node:fs';
17
+
18
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
19
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
20
+
21
+ const browser = await chromium.launch({ headless: true });
22
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
23
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
24
+ await ctx.addInitScript(() => {
25
+ try {
26
+ localStorage.setItem('anet-theme', 'cyber');
27
+ localStorage.setItem('anet-topo-layout', 'ring');
28
+ sessionStorage.setItem('anet_v3_auth', '1');
29
+ } 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) => ({
36
+ alias, status: 'idle', 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
+ // workingCount=0 (all idle) so highlight is visible
41
+ await route.fulfill({ response: r, json: { ...b, sessions: [
42
+ mk('alpha·1'), mk('alpha·2'), mk('alpha·3'),
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
+ const page = await ctx.newPage();
48
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
49
+ await page.waitForSelector('[data-topo-hub-highlight]', { timeout: 15000 });
50
+ await page.waitForTimeout(800);
51
+
52
+ const restRead = async () => page.evaluate(() => {
53
+ const el = document.querySelector('[data-topo-hub-highlight]');
54
+ if (!el) return null;
55
+ const cs = getComputedStyle(el);
56
+ return {
57
+ attrR: el.getAttribute('data-topo-hub-highlight-radius'),
58
+ attrHovered: el.getAttribute('data-topo-hub-highlight-hovered'),
59
+ attrVisible: el.getAttribute('data-topo-hub-highlight-visible'),
60
+ cssR: cs.r, // computed CSS `r` value
61
+ transition: cs.transition,
62
+ };
63
+ });
64
+
65
+ // Phase 1: rest
66
+ const rest = await restRead();
67
+
68
+ // Phase 2: hover hub (via mouse.move to highlight bbox center —
69
+ // banked path from R527 hub-digit-ls test)
70
+ const bbox = await page.locator('[data-topo-hub-highlight]').first().boundingBox();
71
+ if (bbox) {
72
+ await page.mouse.move(bbox.x + bbox.width / 2, bbox.y + bbox.height / 2);
73
+ }
74
+ await page.waitForTimeout(400);
75
+ const hover = await restRead();
76
+
77
+ await browser.close();
78
+
79
+ // Source regex
80
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
81
+ const sourceHighlightRTernary =
82
+ /const highlightR = !reducedMotion && hoveredHub \? 6 : 5\.5;/.test(src);
83
+ const sourceStyleR =
84
+ /r: `\$\{highlightR\}px`,/.test(src);
85
+ const sourceTransitionR =
86
+ /transition: 'opacity 300ms ease-out, fill 200ms ease-out, r 200ms ease-out'/.test(src);
87
+ const sourceAttrDynamic =
88
+ /data-topo-hub-highlight-radius=\{highlightR\}/.test(src);
89
+
90
+ const approxEq = (a, b, tol = 0.1) => Math.abs(a - b) < tol;
91
+
92
+ const results = {
93
+ rest_attr_55: rest?.attrR === '5.5',
94
+ rest_hovered_false: rest?.attrHovered === 'false',
95
+ rest_visible_true: rest?.attrVisible === 'true', // workingCount=0
96
+ rest_css_r_55: approxEq(parseFloat(rest?.cssR || '0'), 5.5),
97
+ rest_transition_has_r: /\br\b/.test(rest?.transition || '') ||
98
+ /\b(r 200ms|r 0\.2s)/.test(rest?.transition || ''),
99
+ hover_attr_6: hover?.attrR === '6',
100
+ hover_hovered_true: hover?.attrHovered === 'true',
101
+ hover_css_r_6: approxEq(parseFloat(hover?.cssR || '0'), 6),
102
+ source_highlightR_ternary: sourceHighlightRTernary,
103
+ source_style_r: sourceStyleR,
104
+ source_transition_r: sourceTransitionR,
105
+ source_attr_dynamic: sourceAttrDynamic,
106
+ };
107
+ const ok = Object.values(results).every(Boolean);
108
+ console.log(`${ok ? '✅' : '❌'} R529 hub-highlight hover r:`,
109
+ JSON.stringify(results, null, 2),
110
+ '\n rest:', JSON.stringify(rest),
111
+ '\n hover:', JSON.stringify(hover));
112
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,144 @@
1
+ /* Round 508 verification: hub-highlight circle joins the focal-recede
2
+ * pattern (2nd anchor after R507's hub digit). When canvas attention
3
+ * is elsewhere, the hub focal cluster (digit + highlight) recedes as
4
+ * a unit:
5
+ * - hub-digit opacity: 1 → 0.85 (R507)
6
+ * - hub-highlight opacity: 0.95 → 0.81 (R508, 0.95 × 0.85)
7
+ * - hub-highlight SMIL breath: active → halted (R508)
8
+ *
9
+ * Also verifies R497 idle-breath still works in the un-receded state
10
+ * — refactor regression check.
11
+ */
12
+ import { chromium } from 'playwright';
13
+ import { readFileSync } from 'node:fs';
14
+
15
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
16
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
17
+
18
+ const browser = await chromium.launch({ headless: true });
19
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
20
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
21
+ await ctx.addInitScript(() => {
22
+ try {
23
+ localStorage.setItem('anet-theme', 'cyber');
24
+ localStorage.setItem('anet-topo-layout', 'ring');
25
+ sessionStorage.setItem('anet_v3_auth', '1');
26
+ } catch {}
27
+ });
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, status) => ({
33
+ alias, status, 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
+ // workingCount === 0 (all idle) so hub-highlight is visible
38
+ await route.fulfill({ response: r, json: { ...b, sessions: [
39
+ mk('alpha·a1', 'idle'),
40
+ mk('alpha·a2', 'idle'),
41
+ ] } });
42
+ });
43
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
44
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
45
+ const page = await ctx.newPage();
46
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
47
+ await page.waitForSelector('[data-topo-hub-highlight]', { timeout: 15000 });
48
+ await page.waitForTimeout(1500);
49
+
50
+ // Phase 1: rest state — idle fleet, no external hover
51
+ // expected: opacity 0.95, breath='true', recede='false', animate present
52
+ const rest = await page.evaluate(() => {
53
+ const circle = document.querySelector('[data-topo-hub-highlight]');
54
+ if (!circle) return null;
55
+ return {
56
+ opacity_attr: circle.getAttribute('data-topo-hub-highlight-opacity'),
57
+ breath_attr: circle.getAttribute('data-topo-hub-highlight-breath'),
58
+ recede_attr: circle.getAttribute('data-topo-hub-highlight-recede'),
59
+ has_animate: !!circle.querySelector('animate[attributeName="opacity"]'),
60
+ };
61
+ });
62
+
63
+ // Phase 2: external hover via R488 synthetic dispatch on a node
64
+ // expected: opacity 0.81 (0.95 * 0.85), breath='false', recede='true',
65
+ // animate ABSENT (halted)
66
+ await page.evaluate(() => {
67
+ const g = document.querySelector('g[data-node]');
68
+ if (!g) return;
69
+ const target = g.querySelector('circle, image, rect') || g;
70
+ ['pointerenter', 'pointerover', 'mouseenter', 'mouseover'].forEach((t) => {
71
+ target.dispatchEvent(new Event(t, { bubbles: true, cancelable: true }));
72
+ });
73
+ });
74
+ await page.waitForTimeout(500);
75
+ const hover = await page.evaluate(() => {
76
+ const circle = document.querySelector('[data-topo-hub-highlight]');
77
+ if (!circle) return null;
78
+ return {
79
+ opacity_attr: circle.getAttribute('data-topo-hub-highlight-opacity'),
80
+ breath_attr: circle.getAttribute('data-topo-hub-highlight-breath'),
81
+ recede_attr: circle.getAttribute('data-topo-hub-highlight-recede'),
82
+ has_animate: !!circle.querySelector('animate[attributeName="opacity"]'),
83
+ };
84
+ });
85
+
86
+ // Phase 3: release hover
87
+ await page.evaluate(() => {
88
+ const g = document.querySelector('g[data-node]');
89
+ if (!g) return;
90
+ const target = g.querySelector('circle, image, rect') || g;
91
+ ['pointerleave', 'pointerout', 'mouseleave', 'mouseout'].forEach((t) => {
92
+ target.dispatchEvent(new Event(t, { bubbles: true, cancelable: true }));
93
+ });
94
+ });
95
+ await page.waitForTimeout(500);
96
+ const release = await page.evaluate(() => {
97
+ const circle = document.querySelector('[data-topo-hub-highlight]');
98
+ if (!circle) return null;
99
+ return {
100
+ opacity_attr: circle.getAttribute('data-topo-hub-highlight-opacity'),
101
+ breath_attr: circle.getAttribute('data-topo-hub-highlight-breath'),
102
+ recede_attr: circle.getAttribute('data-topo-hub-highlight-recede'),
103
+ has_animate: !!circle.querySelector('animate[attributeName="opacity"]'),
104
+ };
105
+ });
106
+
107
+ await browser.close();
108
+
109
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
110
+ const sourceIIFE = /const hubRecede = !!\(\(hoveredAlias \|\| hoveredEdgeKey \|\| hoveredGroupLabel/.test(src);
111
+ const sourceOpacityMultiplier = /const resolvedOpacity = hubRecede \? baseOpacity \* 0\.85 : baseOpacity;/.test(src);
112
+ // R511 extension: breathActive now also halts on hoveredHub (hub-hover
113
+ // amplify state added to the 3-state opacity ladder). Regex updated to
114
+ // match the extended formula. Runtime DOM contract (rest = breath true)
115
+ // unchanged.
116
+ const sourceBreathGate = /const breathActive = !reducedMotion && workingCount === 0 && !hubRecede && !hoveredHub;/.test(src);
117
+ const sourceAnimateGate = /\{breathActive && \(\s*<animate/.test(src);
118
+
119
+ const approxEq = (a, b) => Math.abs(parseFloat(a) - b) < 0.01;
120
+
121
+ const results = {
122
+ rest_opacity_095: rest && approxEq(rest.opacity_attr, 0.95),
123
+ rest_breath_true: rest && rest.breath_attr === 'true',
124
+ rest_recede_false: rest && rest.recede_attr === 'false',
125
+ rest_has_animate: rest && rest.has_animate,
126
+ hover_opacity_081: hover && approxEq(hover.opacity_attr, 0.81),
127
+ hover_breath_false: hover && hover.breath_attr === 'false',
128
+ hover_recede_true: hover && hover.recede_attr === 'true',
129
+ hover_no_animate: hover && !hover.has_animate,
130
+ release_opacity_095: release && approxEq(release.opacity_attr, 0.95),
131
+ release_breath_true: release && release.breath_attr === 'true',
132
+ release_recede_false:release && release.recede_attr === 'false',
133
+ release_has_animate: release && release.has_animate,
134
+ source_iife_wired: sourceIIFE,
135
+ source_multiplier_wired: sourceOpacityMultiplier,
136
+ source_breath_gate_wired: sourceBreathGate,
137
+ source_animate_gate_wired:sourceAnimateGate,
138
+ };
139
+ const ok = Object.values(results).every(Boolean);
140
+ console.log(`${ok ? '✅' : '❌'} R508 hub-highlight recede:`, JSON.stringify(results),
141
+ '\n rest:', JSON.stringify(rest),
142
+ '\n hover:', JSON.stringify(hover),
143
+ '\n release:', JSON.stringify(release));
144
+ process.exit(ok ? 0 : 1);