heyiam 0.2.29 → 0.3.1

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 (186) hide show
  1. package/README.md +45 -0
  2. package/dist/auth.js +29 -3
  3. package/dist/config.js +10 -1
  4. package/dist/db.js +0 -1
  5. package/dist/export.js +124 -27
  6. package/dist/format-utils.js +5 -0
  7. package/dist/github.js +381 -0
  8. package/dist/index.js +168 -0
  9. package/dist/mount.js +300 -102
  10. package/dist/parsers/claude.js +2 -28
  11. package/dist/parsers/codex.js +2 -26
  12. package/dist/parsers/cursor.js +2 -26
  13. package/dist/parsers/duration.js +35 -0
  14. package/dist/parsers/gemini.js +2 -20
  15. package/dist/parsers/index.js +22 -3
  16. package/dist/parsers/types.js +0 -1
  17. package/dist/public/assets/index-Coilyhtr.css +1 -0
  18. package/dist/public/assets/index-D0noVMFu.js +44 -0
  19. package/dist/public/index.html +2 -2
  20. package/dist/redact.js +4 -104
  21. package/dist/render/build-render-data.js +9 -2
  22. package/dist/render/index.js +32 -5
  23. package/dist/render/liquid.js +147 -7
  24. package/dist/render/mock-data.js +303 -0
  25. package/dist/render/templates/aurora/portfolio.liquid +192 -0
  26. package/dist/render/templates/aurora/project.liquid +260 -0
  27. package/dist/render/templates/aurora/session.liquid +223 -0
  28. package/dist/render/templates/aurora/styles.css +1184 -0
  29. package/dist/render/templates/bauhaus/portfolio.liquid +169 -0
  30. package/dist/render/templates/bauhaus/project.liquid +300 -0
  31. package/dist/render/templates/bauhaus/session.liquid +333 -0
  32. package/dist/render/templates/bauhaus/styles.css +1645 -0
  33. package/dist/render/templates/blueprint/portfolio.liquid +153 -0
  34. package/dist/render/templates/blueprint/project.liquid +286 -0
  35. package/dist/render/templates/blueprint/session.liquid +248 -0
  36. package/dist/render/templates/blueprint/styles.css +1289 -0
  37. package/dist/render/templates/canvas/portfolio.liquid +203 -0
  38. package/dist/render/templates/canvas/project.liquid +235 -0
  39. package/dist/render/templates/canvas/session.liquid +223 -0
  40. package/dist/render/templates/canvas/styles.css +1440 -0
  41. package/dist/render/templates/carbon/portfolio.liquid +160 -0
  42. package/dist/render/templates/carbon/project.liquid +249 -0
  43. package/dist/render/templates/carbon/session.liquid +190 -0
  44. package/dist/render/templates/carbon/styles.css +1097 -0
  45. package/dist/render/templates/chalk/portfolio.liquid +189 -0
  46. package/dist/render/templates/chalk/project.liquid +245 -0
  47. package/dist/render/templates/chalk/session.liquid +215 -0
  48. package/dist/render/templates/chalk/styles.css +1161 -0
  49. package/dist/render/templates/circuit/portfolio.liquid +152 -0
  50. package/dist/render/templates/circuit/project.liquid +247 -0
  51. package/dist/render/templates/circuit/session.liquid +205 -0
  52. package/dist/render/templates/circuit/styles.css +1409 -0
  53. package/dist/render/templates/cosmos/portfolio.liquid +222 -0
  54. package/dist/render/templates/cosmos/project.liquid +327 -0
  55. package/dist/render/templates/cosmos/session.liquid +239 -0
  56. package/dist/render/templates/cosmos/styles.css +1157 -0
  57. package/dist/render/templates/daylight/portfolio.liquid +207 -0
  58. package/dist/render/templates/daylight/project.liquid +229 -0
  59. package/dist/render/templates/daylight/session.liquid +219 -0
  60. package/dist/render/templates/daylight/styles.css +1315 -0
  61. package/dist/render/templates/editorial/portfolio.liquid +110 -0
  62. package/dist/render/templates/editorial/project.liquid +202 -0
  63. package/dist/render/templates/editorial/session.liquid +171 -0
  64. package/dist/render/templates/editorial/styles.css +826 -0
  65. package/dist/render/templates/ember/portfolio.liquid +306 -0
  66. package/dist/render/templates/ember/project.liquid +232 -0
  67. package/dist/render/templates/ember/session.liquid +202 -0
  68. package/dist/render/templates/ember/styles.css +1289 -0
  69. package/dist/render/templates/glacier/portfolio.liquid +261 -0
  70. package/dist/render/templates/glacier/project.liquid +288 -0
  71. package/dist/render/templates/glacier/session.liquid +217 -0
  72. package/dist/render/templates/glacier/styles.css +1204 -0
  73. package/dist/render/templates/grid/portfolio.liquid +255 -0
  74. package/dist/render/templates/grid/project.liquid +306 -0
  75. package/dist/render/templates/grid/session.liquid +260 -0
  76. package/dist/render/templates/grid/styles.css +1445 -0
  77. package/dist/render/templates/kinetic/portfolio.liquid +158 -0
  78. package/dist/render/templates/kinetic/project.liquid +242 -0
  79. package/dist/render/templates/kinetic/session.liquid +228 -0
  80. package/dist/render/templates/kinetic/styles.css +948 -0
  81. package/dist/render/templates/meridian/portfolio.liquid +243 -0
  82. package/dist/render/templates/meridian/project.liquid +376 -0
  83. package/dist/render/templates/meridian/session.liquid +298 -0
  84. package/dist/render/templates/meridian/styles.css +1375 -0
  85. package/dist/render/templates/minimal/portfolio.liquid +71 -0
  86. package/dist/render/templates/minimal/project.liquid +154 -0
  87. package/dist/render/templates/minimal/session.liquid +140 -0
  88. package/dist/render/templates/minimal/styles.css +529 -0
  89. package/dist/render/templates/mono/portfolio.liquid +281 -0
  90. package/dist/render/templates/mono/project.liquid +275 -0
  91. package/dist/render/templates/mono/session.liquid +276 -0
  92. package/dist/render/templates/mono/styles.css +1022 -0
  93. package/dist/render/templates/neon/portfolio.liquid +207 -0
  94. package/dist/render/templates/neon/project.liquid +225 -0
  95. package/dist/render/templates/neon/session.liquid +195 -0
  96. package/dist/render/templates/neon/styles.css +1271 -0
  97. package/dist/render/templates/noir/portfolio.liquid +137 -0
  98. package/dist/render/templates/noir/project.liquid +220 -0
  99. package/dist/render/templates/noir/session.liquid +241 -0
  100. package/dist/render/templates/noir/styles.css +1229 -0
  101. package/dist/render/templates/obsidian/portfolio.liquid +247 -0
  102. package/dist/render/templates/obsidian/project.liquid +280 -0
  103. package/dist/render/templates/obsidian/session.liquid +241 -0
  104. package/dist/render/templates/obsidian/styles.css +1407 -0
  105. package/dist/render/templates/paper/portfolio.liquid +257 -0
  106. package/dist/render/templates/paper/project.liquid +235 -0
  107. package/dist/render/templates/paper/session.liquid +271 -0
  108. package/dist/render/templates/paper/styles.css +1513 -0
  109. package/dist/render/templates/parallax/portfolio.liquid +295 -0
  110. package/dist/render/templates/parallax/project.liquid +275 -0
  111. package/dist/render/templates/parallax/session.liquid +295 -0
  112. package/dist/render/templates/parallax/styles.css +1880 -0
  113. package/dist/render/templates/parchment/portfolio.liquid +280 -0
  114. package/dist/render/templates/parchment/project.liquid +289 -0
  115. package/dist/render/templates/parchment/session.liquid +346 -0
  116. package/dist/render/templates/parchment/styles.css +1401 -0
  117. package/dist/render/templates/partials/_beats.liquid +16 -0
  118. package/dist/render/templates/partials/_breadcrumb.liquid +9 -0
  119. package/dist/render/templates/partials/_footer.liquid +7 -0
  120. package/dist/render/templates/partials/_growth-chart.liquid +7 -0
  121. package/dist/render/templates/partials/_key-decisions.liquid +20 -0
  122. package/dist/render/templates/partials/_links.liquid +16 -0
  123. package/dist/render/templates/partials/_narrative.liquid +8 -0
  124. package/dist/render/templates/partials/_phases.liquid +20 -0
  125. package/dist/render/templates/partials/_portfolio-header.liquid +20 -0
  126. package/dist/render/templates/partials/_portfolio-projects.liquid +16 -0
  127. package/dist/render/templates/partials/_portfolio-stats.liquid +19 -0
  128. package/dist/render/templates/partials/_qa.liquid +13 -0
  129. package/dist/render/templates/partials/_screenshot.liquid +15 -0
  130. package/dist/render/templates/partials/_session-cards.liquid +30 -0
  131. package/dist/render/templates/partials/_session-header.liquid +39 -0
  132. package/dist/render/templates/partials/_session-sidebar.liquid +30 -0
  133. package/dist/render/templates/partials/_skills.liquid +12 -0
  134. package/dist/render/templates/partials/_source-breakdown.liquid +22 -0
  135. package/dist/render/templates/partials/_stats.liquid +38 -0
  136. package/dist/render/templates/partials/_work-timeline.liquid +7 -0
  137. package/dist/render/templates/project.liquid +7 -4
  138. package/dist/render/templates/radar/portfolio.liquid +223 -0
  139. package/dist/render/templates/radar/project.liquid +278 -0
  140. package/dist/render/templates/radar/session.liquid +300 -0
  141. package/dist/render/templates/radar/styles.css +1055 -0
  142. package/dist/render/templates/showcase/portfolio.liquid +221 -0
  143. package/dist/render/templates/showcase/project.liquid +237 -0
  144. package/dist/render/templates/showcase/session.liquid +210 -0
  145. package/dist/render/templates/showcase/styles.css +1284 -0
  146. package/dist/render/templates/signal/portfolio.liquid +217 -0
  147. package/dist/render/templates/signal/project.liquid +278 -0
  148. package/dist/render/templates/signal/session.liquid +282 -0
  149. package/dist/render/templates/signal/styles.css +1401 -0
  150. package/dist/render/templates/strata/portfolio.liquid +180 -0
  151. package/dist/render/templates/strata/project.liquid +282 -0
  152. package/dist/render/templates/strata/session.liquid +261 -0
  153. package/dist/render/templates/strata/styles.css +1354 -0
  154. package/dist/render/templates/styles.css +1190 -0
  155. package/dist/render/templates/terminal/portfolio.liquid +102 -0
  156. package/dist/render/templates/terminal/project.liquid +161 -0
  157. package/dist/render/templates/terminal/session.liquid +145 -0
  158. package/dist/render/templates/terminal/styles.css +497 -0
  159. package/dist/render/templates/verdant/portfolio.liquid +321 -0
  160. package/dist/render/templates/verdant/project.liquid +309 -0
  161. package/dist/render/templates/verdant/session.liquid +237 -0
  162. package/dist/render/templates/verdant/styles.css +1261 -0
  163. package/dist/render/templates/zen/portfolio.liquid +124 -0
  164. package/dist/render/templates/zen/project.liquid +187 -0
  165. package/dist/render/templates/zen/session.liquid +203 -0
  166. package/dist/render/templates/zen/styles.css +1211 -0
  167. package/dist/render/templates.js +90 -0
  168. package/dist/routes/auth.js +7 -3
  169. package/dist/routes/context.js +17 -10
  170. package/dist/routes/delete.js +195 -0
  171. package/dist/routes/enhance.js +57 -40
  172. package/dist/routes/export.js +14 -4
  173. package/dist/routes/github.js +254 -0
  174. package/dist/routes/index.js +2 -0
  175. package/dist/routes/portfolio-render-data.js +160 -0
  176. package/dist/routes/preview.js +555 -108
  177. package/dist/routes/projects.js +61 -24
  178. package/dist/routes/publish.js +320 -31
  179. package/dist/routes/settings.js +194 -1
  180. package/dist/routes/sse.js +9 -0
  181. package/dist/search.js +6 -0
  182. package/dist/server.js +11 -3
  183. package/dist/settings.js +112 -9
  184. package/package.json +3 -4
  185. package/dist/public/assets/index-CC9G8EF1.js +0 -21
  186. package/dist/public/assets/index-Dalqz2mC.css +0 -1
