@hexclave/next 1.0.19 → 1.0.21

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 (105) hide show
  1. package/dist/components/credential-sign-in.js +5 -1
  2. package/dist/components/credential-sign-in.js.map +1 -1
  3. package/dist/components/team-switcher.js +3 -5
  4. package/dist/components/team-switcher.js.map +1 -1
  5. package/dist/components-page/auth-page.js +2 -2
  6. package/dist/components-page/auth-page.js.map +1 -1
  7. package/dist/components-page/forgot-password.js +5 -1
  8. package/dist/components-page/forgot-password.js.map +1 -1
  9. package/dist/components-page/oauth-callback.js +6 -1
  10. package/dist/components-page/oauth-callback.js.map +1 -1
  11. package/dist/components-page/team-creation.js +2 -3
  12. package/dist/components-page/team-creation.js.map +1 -1
  13. package/dist/esm/components/credential-sign-in.js +5 -1
  14. package/dist/esm/components/credential-sign-in.js.map +1 -1
  15. package/dist/esm/components/team-switcher.js +3 -5
  16. package/dist/esm/components/team-switcher.js.map +1 -1
  17. package/dist/esm/components-page/auth-page.js +2 -2
  18. package/dist/esm/components-page/auth-page.js.map +1 -1
  19. package/dist/esm/components-page/forgot-password.js +5 -1
  20. package/dist/esm/components-page/forgot-password.js.map +1 -1
  21. package/dist/esm/components-page/oauth-callback.js +6 -1
  22. package/dist/esm/components-page/oauth-callback.js.map +1 -1
  23. package/dist/esm/components-page/team-creation.js +2 -3
  24. package/dist/esm/components-page/team-creation.js.map +1 -1
  25. package/dist/esm/generated/quetzal-translations.d.ts +2 -2
  26. package/dist/esm/lib/hexclave-app/apps/implementations/admin-app-impl.d.ts.map +1 -1
  27. package/dist/esm/lib/hexclave-app/apps/implementations/client-app-impl.cross-domain.test.js +34 -5
  28. package/dist/esm/lib/hexclave-app/apps/implementations/client-app-impl.cross-domain.test.js.map +1 -1
  29. package/dist/esm/lib/hexclave-app/apps/implementations/client-app-impl.d.ts +1 -0
  30. package/dist/esm/lib/hexclave-app/apps/implementations/client-app-impl.d.ts.map +1 -1
  31. package/dist/esm/lib/hexclave-app/apps/implementations/client-app-impl.js +24 -4
  32. package/dist/esm/lib/hexclave-app/apps/implementations/client-app-impl.js.map +1 -1
  33. package/dist/esm/lib/hexclave-app/apps/implementations/common.js +1 -1
  34. package/dist/esm/lib/hexclave-app/apps/implementations/event-tracker.d.ts +1 -0
  35. package/dist/esm/lib/hexclave-app/apps/implementations/event-tracker.d.ts.map +1 -1
  36. package/dist/esm/lib/hexclave-app/apps/implementations/event-tracker.js +17 -13
  37. package/dist/esm/lib/hexclave-app/apps/implementations/event-tracker.js.map +1 -1
  38. package/dist/esm/lib/hexclave-app/apps/implementations/event-tracker.test.js +4 -8
  39. package/dist/esm/lib/hexclave-app/apps/implementations/event-tracker.test.js.map +1 -1
  40. package/dist/esm/lib/hexclave-app/apps/implementations/server-app-impl.d.ts +1 -1
  41. package/dist/esm/lib/hexclave-app/apps/implementations/server-app-impl.js +2 -2
  42. package/dist/esm/lib/hexclave-app/apps/implementations/server-app-impl.js.map +1 -1
  43. package/dist/esm/lib/hexclave-app/apps/implementations/session-replay.d.ts +3 -1
  44. package/dist/esm/lib/hexclave-app/apps/implementations/session-replay.d.ts.map +1 -1
  45. package/dist/esm/lib/hexclave-app/apps/implementations/session-replay.js +19 -13
  46. package/dist/esm/lib/hexclave-app/apps/implementations/session-replay.js.map +1 -1
  47. package/dist/esm/lib/hexclave-app/apps/implementations/session-replay.test.js +4 -9
  48. package/dist/esm/lib/hexclave-app/apps/implementations/session-replay.test.js.map +1 -1
  49. package/dist/esm/lib/hexclave-app/apps/interfaces/client-app.d.ts +3 -2
  50. package/dist/esm/lib/hexclave-app/apps/interfaces/client-app.d.ts.map +1 -1
  51. package/dist/esm/lib/hexclave-app/apps/interfaces/client-app.js.map +1 -1
  52. package/dist/esm/lib/hexclave-app/project-configs/index.d.ts +7 -0
  53. package/dist/esm/lib/hexclave-app/project-configs/index.d.ts.map +1 -1
  54. package/dist/esm/lib/hexclave-app/projects/index.d.ts.map +1 -1
  55. package/dist/esm/lib/hexclave-app/projects/index.js +1 -1
  56. package/dist/esm/lib/hexclave-app/projects/index.js.map +1 -1
  57. package/dist/generated/quetzal-translations.d.ts +2 -2
  58. package/dist/lib/hexclave-app/apps/implementations/admin-app-impl.d.ts.map +1 -1
  59. package/dist/lib/hexclave-app/apps/implementations/client-app-impl.cross-domain.test.js +34 -5
  60. package/dist/lib/hexclave-app/apps/implementations/client-app-impl.cross-domain.test.js.map +1 -1
  61. package/dist/lib/hexclave-app/apps/implementations/client-app-impl.d.ts +1 -0
  62. package/dist/lib/hexclave-app/apps/implementations/client-app-impl.d.ts.map +1 -1
  63. package/dist/lib/hexclave-app/apps/implementations/client-app-impl.js +24 -4
  64. package/dist/lib/hexclave-app/apps/implementations/client-app-impl.js.map +1 -1
  65. package/dist/lib/hexclave-app/apps/implementations/common.js +1 -1
  66. package/dist/lib/hexclave-app/apps/implementations/event-tracker.d.ts +1 -0
  67. package/dist/lib/hexclave-app/apps/implementations/event-tracker.d.ts.map +1 -1
  68. package/dist/lib/hexclave-app/apps/implementations/event-tracker.js +16 -12
  69. package/dist/lib/hexclave-app/apps/implementations/event-tracker.js.map +1 -1
  70. package/dist/lib/hexclave-app/apps/implementations/event-tracker.test.js +4 -8
  71. package/dist/lib/hexclave-app/apps/implementations/event-tracker.test.js.map +1 -1
  72. package/dist/lib/hexclave-app/apps/implementations/server-app-impl.d.ts +1 -1
  73. package/dist/lib/hexclave-app/apps/implementations/server-app-impl.js +2 -2
  74. package/dist/lib/hexclave-app/apps/implementations/server-app-impl.js.map +1 -1
  75. package/dist/lib/hexclave-app/apps/implementations/session-replay.d.ts +3 -1
  76. package/dist/lib/hexclave-app/apps/implementations/session-replay.d.ts.map +1 -1
  77. package/dist/lib/hexclave-app/apps/implementations/session-replay.js +19 -12
  78. package/dist/lib/hexclave-app/apps/implementations/session-replay.js.map +1 -1
  79. package/dist/lib/hexclave-app/apps/implementations/session-replay.test.js +4 -9
  80. package/dist/lib/hexclave-app/apps/implementations/session-replay.test.js.map +1 -1
  81. package/dist/lib/hexclave-app/apps/interfaces/client-app.d.ts +3 -2
  82. package/dist/lib/hexclave-app/apps/interfaces/client-app.d.ts.map +1 -1
  83. package/dist/lib/hexclave-app/apps/interfaces/client-app.js.map +1 -1
  84. package/dist/lib/hexclave-app/project-configs/index.d.ts +7 -0
  85. package/dist/lib/hexclave-app/project-configs/index.d.ts.map +1 -1
  86. package/dist/lib/hexclave-app/projects/index.d.ts.map +1 -1
  87. package/dist/lib/hexclave-app/projects/index.js +1 -1
  88. package/dist/lib/hexclave-app/projects/index.js.map +1 -1
  89. package/package.json +4 -4
  90. package/src/components/credential-sign-in.tsx +8 -1
  91. package/src/components/team-switcher.tsx +3 -5
  92. package/src/components-page/auth-page.tsx +2 -2
  93. package/src/components-page/forgot-password.tsx +7 -1
  94. package/src/components-page/oauth-callback.tsx +9 -1
  95. package/src/components-page/team-creation.tsx +2 -3
  96. package/src/lib/hexclave-app/apps/implementations/client-app-impl.cross-domain.test.ts +36 -0
  97. package/src/lib/hexclave-app/apps/implementations/client-app-impl.ts +43 -4
  98. package/src/lib/hexclave-app/apps/implementations/event-tracker.test.ts +5 -13
  99. package/src/lib/hexclave-app/apps/implementations/event-tracker.ts +19 -14
  100. package/src/lib/hexclave-app/apps/implementations/server-app-impl.ts +2 -2
  101. package/src/lib/hexclave-app/apps/implementations/session-replay.test.ts +4 -20
  102. package/src/lib/hexclave-app/apps/implementations/session-replay.ts +19 -12
  103. package/src/lib/hexclave-app/apps/interfaces/client-app.ts +3 -2
  104. package/src/lib/hexclave-app/project-configs/index.ts +8 -0
  105. package/src/lib/hexclave-app/projects/index.ts +13 -11
@@ -14,7 +14,7 @@ import { envVars } from "../../../../generated/env.js";
14
14
  import { resolveHandlerUrls } from "../../url-targets.js";
15
15
 
16
16
  //#region src/lib/hexclave-app/apps/implementations/common.ts
17
- const clientVersion = "js @hexclave/next@1.0.19";
17
+ const clientVersion = "js @hexclave/next@1.0.21";
18
18
  if (clientVersion.startsWith("STACK_COMPILE_TIME")) throw new HexclaveAssertionError("Client version was not replaced. Something went wrong during build!");