@@ -0,0 +1,300 @@
1
+ <div class="heyiam-session radar" data-render-version="2" data-template="radar">
2
+
3
+ {%- comment -%} Breadcrumb {%- endcomment -%}
4
+ <nav class="radar-breadcrumb" aria-label="Breadcrumb">
5
+ <a href="/{{ user.username }}">{{ user.username }}</a>
6
+ <span class="radar-breadcrumb__sep">/</span>
7
+ <a href="/{{ user.username }}/{{ projectSlug }}">{{ projectSlug }}</a>
8
+ <span class="radar-breadcrumb__sep">/</span>
9
+ <span>{{ session.slug }}</span>
10
+ </nav>
11
+
12
+ {%- comment -%} Title {%- endcomment -%}
13
+ <section class="radar-section" aria-label="Session title">
14
+ <h1 class="radar-title">{{ session.title }}</h1>
15
+ </section>
16
+
17
+ {%- comment -%} Session Meta {%- endcomment -%}
18
+ <section class="radar-section" aria-label="Session metadata" id="stats-section">
19
+ <div class="radar-label">Session Data</div>
20
+ <div class="stat-grid-inline">
21
+ {% if session.recordedAt %}
22
+ <div class="stat-card-sm">
23
+ <div class="stat-card-sm__label">Date</div>
24
+ <div class="stat-card-sm__value" style="font-size: 0.875rem;">{{ session.recordedAt | formatDateShort }}</div>
25
+ </div>
26
+ {% endif %}
27
+ {% if session.sourceTool %}
28
+ <div class="stat-card-sm">
29
+ <div class="stat-card-sm__label">Source</div>
30
+ <div class="stat-card-sm__value" style="font-size: 0.875rem;">{{ session.sourceTool }}</div>
31
+ </div>
32
+ {% endif %}
33
+ {% if session.durationMinutes %}
34
+ <div class="stat-card-sm">
35
+ <div class="stat-card-sm__label">Duration</div>
36
+ <div class="stat-card-sm__value">{{ session.durationMinutes | formatDuration }}</div>
37
+ </div>
38
+ {% endif %}
39
+ {% if session.wallClockMinutes %}
40
+ <div class="stat-card-sm">
41
+ <div class="stat-card-sm__label">Wall Clock</div>
42
+ <div class="stat-card-sm__value">{{ session.wallClockMinutes | formatDuration }}</div>
43
+ </div>
44
+ {% endif %}
45
+ {% if session.turns %}
46
+ <div class="stat-card-sm">
47
+ <div class="stat-card-sm__label">Turns</div>
48
+ <div class="stat-card-sm__value">{{ session.turns }}</div>
49
+ </div>
50
+ {% endif %}
51
+ {% if session.locChanged %}
52
+ <div class="stat-card-sm">
53
+ <div class="stat-card-sm__label">LOC</div>
54
+ <div class="stat-card-sm__value">{{ session.locChanged | formatLoc }}</div>
55
+ </div>
56
+ {% endif %}
57
+ {% if session.filesChanged %}
58
+ <div class="stat-card-sm">
59
+ <div class="stat-card-sm__label">Files</div>
60
+ <div class="stat-card-sm__value">{{ session.filesChanged }}</div>
61
+ </div>
62
+ {% endif %}
63
+ </div>
64
+ </section>
65
+
66
+ {%- comment -%} Dev Take {%- endcomment -%}
67
+ {% if session.devTake != blank %}
68
+ <section class="radar-section" aria-label="Developer commentary">
69
+ <div class="radar-label">Dev Take</div>
70
+ <div class="dev-take">
71
+ <p>&ldquo;{{ session.devTake }}&rdquo;</p>
72
+ </div>
73
+ </section>
74
+ {% endif %}
75
+
76
+ {%- comment -%} Main Content + Sidebar {%- endcomment -%}
77
+ <section class="radar-section" aria-label="Session details">
78
+ <div class="session-layout">
79
+
80
+ {%- comment -%} Main Column {%- endcomment -%}
81
+ <div>
82
+ {%- comment -%} Execution Path / Beats {%- endcomment -%}
83
+ {% if session.beats.size > 0 %}
84
+ <div class="radar-label">Execution Path</div>
85
+ <ol class="beat-list" role="list">
86
+ <div class="beat-list__line"></div>
87
+ {% for beat in session.beats %}
88
+ <li class="beat-item">
89
+ <div class="beat-item__dot"></div>
90
+ <div class="beat-item__number">Beat {{ beat.stepNumber }}</div>
91
+ <div class="beat-item__title">{{ beat.title }}</div>
92
+ {% if beat.body != blank %}
93
+ <div class="beat-item__desc">{{ beat.body }}</div>
94
+ {% endif %}
95
+ </li>
96
+ {% endfor %}
97
+ </ol>
98
+ {% endif %}
99
+
100
+ {%- comment -%} Narrative {%- endcomment -%}
101
+ {% if session.narrative != blank %}
102
+ <div style="margin-top: 2rem;">
103
+ <div class="radar-label">Narrative</div>
104
+ <div class="radar-narrative">
105
+ {{ session.narrative }}
106
+ </div>
107
+ </div>
108
+ {% endif %}
109
+
110
+ {%- comment -%} Highlights {%- endcomment -%}
111
+ {% if session.highlights.size > 0 %}
112
+ <div style="margin-top: 2rem;">
113
+ <div class="radar-label">Highlights</div>
114
+ <ul class="decision-list">
115
+ {% for h in session.highlights %}
116
+ <li class="decision-item">
117
+ <div class="decision-bullet">
118
+ <div>
119
+ <div class="decision-desc">{{ h }}</div>
120
+ </div>
121
+ </div>
122
+ </li>
123
+ {% endfor %}
124
+ </ul>
125
+ </div>
126
+ {% endif %}
127
+ </div>
128
+
129
+ {%- comment -%} Sidebar {%- endcomment -%}
130
+ <aside aria-label="Session sidebar">
131
+
132
+ {%- comment -%} Tools Used {%- endcomment -%}
133
+ {% if session.toolBreakdown.size > 0 %}
134
+ <div class="sidebar-card">
135
+ <div class="radar-label">Tools Used</div>
136
+ <table class="tools-table">
137
+ <thead>
138
+ <tr><th>Tool</th><th style="text-align: right;">Count</th></tr>
139
+ </thead>
140
+ <tbody>
141
+ {% for t in session.toolBreakdown %}
142
+ <tr><td>{{ t.tool }}</td><td>{{ t.count }}</td></tr>
143
+ {% endfor %}
144
+ </tbody>
145
+ </table>
146
+ </div>
147
+ {% endif %}
148
+
149
+ {%- comment -%} Files Changed {%- endcomment -%}
150
+ {% if session.topFiles.size > 0 %}
151
+ <div class="sidebar-card">
152
+ <div class="radar-label">Files Changed</div>
153
+ <table class="files-table">
154
+ <thead>
155
+ <tr><th>File</th><th style="text-align: right;">+/-</th></tr>
156
+ </thead>
157
+ <tbody>
158
+ {% for f in session.topFiles %}
159
+ <tr><td>{{ f.path }}</td><td>+{{ f.additions }}{% if f.deletions > 0 %}/&minus;{{ f.deletions }}{% endif %}</td></tr>
160
+ {% endfor %}
161
+ </tbody>
162
+ </table>
163
+ </div>
164
+ {% endif %}
165
+
166
+ {%- comment -%} Skills {%- endcomment -%}
167
+ {% if session.skills.size > 0 %}
168
+ <div class="sidebar-card">
169
+ <div class="radar-label">Skills</div>
170
+ <div class="chip-list">
171
+ {% for skill in session.skills %}
172
+ <span class="chip">{{ skill }}</span>
173
+ {% endfor %}
174
+ </div>
175
+ </div>
176
+ {% endif %}
177
+
178
+ </aside>
179
+ </div>
180
+ </section>
181
+
182
+ {%- comment -%} Q&A {%- endcomment -%}
183
+ {% if session.qaPairs.size > 0 %}
184
+ <section class="radar-section" aria-label="Questions and answers">
185
+ <div class="radar-label">Q&amp;A</div>
186
+ <div>
187
+ {% for qa in session.qaPairs %}
188
+ <div class="qa-item">
189
+ <div class="qa-question">{{ qa.question }}</div>
190
+ <div class="qa-answer">{{ qa.answer }}</div>
191
+ </div>
192
+ {% endfor %}
193
+ </div>
194
+ </section>
195
+ {% endif %}
196
+
197
+ {%- comment -%} Agent Summary {%- endcomment -%}
198
+ {% if session.agentSummary != blank or session.agents.size > 0 %}
199
+ <section class="radar-section" aria-label="Agent summary" id="agent-section">
200
+ <div class="radar-label">Agent Summary</div>
201
+ {% if session.agents.size > 0 %}
202
+ <div class="agent-layout">
203
+ <div>
204
+ <table class="agent-table">
205
+ <thead>
206
+ <tr><th>Role</th><th style="text-align: right;">Duration</th><th style="text-align: right;">LOC</th></tr>
207
+ </thead>
208
+ <tbody>
209
+ {% for agent in session.agents %}
210
+ <tr>
211
+ <td><span class="agent-dot agent-dot--{{ agent.role | downcase }}"></span>{{ agent.role }}</td>
212
+ <td>{{ agent.durationMinutes | formatDuration }}</td>
213
+ <td>{{ agent.locChanged | formatLoc }}</td>
214
+ </tr>
215
+ {% endfor %}
216
+ </tbody>
217
+ </table>
218
+ </div>
219
+ <div>
220
+ <div class="agent-bar-chart">
221
+ {% assign maxAgentDuration = session.agents[0].durationMinutes %}
222
+ {% for agent in session.agents %}
223
+ {% if agent.durationMinutes > maxAgentDuration %}{% assign maxAgentDuration = agent.durationMinutes %}{% endif %}
224
+ {% endfor %}
225
+ {% for agent in session.agents %}
226
+ <div class="agent-bar-row">
227
+ <div class="agent-bar-row__label">{{ agent.role }}</div>
228
+ <div class="agent-bar-row__bar" style="width: {{ agent.durationMinutes | times: 100.0 | divided_by: maxAgentDuration | round }}%; background: var(--accent);"></div>
229
+ <div class="agent-bar-row__value">{{ agent.durationMinutes | formatDuration }}</div>
230
+ </div>
231
+ {% endfor %}
232
+ </div>
233
+ </div>
234
+ </div>
235
+ {% endif %}
236
+ {% if session.agentSummary != blank %}
237
+ <div class="radar-narrative" {% if session.agents.size > 0 %}style="margin-top: 1.5rem;"{% endif %}>
238
+ {{ session.agentSummary }}
239
+ </div>
240
+ {% endif %}
241
+ </section>
242
+ {% endif %}
243
+
244
+ {%- comment -%} Footer {%- endcomment -%}
245
+ <section class="radar-section" aria-label="Footer" data-radar-section="footer">
246
+ <footer class="radar-footer">
247
+ <div class="radar-footer__text">Built with heyi.am</div>
248
+ </footer>
249
+ </section>
250
+
251
+ {%- comment -%} Radar Floating Nav {%- endcomment -%}
252
+ <nav class="radar-nav" aria-label="Section navigation">
253
+ <div class="radar-nav__ring"></div>
254
+ <div class="radar-nav__sweep"></div>
255
+ <div class="radar-nav__center"></div>
256
+ </nav>
257
+ <script>
258
+ (function() {
259
+ 'use strict';
260
+ var nav = document.querySelector('.radar-nav');
261
+ if (!nav) return;
262
+ var sections = document.querySelectorAll('.radar-section');
263
+ var dots = [];
264
+ sections.forEach(function(sec, i) {
265
+ sec.setAttribute('data-radar-section', i);
266
+ var label = sec.getAttribute('aria-label') || 'Section ' + (i + 1);
267
+ var btn = document.createElement('button');
268
+ btn.className = 'radar-nav__dot';
269
+ btn.setAttribute('data-radar-dot', i);
270
+ btn.setAttribute('aria-label', 'Navigate to ' + label);
271
+ nav.appendChild(btn);
272
+ dots.push(btn);
273
+ });
274
+ var total = dots.length;
275
+ dots.forEach(function(dot, i) {
276
+ var angle = (i / total) * 2 * Math.PI - Math.PI / 2;
277
+ var r = 54;
278
+ dot.style.left = (60 + r * Math.cos(angle) - 3) + 'px';
279
+ dot.style.top = (60 + r * Math.sin(angle) - 3) + 'px';
280
+ });
281
+ var observer = new IntersectionObserver(function(entries) {
282
+ entries.forEach(function(entry) {
283
+ var idx = entry.target.getAttribute('data-radar-section');
284
+ var dot = nav.querySelector('[data-radar-dot="' + idx + '"]');
285
+ if (dot) dot.classList.toggle('radar-nav__dot--active', entry.isIntersecting);
286
+ });
287
+ }, { threshold: 0.3 });
288
+ sections.forEach(function(s) { observer.observe(s); });
289
+ dots.forEach(function(dot) {
290
+ dot.addEventListener('click', function() {
291
+ var idx = this.getAttribute('data-radar-dot');
292
+ var target = document.querySelector('[data-radar-section="' + idx + '"]');
293
+ if (target) target.scrollIntoView({ behavior: 'smooth', block: 'start' });
294
+ });
295
+ });
296
+ })();
297
+ </script>
298
+
299
+ </div>
300
+