19
19
  const replaceHexclavePortPrefix = (input) => {
20
20
  if (!input) return input;
@@ -49,6 +49,7 @@ declare class EventTracker {
49
49
  private _setupPageHideListeners;
50
50
  private _teardown;
51
51
  private _flush;
52
+ private _disable;
52
53
  private _tick;
53
54
  }
54
55
  //#endregion
@@ -1 +1 @@
1
- {"version":3,"file":"event-tracker.d.ts","names":[],"sources":["../../../../../../src/lib/hexclave-app/apps/implementations/event-tracker.ts"],"mappings":";;;KAoGY,gBAAA;EACV,SAAA;EACA,SAAA,GAAY,IAAA,UAAc,OAAA;IAAW,SAAA;EAAA,MAAyB,OAAA,CAAQ,MAAA,CAAO,QAAA,EAAU,KAAA;AAAA;AAAA,cAS5E,YAAA;EAAA,QACH,QAAA;EAAA,QACA,UAAA;EAAA,QACA,SAAA;EAAA,QACA,gBAAA;EAAA,QACA,WAAA;EAAA,QACA,OAAA;EAAA,QACA,YAAA;EAAA,QACA,QAAA;EAAA,iBACS,uBAAA;EAAA,iBACA,KAAA;EAAA,QAET,kBAAA;EAAA,QACA,qBAAA;EAAA,QAEA,eAAA;EAAA,QACA,0BAAA;EAAA,QAGA,mBAAA;EAAA,QACA,iBAAA;EAAA,QACA,eAAA;EAAA,QACA,yBAAA;EAAA,QACA,yBAAA;cAEI,IAAA,EAAM,gBAAA;EAKlB,KAAA,CAAA;EAsBA,IAAA,CAAA;EAUA,WAAA,CAAA;EAAA,QAMQ,UAAA;EAAA,QASA,gBAAA;EAAA,QA4BA,qBAAA;EAAA,iBA4BS,WAAA;EAAA,QAIT,cAAA;EAAA,QA6CA,sBAAA;EAAA,iBAWS,eAAA;EAAA,QAiDT,kBAAA;EAAA,iBAIS,kBAAA;EAAA,iBAIA,2BAAA;EAAA,iBAIA,4BAAA;EAAA,QAIT,wBAAA;EAAA,QA2BA,gBAAA;EAAA,QAoBA,2BAAA;EAAA,iBAeS,WAAA;EAAA,QAIT,uBAAA;EAAA,QASA,SAAA;EAAA,QA2BM,MAAA;EAAA,QAuDN,KAAA;AAAA"}
1
+ {"version":3,"file":"event-tracker.d.ts","names":[],"sources":["../../../../../../src/lib/hexclave-app/apps/implementations/event-tracker.ts"],"mappings":";;;KAwGY,gBAAA;EACV,SAAA;EACA,SAAA,GAAY,IAAA,UAAc,OAAA;IAAW,SAAA;EAAA,MAAyB,OAAA,CAAQ,MAAA,CAAO,QAAA,EAAU,KAAA;AAAA;AAAA,cAS5E,YAAA;EAAA,QACH,QAAA;EAAA,QACA,UAAA;EAAA,QACA,SAAA;EAAA,QACA,gBAAA;EAAA,QACA,WAAA;EAAA,QACA,OAAA;EAAA,QACA,YAAA;EAAA,QACA,QAAA;EAAA,iBACS,uBAAA;EAAA,iBACA,KAAA;EAAA,QAET,kBAAA;EAAA,QACA,qBAAA;EAAA,QAEA,eAAA;EAAA,QACA,0BAAA;EAAA,QAGA,mBAAA;EAAA,QACA,iBAAA;EAAA,QACA,eAAA;EAAA,QACA,yBAAA;EAAA,QACA,yBAAA;cAEI,IAAA,EAAM,gBAAA;EAKlB,KAAA,CAAA;EAsBA,IAAA,CAAA;EAUA,WAAA,CAAA;EAAA,QAMQ,UAAA;EAAA,QASA,gBAAA;EAAA,QA4BA,qBAAA;EAAA,iBA4BS,WAAA;EAAA,QAIT,cAAA;EAAA,QA6CA,sBAAA;EAAA,iBAWS,eAAA;EAAA,QAiDT,kBAAA;EAAA,iBAIS,kBAAA;EAAA,iBAIA,2BAAA;EAAA,iBAIA,4BAAA;EAAA,QAIT,wBAAA;EAAA,QA2BA,gBAAA;EAAA,QAoBA,2BAAA;EAAA,iBAeS,WAAA;EAAA,QAIT,uBAAA;EAAA,QASA,SAAA;EAAA,QA2BM,MAAA;EAAA,QA+CN,QAAA;EAAA,QASA,KAAA;AAAA"}
@@ -3,7 +3,7 @@ import { cssEscapeIdent } from "@hexclave/shared/dist/utils/dom";
3
3
  import { ELEMENTS_CHAIN_MAX_DEPTH, buildElementsChain } from "@hexclave/shared/dist/utils/elements-chain";
4
4
  import { runAsynchronously } from "@hexclave/shared/dist/utils/promises";
5
5
  import { isBrowserLike } from "@hexclave/shared/dist/utils/env";
6
- import { generateUuid } from "./session-replay.js";
6
+ import { generateUuid, isAnalyticsNotEnabledError } from "./session-replay.js";
7
7
 
8
8
  //#region src/lib/hexclave-app/apps/implementations/event-tracker.ts
9
9
  const FLUSH_INTERVAL_MS = 1e4;
@@ -19,6 +19,9 @@ function hasHistoryMethods(value) {
19
19
  if (!("pushState" in value) || !("replaceState" in value)) return false;
20
20
  return typeof value.pushState === "function" && typeof value.replaceState === "function";
21
21
  }
22
+ function getTextSnippet(textContent) {
23
+ return textContent == null ? "" : textContent.trim().substring(0, 200);
24
+ }
22
25
  const CLICKMAP_SCALE_FACTOR = 16;
23
26
  const DEAD_CLICK_SCROLL_THRESHOLD_MS = 100;
24
27
  const DEAD_CLICK_SELECTION_CHANGED_THRESHOLD_MS = 100;
@@ -83,7 +86,7 @@ var EventTracker = class {
83
86
  event_at_ms: Date.now(),
84
87
  data: {
85
88
  tag_name: target.tagName.toLowerCase(),
86
- text: target.textContent.trim().substring(0, 200),
89
+ text: getTextSnippet(target.textContent),
87
90
  href: this._findNearestAnchorHref(target),
88
91
  selector: this._buildSelector(target),
89
92
  elements_chain: buildElementsChain(target),
@@ -331,21 +334,22 @@ var EventTracker = class {
331
334
  };
332
335
  const res = await this._deps.sendBatch(JSON.stringify(payload), { keepalive: options.keepalive });
333
336
  if (res.status === "error") {
337
+ if (isAnalyticsNotEnabledError(res.error)) {
338
+ this._disable();
339
+ return;
340
+ }
334
341
  console.warn("EventTracker flush failed:", res.error);
335
342
  return;
336
343
  }
337
- if (!res.data.ok) {
338
- if ((res.data.headers.get("x-hexclave-known-error") ?? res.data.headers.get("x-stack-known-error")) === "ANALYTICS_NOT_ENABLED") {
339
- this._disabled = true;
340
- if (this._flushTimer !== null) {
341
- clearInterval(this._flushTimer);
342
- this._flushTimer = null;
343
- }
344
- this._teardown();
345
- return;
346
- }
347
- console.warn("EventTracker flush failed:", res.data.status, await res.data.text());
344
+ if (!res.data.ok) console.warn("EventTracker flush failed:", res.data.status, await res.data.text());
345
+ }
346
+ _disable() {
347
+ this._disabled = true;
348
+ if (this._flushTimer !== null) {
349
+ clearInterval(this._flushTimer);
350
+ this._flushTimer = null;
348
351
  }
352
+ this._teardown();
349
353
  }
350
354
  _tick() {
351
355
  if (this._cancelled) return;
@@ -1 +1 @@
1
- {"version":3,"file":"event-tracker.js","names":[],"sources":["../../../../../../src/lib/hexclave-app/apps/implementations/event-tracker.ts"],"sourcesContent":["\n//===========================================\n// THIS FILE IS AUTO-GENERATED FROM TEMPLATE. DO NOT EDIT IT DIRECTLY UNLESS YOU ALSO EDIT THE CORRESPONDING FILE IN packages/template\n//===========================================\nimport { isBrowserLike } from \"@hexclave/shared/dist/utils/env\";\nimport { CLICKMAP_ROOT_ID, DEV_TOOL_ROOT_ID } from \"@hexclave/shared/dist/utils/dev-tool\";\nimport { cssEscapeIdent } from \"@hexclave/shared/dist/utils/dom\";\nimport { buildElementsChain, ELEMENTS_CHAIN_MAX_DEPTH } from \"@hexclave/shared/dist/utils/elements-chain\";\nimport { runAsynchronously } from \"@hexclave/shared/dist/utils/promises\";\nimport { Result } from \"@hexclave/shared/dist/utils/results\";\nimport { generateUuid } from \"./session-replay\";\n\nconst FLUSH_INTERVAL_MS = 10_000;\nconst MAX_EVENTS_PER_BATCH = 50;\nconst MAX_APPROX_BYTES_PER_BATCH = 64_000;\n\nfunction hasScreenDimensions(value: unknown): value is { width: number, height: number } {\n if (value == null || typeof value !== \"object\") {\n return false;\n }\n if (!(\"width\" in value) || !(\"height\" in value)) {\n return false;\n }\n return typeof value.width === \"number\" && typeof value.height === \"number\";\n}\n\nfunction hasHistoryMethods(value: unknown): value is { pushState: History[\"pushState\"], replaceState: History[\"replaceState\"] } {\n if (value == null || typeof value !== \"object\") {\n return false;\n }\n if (!(\"pushState\" in value) || !(\"replaceState\" in value)) {\n return false;\n }\n return typeof value.pushState === \"function\" && typeof value.replaceState === \"function\";\n}\n\n// Pixel quantization factor for x/y/viewport in stored click events. Matches the\n// SCALE_FACTOR used by the ClickHouse clickmap_events MV — keep them in sync.\nconst CLICKMAP_SCALE_FACTOR = 16;\n\n// Dead-click detection (PostHog-style). Whether an element has a click handler\n// is unknowable from page script, so a click is classified by its observable\n// consequences instead: it is \"alive\" if the page scrolled, the text selection\n// changed, or the tab visibility changed (a new tab opened) almost\n// immediately, or if the DOM mutated within a couple of seconds — and \"dead\"\n// if none of that happened by the absolute timeout.\n//\n// The $click event is buffered immediately like any other event (so\n// event_at_ms, ordering, and every query are untouched) and the sweep sets\n// data.dead=1 on it in place if nothing observable happened. _flush holds\n// back clicks that are still unclassified — classification always finishes\n// well within one FLUSH_INTERVAL_MS, so a held click rides the next flush at\n// the latest. A keepalive flush (pagehide/stop) sends them unmarked: a click\n// still pending when the page unloads led to that navigation, alive by\n// definition.\n//\n// NOTE — blocker for any future real-time / \"live clicks\" view: a click that\n// is still unclassified when its natural flush fires arrives up to one extra\n// FLUSH_INTERVAL_MS late. A surface showing clicks as they happen must either\n// accept that lag or emit a provisional $click plus a later dead-click\n// reconciliation event.\nconst DEAD_CLICK_SCROLL_THRESHOLD_MS = 100;\nconst DEAD_CLICK_SELECTION_CHANGED_THRESHOLD_MS = 100;\nconst DEAD_CLICK_VISIBILITY_CHANGE_THRESHOLD_MS = 100;\nconst DEAD_CLICK_MUTATION_THRESHOLD_MS = 2_500;\n// 1.1x the mutation threshold, mirroring posthog-js: every signal window has\n// closed before a click is declared dead.\nconst DEAD_CLICK_ABSOLUTE_TIMEOUT_MS = 2_750;\nconst DEAD_CLICK_CHECK_INTERVAL_MS = 1_000;\n// Backstop against click storms (e.g. rage clicks on a dead element): past the\n// cap, clicks are simply not classified rather than not recorded.\nconst DEAD_CLICK_MAX_PENDING = 50;\n\nfunction isPointerTargetFixed(element: Element): boolean {\n let current: Element | null = element;\n let depth = 0;\n while (current != null && depth < ELEMENTS_CHAIN_MAX_DEPTH * 2) {\n const style = window.getComputedStyle(current);\n if (style.position === \"fixed\" || style.position === \"sticky\") {\n return true;\n }\n current = current.parentElement;\n depth += 1;\n }\n return false;\n}\n\n// Clicks on Hexclave's own in-page UI (the dev tool and the standalone\n// clickmap overlay) must never be ingested as analytics events.\nfunction isInsideHexclaveUi(element: Element): boolean {\n return element.closest(`#${cssEscapeIdent(DEV_TOOL_ROOT_ID)}, #${cssEscapeIdent(CLICKMAP_ROOT_ID)}`) != null;\n}\n\n// Mutation-record targets can be text/comment nodes; resolve to the nearest\n// element before asking whether the mutation came from Hexclave's own UI.\nfunction isInsideHexclaveUiNode(node: Node | null): boolean {\n const element = node instanceof Element ? node : node?.parentElement ?? null;\n return element != null && isInsideHexclaveUi(element);\n}\n\nexport type EventTrackerDeps = {\n projectId: string,\n sendBatch: (body: string, options: { keepalive: boolean }) => Promise<Result<Response, Error>>,\n};\n\ntype TrackedEvent = {\n event_type: \"$page-view\" | \"$click\",\n event_at_ms: number,\n data: Record<string, unknown>,\n};\n\nexport class EventTracker {\n private _started = false;\n private _cancelled = false;\n private _disabled = false;\n private _detachListeners: (() => void) | null = null;\n private _flushTimer: ReturnType<typeof setInterval> | null = null;\n private _events: TrackedEvent[] = [];\n private _approxBytes = 0;\n private _lastUrl: string | null = null;\n private readonly _sessionReplaySegmentId: string;\n private readonly _deps: EventTrackerDeps;\n\n private _originalPushState: History[\"pushState\"] | null = null;\n private _originalReplaceState: History[\"replaceState\"] | null = null;\n\n private _deadClickTimer: ReturnType<typeof setInterval> | null = null;\n private _deadClickMutationObserver: MutationObserver | null = null;\n // Buffered $click events still awaiting dead-click classification. Always a\n // subset of _events — _flush holds these back until the sweep resolves them.\n private _unclassifiedClicks = new Set<TrackedEvent>();\n private _lastMutationAtMs: number | null = null;\n private _lastScrollAtMs: number | null = null;\n private _lastSelectionChangedAtMs: number | null = null;\n private _lastVisibilityChangeAtMs: number | null = null;\n\n constructor(deps: EventTrackerDeps) {\n this._deps = deps;\n this._sessionReplaySegmentId = generateUuid();\n }\n\n start() {\n if (this._started) return;\n if (!isBrowserLike()) return;\n if (\n typeof window.addEventListener !== \"function\"\n || typeof window.removeEventListener !== \"function\"\n || typeof document.addEventListener !== \"function\"\n || typeof document.removeEventListener !== \"function\"\n || !hasScreenDimensions(window.screen)\n ) {\n return;\n }\n this._started = true;\n\n this._setupPageViewCapture();\n this._setupClickCapture();\n this._setupDeadClickDetection();\n this._setupPageHideListeners();\n\n this._flushTimer = setInterval(() => this._tick(), FLUSH_INTERVAL_MS);\n }\n\n stop() {\n this._cancelled = true;\n if (this._flushTimer !== null) {\n clearInterval(this._flushTimer);\n this._flushTimer = null;\n }\n runAsynchronously(() => this._flush({ keepalive: true }));\n this._teardown();\n }\n\n clearBuffer() {\n this._events = [];\n this._approxBytes = 0;\n this._unclassifiedClicks.clear();\n }\n\n private _pushEvent(event: TrackedEvent) {\n if (this._disabled) return;\n this._events.push(event);\n this._approxBytes += JSON.stringify(event).length;\n if (this._events.length >= MAX_EVENTS_PER_BATCH || this._approxBytes >= MAX_APPROX_BYTES_PER_BATCH) {\n runAsynchronously(() => this._flush({ keepalive: false }));\n }\n }\n\n private _capturePageView(entryType: \"initial\" | \"push\" | \"replace\" | \"pop\") {\n const screenObject = window.screen;\n if (!hasScreenDimensions(screenObject)) {\n return;\n }\n\n const url = window.location.href;\n if (url === this._lastUrl && entryType !== \"initial\") return;\n this._lastUrl = url;\n\n this._pushEvent({\n event_type: \"$page-view\",\n event_at_ms: Date.now(),\n data: {\n url,\n path: window.location.pathname,\n referrer: document.referrer,\n title: document.title,\n entry_type: entryType,\n viewport_width: window.innerWidth,\n viewport_height: window.innerHeight,\n screen_width: screenObject.width,\n screen_height: screenObject.height,\n user_agent: typeof navigator !== \"undefined\" ? navigator.userAgent : null,\n },\n });\n }\n\n private _setupPageViewCapture() {\n // Fire initial page-view\n this._capturePageView(\"initial\");\n const historyObject = window.history;\n if (!hasHistoryMethods(historyObject)) {\n return;\n }\n const originalPushState = historyObject.pushState;\n const originalReplaceState = historyObject.replaceState;\n\n // Monkey-patch history.pushState\n this._originalPushState = (...args: Parameters<History[\"pushState\"]>) => originalPushState.apply(historyObject, args);\n historyObject.pushState = (...args: Parameters<History[\"pushState\"]>) => {\n this._originalPushState!(...args);\n this._capturePageView(\"push\");\n };\n\n // Monkey-patch history.replaceState\n this._originalReplaceState = (...args: Parameters<History[\"replaceState\"]>) => originalReplaceState.apply(historyObject, args);\n historyObject.replaceState = (...args: Parameters<History[\"replaceState\"]>) => {\n this._originalReplaceState!(...args);\n this._capturePageView(\"replace\");\n };\n\n // Listen for popstate (back/forward navigation)\n window.addEventListener(\"popstate\", this._onPopState);\n }\n\n private readonly _onPopState = () => {\n this._capturePageView(\"pop\");\n };\n\n private _buildSelector(element: Element): string {\n const parts: string[] = [];\n let current: Element | null = element;\n let depth = 0;\n\n while (current && depth < 8 && current !== document.documentElement) {\n let part = current.tagName.toLowerCase();\n let testIdAttr = \"data-testid\";\n let testId = current.getAttribute(\"data-testid\");\n if (testId == null) {\n testIdAttr = \"data-test-id\";\n testId = current.getAttribute(\"data-test-id\");\n }\n if (testId != null && testId.trim() !== \"\") {\n part += `[${testIdAttr}=\"${testId.replace(/\"/g, '\\\\\"')}\"]`;\n parts.unshift(part);\n break;\n }\n if (current.id !== \"\") {\n part += `#${cssEscapeIdent(current.id)}`;\n parts.unshift(part);\n break;\n }\n if (current.className && typeof current.className === \"string\") {\n const classes = current.className.trim().split(/\\s+/).filter(Boolean).slice(0, 4);\n if (classes.length > 0) {\n part += `.${classes.map(cssEscapeIdent).join(\".\")}`;\n }\n }\n const parent: Element | null = current.parentElement;\n if (parent != null) {\n const tagName = current.tagName;\n const siblings = Array.from(parent.children).filter((child) => child.tagName === tagName);\n if (siblings.length > 1) {\n part += `:nth-of-type(${siblings.indexOf(current) + 1})`;\n }\n }\n parts.unshift(part);\n current = parent;\n depth++;\n }\n\n return parts.join(\" > \");\n }\n\n private _findNearestAnchorHref(element: Element): string | null {\n let current: Element | null = element;\n while (current) {\n if (current.tagName === \"A\" && current.hasAttribute(\"href\")) {\n return current.getAttribute(\"href\");\n }\n current = current.parentElement;\n }\n return null;\n }\n\n private readonly _onClickCapture = (event: MouseEvent) => {\n const target = event.target;\n if (!(target instanceof Element)) return;\n if (isInsideHexclaveUi(target)) return;\n\n const viewportWidth = window.innerWidth;\n const viewportHeight = window.innerHeight;\n const pointerTargetFixed = isPointerTargetFixed(target);\n // Pre-scale at ingest so old + new rows land in identical buckets in CH.\n const xScaled = Math.round(event.pageX / CLICKMAP_SCALE_FACTOR);\n const yScaled = Math.round(event.pageY / CLICKMAP_SCALE_FACTOR);\n const clientYScaled = Math.round(event.clientY / CLICKMAP_SCALE_FACTOR);\n const relativeX = viewportWidth > 0 ? event.clientX / viewportWidth : 0;\n\n const clickEvent: TrackedEvent = {\n event_type: \"$click\",\n event_at_ms: Date.now(),\n data: {\n tag_name: target.tagName.toLowerCase(),\n text: target.textContent.trim().substring(0, 200),\n href: this._findNearestAnchorHref(target),\n selector: this._buildSelector(target),\n elements_chain: buildElementsChain(target),\n pointer_target_fixed: pointerTargetFixed ? 1 : 0,\n url: window.location.href,\n path: window.location.pathname,\n title: document.title,\n x: event.clientX,\n y: event.clientY,\n page_x: event.pageX,\n page_y: event.pageY,\n x_scaled: xScaled,\n y_scaled: yScaled,\n client_y_scaled: clientYScaled,\n pointer_relative_x: relativeX,\n viewport_width: viewportWidth,\n viewport_height: viewportHeight,\n scale_factor: CLICKMAP_SCALE_FACTOR,\n },\n };\n\n // Register for dead-click classification before buffering, so a\n // size-triggered flush from this very push already holds the click back.\n if (this._deadClickTimer !== null && this._unclassifiedClicks.size < DEAD_CLICK_MAX_PENDING) {\n this._unclassifiedClicks.add(clickEvent);\n }\n this._pushEvent(clickEvent);\n };\n\n private _setupClickCapture() {\n document.addEventListener(\"click\", this._onClickCapture, { capture: true });\n }\n\n private readonly _onDeadClickScroll = () => {\n this._lastScrollAtMs = Date.now();\n };\n\n private readonly _onDeadClickSelectionChange = () => {\n this._lastSelectionChangedAtMs = Date.now();\n };\n\n private readonly _onDeadClickVisibilityChange = () => {\n this._lastVisibilityChangeAtMs = Date.now();\n };\n\n private _setupDeadClickDetection() {\n if (typeof MutationObserver !== \"function\") return;\n\n this._deadClickMutationObserver = new MutationObserver((mutations) => {\n // The dev tool and the clickmap overlay rewrite their own DOM constantly\n // while open; their mutations must not mark host-page clicks as alive.\n if (mutations.every((mutation) => isInsideHexclaveUiNode(mutation.target))) {\n return;\n }\n this._lastMutationAtMs = Date.now();\n });\n this._deadClickMutationObserver.observe(document.documentElement, {\n childList: true,\n attributes: true,\n characterData: true,\n subtree: true,\n });\n\n // Capture phase so scrolls inside nested scroll containers count, not just\n // the document itself (scroll events don't bubble).\n document.addEventListener(\"scroll\", this._onDeadClickScroll, { capture: true, passive: true });\n document.addEventListener(\"selectionchange\", this._onDeadClickSelectionChange);\n document.addEventListener(\"visibilitychange\", this._onDeadClickVisibilityChange);\n\n this._deadClickTimer = setInterval(() => this._checkDeadClicks(), DEAD_CLICK_CHECK_INTERVAL_MS);\n }\n\n private _checkDeadClicks() {\n const nowMs = Date.now();\n for (const click of this._unclassifiedClicks) {\n const signalWithin = (signalAtMs: number | null, thresholdMs: number) =>\n signalAtMs != null && signalAtMs >= click.event_at_ms && signalAtMs - click.event_at_ms < thresholdMs;\n\n const isAlive = signalWithin(this._lastScrollAtMs, DEAD_CLICK_SCROLL_THRESHOLD_MS)\n || signalWithin(this._lastSelectionChangedAtMs, DEAD_CLICK_SELECTION_CHANGED_THRESHOLD_MS)\n || signalWithin(this._lastVisibilityChangeAtMs, DEAD_CLICK_VISIBILITY_CHANGE_THRESHOLD_MS)\n || signalWithin(this._lastMutationAtMs, DEAD_CLICK_MUTATION_THRESHOLD_MS);\n if (isAlive) {\n this._unclassifiedClicks.delete(click);\n } else if (nowMs - click.event_at_ms >= DEAD_CLICK_ABSOLUTE_TIMEOUT_MS) {\n // The already-buffered event is marked in place — no second event.\n click.data.dead = 1;\n this._unclassifiedClicks.delete(click);\n }\n }\n }\n\n private _teardownDeadClickDetection() {\n if (this._deadClickTimer !== null) {\n clearInterval(this._deadClickTimer);\n this._deadClickTimer = null;\n }\n if (this._deadClickMutationObserver !== null) {\n this._deadClickMutationObserver.disconnect();\n this._deadClickMutationObserver = null;\n }\n document.removeEventListener(\"scroll\", this._onDeadClickScroll, { capture: true });\n document.removeEventListener(\"selectionchange\", this._onDeadClickSelectionChange);\n document.removeEventListener(\"visibilitychange\", this._onDeadClickVisibilityChange);\n this._unclassifiedClicks.clear();\n }\n\n private readonly _onPageHide = () => {\n runAsynchronously(() => this._flush({ keepalive: true }));\n };\n\n private _setupPageHideListeners() {\n window.addEventListener(\"pagehide\", this._onPageHide);\n document.addEventListener(\"visibilitychange\", this._onPageHide);\n this._detachListeners = () => {\n window.removeEventListener(\"pagehide\", this._onPageHide);\n document.removeEventListener(\"visibilitychange\", this._onPageHide);\n };\n }\n\n private _teardown() {\n if (this._detachListeners) {\n this._detachListeners();\n this._detachListeners = null;\n }\n\n // Restore history methods\n const historyObject = window.history;\n if (hasHistoryMethods(historyObject)) {\n if (this._originalPushState) {\n historyObject.pushState = this._originalPushState;\n }\n if (this._originalReplaceState) {\n historyObject.replaceState = this._originalReplaceState;\n }\n }\n this._originalPushState = null;\n this._originalReplaceState = null;\n\n window.removeEventListener(\"popstate\", this._onPopState);\n document.removeEventListener(\"click\", this._onClickCapture, { capture: true });\n this._teardownDeadClickDetection();\n\n this._events = [];\n this._approxBytes = 0;\n }\n\n private async _flush(options: { keepalive: boolean }) {\n if (this._disabled) return;\n\n // A keepalive flush means the page is unloading — a click still awaiting\n // dead-click classification led to that unload, so it is alive by\n // definition and ships unmarked.\n if (options.keepalive) {\n this._unclassifiedClicks.clear();\n }\n\n // Clicks still awaiting classification stay buffered so the sweep can\n // mark them dead in place; classification finishes well within one flush\n // interval, so they ride the next flush at the latest.\n const events = this._events.filter((event) => !this._unclassifiedClicks.has(event));\n if (events.length === 0) return;\n this._events = this._events.filter((event) => this._unclassifiedClicks.has(event));\n this._approxBytes = this._events.reduce((total, event) => total + JSON.stringify(event).length, 0);\n\n const nowMs = Date.now();\n\n const batchId = generateUuid();\n const payload = {\n session_replay_segment_id: this._sessionReplaySegmentId,\n batch_id: batchId,\n sent_at_ms: nowMs,\n events,\n };\n\n const res = await this._deps.sendBatch(\n JSON.stringify(payload),\n { keepalive: options.keepalive },\n );\n\n if (res.status === \"error\") {\n console.warn(\"EventTracker flush failed:\", res.error);\n return;\n }\n\n if (!res.data.ok) {\n // If the server tells us analytics is not enabled for this project,\n // silently disable the tracker — no point retrying or warning the user.\n const knownError = res.data.headers.get(\"x-hexclave-known-error\") ?? res.data.headers.get(\"x-stack-known-error\");\n if (knownError === \"ANALYTICS_NOT_ENABLED\") {\n this._disabled = true;\n if (this._flushTimer !== null) {\n clearInterval(this._flushTimer);\n this._flushTimer = null;\n }\n this._teardown();\n return;\n }\n console.warn(\"EventTracker flush failed:\", res.data.status, await res.data.text());\n }\n }\n\n private _tick() {\n if (this._cancelled) return;\n if (this._events.length > 0) {\n runAsynchronously(() => this._flush({ keepalive: false }));\n }\n }\n}\n"],"mappings":";;;;;;;;AAYA,MAAM,oBAAoB;AAC1B,MAAM,uBAAuB;AAC7B,MAAM,6BAA6B;AAEnC,SAAS,oBAAoB,OAA4D;AACvF,KAAI,SAAS,QAAQ,OAAO,UAAU,SACpC,QAAO;AAET,KAAI,EAAE,WAAW,UAAU,EAAE,YAAY,OACvC,QAAO;AAET,QAAO,OAAO,MAAM,UAAU,YAAY,OAAO,MAAM,WAAW;;AAGpE,SAAS,kBAAkB,OAAqG;AAC9H,KAAI,SAAS,QAAQ,OAAO,UAAU,SACpC,QAAO;AAET,KAAI,EAAE,eAAe,UAAU,EAAE,kBAAkB,OACjD,QAAO;AAET,QAAO,OAAO,MAAM,cAAc,cAAc,OAAO,MAAM,iBAAiB;;AAKhF,MAAM,wBAAwB;AAuB9B,MAAM,iCAAiC;AACvC,MAAM,4CAA4C;AAClD,MAAM,4CAA4C;AAClD,MAAM,mCAAmC;AAGzC,MAAM,iCAAiC;AACvC,MAAM,+BAA+B;AAGrC,MAAM,yBAAyB;AAE/B,SAAS,qBAAqB,SAA2B;CACvD,IAAI,UAA0B;CAC9B,IAAI,QAAQ;AACZ,QAAO,WAAW,QAAQ,QAAQ,2BAA2B,GAAG;EAC9D,MAAM,QAAQ,OAAO,iBAAiB,QAAQ;AAC9C,MAAI,MAAM,aAAa,WAAW,MAAM,aAAa,SACnD,QAAO;AAET,YAAU,QAAQ;AAClB,WAAS;;AAEX,QAAO;;AAKT,SAAS,mBAAmB,SAA2B;AACrD,QAAO,QAAQ,QAAQ,IAAI,eAAe,iBAAiB,CAAC,KAAK,eAAe,iBAAiB,GAAG,IAAI;;AAK1G,SAAS,uBAAuB,MAA4B;CAC1D,MAAM,UAAU,gBAAgB,UAAU,OAAO,MAAM,iBAAiB;AACxE,QAAO,WAAW,QAAQ,mBAAmB,QAAQ;;AAcvD,IAAa,eAAb,MAA0B;CAyBxB,YAAY,MAAwB;kBAxBjB;oBACE;mBACD;0BAC4B;qBACa;iBAC3B,EAAE;sBACb;kBACW;4BAIwB;+BACM;yBAEC;oCACH;6CAGhC,IAAI,KAAmB;2BACV;yBACF;mCACU;mCACA;2BA8Gd;AACnC,QAAK,iBAAiB,MAAM;;0BA2DM,UAAsB;GACxD,MAAM,SAAS,MAAM;AACrB,OAAI,EAAE,kBAAkB,SAAU;AAClC,OAAI,mBAAmB,OAAO,CAAE;GAEhC,MAAM,gBAAgB,OAAO;GAC7B,MAAM,iBAAiB,OAAO;GAC9B,MAAM,qBAAqB,qBAAqB,OAAO;GAEvD,MAAM,UAAU,KAAK,MAAM,MAAM,QAAQ,sBAAsB;GAC/D,MAAM,UAAU,KAAK,MAAM,MAAM,QAAQ,sBAAsB;GAC/D,MAAM,gBAAgB,KAAK,MAAM,MAAM,UAAU,sBAAsB;GACvE,MAAM,YAAY,gBAAgB,IAAI,MAAM,UAAU,gBAAgB;GAEtE,MAAM,aAA2B;IAC/B,YAAY;IACZ,aAAa,KAAK,KAAK;IACvB,MAAM;KACJ,UAAU,OAAO,QAAQ,aAAa;KACtC,MAAM,OAAO,YAAY,MAAM,CAAC,UAAU,GAAG,IAAI;KACjD,MAAM,KAAK,uBAAuB,OAAO;KACzC,UAAU,KAAK,eAAe,OAAO;KACrC,gBAAgB,mBAAmB,OAAO;KAC1C,sBAAsB,qBAAqB,IAAI;KAC/C,KAAK,OAAO,SAAS;KACrB,MAAM,OAAO,SAAS;KACtB,OAAO,SAAS;KAChB,GAAG,MAAM;KACT,GAAG,MAAM;KACT,QAAQ,MAAM;KACd,QAAQ,MAAM;KACd,UAAU;KACV,UAAU;KACV,iBAAiB;KACjB,oBAAoB;KACpB,gBAAgB;KAChB,iBAAiB;KACjB,cAAc;KACf;IACF;AAID,OAAI,KAAK,oBAAoB,QAAQ,KAAK,oBAAoB,OAAO,uBACnE,MAAK,oBAAoB,IAAI,WAAW;AAE1C,QAAK,WAAW,WAAW;;kCAOe;AAC1C,QAAK,kBAAkB,KAAK,KAAK;;2CAGkB;AACnD,QAAK,4BAA4B,KAAK,KAAK;;4CAGS;AACpD,QAAK,4BAA4B,KAAK,KAAK;;2BAiER;AACnC,2BAAwB,KAAK,OAAO,EAAE,WAAW,MAAM,CAAC,CAAC;;AAvSzD,OAAK,QAAQ;AACb,OAAK,0BAA0B,cAAc;;CAG/C,QAAQ;AACN,MAAI,KAAK,SAAU;AACnB,MAAI,CAAC,eAAe,CAAE;AACtB,MACE,OAAO,OAAO,qBAAqB,cAChC,OAAO,OAAO,wBAAwB,cACtC,OAAO,SAAS,qBAAqB,cACrC,OAAO,SAAS,wBAAwB,cACxC,CAAC,oBAAoB,OAAO,OAAO,CAEtC;AAEF,OAAK,WAAW;AAEhB,OAAK,uBAAuB;AAC5B,OAAK,oBAAoB;AACzB,OAAK,0BAA0B;AAC/B,OAAK,yBAAyB;AAE9B,OAAK,cAAc,kBAAkB,KAAK,OAAO,EAAE,kBAAkB;;CAGvE,OAAO;AACL,OAAK,aAAa;AAClB,MAAI,KAAK,gBAAgB,MAAM;AAC7B,iBAAc,KAAK,YAAY;AAC/B,QAAK,cAAc;;AAErB,0BAAwB,KAAK,OAAO,EAAE,WAAW,MAAM,CAAC,CAAC;AACzD,OAAK,WAAW;;CAGlB,cAAc;AACZ,OAAK,UAAU,EAAE;AACjB,OAAK,eAAe;AACpB,OAAK,oBAAoB,OAAO;;CAGlC,AAAQ,WAAW,OAAqB;AACtC,MAAI,KAAK,UAAW;AACpB,OAAK,QAAQ,KAAK,MAAM;AACxB,OAAK,gBAAgB,KAAK,UAAU,MAAM,CAAC;AAC3C,MAAI,KAAK,QAAQ,UAAU,wBAAwB,KAAK,gBAAgB,2BACtE,yBAAwB,KAAK,OAAO,EAAE,WAAW,OAAO,CAAC,CAAC;;CAI9D,AAAQ,iBAAiB,WAAmD;EAC1E,MAAM,eAAe,OAAO;AAC5B,MAAI,CAAC,oBAAoB,aAAa,CACpC;EAGF,MAAM,MAAM,OAAO,SAAS;AAC5B,MAAI,QAAQ,KAAK,YAAY,cAAc,UAAW;AACtD,OAAK,WAAW;AAEhB,OAAK,WAAW;GACd,YAAY;GACZ,aAAa,KAAK,KAAK;GACvB,MAAM;IACJ;IACA,MAAM,OAAO,SAAS;IACtB,UAAU,SAAS;IACnB,OAAO,SAAS;IAChB,YAAY;IACZ,gBAAgB,OAAO;IACvB,iBAAiB,OAAO;IACxB,cAAc,aAAa;IAC3B,eAAe,aAAa;IAC5B,YAAY,OAAO,cAAc,cAAc,UAAU,YAAY;IACtE;GACF,CAAC;;CAGJ,AAAQ,wBAAwB;AAE9B,OAAK,iBAAiB,UAAU;EAChC,MAAM,gBAAgB,OAAO;AAC7B,MAAI,CAAC,kBAAkB,cAAc,CACnC;EAEF,MAAM,oBAAoB,cAAc;EACxC,MAAM,uBAAuB,cAAc;AAG3C,OAAK,sBAAsB,GAAG,SAA2C,kBAAkB,MAAM,eAAe,KAAK;AACrH,gBAAc,aAAa,GAAG,SAA2C;AACvE,QAAK,mBAAoB,GAAG,KAAK;AACjC,QAAK,iBAAiB,OAAO;;AAI/B,OAAK,yBAAyB,GAAG,SAA8C,qBAAqB,MAAM,eAAe,KAAK;AAC9H,gBAAc,gBAAgB,GAAG,SAA8C;AAC7E,QAAK,sBAAuB,GAAG,KAAK;AACpC,QAAK,iBAAiB,UAAU;;AAIlC,SAAO,iBAAiB,YAAY,KAAK,YAAY;;CAOvD,AAAQ,eAAe,SAA0B;EAC/C,MAAM,QAAkB,EAAE;EAC1B,IAAI,UAA0B;EAC9B,IAAI,QAAQ;AAEZ,SAAO,WAAW,QAAQ,KAAK,YAAY,SAAS,iBAAiB;GACnE,IAAI,OAAO,QAAQ,QAAQ,aAAa;GACxC,IAAI,aAAa;GACjB,IAAI,SAAS,QAAQ,aAAa,cAAc;AAChD,OAAI,UAAU,MAAM;AAClB,iBAAa;AACb,aAAS,QAAQ,aAAa,eAAe;;AAE/C,OAAI,UAAU,QAAQ,OAAO,MAAM,KAAK,IAAI;AAC1C,YAAQ,IAAI,WAAW,IAAI,OAAO,QAAQ,MAAM,OAAM,CAAC;AACvD,UAAM,QAAQ,KAAK;AACnB;;AAEF,OAAI,QAAQ,OAAO,IAAI;AACrB,YAAQ,IAAI,eAAe,QAAQ,GAAG;AACtC,UAAM,QAAQ,KAAK;AACnB;;AAEF,OAAI,QAAQ,aAAa,OAAO,QAAQ,cAAc,UAAU;IAC9D,MAAM,UAAU,QAAQ,UAAU,MAAM,CAAC,MAAM,MAAM,CAAC,OAAO,QAAQ,CAAC,MAAM,GAAG,EAAE;AACjF,QAAI,QAAQ,SAAS,EACnB,SAAQ,IAAI,QAAQ,IAAI,eAAe,CAAC,KAAK,IAAI;;GAGrD,MAAM,SAAyB,QAAQ;AACvC,OAAI,UAAU,MAAM;IAClB,MAAM,UAAU,QAAQ;IACxB,MAAM,WAAW,MAAM,KAAK,OAAO,SAAS,CAAC,QAAQ,UAAU,MAAM,YAAY,QAAQ;AACzF,QAAI,SAAS,SAAS,EACpB,SAAQ,gBAAgB,SAAS,QAAQ,QAAQ,GAAG,EAAE;;AAG1D,SAAM,QAAQ,KAAK;AACnB,aAAU;AACV;;AAGF,SAAO,MAAM,KAAK,MAAM;;CAG1B,AAAQ,uBAAuB,SAAiC;EAC9D,IAAI,UAA0B;AAC9B,SAAO,SAAS;AACd,OAAI,QAAQ,YAAY,OAAO,QAAQ,aAAa,OAAO,CACzD,QAAO,QAAQ,aAAa,OAAO;AAErC,aAAU,QAAQ;;AAEpB,SAAO;;CAoDT,AAAQ,qBAAqB;AAC3B,WAAS,iBAAiB,SAAS,KAAK,iBAAiB,EAAE,SAAS,MAAM,CAAC;;CAe7E,AAAQ,2BAA2B;AACjC,MAAI,OAAO,qBAAqB,WAAY;AAE5C,OAAK,6BAA6B,IAAI,kBAAkB,cAAc;AAGpE,OAAI,UAAU,OAAO,aAAa,uBAAuB,SAAS,OAAO,CAAC,CACxE;AAEF,QAAK,oBAAoB,KAAK,KAAK;IACnC;AACF,OAAK,2BAA2B,QAAQ,SAAS,iBAAiB;GAChE,WAAW;GACX,YAAY;GACZ,eAAe;GACf,SAAS;GACV,CAAC;AAIF,WAAS,iBAAiB,UAAU,KAAK,oBAAoB;GAAE,SAAS;GAAM,SAAS;GAAM,CAAC;AAC9F,WAAS,iBAAiB,mBAAmB,KAAK,4BAA4B;AAC9E,WAAS,iBAAiB,oBAAoB,KAAK,6BAA6B;AAEhF,OAAK,kBAAkB,kBAAkB,KAAK,kBAAkB,EAAE,6BAA6B;;CAGjG,AAAQ,mBAAmB;EACzB,MAAM,QAAQ,KAAK,KAAK;AACxB,OAAK,MAAM,SAAS,KAAK,qBAAqB;GAC5C,MAAM,gBAAgB,YAA2B,gBAC/C,cAAc,QAAQ,cAAc,MAAM,eAAe,aAAa,MAAM,cAAc;AAM5F,OAJgB,aAAa,KAAK,iBAAiB,+BAA+B,IAC7E,aAAa,KAAK,2BAA2B,0CAA0C,IACvF,aAAa,KAAK,2BAA2B,0CAA0C,IACvF,aAAa,KAAK,mBAAmB,iCAAiC,CAEzE,MAAK,oBAAoB,OAAO,MAAM;YAC7B,QAAQ,MAAM,eAAe,gCAAgC;AAEtE,UAAM,KAAK,OAAO;AAClB,SAAK,oBAAoB,OAAO,MAAM;;;;CAK5C,AAAQ,8BAA8B;AACpC,MAAI,KAAK,oBAAoB,MAAM;AACjC,iBAAc,KAAK,gBAAgB;AACnC,QAAK,kBAAkB;;AAEzB,MAAI,KAAK,+BAA+B,MAAM;AAC5C,QAAK,2BAA2B,YAAY;AAC5C,QAAK,6BAA6B;;AAEpC,WAAS,oBAAoB,UAAU,KAAK,oBAAoB,EAAE,SAAS,MAAM,CAAC;AAClF,WAAS,oBAAoB,mBAAmB,KAAK,4BAA4B;AACjF,WAAS,oBAAoB,oBAAoB,KAAK,6BAA6B;AACnF,OAAK,oBAAoB,OAAO;;CAOlC,AAAQ,0BAA0B;AAChC,SAAO,iBAAiB,YAAY,KAAK,YAAY;AACrD,WAAS,iBAAiB,oBAAoB,KAAK,YAAY;AAC/D,OAAK,yBAAyB;AAC5B,UAAO,oBAAoB,YAAY,KAAK,YAAY;AACxD,YAAS,oBAAoB,oBAAoB,KAAK,YAAY;;;CAItE,AAAQ,YAAY;AAClB,MAAI,KAAK,kBAAkB;AACzB,QAAK,kBAAkB;AACvB,QAAK,mBAAmB;;EAI1B,MAAM,gBAAgB,OAAO;AAC7B,MAAI,kBAAkB,cAAc,EAAE;AACpC,OAAI,KAAK,mBACP,eAAc,YAAY,KAAK;AAEjC,OAAI,KAAK,sBACP,eAAc,eAAe,KAAK;;AAGtC,OAAK,qBAAqB;AAC1B,OAAK,wBAAwB;AAE7B,SAAO,oBAAoB,YAAY,KAAK,YAAY;AACxD,WAAS,oBAAoB,SAAS,KAAK,iBAAiB,EAAE,SAAS,MAAM,CAAC;AAC9E,OAAK,6BAA6B;AAElC,OAAK,UAAU,EAAE;AACjB,OAAK,eAAe;;CAGtB,MAAc,OAAO,SAAiC;AACpD,MAAI,KAAK,UAAW;AAKpB,MAAI,QAAQ,UACV,MAAK,oBAAoB,OAAO;EAMlC,MAAM,SAAS,KAAK,QAAQ,QAAQ,UAAU,CAAC,KAAK,oBAAoB,IAAI,MAAM,CAAC;AACnF,MAAI,OAAO,WAAW,EAAG;AACzB,OAAK,UAAU,KAAK,QAAQ,QAAQ,UAAU,KAAK,oBAAoB,IAAI,MAAM,CAAC;AAClF,OAAK,eAAe,KAAK,QAAQ,QAAQ,OAAO,UAAU,QAAQ,KAAK,UAAU,MAAM,CAAC,QAAQ,EAAE;EAElG,MAAM,QAAQ,KAAK,KAAK;EAExB,MAAM,UAAU,cAAc;EAC9B,MAAM,UAAU;GACd,2BAA2B,KAAK;GAChC,UAAU;GACV,YAAY;GACZ;GACD;EAED,MAAM,MAAM,MAAM,KAAK,MAAM,UAC3B,KAAK,UAAU,QAAQ,EACvB,EAAE,WAAW,QAAQ,WAAW,CACjC;AAED,MAAI,IAAI,WAAW,SAAS;AAC1B,WAAQ,KAAK,8BAA8B,IAAI,MAAM;AACrD;;AAGF,MAAI,CAAC,IAAI,KAAK,IAAI;AAIhB,QADmB,IAAI,KAAK,QAAQ,IAAI,yBAAyB,IAAI,IAAI,KAAK,QAAQ,IAAI,sBAAsB,MAC7F,yBAAyB;AAC1C,SAAK,YAAY;AACjB,QAAI,KAAK,gBAAgB,MAAM;AAC7B,mBAAc,KAAK,YAAY;AAC/B,UAAK,cAAc;;AAErB,SAAK,WAAW;AAChB;;AAEF,WAAQ,KAAK,8BAA8B,IAAI,KAAK,QAAQ,MAAM,IAAI,KAAK,MAAM,CAAC;;;CAItF,AAAQ,QAAQ;AACd,MAAI,KAAK,WAAY;AACrB,MAAI,KAAK,QAAQ,SAAS,EACxB,yBAAwB,KAAK,OAAO,EAAE,WAAW,OAAO,CAAC,CAAC"}
1
+ {"version":3,"file":"event-tracker.js","names":[],"sources":["../../../../../../src/lib/hexclave-app/apps/implementations/event-tracker.ts"],"sourcesContent":["\n//===========================================\n// THIS FILE IS AUTO-GENERATED FROM TEMPLATE. DO NOT EDIT IT DIRECTLY UNLESS YOU ALSO EDIT THE CORRESPONDING FILE IN packages/template\n//===========================================\nimport { isBrowserLike } from \"@hexclave/shared/dist/utils/env\";\nimport { CLICKMAP_ROOT_ID, DEV_TOOL_ROOT_ID } from \"@hexclave/shared/dist/utils/dev-tool\";\nimport { cssEscapeIdent } from \"@hexclave/shared/dist/utils/dom\";\nimport { buildElementsChain, ELEMENTS_CHAIN_MAX_DEPTH } from \"@hexclave/shared/dist/utils/elements-chain\";\nimport { runAsynchronously } from \"@hexclave/shared/dist/utils/promises\";\nimport { Result } from \"@hexclave/shared/dist/utils/results\";\nimport { generateUuid, isAnalyticsNotEnabledError } from \"./session-replay\";\n\nconst FLUSH_INTERVAL_MS = 10_000;\nconst MAX_EVENTS_PER_BATCH = 50;\nconst MAX_APPROX_BYTES_PER_BATCH = 64_000;\n\nfunction hasScreenDimensions(value: unknown): value is { width: number, height: number } {\n if (value == null || typeof value !== \"object\") {\n return false;\n }\n if (!(\"width\" in value) || !(\"height\" in value)) {\n return false;\n }\n return typeof value.width === \"number\" && typeof value.height === \"number\";\n}\n\nfunction hasHistoryMethods(value: unknown): value is { pushState: History[\"pushState\"], replaceState: History[\"replaceState\"] } {\n if (value == null || typeof value !== \"object\") {\n return false;\n }\n if (!(\"pushState\" in value) || !(\"replaceState\" in value)) {\n return false;\n }\n return typeof value.pushState === \"function\" && typeof value.replaceState === \"function\";\n}\n\nfunction getTextSnippet(textContent: string | null): string {\n return textContent == null ? \"\" : textContent.trim().substring(0, 200);\n}\n\n// Pixel quantization factor for x/y/viewport in stored click events. Matches the\n// SCALE_FACTOR used by the ClickHouse clickmap_events MV — keep them in sync.\nconst CLICKMAP_SCALE_FACTOR = 16;\n\n// Dead-click detection (PostHog-style). Whether an element has a click handler\n// is unknowable from page script, so a click is classified by its observable\n// consequences instead: it is \"alive\" if the page scrolled, the text selection\n// changed, or the tab visibility changed (a new tab opened) almost\n// immediately, or if the DOM mutated within a couple of seconds — and \"dead\"\n// if none of that happened by the absolute timeout.\n//\n// The $click event is buffered immediately like any other event (so\n// event_at_ms, ordering, and every query are untouched) and the sweep sets\n// data.dead=1 on it in place if nothing observable happened. _flush holds\n// back clicks that are still unclassified — classification always finishes\n// well within one FLUSH_INTERVAL_MS, so a held click rides the next flush at\n// the latest. A keepalive flush (pagehide/stop) sends them unmarked: a click\n// still pending when the page unloads led to that navigation, alive by\n// definition.\n//\n// NOTE — blocker for any future real-time / \"live clicks\" view: a click that\n// is still unclassified when its natural flush fires arrives up to one extra\n// FLUSH_INTERVAL_MS late. A surface showing clicks as they happen must either\n// accept that lag or emit a provisional $click plus a later dead-click\n// reconciliation event.\nconst DEAD_CLICK_SCROLL_THRESHOLD_MS = 100;\nconst DEAD_CLICK_SELECTION_CHANGED_THRESHOLD_MS = 100;\nconst DEAD_CLICK_VISIBILITY_CHANGE_THRESHOLD_MS = 100;\nconst DEAD_CLICK_MUTATION_THRESHOLD_MS = 2_500;\n// 1.1x the mutation threshold, mirroring posthog-js: every signal window has\n// closed before a click is declared dead.\nconst DEAD_CLICK_ABSOLUTE_TIMEOUT_MS = 2_750;\nconst DEAD_CLICK_CHECK_INTERVAL_MS = 1_000;\n// Backstop against click storms (e.g. rage clicks on a dead element): past the\n// cap, clicks are simply not classified rather than not recorded.\nconst DEAD_CLICK_MAX_PENDING = 50;\n\nfunction isPointerTargetFixed(element: Element): boolean {\n let current: Element | null = element;\n let depth = 0;\n while (current != null && depth < ELEMENTS_CHAIN_MAX_DEPTH * 2) {\n const style = window.getComputedStyle(current);\n if (style.position === \"fixed\" || style.position === \"sticky\") {\n return true;\n }\n current = current.parentElement;\n depth += 1;\n }\n return false;\n}\n\n// Clicks on Hexclave's own in-page UI (the dev tool and the standalone\n// clickmap overlay) must never be ingested as analytics events.\nfunction isInsideHexclaveUi(element: Element): boolean {\n return element.closest(`#${cssEscapeIdent(DEV_TOOL_ROOT_ID)}, #${cssEscapeIdent(CLICKMAP_ROOT_ID)}`) != null;\n}\n\n// Mutation-record targets can be text/comment nodes; resolve to the nearest\n// element before asking whether the mutation came from Hexclave's own UI.\nfunction isInsideHexclaveUiNode(node: Node | null): boolean {\n const element = node instanceof Element ? node : node?.parentElement ?? null;\n return element != null && isInsideHexclaveUi(element);\n}\n\nexport type EventTrackerDeps = {\n projectId: string,\n sendBatch: (body: string, options: { keepalive: boolean }) => Promise<Result<Response, Error>>,\n};\n\ntype TrackedEvent = {\n event_type: \"$page-view\" | \"$click\",\n event_at_ms: number,\n data: Record<string, unknown>,\n};\n\nexport class EventTracker {\n private _started = false;\n private _cancelled = false;\n private _disabled = false;\n private _detachListeners: (() => void) | null = null;\n private _flushTimer: ReturnType<typeof setInterval> | null = null;\n private _events: TrackedEvent[] = [];\n private _approxBytes = 0;\n private _lastUrl: string | null = null;\n private readonly _sessionReplaySegmentId: string;\n private readonly _deps: EventTrackerDeps;\n\n private _originalPushState: History[\"pushState\"] | null = null;\n private _originalReplaceState: History[\"replaceState\"] | null = null;\n\n private _deadClickTimer: ReturnType<typeof setInterval> | null = null;\n private _deadClickMutationObserver: MutationObserver | null = null;\n // Buffered $click events still awaiting dead-click classification. Always a\n // subset of _events — _flush holds these back until the sweep resolves them.\n private _unclassifiedClicks = new Set<TrackedEvent>();\n private _lastMutationAtMs: number | null = null;\n private _lastScrollAtMs: number | null = null;\n private _lastSelectionChangedAtMs: number | null = null;\n private _lastVisibilityChangeAtMs: number | null = null;\n\n constructor(deps: EventTrackerDeps) {\n this._deps = deps;\n this._sessionReplaySegmentId = generateUuid();\n }\n\n start() {\n if (this._started) return;\n if (!isBrowserLike()) return;\n if (\n typeof window.addEventListener !== \"function\"\n || typeof window.removeEventListener !== \"function\"\n || typeof document.addEventListener !== \"function\"\n || typeof document.removeEventListener !== \"function\"\n || !hasScreenDimensions(window.screen)\n ) {\n return;\n }\n this._started = true;\n\n this._setupPageViewCapture();\n this._setupClickCapture();\n this._setupDeadClickDetection();\n this._setupPageHideListeners();\n\n this._flushTimer = setInterval(() => this._tick(), FLUSH_INTERVAL_MS);\n }\n\n stop() {\n this._cancelled = true;\n if (this._flushTimer !== null) {\n clearInterval(this._flushTimer);\n this._flushTimer = null;\n }\n runAsynchronously(() => this._flush({ keepalive: true }));\n this._teardown();\n }\n\n clearBuffer() {\n this._events = [];\n this._approxBytes = 0;\n this._unclassifiedClicks.clear();\n }\n\n private _pushEvent(event: TrackedEvent) {\n if (this._disabled) return;\n this._events.push(event);\n this._approxBytes += JSON.stringify(event).length;\n if (this._events.length >= MAX_EVENTS_PER_BATCH || this._approxBytes >= MAX_APPROX_BYTES_PER_BATCH) {\n runAsynchronously(() => this._flush({ keepalive: false }));\n }\n }\n\n private _capturePageView(entryType: \"initial\" | \"push\" | \"replace\" | \"pop\") {\n const screenObject = window.screen;\n if (!hasScreenDimensions(screenObject)) {\n return;\n }\n\n const url = window.location.href;\n if (url === this._lastUrl && entryType !== \"initial\") return;\n this._lastUrl = url;\n\n this._pushEvent({\n event_type: \"$page-view\",\n event_at_ms: Date.now(),\n data: {\n url,\n path: window.location.pathname,\n referrer: document.referrer,\n title: document.title,\n entry_type: entryType,\n viewport_width: window.innerWidth,\n viewport_height: window.innerHeight,\n screen_width: screenObject.width,\n screen_height: screenObject.height,\n user_agent: typeof navigator !== \"undefined\" ? navigator.userAgent : null,\n },\n });\n }\n\n private _setupPageViewCapture() {\n // Fire initial page-view\n this._capturePageView(\"initial\");\n const historyObject = window.history;\n if (!hasHistoryMethods(historyObject)) {\n return;\n }\n const originalPushState = historyObject.pushState;\n const originalReplaceState = historyObject.replaceState;\n\n // Monkey-patch history.pushState\n this._originalPushState = (...args: Parameters<History[\"pushState\"]>) => originalPushState.apply(historyObject, args);\n historyObject.pushState = (...args: Parameters<History[\"pushState\"]>) => {\n this._originalPushState!(...args);\n this._capturePageView(\"push\");\n };\n\n // Monkey-patch history.replaceState\n this._originalReplaceState = (...args: Parameters<History[\"replaceState\"]>) => originalReplaceState.apply(historyObject, args);\n historyObject.replaceState = (...args: Parameters<History[\"replaceState\"]>) => {\n this._originalReplaceState!(...args);\n this._capturePageView(\"replace\");\n };\n\n // Listen for popstate (back/forward navigation)\n window.addEventListener(\"popstate\", this._onPopState);\n }\n\n private readonly _onPopState = () => {\n this._capturePageView(\"pop\");\n };\n\n private _buildSelector(element: Element): string {\n const parts: string[] = [];\n let current: Element | null = element;\n let depth = 0;\n\n while (current && depth < 8 && current !== document.documentElement) {\n let part = current.tagName.toLowerCase();\n let testIdAttr = \"data-testid\";\n let testId = current.getAttribute(\"data-testid\");\n if (testId == null) {\n testIdAttr = \"data-test-id\";\n testId = current.getAttribute(\"data-test-id\");\n }\n if (testId != null && testId.trim() !== \"\") {\n part += `[${testIdAttr}=\"${testId.replace(/\"/g, '\\\\\"')}\"]`;\n parts.unshift(part);\n break;\n }\n if (current.id !== \"\") {\n part += `#${cssEscapeIdent(current.id)}`;\n parts.unshift(part);\n break;\n }\n if (current.className && typeof current.className === \"string\") {\n const classes = current.className.trim().split(/\\s+/).filter(Boolean).slice(0, 4);\n if (classes.length > 0) {\n part += `.${classes.map(cssEscapeIdent).join(\".\")}`;\n }\n }\n const parent: Element | null = current.parentElement;\n if (parent != null) {\n const tagName = current.tagName;\n const siblings = Array.from(parent.children).filter((child) => child.tagName === tagName);\n if (siblings.length > 1) {\n part += `:nth-of-type(${siblings.indexOf(current) + 1})`;\n }\n }\n parts.unshift(part);\n current = parent;\n depth++;\n }\n\n return parts.join(\" > \");\n }\n\n private _findNearestAnchorHref(element: Element): string | null {\n let current: Element | null = element;\n while (current) {\n if (current.tagName === \"A\" && current.hasAttribute(\"href\")) {\n return current.getAttribute(\"href\");\n }\n current = current.parentElement;\n }\n return null;\n }\n\n private readonly _onClickCapture = (event: MouseEvent) => {\n const target = event.target;\n if (!(target instanceof Element)) return;\n if (isInsideHexclaveUi(target)) return;\n\n const viewportWidth = window.innerWidth;\n const viewportHeight = window.innerHeight;\n const pointerTargetFixed = isPointerTargetFixed(target);\n // Pre-scale at ingest so old + new rows land in identical buckets in CH.\n const xScaled = Math.round(event.pageX / CLICKMAP_SCALE_FACTOR);\n const yScaled = Math.round(event.pageY / CLICKMAP_SCALE_FACTOR);\n const clientYScaled = Math.round(event.clientY / CLICKMAP_SCALE_FACTOR);\n const relativeX = viewportWidth > 0 ? event.clientX / viewportWidth : 0;\n\n const clickEvent: TrackedEvent = {\n event_type: \"$click\",\n event_at_ms: Date.now(),\n data: {\n tag_name: target.tagName.toLowerCase(),\n text: getTextSnippet(target.textContent),\n href: this._findNearestAnchorHref(target),\n selector: this._buildSelector(target),\n elements_chain: buildElementsChain(target),\n pointer_target_fixed: pointerTargetFixed ? 1 : 0,\n url: window.location.href,\n path: window.location.pathname,\n title: document.title,\n x: event.clientX,\n y: event.clientY,\n page_x: event.pageX,\n page_y: event.pageY,\n x_scaled: xScaled,\n y_scaled: yScaled,\n client_y_scaled: clientYScaled,\n pointer_relative_x: relativeX,\n viewport_width: viewportWidth,\n viewport_height: viewportHeight,\n scale_factor: CLICKMAP_SCALE_FACTOR,\n },\n };\n\n // Register for dead-click classification before buffering, so a\n // size-triggered flush from this very push already holds the click back.\n if (this._deadClickTimer !== null && this._unclassifiedClicks.size < DEAD_CLICK_MAX_PENDING) {\n this._unclassifiedClicks.add(clickEvent);\n }\n this._pushEvent(clickEvent);\n };\n\n private _setupClickCapture() {\n document.addEventListener(\"click\", this._onClickCapture, { capture: true });\n }\n\n private readonly _onDeadClickScroll = () => {\n this._lastScrollAtMs = Date.now();\n };\n\n private readonly _onDeadClickSelectionChange = () => {\n this._lastSelectionChangedAtMs = Date.now();\n };\n\n private readonly _onDeadClickVisibilityChange = () => {\n this._lastVisibilityChangeAtMs = Date.now();\n };\n\n private _setupDeadClickDetection() {\n if (typeof MutationObserver !== \"function\") return;\n\n this._deadClickMutationObserver = new MutationObserver((mutations) => {\n // The dev tool and the clickmap overlay rewrite their own DOM constantly\n // while open; their mutations must not mark host-page clicks as alive.\n if (mutations.every((mutation) => isInsideHexclaveUiNode(mutation.target))) {\n return;\n }\n this._lastMutationAtMs = Date.now();\n });\n this._deadClickMutationObserver.observe(document.documentElement, {\n childList: true,\n attributes: true,\n characterData: true,\n subtree: true,\n });\n\n // Capture phase so scrolls inside nested scroll containers count, not just\n // the document itself (scroll events don't bubble).\n document.addEventListener(\"scroll\", this._onDeadClickScroll, { capture: true, passive: true });\n document.addEventListener(\"selectionchange\", this._onDeadClickSelectionChange);\n document.addEventListener(\"visibilitychange\", this._onDeadClickVisibilityChange);\n\n this._deadClickTimer = setInterval(() => this._checkDeadClicks(), DEAD_CLICK_CHECK_INTERVAL_MS);\n }\n\n private _checkDeadClicks() {\n const nowMs = Date.now();\n for (const click of this._unclassifiedClicks) {\n const signalWithin = (signalAtMs: number | null, thresholdMs: number) =>\n signalAtMs != null && signalAtMs >= click.event_at_ms && signalAtMs - click.event_at_ms < thresholdMs;\n\n const isAlive = signalWithin(this._lastScrollAtMs, DEAD_CLICK_SCROLL_THRESHOLD_MS)\n || signalWithin(this._lastSelectionChangedAtMs, DEAD_CLICK_SELECTION_CHANGED_THRESHOLD_MS)\n || signalWithin(this._lastVisibilityChangeAtMs, DEAD_CLICK_VISIBILITY_CHANGE_THRESHOLD_MS)\n || signalWithin(this._lastMutationAtMs, DEAD_CLICK_MUTATION_THRESHOLD_MS);\n if (isAlive) {\n this._unclassifiedClicks.delete(click);\n } else if (nowMs - click.event_at_ms >= DEAD_CLICK_ABSOLUTE_TIMEOUT_MS) {\n // The already-buffered event is marked in place — no second event.\n click.data.dead = 1;\n this._unclassifiedClicks.delete(click);\n }\n }\n }\n\n private _teardownDeadClickDetection() {\n if (this._deadClickTimer !== null) {\n clearInterval(this._deadClickTimer);\n this._deadClickTimer = null;\n }\n if (this._deadClickMutationObserver !== null) {\n this._deadClickMutationObserver.disconnect();\n this._deadClickMutationObserver = null;\n }\n document.removeEventListener(\"scroll\", this._onDeadClickScroll, { capture: true });\n document.removeEventListener(\"selectionchange\", this._onDeadClickSelectionChange);\n document.removeEventListener(\"visibilitychange\", this._onDeadClickVisibilityChange);\n this._unclassifiedClicks.clear();\n }\n\n private readonly _onPageHide = () => {\n runAsynchronously(() => this._flush({ keepalive: true }));\n };\n\n private _setupPageHideListeners() {\n window.addEventListener(\"pagehide\", this._onPageHide);\n document.addEventListener(\"visibilitychange\", this._onPageHide);\n this._detachListeners = () => {\n window.removeEventListener(\"pagehide\", this._onPageHide);\n document.removeEventListener(\"visibilitychange\", this._onPageHide);\n };\n }\n\n private _teardown() {\n if (this._detachListeners) {\n this._detachListeners();\n this._detachListeners = null;\n }\n\n // Restore history methods\n const historyObject = window.history;\n if (hasHistoryMethods(historyObject)) {\n if (this._originalPushState) {\n historyObject.pushState = this._originalPushState;\n }\n if (this._originalReplaceState) {\n historyObject.replaceState = this._originalReplaceState;\n }\n }\n this._originalPushState = null;\n this._originalReplaceState = null;\n\n window.removeEventListener(\"popstate\", this._onPopState);\n document.removeEventListener(\"click\", this._onClickCapture, { capture: true });\n this._teardownDeadClickDetection();\n\n this._events = [];\n this._approxBytes = 0;\n }\n\n private async _flush(options: { keepalive: boolean }) {\n if (this._disabled) return;\n\n // A keepalive flush means the page is unloading — a click still awaiting\n // dead-click classification led to that unload, so it is alive by\n // definition and ships unmarked.\n if (options.keepalive) {\n this._unclassifiedClicks.clear();\n }\n\n // Clicks still awaiting classification stay buffered so the sweep can\n // mark them dead in place; classification finishes well within one flush\n // interval, so they ride the next flush at the latest.\n const events = this._events.filter((event) => !this._unclassifiedClicks.has(event));\n if (events.length === 0) return;\n this._events = this._events.filter((event) => this._unclassifiedClicks.has(event));\n this._approxBytes = this._events.reduce((total, event) => total + JSON.stringify(event).length, 0);\n\n const nowMs = Date.now();\n\n const batchId = generateUuid();\n const payload = {\n session_replay_segment_id: this._sessionReplaySegmentId,\n batch_id: batchId,\n sent_at_ms: nowMs,\n events,\n };\n\n const res = await this._deps.sendBatch(\n JSON.stringify(payload),\n { keepalive: options.keepalive },\n );\n\n if (res.status === \"error\") {\n if (isAnalyticsNotEnabledError(res.error)) {\n this._disable();\n return;\n }\n console.warn(\"EventTracker flush failed:\", res.error);\n return;\n }\n\n if (!res.data.ok) {\n console.warn(\"EventTracker flush failed:\", res.data.status, await res.data.text());\n }\n }\n\n private _disable() {\n this._disabled = true;\n if (this._flushTimer !== null) {\n clearInterval(this._flushTimer);\n this._flushTimer = null;\n }\n this._teardown();\n }\n\n private _tick() {\n if (this._cancelled) return;\n if (this._events.length > 0) {\n runAsynchronously(() => this._flush({ keepalive: false }));\n }\n }\n}\n"],"mappings":"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"}
@@ -1,5 +1,6 @@
1
1
  import { afterEach, describe, expect, it, vi } from "vitest";
2
2
  import { Result } from "@hexclave/shared/dist/utils/results";
3
+ import { KnownErrors } from "@hexclave/shared/dist/known-errors";
3
4
  import { EventTracker } from "./event-tracker.js";
4
5
 
5
6
  //#region src/lib/hexclave-app/apps/implementations/event-tracker.test.ts
@@ -301,7 +302,7 @@ describe("EventTracker", () => {
301
302
  tracker.stop();
302
303
  }
303
304
  });
304
- it("silently disables when server responds with ANALYTICS_NOT_ENABLED", async () => {
305
+ it("silently disables when client interface returns ANALYTICS_NOT_ENABLED as an error", async () => {
305
306
  vi.useFakeTimers();
306
307
  document.body.innerHTML = "<button>Click me</button>";
307
308
  const warnSpy = vi.spyOn(console, "warn").mockImplementation(() => {});
@@ -310,13 +311,7 @@ describe("EventTracker", () => {
310
311
  projectId: "internal",
311
312
  sendBatch: async (body) => {
312
313
  sentBodies.push(body);
313
- return Result.ok(new Response(JSON.stringify({
314
- code: "ANALYTICS_NOT_ENABLED",
315
- error: "Analytics is not enabled for this project."
316
- }), {
317
- status: 400,
318
- headers: { "x-stack-known-error": "ANALYTICS_NOT_ENABLED" }
319
- }));
314
+ return Result.error(new KnownErrors.AnalyticsNotEnabled());
320
315
  }
321
316
  });
322
317
  try {
@@ -324,6 +319,7 @@ describe("EventTracker", () => {
324
319
  await advancePastFlush();
325
320
  expect(sentBodies).toHaveLength(1);
326
321
  expect(warnSpy).not.toHaveBeenCalled();
322
+ expect(tracker._flushTimer).toBeNull();
327
323
  document.querySelector("button")?.dispatchEvent(new MouseEvent("click", { bubbles: true }));
328
324
  await advancePastFlush();
329
325
  expect(sentBodies).toHaveLength(1);
@@ -1 +1 @@
1
- {"version":3,"file":"event-tracker.test.js","names":[],"sources":["../../../../../../src/lib/hexclave-app/apps/implementations/event-tracker.test.ts"],"sourcesContent":["\n//===========================================\n// THIS FILE IS AUTO-GENERATED FROM TEMPLATE. DO NOT EDIT IT DIRECTLY UNLESS YOU ALSO EDIT THE CORRESPONDING FILE IN packages/template\n//===========================================\n// @vitest-environment jsdom\n\nimport { Result } from \"@hexclave/shared/dist/utils/results\";\nimport { afterEach, describe, expect, it, vi } from \"vitest\";\nimport { EventTracker } from \"./event-tracker\";\n\nasync function advancePastFlush() {\n await vi.advanceTimersByTimeAsync(10_000);\n await Promise.resolve();\n}\n\nfunction getSentEventTypes(sentBodies: string[]) {\n const [body] = sentBodies;\n\n const payload = JSON.parse(body);\n if (typeof payload !== \"object\" || payload === null || !(\"events\" in payload) || !Array.isArray(payload.events)) {\n throw new Error(\"Expected analytics batch payload to include an events array.\");\n }\n\n return (payload.events as { event_type: string }[]).map((event) => event.event_type);\n}\n\ndescribe(\"EventTracker\", () => {\n afterEach(() => {\n vi.useRealTimers();\n });\n\n it(\"captures events when browser globals are exposed as accessor descriptors\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button>Open project</button>\";\n\n const screenDescriptor = Object.getOwnPropertyDescriptor(window, \"screen\");\n const historyDescriptor = Object.getOwnPropertyDescriptor(window, \"history\");\n expect(screenDescriptor?.value).toBeUndefined();\n expect(historyDescriptor?.value).toBeUndefined();\n expect(screenDescriptor?.get).toBeTypeOf(\"function\");\n expect(historyDescriptor?.get).toBeTypeOf(\"function\");\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n document.querySelector(\"button\")?.dispatchEvent(new MouseEvent(\"click\", {\n bubbles: true,\n clientX: 12,\n clientY: 34,\n }));\n\n await advancePastFlush();\n\n // Dead-click classification marks the buffered $click in place —\n // exactly one click event either way.\n expect(getSentEventTypes(sentBodies)).toMatchInlineSnapshot(`\n [\n \"$page-view\",\n \"$click\",\n ]\n `);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"emits a PostHog-style elements_chain plus scaled pointer coords for $click\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = `\n <main>\n <section class=\"card panel\">\n <button id=\"save-btn\" data-testid=\"save\" aria-label=\"Save project\">Save changes</button>\n </section>\n </main>\n `;\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n const button = document.querySelector(\"#save-btn\");\n if (button == null) throw new Error(\"button missing\");\n button.dispatchEvent(new MouseEvent(\"click\", {\n bubbles: true,\n clientX: 100,\n clientY: 200,\n }));\n\n await advancePastFlush();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, data: Record<string, unknown> }[] };\n const click = payload.events.find((event) => event.event_type === \"$click\");\n if (click == null) throw new Error(\"no $click event captured\");\n\n // elements_chain encodes the target leaf plus a few ancestors. Leaf is\n // first; segments are `;`-delimited. Assert against substrings rather\n // than the full string so jsdom layout quirks don't make this flaky.\n const chain = click.data.elements_chain;\n expect(typeof chain).toBe(\"string\");\n expect(chain).toContain('button');\n expect(chain).toContain('attr__id=\"save-btn\"');\n expect(chain).toContain('attr__data-testid=\"save\"');\n expect(chain).toContain('attr__aria-label=\"Save project\"');\n expect(chain).toContain('text=\"Save changes\"');\n // Ancestor section is in the chain too.\n expect(chain).toContain(\"section\");\n\n // Pre-scaled coords land in clickmap_events.pointer_*. SCALE_FACTOR=16.\n expect(click.data.x_scaled).toBe(Math.round(100 / 16));\n expect(click.data.y_scaled).toBe(Math.round(200 / 16));\n expect(click.data.client_y_scaled).toBe(Math.round(200 / 16));\n expect(click.data.scale_factor).toBe(16);\n expect(click.data.pointer_relative_x).toBeCloseTo(100 / window.innerWidth, 4);\n expect(click.data.pointer_target_fixed).toBe(0);\n\n // Legacy CSS selector still emitted for back-compat. The builder prefers\n // data-testid over id, so we assert against that anchor rather than #id.\n expect(click.data.selector).toContain('data-testid=\"save\"');\n expect(click.data.tag_name).toBe(\"button\");\n } finally {\n tracker.stop();\n }\n });\n\n it(\"ignores clicks inside the Hexclave dev tool\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = `\n <div id=\"__hexclave-dev-tool-root\">\n <button>Clickmap toolbar control</button>\n </div>\n `;\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n document.querySelector(\"button\")?.dispatchEvent(new MouseEvent(\"click\", {\n bubbles: true,\n clientX: 100,\n clientY: 200,\n }));\n\n await advancePastFlush();\n\n expect(getSentEventTypes(sentBodies)).toMatchInlineSnapshot(`\n [\n \"$page-view\",\n ]\n `);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"flags pointer_target_fixed when the target sits under a fixed-position ancestor\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = `\n <header style=\"position: fixed; top: 0\">\n <button id=\"cta\">Sign up</button>\n </header>\n `;\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n document.querySelector(\"#cta\")?.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n await advancePastFlush();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, data: Record<string, unknown> }[] };\n const click = payload.events.find((event) => event.event_type === \"$click\");\n expect(click?.data.pointer_target_fixed).toBe(1);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"flags a click with no observable effect as dead on its single $click event\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button id=\\\"dead\\\">Does nothing</button>\";\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n const clickAtMs = Date.now();\n document.querySelector(\"#dead\")?.dispatchEvent(new MouseEvent(\"click\", {\n bubbles: true,\n clientX: 10,\n clientY: 20,\n }));\n\n await advancePastFlush();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, event_at_ms: number, data: Record<string, unknown> }[] };\n const clicks = payload.events.filter((event) => event.event_type === \"$click\");\n expect(clicks).toHaveLength(1);\n const click = clicks[0];\n\n // One event per physical click: the buffered $click is marked dead in\n // place, still timestamped at the original click rather than at\n // classification time (~3s later).\n expect(click.data.dead).toBe(1);\n expect(click.event_at_ms).toBe(clickAtMs);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"does not flag a click as dead when it mutates the DOM\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button id=\\\"live\\\">Adds content</button><div id=\\\"out\\\"></div>\";\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n const button = document.querySelector(\"#live\");\n if (button == null) throw new Error(\"button missing\");\n button.addEventListener(\"click\", () => {\n document.querySelector(\"#out\")?.appendChild(document.createElement(\"p\"));\n });\n button.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n // Let the MutationObserver microtask run so the mutation is recorded\n // before the dead-click sweeps start.\n await Promise.resolve();\n\n await advancePastFlush();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, data: Record<string, unknown> }[] };\n const clicks = payload.events.filter((event) => event.event_type === \"$click\");\n expect(clicks).toHaveLength(1);\n expect(clicks[0].data.dead).toBeUndefined();\n } finally {\n tracker.stop();\n }\n });\n\n it(\"drains held clicks as alive on pagehide so navigation clicks are never lost\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<a id=\\\"nav\\\" href=\\\"/pricing\\\">Pricing</a>\";\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n const clickAtMs = Date.now();\n document.querySelector(\"#nav\")?.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n\n // Navigation fires pagehide well before any classification sweep — the\n // keepalive flush ships the still-unclassified click as a plain (alive)\n // $click.\n window.dispatchEvent(new Event(\"pagehide\"));\n await Promise.resolve();\n await Promise.resolve();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, event_at_ms: number, data: Record<string, unknown> }[] };\n const clicks = payload.events.filter((event) => event.event_type === \"$click\");\n expect(clicks).toHaveLength(1);\n expect(clicks[0].data.dead).toBeUndefined();\n expect(clicks[0].event_at_ms).toBe(clickAtMs);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"holds an unclassified click out of a flush and ships it on the next one\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button id=\\\"late\\\">Late click</button>\";\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n // Click 500ms before the 10s flush tick: classification cannot finish\n // in time, so the flush must hold the click back rather than send it\n // unclassified.\n await vi.advanceTimersByTimeAsync(9_500);\n document.querySelector(\"#late\")?.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n await vi.advanceTimersByTimeAsync(500);\n\n expect(getSentEventTypes(sentBodies)).toMatchInlineSnapshot(`\n [\n \"$page-view\",\n ]\n `);\n\n // By the next flush the sweep has classified it (dead — nothing\n // observable happened) and it ships marked.\n await vi.advanceTimersByTimeAsync(10_000);\n const second = JSON.parse(sentBodies[1] ?? \"{}\") as { events: { event_type: string, data: Record<string, unknown> }[] };\n expect(second.events.map((event) => event.event_type)).toMatchInlineSnapshot(`\n [\n \"$click\",\n ]\n `);\n expect(second.events[0].data.dead).toBe(1);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"captures client-side navigations when history is exposed as an accessor descriptor\", async () => {\n vi.useFakeTimers();\n\n const historyDescriptor = Object.getOwnPropertyDescriptor(window, \"history\");\n expect(historyDescriptor?.value).toBeUndefined();\n expect(historyDescriptor?.get).toBeTypeOf(\"function\");\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n window.history.pushState({}, \"\", \"/projects/test-project\");\n\n await advancePastFlush();\n\n expect(getSentEventTypes(sentBodies)).toMatchInlineSnapshot(`\n [\n \"$page-view\",\n \"$page-view\",\n ]\n `);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"silently disables when server responds with ANALYTICS_NOT_ENABLED\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button>Click me</button>\";\n\n const warnSpy = vi.spyOn(console, \"warn\").mockImplementation(() => {});\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response(\n JSON.stringify({ code: \"ANALYTICS_NOT_ENABLED\", error: \"Analytics is not enabled for this project.\" }),\n {\n status: 400,\n headers: { \"x-stack-known-error\": \"ANALYTICS_NOT_ENABLED\" },\n },\n ));\n },\n });\n\n try {\n tracker.start();\n\n // First flush sends the initial page-view event; server rejects it.\n await advancePastFlush();\n expect(sentBodies).toHaveLength(1);\n\n // No console.warn should have been emitted.\n expect(warnSpy).not.toHaveBeenCalled();\n\n // After disabling, new events should not accumulate or trigger further\n // flushes.\n document.querySelector(\"button\")?.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n await advancePastFlush();\n expect(sentBodies).toHaveLength(1);\n } finally {\n tracker.stop();\n warnSpy.mockRestore();\n }\n });\n});\n"],"mappings":"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"}
1
+ {"version":3,"file":"event-tracker.test.js","names":[],"sources":["../../../../../../src/lib/hexclave-app/apps/implementations/event-tracker.test.ts"],"sourcesContent":["\n//===========================================\n// THIS FILE IS AUTO-GENERATED FROM TEMPLATE. DO NOT EDIT IT DIRECTLY UNLESS YOU ALSO EDIT THE CORRESPONDING FILE IN packages/template\n//===========================================\n// @vitest-environment jsdom\n\nimport { KnownErrors } from \"@hexclave/shared/dist/known-errors\";\nimport { Result } from \"@hexclave/shared/dist/utils/results\";\nimport { afterEach, describe, expect, it, vi } from \"vitest\";\nimport { EventTracker } from \"./event-tracker\";\n\nasync function advancePastFlush() {\n await vi.advanceTimersByTimeAsync(10_000);\n await Promise.resolve();\n}\n\nfunction getSentEventTypes(sentBodies: string[]) {\n const [body] = sentBodies;\n\n const payload = JSON.parse(body);\n if (typeof payload !== \"object\" || payload === null || !(\"events\" in payload) || !Array.isArray(payload.events)) {\n throw new Error(\"Expected analytics batch payload to include an events array.\");\n }\n\n return (payload.events as { event_type: string }[]).map((event) => event.event_type);\n}\n\ndescribe(\"EventTracker\", () => {\n afterEach(() => {\n vi.useRealTimers();\n });\n\n it(\"captures events when browser globals are exposed as accessor descriptors\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button>Open project</button>\";\n\n const screenDescriptor = Object.getOwnPropertyDescriptor(window, \"screen\");\n const historyDescriptor = Object.getOwnPropertyDescriptor(window, \"history\");\n expect(screenDescriptor?.value).toBeUndefined();\n expect(historyDescriptor?.value).toBeUndefined();\n expect(screenDescriptor?.get).toBeTypeOf(\"function\");\n expect(historyDescriptor?.get).toBeTypeOf(\"function\");\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n document.querySelector(\"button\")?.dispatchEvent(new MouseEvent(\"click\", {\n bubbles: true,\n clientX: 12,\n clientY: 34,\n }));\n\n await advancePastFlush();\n\n // Dead-click classification marks the buffered $click in place —\n // exactly one click event either way.\n expect(getSentEventTypes(sentBodies)).toMatchInlineSnapshot(`\n [\n \"$page-view\",\n \"$click\",\n ]\n `);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"emits a PostHog-style elements_chain plus scaled pointer coords for $click\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = `\n <main>\n <section class=\"card panel\">\n <button id=\"save-btn\" data-testid=\"save\" aria-label=\"Save project\">Save changes</button>\n </section>\n </main>\n `;\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n const button = document.querySelector(\"#save-btn\");\n if (button == null) throw new Error(\"button missing\");\n button.dispatchEvent(new MouseEvent(\"click\", {\n bubbles: true,\n clientX: 100,\n clientY: 200,\n }));\n\n await advancePastFlush();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, data: Record<string, unknown> }[] };\n const click = payload.events.find((event) => event.event_type === \"$click\");\n if (click == null) throw new Error(\"no $click event captured\");\n\n // elements_chain encodes the target leaf plus a few ancestors. Leaf is\n // first; segments are `;`-delimited. Assert against substrings rather\n // than the full string so jsdom layout quirks don't make this flaky.\n const chain = click.data.elements_chain;\n expect(typeof chain).toBe(\"string\");\n expect(chain).toContain('button');\n expect(chain).toContain('attr__id=\"save-btn\"');\n expect(chain).toContain('attr__data-testid=\"save\"');\n expect(chain).toContain('attr__aria-label=\"Save project\"');\n expect(chain).toContain('text=\"Save changes\"');\n // Ancestor section is in the chain too.\n expect(chain).toContain(\"section\");\n\n // Pre-scaled coords land in clickmap_events.pointer_*. SCALE_FACTOR=16.\n expect(click.data.x_scaled).toBe(Math.round(100 / 16));\n expect(click.data.y_scaled).toBe(Math.round(200 / 16));\n expect(click.data.client_y_scaled).toBe(Math.round(200 / 16));\n expect(click.data.scale_factor).toBe(16);\n expect(click.data.pointer_relative_x).toBeCloseTo(100 / window.innerWidth, 4);\n expect(click.data.pointer_target_fixed).toBe(0);\n\n // Legacy CSS selector still emitted for back-compat. The builder prefers\n // data-testid over id, so we assert against that anchor rather than #id.\n expect(click.data.selector).toContain('data-testid=\"save\"');\n expect(click.data.tag_name).toBe(\"button\");\n } finally {\n tracker.stop();\n }\n });\n\n it(\"ignores clicks inside the Hexclave dev tool\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = `\n <div id=\"__hexclave-dev-tool-root\">\n <button>Clickmap toolbar control</button>\n </div>\n `;\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n document.querySelector(\"button\")?.dispatchEvent(new MouseEvent(\"click\", {\n bubbles: true,\n clientX: 100,\n clientY: 200,\n }));\n\n await advancePastFlush();\n\n expect(getSentEventTypes(sentBodies)).toMatchInlineSnapshot(`\n [\n \"$page-view\",\n ]\n `);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"flags pointer_target_fixed when the target sits under a fixed-position ancestor\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = `\n <header style=\"position: fixed; top: 0\">\n <button id=\"cta\">Sign up</button>\n </header>\n `;\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n document.querySelector(\"#cta\")?.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n await advancePastFlush();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, data: Record<string, unknown> }[] };\n const click = payload.events.find((event) => event.event_type === \"$click\");\n expect(click?.data.pointer_target_fixed).toBe(1);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"flags a click with no observable effect as dead on its single $click event\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button id=\\\"dead\\\">Does nothing</button>\";\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n const clickAtMs = Date.now();\n document.querySelector(\"#dead\")?.dispatchEvent(new MouseEvent(\"click\", {\n bubbles: true,\n clientX: 10,\n clientY: 20,\n }));\n\n await advancePastFlush();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, event_at_ms: number, data: Record<string, unknown> }[] };\n const clicks = payload.events.filter((event) => event.event_type === \"$click\");\n expect(clicks).toHaveLength(1);\n const click = clicks[0];\n\n // One event per physical click: the buffered $click is marked dead in\n // place, still timestamped at the original click rather than at\n // classification time (~3s later).\n expect(click.data.dead).toBe(1);\n expect(click.event_at_ms).toBe(clickAtMs);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"does not flag a click as dead when it mutates the DOM\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button id=\\\"live\\\">Adds content</button><div id=\\\"out\\\"></div>\";\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n const button = document.querySelector(\"#live\");\n if (button == null) throw new Error(\"button missing\");\n button.addEventListener(\"click\", () => {\n document.querySelector(\"#out\")?.appendChild(document.createElement(\"p\"));\n });\n button.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n // Let the MutationObserver microtask run so the mutation is recorded\n // before the dead-click sweeps start.\n await Promise.resolve();\n\n await advancePastFlush();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, data: Record<string, unknown> }[] };\n const clicks = payload.events.filter((event) => event.event_type === \"$click\");\n expect(clicks).toHaveLength(1);\n expect(clicks[0].data.dead).toBeUndefined();\n } finally {\n tracker.stop();\n }\n });\n\n it(\"drains held clicks as alive on pagehide so navigation clicks are never lost\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<a id=\\\"nav\\\" href=\\\"/pricing\\\">Pricing</a>\";\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n const clickAtMs = Date.now();\n document.querySelector(\"#nav\")?.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n\n // Navigation fires pagehide well before any classification sweep — the\n // keepalive flush ships the still-unclassified click as a plain (alive)\n // $click.\n window.dispatchEvent(new Event(\"pagehide\"));\n await Promise.resolve();\n await Promise.resolve();\n\n const payload = JSON.parse(sentBodies[0] ?? \"{}\") as { events: { event_type: string, event_at_ms: number, data: Record<string, unknown> }[] };\n const clicks = payload.events.filter((event) => event.event_type === \"$click\");\n expect(clicks).toHaveLength(1);\n expect(clicks[0].data.dead).toBeUndefined();\n expect(clicks[0].event_at_ms).toBe(clickAtMs);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"holds an unclassified click out of a flush and ships it on the next one\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button id=\\\"late\\\">Late click</button>\";\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n // Click 500ms before the 10s flush tick: classification cannot finish\n // in time, so the flush must hold the click back rather than send it\n // unclassified.\n await vi.advanceTimersByTimeAsync(9_500);\n document.querySelector(\"#late\")?.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n await vi.advanceTimersByTimeAsync(500);\n\n expect(getSentEventTypes(sentBodies)).toMatchInlineSnapshot(`\n [\n \"$page-view\",\n ]\n `);\n\n // By the next flush the sweep has classified it (dead — nothing\n // observable happened) and it ships marked.\n await vi.advanceTimersByTimeAsync(10_000);\n const second = JSON.parse(sentBodies[1] ?? \"{}\") as { events: { event_type: string, data: Record<string, unknown> }[] };\n expect(second.events.map((event) => event.event_type)).toMatchInlineSnapshot(`\n [\n \"$click\",\n ]\n `);\n expect(second.events[0].data.dead).toBe(1);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"captures client-side navigations when history is exposed as an accessor descriptor\", async () => {\n vi.useFakeTimers();\n\n const historyDescriptor = Object.getOwnPropertyDescriptor(window, \"history\");\n expect(historyDescriptor?.value).toBeUndefined();\n expect(historyDescriptor?.get).toBeTypeOf(\"function\");\n\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.ok(new Response());\n },\n });\n\n try {\n tracker.start();\n window.history.pushState({}, \"\", \"/projects/test-project\");\n\n await advancePastFlush();\n\n expect(getSentEventTypes(sentBodies)).toMatchInlineSnapshot(`\n [\n \"$page-view\",\n \"$page-view\",\n ]\n `);\n } finally {\n tracker.stop();\n }\n });\n\n it(\"silently disables when client interface returns ANALYTICS_NOT_ENABLED as an error\", async () => {\n vi.useFakeTimers();\n document.body.innerHTML = \"<button>Click me</button>\";\n\n const warnSpy = vi.spyOn(console, \"warn\").mockImplementation(() => {});\n const sentBodies: string[] = [];\n const tracker = new EventTracker({\n projectId: \"internal\",\n sendBatch: async (body) => {\n sentBodies.push(body);\n return Result.error(new KnownErrors.AnalyticsNotEnabled());\n },\n });\n\n try {\n tracker.start();\n\n await advancePastFlush();\n expect(sentBodies).toHaveLength(1);\n expect(warnSpy).not.toHaveBeenCalled();\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n expect((tracker as any)._flushTimer).toBeNull();\n\n document.querySelector(\"button\")?.dispatchEvent(new MouseEvent(\"click\", { bubbles: true }));\n await advancePastFlush();\n expect(sentBodies).toHaveLength(1);\n } finally {\n tracker.stop();\n warnSpy.mockRestore();\n }\n });\n});\n"],"mappings":"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"}
@@ -70,7 +70,7 @@ declare class _HexclaveServerAppImplIncomplete<HasTokenStore extends boolean, Pr
70
70
  protected _serverNotificationCategoryFromCrud(userId: string, crud: NotificationPreferenceCrud['Server']['Read']): NotificationCategory;
71
71
  protected _serverOAuthProviderFromCrud(crud: OAuthProviderCrud['Server']['Read']): {
72
72
  id: string;
73
- type: "google" | "github" | "microsoft" | "spotify" | "facebook" | "discord" | "gitlab" | "bitbucket" | "linkedin" | "apple" | "x" | "twitch";
73
+ type: "x" | "google" | "github" | "microsoft" | "spotify" | "facebook" | "discord" | "gitlab" | "bitbucket" | "linkedin" | "apple" | "twitch";
74
74
  userId: string;
75
75
  accountId: string;
76
76
  email: string | undefined;
@@ -107,7 +107,7 @@ var _HexclaveServerAppImplIncomplete = class extends _HexclaveClientAppImplIncom
107
107
  isPrimary: crud.is_primary,
108
108
  usedForAuth: crud.used_for_auth,
109
109
  async sendVerificationEmail(options) {
110
- await app._interface.sendServerContactChannelVerificationEmail(userId, crud.id, options?.callbackUrl ?? constructRedirectUrl(app.urls.emailVerification, "callbackUrl"));
110
+ await app._interface.sendServerContactChannelVerificationEmail(userId, crud.id, options?.callbackUrl ?? constructRedirectUrl(app._getUrls().emailVerification, "callbackUrl"));
111
111
  },
112
112
  async update(data) {
113
113
  await app._interface.updateServerContactChannel(userId, crud.id, serverContactChannelUpdateOptionsToCrud(data));
@@ -860,7 +860,7 @@ var _HexclaveServerAppImplIncomplete = class extends _HexclaveClientAppImplIncom
860
860
  await app._interface.sendServerTeamInvitation({
861
861
  teamId: crud.id,
862
862
  email: options.email,
863
- callbackUrl: options.callbackUrl ?? constructRedirectUrl(app.urls.teamInvitation, "callbackUrl")
863
+ callbackUrl: options.callbackUrl ?? constructRedirectUrl(app._getUrls().teamInvitation, "callbackUrl")
864
864
  });
865
865
  await app._serverTeamInvitationsCache.refresh([crud.id]);
866
866
  },