alphana-sdk 0.3.3 → 0.4.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/alphana-sdk.global.js +1 -28
- package/dist/index.d.mts +5 -15
- package/dist/index.d.ts +5 -15
- package/dist/index.js +2 -2
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +2 -2
- package/dist/index.mjs.map +1 -1
- package/dist/react/index.d.mts +5 -15
- package/dist/react/index.d.ts +5 -15
- package/dist/react/index.js +1 -1
- package/dist/react/index.js.map +1 -1
- package/dist/react/index.mjs +1 -1
- package/dist/react/index.mjs.map +1 -1
- package/package.json +1 -1
package/dist/react/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/react/provider.tsx","../../src/utils/session.ts","../../src/utils/geo.ts","../../src/core/navigation.ts","../../src/core/time.ts","../../src/utils/throttle.ts","../../src/core/heatmap.ts","../../src/core/logger.ts","../../src/core/snapshot.ts","../../src/tracker.ts","../../src/react/context.tsx","../../src/react/hooks.ts"],"sourcesContent":["import { useEffect, useRef, type ReactNode } from \"react\";\nimport { UserTracker } from \"../tracker\";\nimport type { TrackerConfig } from \"../types\";\nimport { TrackerContext } from \"./context\";\n\nexport interface UserTrackerProviderProps {\n /** Tracker configuration. Captured on first render — changes are ignored. */\n config?: TrackerConfig;\n children: ReactNode;\n}\n\n/**\n * Wraps your application (or a subtree) and provides a `UserTracker` instance\n * via React context.\n *\n * The tracker is created once, initialized on mount, and destroyed on unmount.\n *\n * **Next.js App Router** — mark your layout wrapper as a Client Component:\n * ```tsx\n * 'use client';\n * import { UserTrackerProvider } from 'user-tracker/react';\n * export default function RootLayout({ children }) {\n * return <UserTrackerProvider config={{ endpoint: '/api/events' }}>{children}</UserTrackerProvider>;\n * }\n * ```\n */\nexport function UserTrackerProvider({\n config = {},\n children,\n}: UserTrackerProviderProps) {\n // Create the tracker instance exactly once (lazy ref initialisation).\n const trackerRef = useRef<UserTracker | null>(null);\n if (trackerRef.current === null) {\n trackerRef.current = new UserTracker(config);\n }\n\n useEffect(() => {\n const tracker = trackerRef.current!;\n tracker.init();\n return () => tracker.destroy();\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n return (\n <TrackerContext.Provider value={trackerRef.current}>\n {children}\n </TrackerContext.Provider>\n );\n}\n","const SESSION_STORAGE_KEY = \"__ut_sid__\";\nconst VISITOR_STORAGE_KEY = \"__ut_vid__\";\n\n/** Generate a RFC-4122 v4 UUID using the native crypto API with a fallback. */\nexport function generateSessionId(): string {\n if (\n typeof crypto !== \"undefined\" &&\n typeof crypto.randomUUID === \"function\"\n ) {\n return crypto.randomUUID();\n }\n // Math.random fallback (not cryptographically secure, but sufficient for analytics)\n return \"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx\".replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0;\n const v = c === \"x\" ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n });\n}\n\n/**\n * Retrieve the session ID from sessionStorage, or create and persist a new one.\n * Falls back to an in-memory ID when sessionStorage is unavailable (e.g. SSR).\n */\nexport function getOrCreateSessionId(): string {\n if (typeof sessionStorage === \"undefined\") return generateSessionId();\n try {\n const existing = sessionStorage.getItem(SESSION_STORAGE_KEY);\n if (existing) return existing;\n const id = generateSessionId();\n sessionStorage.setItem(SESSION_STORAGE_KEY, id);\n return id;\n } catch {\n return generateSessionId();\n }\n}\n\n/**\n * Retrieve the visitor ID from localStorage (persists across sessions/tabs),\n * or create and store a new one. Returns a temporary in-memory ID when\n * localStorage is unavailable.\n */\nexport function getOrCreateVisitorId(): string {\n if (typeof localStorage === \"undefined\") return generateSessionId();\n try {\n const existing = localStorage.getItem(VISITOR_STORAGE_KEY);\n if (existing) return existing;\n const id = generateSessionId();\n localStorage.setItem(VISITOR_STORAGE_KEY, id);\n return id;\n } catch {\n return generateSessionId();\n }\n}\n","import type { GeoLocation } from \"../types\";\n\n/**\n * Resolves the visitor's approximate location from their public IP address\n * using the ipapi.co free-tier JSON endpoint (no API-key required, up to\n * 1 000 requests/day on the free plan).\n *\n * Runs silently — returns `null` on any network error, rate-limit, or\n * reserved/private IP so that tracking is never blocked.\n */\nexport async function fetchLocation(): Promise<GeoLocation | null> {\n try {\n const res = await fetch(\"https://ipapi.co/json/\", {\n method: \"GET\",\n headers: { Accept: \"application/json\" },\n });\n if (!res.ok) return null;\n const d = (await res.json()) as Record<string, unknown>;\n // ipapi returns { \"error\": true, \"reason\": \"...\" } for private/reserved IPs\n if (d[\"error\"]) return null;\n return {\n country: typeof d[\"country_code\"] === \"string\" ? d[\"country_code\"] : \"\",\n countryName:\n typeof d[\"country_name\"] === \"string\" ? d[\"country_name\"] : \"\",\n city: typeof d[\"city\"] === \"string\" ? d[\"city\"] : undefined,\n region: typeof d[\"region\"] === \"string\" ? d[\"region\"] : undefined,\n latitude: typeof d[\"latitude\"] === \"number\" ? d[\"latitude\"] : undefined,\n longitude:\n typeof d[\"longitude\"] === \"number\" ? d[\"longitude\"] : undefined,\n };\n } catch {\n return null;\n }\n}\n","import type { TrackerEvent } from \"../types\";\n\ntype EmitFn = (event: TrackerEvent) => void;\n\ninterface NavigationPluginOptions {\n emit: EmitFn;\n sessionId: string;\n}\n\n/**\n * Tracks SPA route changes by monkey-patching history.pushState /\n * history.replaceState and listening to the popstate event.\n *\n * For every navigation it:\n * 1. Emits a `pageview` event.\n * 2. Dispatches the custom DOM event `tracker:navigate` so that other\n * plugins (TimePlugin, HeatmapPlugin) can react without having to\n * duplicate the pushState patching.\n * 3. Detects U-turns: if the user leaves a page in ≤5 s, a `uturn` event\n * is emitted with the time-on-page and destination path.\n *\n * Next.js App Router note:\n * The App Router manages navigation internally; use `usePageView(pathname)`\n * from `user-tracker/react` together with `usePathname()` instead.\n */\nexport class NavigationPlugin {\n private readonly emit: EmitFn;\n private readonly sessionId: string;\n private previousPath = \"\";\n private originalPushState: typeof history.pushState | null = null;\n private originalReplaceState: typeof history.replaceState | null = null;\n private pageEntryTime = 0;\n private static readonly UTURN_THRESHOLD_MS = 5_000;\n\n constructor({ emit, sessionId }: NavigationPluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n }\n\n init(): void {\n // Record the initial page view on load.\n this.recordPageView(window.location.pathname + window.location.search);\n\n window.addEventListener(\"popstate\", this.handlePopState);\n\n // Patch pushState\n this.originalPushState = history.pushState.bind(history);\n const origPush = this.originalPushState;\n history.pushState = (state, title, url): void => {\n origPush(state, title, url);\n this.handleNavigation();\n };\n\n // Patch replaceState\n this.originalReplaceState = history.replaceState.bind(history);\n const origReplace = this.originalReplaceState;\n history.replaceState = (state, title, url): void => {\n origReplace(state, title, url);\n this.handleNavigation();\n };\n }\n\n destroy(): void {\n window.removeEventListener(\"popstate\", this.handlePopState);\n if (this.originalPushState) history.pushState = this.originalPushState;\n if (this.originalReplaceState)\n history.replaceState = this.originalReplaceState;\n }\n\n // Arrow property → always bound to `this`, safe to use as event listener.\n private handlePopState = (): void => {\n this.handleNavigation();\n };\n\n private handleNavigation(): void {\n const newPath = window.location.pathname + window.location.search;\n if (newPath === this.previousPath) return; // hash-only or duplicate call\n\n // U-turn detection: user left the previous page very quickly.\n if (this.previousPath && this.pageEntryTime > 0) {\n const timeOnPage = Date.now() - this.pageEntryTime;\n if (timeOnPage <= NavigationPlugin.UTURN_THRESHOLD_MS) {\n this.emit({\n type: \"uturn\",\n data: {\n fromPath: this.previousPath,\n toPath: newPath,\n timeOnPageMs: timeOnPage,\n timestamp: Date.now(),\n sessionId: this.sessionId,\n },\n });\n }\n }\n\n this.recordPageView(newPath);\n }\n\n private recordPageView(path: string): void {\n this.previousPath = path;\n this.pageEntryTime = Date.now();\n\n this.emit({\n type: \"pageview\",\n data: {\n path,\n title: document.title,\n timestamp: Date.now(),\n sessionId: this.sessionId,\n referrer: document.referrer || undefined,\n },\n });\n\n // Notify other plugins via a custom DOM event (synchronous dispatch).\n window.dispatchEvent(\n new CustomEvent(\"tracker:navigate\", {\n detail: { path, title: document.title },\n }),\n );\n }\n}\n","import type { TrackerEvent } from \"../types\";\n\ntype EmitFn = (event: TrackerEvent) => void;\n\ninterface TimePluginOptions {\n emit: EmitFn;\n sessionId: string;\n}\n\n/**\n * Tracks the time a user spends on each page.\n *\n * - Starts a timer when the page becomes active (init / tab focus).\n * - Stops and emits a `timespent` event when:\n * • The user navigates away (tracker:navigate)\n * • The tab is hidden (visibilitychange)\n * • The page is unloading (beforeunload / pagehide)\n * - Resumes timing when the tab becomes visible again.\n */\nexport class TimePlugin {\n private readonly emit: EmitFn;\n private readonly sessionId: string;\n private currentPath = \"\";\n private startTime = 0;\n private tracking = false;\n\n constructor({ emit, sessionId }: TimePluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n }\n\n init(): void {\n this.currentPath = window.location.pathname + window.location.search;\n this.startTracking();\n\n window.addEventListener(\"tracker:navigate\", this.handleNavigate);\n document.addEventListener(\"visibilitychange\", this.handleVisibilityChange);\n window.addEventListener(\"beforeunload\", this.handleUnload);\n window.addEventListener(\"pagehide\", this.handleUnload);\n }\n\n destroy(): void {\n this.stopTracking();\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n document.removeEventListener(\n \"visibilitychange\",\n this.handleVisibilityChange,\n );\n window.removeEventListener(\"beforeunload\", this.handleUnload);\n window.removeEventListener(\"pagehide\", this.handleUnload);\n }\n\n private startTracking(): void {\n this.startTime = Date.now();\n this.tracking = true;\n }\n\n private stopTracking(): void {\n if (!this.tracking || !this.currentPath) return;\n const duration = Date.now() - this.startTime;\n if (duration < 100) {\n this.tracking = false;\n return; // Ignore sub-100 ms blips (e.g. rapid navigations).\n }\n this.emit({\n type: \"timespent\",\n data: {\n path: this.currentPath,\n duration,\n sessionId: this.sessionId,\n timestamp: Date.now(),\n },\n });\n this.tracking = false;\n }\n\n private handleNavigate = (e: CustomEvent<{ path: string }>): void => {\n this.stopTracking();\n this.currentPath = e.detail.path;\n this.startTracking();\n };\n\n private handleVisibilityChange = (): void => {\n if (document.hidden) {\n this.stopTracking();\n } else {\n this.startTracking();\n }\n };\n\n private handleUnload = (): void => {\n this.stopTracking();\n };\n}\n","/**\n * Returns a function that invokes `fn` at most once every `delay` ms.\n * The first call in a new window is executed immediately.\n */\nexport function throttle<Args extends unknown[]>(\n fn: (...args: Args) => void,\n delay: number,\n): (...args: Args) => void {\n let lastCall = 0;\n return (...args: Args): void => {\n const now = Date.now();\n if (now - lastCall >= delay) {\n lastCall = now;\n fn(...args);\n }\n };\n}\n","import type { TrackerEvent, HeatmapPoint } from \"../types\";\nimport { throttle } from \"../utils/throttle\";\n\ntype EmitFn = (event: TrackerEvent) => void;\n\ninterface HeatmapPluginOptions {\n emit: EmitFn;\n sessionId: string;\n /** Fraction of mousemove / scroll events to sample (0–1). Default: 0.3 */\n sampleRate?: number;\n /** Maximum points stored per page before recording stops. Default: 2000 */\n maxPoints?: number;\n /**\n * If provided, heatmap data will only be collected for paths in this list.\n * An empty array or undefined means all pages are tracked.\n */\n allowedPaths?: string[];\n}\n\n/**\n * Collects mouse-move, click, and scroll positions for heatmap analysis.\n *\n * Also detects rage clicks (≥3 clicks within 1 s in a 50 px radius) and emits\n * a `rageclik` event so the backend can surface problematic UI hotspots.\n *\n * Click events include an optional `target` field containing a human-readable\n * label for the clicked element (aria-label › id › button/link text).\n */\nexport class HeatmapPlugin {\n private readonly emit: EmitFn;\n private readonly sessionId: string;\n private readonly sampleRate: number;\n private readonly maxPoints: number;\n private readonly allowedPaths: Set<string> | null;\n private currentPath = \"\";\n private pointCounts: Record<string, number> = {};\n\n // Rage-click detection state\n private recentClicks: { x: number; y: number; t: number }[] = [];\n private static readonly RAGE_THRESHOLD = 3;\n private static readonly RAGE_WINDOW_MS = 1_000;\n private static readonly RAGE_RADIUS_PX = 50;\n\n private readonly throttledMouseMove: (e: MouseEvent) => void;\n private readonly throttledScroll: () => void;\n\n constructor({\n emit,\n sessionId,\n sampleRate = 0.3,\n maxPoints = 2000,\n allowedPaths,\n }: HeatmapPluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n this.sampleRate = sampleRate;\n this.maxPoints = maxPoints;\n this.allowedPaths =\n allowedPaths && allowedPaths.length > 0 ? new Set(allowedPaths) : null;\n\n this.throttledMouseMove = throttle(this.handleMouseMove, 50);\n this.throttledScroll = throttle(this.handleScroll, 100);\n }\n\n init(): void {\n this.currentPath = window.location.pathname + window.location.search;\n\n document.addEventListener(\"mousemove\", this.throttledMouseMove);\n document.addEventListener(\"click\", this.handleClick);\n window.addEventListener(\"scroll\", this.throttledScroll, { passive: true });\n window.addEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n destroy(): void {\n document.removeEventListener(\"mousemove\", this.throttledMouseMove);\n document.removeEventListener(\"click\", this.handleClick);\n window.removeEventListener(\"scroll\", this.throttledScroll);\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n private canRecord(): boolean {\n if (\n this.allowedPaths !== null &&\n !this.allowedPaths.has(this.currentPath)\n ) {\n return false;\n }\n return (this.pointCounts[this.currentPath] ?? 0) < this.maxPoints;\n }\n\n private recordPoint(\n point: Omit<HeatmapPoint, \"path\" | \"timestamp\" | \"sessionId\">,\n ): void {\n if (!this.canRecord()) return;\n this.pointCounts[this.currentPath] =\n (this.pointCounts[this.currentPath] ?? 0) + 1;\n this.emit({\n type: \"heatmap\",\n data: {\n ...point,\n path: this.currentPath,\n timestamp: Date.now(),\n },\n });\n }\n\n /** Extract a human-readable label for the clicked element. */\n private getClickTarget(e: MouseEvent): string | undefined {\n const el = e.target as HTMLElement | null;\n if (!el) return undefined;\n const label =\n el.getAttribute(\"aria-label\") ||\n el.closest(\"[aria-label]\")?.getAttribute(\"aria-label\") ||\n el.getAttribute(\"data-track-label\") ||\n el.getAttribute(\"id\") ||\n (el instanceof HTMLButtonElement || el instanceof HTMLAnchorElement\n ? el.innerText?.trim().slice(0, 60)\n : el.closest(\"button, a\")?.textContent?.trim().slice(0, 60));\n return label || undefined;\n }\n\n /** Detect rage-click bursts and emit a `rageclik` event when found. */\n private checkRageClick(x: number, y: number, path: string): void {\n const now = Date.now();\n this.recentClicks = this.recentClicks.filter(\n (c) => now - c.t < HeatmapPlugin.RAGE_WINDOW_MS,\n );\n this.recentClicks.push({ x, y, t: now });\n\n const nearby = this.recentClicks.filter(\n (c) => Math.hypot(c.x - x, c.y - y) <= HeatmapPlugin.RAGE_RADIUS_PX,\n );\n\n if (nearby.length >= HeatmapPlugin.RAGE_THRESHOLD) {\n this.emit({\n type: \"rageclik\",\n data: {\n path,\n x,\n y,\n count: nearby.length,\n timestamp: now,\n sessionId: this.sessionId,\n },\n });\n // Reset to avoid continuously re-triggering in the same burst.\n this.recentClicks = [];\n }\n }\n\n private handleMouseMove = (e: MouseEvent): void => {\n if (Math.random() > this.sampleRate) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = e.clientY + window.scrollY;\n this.recordPoint({\n x: e.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (e.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n type: \"move\",\n });\n };\n\n private handleClick = (e: MouseEvent): void => {\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = e.clientY + window.scrollY;\n const target = this.getClickTarget(e);\n this.recordPoint({\n x: e.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (e.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n type: \"click\",\n ...(target ? { target } : {}),\n });\n this.checkRageClick(e.clientX, absY, this.currentPath);\n };\n\n private handleScroll = (): void => {\n if (Math.random() > this.sampleRate) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const vw = window.innerWidth;\n const scrollX = window.scrollX;\n const scrollY = window.scrollY;\n // Record the scroll position (top edge of the visible viewport).\n // Using scrollY instead of the viewport centre keeps scroll dots aligned\n // with the scroll bar position, matching how click/move events are anchored\n // to their exact page coordinates.\n const absX = scrollX + vw / 2;\n this.recordPoint({\n x: vw / 2,\n y: scrollY,\n xPct: pageWidth > 0 ? (absX / pageWidth) * 100 : 50,\n yPct: pageHeight > 0 ? (scrollY / pageHeight) * 100 : 0,\n type: \"scroll\",\n });\n };\n\n private handleNavigate = (e: CustomEvent<{ path: string }>): void => {\n this.currentPath = e.detail.path;\n };\n}\n","export type LogLevel = \"info\" | \"warn\" | \"error\";\n\nexport interface LogEntry {\n sessionId?: string;\n appId?: string;\n level: LogLevel;\n message: string;\n url?: string;\n stack?: string;\n meta?: Record<string, unknown>;\n timestamp: number;\n}\n\ntype ConsoleFn = (...args: unknown[]) => void;\n\n/**\n * Automatically captures console.info/warn/error output and unhandled errors,\n * then ships them to the backend `/logs/ingest` endpoint.\n *\n * All console methods are restored exactly in `destroy()`.\n */\nexport class LogCapture {\n private readonly endpoint: string;\n private readonly sessionId: string;\n private readonly appId?: string;\n private readonly authHeaders: Record<string, string>;\n\n // Original console methods preserved so we can restore them.\n private origInfo!: ConsoleFn;\n private origWarn!: ConsoleFn;\n private origError!: ConsoleFn;\n\n private prevOnError: OnErrorEventHandler = null;\n private prevOnUnhandledRejection:\n | ((e: PromiseRejectionEvent) => void)\n | null = null;\n\n private initialized = false;\n\n constructor(options: {\n endpoint: string;\n sessionId: string;\n secretKey?: string;\n appId?: string;\n }) {\n // Derive the API base URL by stripping everything from the last path\n // segment that isn't a versioning prefix. The tracker config `endpoint`\n // is the *events* URL (e.g. http://host/api/events), but logs live at\n // http://host/api/logs/ingest, so we walk up until we reach the common\n // base (i.e. remove the final segment).\n try {\n const u = new URL(options.endpoint);\n // Remove the last non-empty path segment (e.g. \"/api/events\" → \"/api\")\n const parts = u.pathname.replace(/\\/$/, \"\").split(\"/\");\n parts.pop();\n u.pathname = parts.join(\"/\") || \"/\";\n this.endpoint = u.toString().replace(/\\/$/, \"\");\n } catch {\n this.endpoint = options.endpoint;\n }\n this.sessionId = options.sessionId;\n this.appId = options.appId;\n this.authHeaders = options.secretKey\n ? { Authorization: `Bearer ${options.secretKey}` }\n : {};\n }\n\n init(): void {\n if (typeof window === \"undefined\" || this.initialized) return;\n\n this.origInfo = console.info.bind(console);\n this.origWarn = console.warn.bind(console);\n this.origError = console.error.bind(console);\n\n console.info = (...args: unknown[]) => {\n this.origInfo(...args);\n this.send(\"info\", this.format(args));\n };\n\n console.warn = (...args: unknown[]) => {\n this.origWarn(...args);\n this.send(\"warn\", this.format(args));\n };\n\n console.error = (...args: unknown[]) => {\n this.origError(...args);\n const [first] = args;\n const stack = first instanceof Error ? first.stack : undefined;\n this.send(\"error\", this.format(args), { stack });\n };\n\n this.prevOnError = window.onerror;\n window.onerror = (msg, src, line, col, err) => {\n this.send(\"error\", String(msg), {\n stack: err?.stack,\n meta: { src, line, col },\n });\n if (typeof this.prevOnError === \"function\") {\n return this.prevOnError(msg, src, line, col, err);\n }\n return false;\n };\n\n this.prevOnUnhandledRejection = (e: PromiseRejectionEvent) => {\n const reason = e.reason;\n const message =\n reason instanceof Error\n ? reason.message\n : String(reason ?? \"Unhandled promise rejection\");\n this.send(\"error\", message, {\n stack: reason instanceof Error ? reason.stack : undefined,\n });\n };\n window.addEventListener(\n \"unhandledrejection\",\n this.prevOnUnhandledRejection,\n );\n\n this.initialized = true;\n }\n\n destroy(): void {\n if (!this.initialized) return;\n console.info = this.origInfo;\n console.warn = this.origWarn;\n console.error = this.origError;\n\n window.onerror = this.prevOnError;\n if (this.prevOnUnhandledRejection) {\n window.removeEventListener(\n \"unhandledrejection\",\n this.prevOnUnhandledRejection,\n );\n }\n this.initialized = false;\n }\n\n /** Manually capture a log entry (e.g. from try/catch). */\n capture(\n level: LogLevel,\n message: string,\n extra?: { stack?: string; meta?: Record<string, unknown> },\n ): void {\n this.send(level, message, extra);\n }\n\n private format(args: unknown[]): string {\n return args\n .map((a) => {\n if (a instanceof Error) return a.message;\n if (typeof a === \"object\") {\n try {\n return JSON.stringify(a);\n } catch {\n return String(a);\n }\n }\n return String(a);\n })\n .join(\" \");\n }\n\n private send(\n level: LogLevel,\n message: string,\n extra?: { stack?: string; meta?: Record<string, unknown> },\n ): void {\n const entry: LogEntry = {\n sessionId: this.sessionId,\n ...(this.appId ? { appId: this.appId } : {}),\n level,\n message,\n url: typeof window !== \"undefined\" ? window.location.href : undefined,\n stack: extra?.stack,\n meta: extra?.meta,\n timestamp: Date.now(),\n };\n const url = `${this.endpoint}/logs/ingest`;\n const body = JSON.stringify(entry);\n\n // Use fetch with keepalive so the request survives page navigation.\n // Errors are logged to the (original, unpatched) console so they are\n // visible in DevTools without creating an infinite log loop.\n void fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...this.authHeaders },\n body,\n keepalive: true,\n }).catch((err: unknown) => {\n // Use the original (pre-patch) error logger to avoid recursion.\n if (this.origError) {\n this.origError(\"[user-tracker] Failed to send log:\", err);\n }\n });\n }\n}\n","import html2canvas from \"html2canvas\";\n\n/**\n * SnapshotPlugin\n *\n * Captures a full-page screenshot via html2canvas and POSTs it to the backend\n * snapshot endpoint once every 5 minutes per page (configurable).\n *\n * Timestamps are persisted in localStorage so the throttle survives\n * page reloads.\n */\n\nconst DEFAULT_INTERVAL_MS = 5 * 60 * 1000; // 5 minutes\nconst STORAGE_KEY = \"__ut_snap_ts__\";\n\nexport class SnapshotPlugin {\n private lastSentPerPath: Record<string, number> = {};\n private readonly snapshotUrl: string;\n private readonly appId?: string;\n private readonly secretKey?: string;\n private readonly intervalMs: number;\n private readonly allowedPaths: Set<string> | null;\n\n constructor(cfg: {\n /** Base events endpoint, e.g. \"https://api.example.com/api/events\" */\n endpoint: string;\n appId?: string;\n secretKey?: string;\n /**\n * Minimum milliseconds between screenshots for the same path.\n * Defaults to 300 000 (5 minutes).\n */\n intervalMs?: number;\n /**\n * If provided, snapshots will only be captured for paths in this list.\n * An empty array or undefined means all pages are captured.\n */\n allowedPaths?: string[];\n }) {\n // Derive the snapshot endpoint from the events endpoint.\n this.snapshotUrl = cfg.endpoint.replace(/\\/events$/, \"/snapshots\");\n this.appId = cfg.appId;\n this.secretKey = cfg.secretKey;\n this.intervalMs = cfg.intervalMs ?? DEFAULT_INTERVAL_MS;\n this.allowedPaths =\n cfg.allowedPaths && cfg.allowedPaths.length > 0\n ? new Set(cfg.allowedPaths)\n : null;\n\n try {\n const stored = localStorage.getItem(STORAGE_KEY);\n if (stored)\n this.lastSentPerPath = JSON.parse(stored) as Record<string, number>;\n } catch {\n // localStorage may be unavailable in some contexts — proceed without it.\n }\n }\n\n /**\n * Capture and send a screenshot for the given path.\n * No-ops if a screenshot was already sent within the last 5 minutes.\n */\n async capture(path: string): Promise<void> {\n if (typeof window === \"undefined\") return;\n\n // Only capture if this path is in the allowed list (when configured)\n if (this.allowedPaths !== null && !this.allowedPaths.has(path)) return;\n\n const last = this.lastSentPerPath[path] ?? 0;\n if (Date.now() - last < this.intervalMs) return;\n\n try {\n // Capture the <html> element so backgrounds set on :root / html are included.\n const root = document.documentElement;\n\n // Use the page's own background colour so sections that use\n // `background: transparent` don't fall back to white.\n const pageBg =\n getComputedStyle(root).backgroundColor ||\n getComputedStyle(document.body).backgroundColor ||\n \"#ffffff\";\n\n const canvas = await html2canvas(root, {\n allowTaint: true,\n useCORS: true,\n logging: false,\n // Full device-pixel-ratio so retina screens stay sharp.\n scale: window.devicePixelRatio || 1,\n // Full page dimensions.\n width: root.scrollWidth,\n height: root.scrollHeight,\n // Use real viewport dimensions so CSS units like dvh/vh are resolved\n // correctly. Setting these to scrollHeight would make 100dvh sections\n // expand to the full page height in the capture.\n windowWidth: window.innerWidth,\n windowHeight: window.innerHeight,\n // Capture from the very top-left regardless of current scroll position.\n scrollX: 0,\n scrollY: 0,\n x: 0,\n y: 0,\n // Provide the resolved background colour so transparent areas are filled correctly.\n backgroundColor: pageBg,\n // foreignObjectRendering causes artifacts in many frameworks — keep off.\n foreignObjectRendering: false,\n // Remove the temporary off-screen clone after rendering.\n removeContainer: true,\n });\n\n // Use a Promise wrapper because toBlob is callback-based.\n const blob = await new Promise<Blob | null>((resolve) =>\n canvas.toBlob(resolve, \"image/png\"),\n );\n if (!blob) return;\n\n const form = new FormData();\n form.append(\"screenshot\", blob, \"screenshot.png\");\n form.append(\"path\", path);\n // Report the logical CSS-pixel dimensions (not DPR-scaled canvas size)\n // so heatmap xPct/yPct coordinates remain accurate.\n form.append(\"width\", String(root.scrollWidth));\n form.append(\"height\", String(root.scrollHeight));\n if (this.appId) form.append(\"appId\", this.appId);\n\n const headers: Record<string, string> = {};\n if (this.secretKey) headers.Authorization = `Bearer ${this.secretKey}`;\n\n await fetch(this.snapshotUrl, { method: \"POST\", headers, body: form });\n\n // Persist timestamp so throttle survives page reloads.\n this.lastSentPerPath[path] = Date.now();\n try {\n localStorage.setItem(STORAGE_KEY, JSON.stringify(this.lastSentPerPath));\n } catch {\n // quota exceeded — non-fatal\n }\n } catch {\n // Never propagate — snapshot failures must not affect the tracked site.\n }\n }\n\n destroy(): void {\n // Nothing to clean up — no timers or listeners.\n }\n}\n","import type {\n TrackerConfig,\n TrackerEvent,\n SessionData,\n PageView,\n HeatmapPoint,\n GeoLocation,\n} from \"./types\";\nimport { generateSessionId, getOrCreateVisitorId } from \"./utils/session\";\nimport { fetchLocation } from \"./utils/geo\";\nimport { NavigationPlugin } from \"./core/navigation\";\nimport { TimePlugin } from \"./core/time\";\nimport { HeatmapPlugin } from \"./core/heatmap\";\nimport { LogCapture } from \"./core/logger\";\nimport { SnapshotPlugin } from \"./core/snapshot\";\n\nexport const DEFAULT_ENDPOINT = \"https://api.alphana.ir/api/events\";\n\nconst DEFAULTS = {\n endpoint: DEFAULT_ENDPOINT,\n trackNavigation: true,\n trackTime: true,\n trackHeatmap: true,\n trackLogs: true,\n trackSnapshots: true,\n mouseSampleRate: 0.3,\n maxHeatmapPoints: 2000,\n batchSize: 20,\n flushInterval: 5_000,\n} as const;\n\ntype SubscriberFn = (event: TrackerEvent) => void;\n\n/**\n * Core tracker class. Framework-agnostic — works in any environment that has\n * a browser DOM (React, Next.js Pages Router, Vite, vanilla JS/TS, etc.).\n *\n * Usage:\n * ```ts\n * const tracker = new UserTracker({ endpoint: 'https://my-api.com/events' });\n * tracker.init(); // call once; safe to call in SSR (no-op server-side)\n * ```\n *\n * Destroy when done (e.g. component unmount):\n * ```ts\n * tracker.destroy();\n * ```\n */\ntype ResolvedConfig = Required<\n Pick<\n TrackerConfig,\n | \"endpoint\"\n | \"trackNavigation\"\n | \"trackTime\"\n | \"trackHeatmap\"\n | \"trackLogs\"\n | \"mouseSampleRate\"\n | \"maxHeatmapPoints\"\n | \"batchSize\"\n | \"flushInterval\"\n >\n> &\n TrackerConfig;\n\nexport class UserTracker {\n private readonly cfg: ResolvedConfig;\n private session: SessionData;\n private navigation?: NavigationPlugin;\n private time?: TimePlugin;\n private heatmap?: HeatmapPlugin;\n /** Public so consumers can call logCapture.capture() for manual log entries. */\n logCapture?: LogCapture;\n private snapshot?: SnapshotPlugin;\n private initialized = false;\n private readonly subscribers = new Set<SubscriberFn>();\n\n /** In-memory queue of events waiting to be flushed. */\n private queue: TrackerEvent[] = [];\n private flushTimer: ReturnType<typeof setInterval> | null = null;\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private location: GeoLocation | null = null;\n\n constructor(config: TrackerConfig = {}) {\n this.cfg = { ...DEFAULTS, ...config } as ResolvedConfig;\n\n // Validate endpoint URL up front so the error is thrown at construction\n // time rather than silently failing during a network request.\n try {\n new URL(this.cfg.endpoint);\n } catch {\n throw new Error(\n `[alpha-tracker] Invalid endpoint URL: \"${this.cfg.endpoint}\"`,\n );\n }\n\n this.session = {\n id: config.sessionId ?? generateSessionId(),\n visitorId: getOrCreateVisitorId(),\n startedAt: Date.now(),\n pageViews: [],\n timeSpent: {},\n heatmap: {},\n };\n }\n\n // ─── Lifecycle ──────────────────────────────────────────────────────────────\n\n /**\n * Attach event listeners and start tracking.\n * Safe to call during SSR — returns `this` immediately if `window` is\n * undefined so it can be chained: `const tracker = new UserTracker(cfg).init()`.\n */\n init(): this {\n if (typeof window === \"undefined\" || this.initialized) return this;\n\n const emit = this.emit.bind(this);\n const { id: sessionId } = this.session;\n\n if (this.cfg.trackNavigation) {\n this.navigation = new NavigationPlugin({ emit, sessionId });\n this.navigation.init();\n }\n\n if (this.cfg.trackTime) {\n this.time = new TimePlugin({ emit, sessionId });\n this.time.init();\n }\n\n if (this.cfg.trackHeatmap) {\n this.heatmap = new HeatmapPlugin({\n emit,\n sessionId,\n sampleRate: this.cfg.mouseSampleRate,\n maxPoints: this.cfg.maxHeatmapPoints,\n allowedPaths: this.cfg.heatmapPages,\n });\n this.heatmap.init();\n }\n\n if (this.cfg.endpoint) {\n // Flush on a regular interval — even if the batch threshold isn't hit.\n this.flushTimer = setInterval(() => {\n if (this.queue.length > 0) void this.flush();\n }, this.cfg.flushInterval);\n\n // Resolve visitor location from IP in the background.\n void fetchLocation().then((loc) => {\n this.location = loc;\n if (loc) this.session.location = loc;\n });\n\n // Flush remaining queue when the tab is hidden or the page is unloaded.\n window.addEventListener(\"visibilitychange\", this.handleVisibilityChange);\n window.addEventListener(\"pagehide\", this.handlePageHide);\n\n // Periodic keep-alive heartbeat every 30 s so the backend knows the\n // session is still active and doesn't expire it prematurely.\n this.heartbeatTimer = setInterval(() => {\n if (document.visibilityState !== \"hidden\") void this.sendHeartbeat();\n }, 30_000);\n\n // Auto-capture console logs and unhandled errors.\n if (this.cfg.trackLogs) {\n this.logCapture = new LogCapture({\n endpoint: this.cfg.endpoint,\n sessionId: this.session.id,\n secretKey: this.cfg.secretKey,\n appId: this.cfg.appId,\n });\n this.logCapture.init();\n }\n\n // Screenshot capture for heatmap background.\n if (this.cfg.trackSnapshots !== false) {\n this.snapshot = new SnapshotPlugin({\n endpoint: this.cfg.endpoint,\n appId: this.cfg.appId,\n secretKey: this.cfg.secretKey,\n intervalMs: this.cfg.snapshotIntervalMs,\n allowedPaths: this.cfg.heatmapPages,\n });\n // Capture immediately on init.\n void this.snapshot.capture(window.location.pathname);\n // Re-capture after each SPA navigation.\n window.addEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n }\n\n this.initialized = true;\n return this;\n }\n\n /** Remove all event listeners, flush remaining queue, and reset state. */\n destroy(): void {\n if (this.flushTimer !== null) {\n clearInterval(this.flushTimer);\n this.flushTimer = null;\n }\n if (this.heartbeatTimer !== null) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n\n if (typeof window !== \"undefined\") {\n window.removeEventListener(\n \"visibilitychange\",\n this.handleVisibilityChange,\n );\n window.removeEventListener(\"pagehide\", this.handlePageHide);\n }\n\n this.navigation?.destroy();\n this.time?.destroy();\n this.heatmap?.destroy();\n this.logCapture?.destroy();\n this.snapshot?.destroy();\n\n if (typeof window !== \"undefined\") {\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n // Best-effort flush of any remaining queued events.\n if (this.queue.length > 0 && this.cfg.endpoint) {\n this.flushBeacon();\n }\n\n this.initialized = false;\n }\n\n private handleVisibilityChange = (): void => {\n if (document.visibilityState === \"hidden\") {\n if (this.queue.length > 0) this.flushBeacon();\n this.sendDeactivate();\n }\n };\n\n private handlePageHide = (): void => {\n if (this.queue.length > 0) this.flushBeacon();\n this.sendDeactivate();\n };\n\n private handleNavigate = (e: Event): void => {\n const path = (e as CustomEvent<{ path: string }>).detail?.path;\n if (path && this.snapshot) {\n void this.snapshot.capture(path);\n }\n };\n\n /**\n * Send a keep-alive heartbeat so the backend knows this session is still\n * active. Called every 30 s while the tab is visible.\n */\n private async sendHeartbeat(): Promise<void> {\n if (!this.cfg.endpoint) return;\n const url = `${this.cfg.endpoint}/heartbeat`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n const body = JSON.stringify({\n sessionId: this.session.id,\n visitorId: this.session.visitorId,\n path: typeof window !== \"undefined\" ? window.location.pathname : \"/\",\n active: true,\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n ...(this.location ? { location: this.location } : {}),\n });\n try {\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body,\n keepalive: true,\n });\n } catch {\n // Silent — heartbeat failure should never surface to the user.\n }\n }\n\n /**\n * Send a synchronous beacon to mark this session as inactive.\n * Called on page unload / tab hidden so the dashboard reflects real-time.\n */\n private sendDeactivate(): void {\n if (!this.cfg.endpoint) return;\n const url = `${this.cfg.endpoint}/heartbeat`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n const body = JSON.stringify({\n sessionId: this.session.id,\n path: typeof window !== \"undefined\" ? window.location.pathname : \"/\",\n active: false,\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n });\n // sendBeacon fires even if the page is being unloaded.\n if (typeof navigator !== \"undefined\" && navigator.sendBeacon) {\n navigator.sendBeacon(url, new Blob([body], { type: \"application/json\" }));\n } else {\n // Fallback for environments without sendBeacon.\n void fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body,\n keepalive: true,\n }).catch(() => undefined);\n }\n }\n\n // ─── Event pipeline ─────────────────────────────────────────────────────────\n\n /** Emit a tracker event. Also used internally by the plugins. */\n emit(event: TrackerEvent): void {\n // 1 – accumulate into session data\n switch (event.type) {\n case \"pageview\":\n this.session.pageViews.push(event.data);\n break;\n\n case \"timespent\": {\n const prev = this.session.timeSpent[event.data.path] ?? 0;\n this.session.timeSpent[event.data.path] = prev + event.data.duration;\n break;\n }\n\n case \"heatmap\": {\n const key = event.data.path;\n if (!this.session.heatmap[key]) this.session.heatmap[key] = [];\n const pts = this.session.heatmap[key];\n if (pts.length < this.cfg.maxHeatmapPoints) pts.push(event.data);\n break;\n }\n\n // rageclik and uturn are forwarded to the backend only; no local state needed.\n case \"rageclik\":\n case \"uturn\":\n break;\n }\n\n // 2 – notify subscribers (used internally by React hooks)\n this.subscribers.forEach((fn) => fn(event));\n\n // 3 – user callback\n this.cfg.onEvent?.(event);\n\n // 4 – enqueue for batched remote sending\n if (this.cfg.endpoint) {\n this.queue.push(event);\n // Auto-flush once the batch size threshold is reached.\n if (this.queue.length >= this.cfg.batchSize) {\n void this.flush();\n }\n }\n }\n\n /**\n * Subscribe to every emitted event. Returns an unsubscribe function.\n *\n * ```ts\n * const unsub = tracker.subscribe(event => console.log(event));\n * // later…\n * unsub();\n * ```\n */\n subscribe(fn: SubscriberFn): () => void {\n this.subscribers.add(fn);\n return () => this.subscribers.delete(fn);\n }\n\n // ─── Manual tracking helpers ────────────────────────────────────────────────\n\n /**\n * Manually record a page view and dispatch `tracker:navigate`.\n * Required for Next.js App Router — call it inside a `useEffect` that\n * depends on `usePathname()`:\n *\n * ```tsx\n * const pathname = usePathname();\n * useEffect(() => { tracker.trackPageView(pathname); }, [pathname]);\n * ```\n */\n trackPageView(path?: string): void {\n const resolvedPath =\n path ??\n (typeof window !== \"undefined\"\n ? window.location.pathname + window.location.search\n : \"/\");\n\n this.emit({\n type: \"pageview\",\n data: {\n path: resolvedPath,\n title: typeof document !== \"undefined\" ? document.title : \"\",\n timestamp: Date.now(),\n sessionId: this.session.id,\n referrer:\n typeof document !== \"undefined\"\n ? document.referrer || undefined\n : undefined,\n },\n });\n\n if (typeof window !== \"undefined\") {\n window.dispatchEvent(\n new CustomEvent(\"tracker:navigate\", {\n detail: { path: resolvedPath, title: document.title },\n }),\n );\n }\n }\n\n // ─── Data accessors ─────────────────────────────────────────────────────────\n\n /** A read-only snapshot of the current session. */\n getSession(): Readonly<SessionData> {\n return this.session;\n }\n\n /** All page views recorded so far. */\n getPageViews(): PageView[] {\n return [...this.session.pageViews];\n }\n\n /** Cumulative milliseconds spent per path. */\n getTimeSpent(): Record<string, number> {\n return { ...this.session.timeSpent };\n }\n\n /** Heatmap points for a specific path. */\n getHeatmapData(path: string): HeatmapPoint[];\n /** Heatmap points for all tracked paths. */\n getHeatmapData(): Record<string, HeatmapPoint[]>;\n getHeatmapData(\n path?: string,\n ): HeatmapPoint[] | Record<string, HeatmapPoint[]> {\n if (path !== undefined) {\n return [...(this.session.heatmap[path] ?? [])];\n }\n return Object.entries(this.session.heatmap).reduce<\n Record<string, HeatmapPoint[]>\n >((acc, [k, v]) => {\n acc[k] = [...v];\n return acc;\n }, {});\n }\n\n // ─── Network ────────────────────────────────────────────────────────────────\n\n /** Drain the queue and POST all pending events to the batch endpoint. */\n private async flush(): Promise<void> {\n if (this.queue.length === 0) return;\n // Splice atomically so new events emitted during the async request don't\n // get lost — they stay in the queue for the next flush.\n const batch = this.queue.splice(0);\n await this.sendBatch(batch);\n }\n\n /**\n * Synchronous best-effort flush via `navigator.sendBeacon`.\n * Used on `pagehide` / `visibilitychange:hidden` where async fetch may be\n * cancelled by the browser before it completes.\n */\n private flushBeacon(): void {\n if (this.queue.length === 0) return;\n const batch = this.queue.splice(0);\n const url = `${this.cfg.endpoint!}/batch`;\n const blob = new Blob([this.buildBatchBody(batch)], {\n type: \"application/json\",\n });\n if (typeof navigator !== \"undefined\" && navigator.sendBeacon) {\n navigator.sendBeacon(url, blob);\n } else {\n // Fallback: fire-and-forget fetch (best effort on platforms without sendBeacon)\n void this.sendBatch(batch);\n }\n }\n\n private buildBatchBody(events: TrackerEvent[]): string {\n return JSON.stringify({\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n visitorId: this.session.visitorId,\n location: this.location ?? undefined,\n events: events.map((e) => ({\n sessionId: this.session.id,\n type: e.type,\n data: e.data,\n })),\n });\n }\n\n private async sendBatch(events: TrackerEvent[]): Promise<void> {\n const url = `${this.cfg.endpoint!}/batch`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n try {\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body: this.buildBatchBody(events),\n keepalive: true,\n });\n } catch {\n // Intentionally silent — analytics must never surface errors to users.\n }\n }\n}\n","import { createContext, useContext } from \"react\";\nimport type { UserTracker } from \"../tracker\";\n\nexport const TrackerContext = createContext<UserTracker | null>(null);\n\n/**\n * Returns the nearest `UserTracker` instance from context.\n * Returns `null` when called outside of a `<UserTrackerProvider>`.\n */\nexport function useTrackerContext(): UserTracker | null {\n return useContext(TrackerContext);\n}\n","import { useEffect, useRef, useState } from \"react\";\nimport type { HeatmapPoint, PageView, TrackerEvent } from \"../types\";\nimport { useTrackerContext } from \"./context\";\n\n// ─── useTracker ───────────────────────────────────────────────────────────────\n\n/**\n * Returns the `UserTracker` instance from the nearest `<UserTrackerProvider>`.\n * Returns `null` when called outside of a provider.\n */\nexport function useTracker() {\n return useTrackerContext();\n}\n\n// ─── usePageView ──────────────────────────────────────────────────────────────\n\n/**\n * Manually records a page view whenever `path` changes.\n *\n * Pass the current pathname — particularly useful with Next.js App Router:\n * ```tsx\n * 'use client';\n * import { usePathname } from 'next/navigation';\n * import { usePageView } from 'user-tracker/react';\n *\n * export function NavigationTracker() {\n * usePageView(usePathname());\n * return null;\n * }\n * ```\n * When no `path` is provided the hook is a no-op (automatic tracking via the\n * NavigationPlugin handles it).\n */\nexport function usePageView(path?: string): void {\n const tracker = useTrackerContext();\n useEffect(() => {\n if (tracker && path !== undefined) {\n tracker.trackPageView(path);\n }\n // Re-fire when the path changes.\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [path]);\n}\n\n// ─── useHeatmapData ───────────────────────────────────────────────────────────\n\n/**\n * Returns a live array of `HeatmapPoint` objects for the given path (defaults\n * to `window.location.pathname`).\n *\n * The state is updated in batches — at most once every `refreshMs` ms — to\n * avoid a re-render on every single mouse move.\n *\n * @param path The page path to query. Defaults to the current pathname.\n * @param refreshMs Minimum interval between state updates. Default: 500.\n */\nexport function useHeatmapData(path?: string, refreshMs = 500): HeatmapPoint[] {\n const tracker = useTrackerContext();\n const [data, setData] = useState<HeatmapPoint[]>([]);\n const pendingRef = useRef(false);\n\n useEffect(() => {\n if (!tracker) return;\n\n const targetPath =\n path ?? (typeof window !== \"undefined\" ? window.location.pathname : \"/\");\n\n const refresh = (): void => {\n setData(tracker.getHeatmapData(targetPath) as HeatmapPoint[]);\n pendingRef.current = false;\n };\n\n // Initial read.\n refresh();\n\n // Re-read after each new heatmap point, debounced by refreshMs.\n const unsub = tracker.subscribe((event: TrackerEvent) => {\n if (event.type === \"heatmap\" && event.data.path === targetPath) {\n if (!pendingRef.current) {\n pendingRef.current = true;\n setTimeout(refresh, refreshMs);\n }\n }\n });\n\n return unsub;\n // refreshMs is intentionally excluded — changing it after mount has no effect.\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [tracker, path]);\n\n return data;\n}\n\n// ─── usePageViews ─────────────────────────────────────────────────────────────\n\n/**\n * Returns a live array of all page views recorded in the current session.\n */\nexport function usePageViews(): PageView[] {\n const tracker = useTrackerContext();\n const [views, setViews] = useState<PageView[]>([]);\n\n useEffect(() => {\n if (!tracker) return;\n\n setViews(tracker.getPageViews());\n\n const unsub = tracker.subscribe((event: TrackerEvent) => {\n if (event.type === \"pageview\") {\n setViews(tracker.getPageViews());\n }\n });\n\n return unsub;\n }, [tracker]);\n\n return views;\n}\n\n// ─── useTimeSpent ─────────────────────────────────────────────────────────────\n\n/**\n * Returns a live record of cumulative milliseconds spent per path.\n */\nexport function useTimeSpent(): Record<string, number> {\n const tracker = useTrackerContext();\n const [time, setTime] = useState<Record<string, number>>({});\n\n useEffect(() => {\n if (!tracker) return;\n\n setTime(tracker.getTimeSpent());\n\n const unsub = tracker.subscribe((event: TrackerEvent) => {\n if (event.type === \"timespent\") {\n setTime(tracker.getTimeSpent());\n }\n });\n\n return unsub;\n }, [tracker]);\n\n return time;\n}\n"],"mappings":"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","names":["useEffect","useRef","VISITOR_STORAGE_KEY","generateSessionId","c","r","getOrCreateVisitorId","generateSessionId","existing","VISITOR_STORAGE_KEY","id","e","fetchLocation","res","d","e","_NavigationPlugin","emit","sessionId","origPush","state","title","url","origReplace","newPath","timeOnPage","path","NavigationPlugin","TimePlugin","emit","sessionId","e","duration","throttle","fn","delay","lastCall","args","now","_HeatmapPlugin","emit","sessionId","sampleRate","maxPoints","allowedPaths","e","pageWidth","pageHeight","absY","target","vw","scrollX","scrollY","absX","throttle","_a","point","_b","_c","_d","el","x","y","path","now","c","nearby","HeatmapPlugin","LogCapture","options","u","parts","args","first","stack","msg","src","line","col","err","e","reason","message","level","extra","a","entry","url","body","html2canvas","DEFAULT_INTERVAL_MS","STORAGE_KEY","SnapshotPlugin","cfg","_a","stored","e","path","last","root","pageBg","canvas","blob","resolve","form","headers","DEFAULT_ENDPOINT","DEFAULTS","UserTracker","config","e","_a","path","generateSessionId","getOrCreateVisitorId","emit","sessionId","NavigationPlugin","TimePlugin","HeatmapPlugin","fetchLocation","loc","LogCapture","SnapshotPlugin","_b","_c","_d","_e","url","authHeaders","body","event","prev","key","pts","fn","resolvedPath","acc","k","v","batch","blob","events","createContext","useContext","TrackerContext","useTrackerContext","jsx","UserTrackerProvider","config","children","trackerRef","useRef","UserTracker","useEffect","tracker","TrackerContext","useEffect","useRef","useState","useTracker","useTrackerContext","usePageView","path","tracker","useEffect","useHeatmapData","refreshMs","data","setData","useState","pendingRef","useRef","targetPath","refresh","event","usePageViews","views","setViews","useTimeSpent","time","setTime"]}
|
|
1
|
+
{"version":3,"sources":["../../src/react/provider.tsx","../../src/utils/session.ts","../../src/utils/geo.ts","../../src/core/navigation.ts","../../src/core/time.ts","../../src/utils/throttle.ts","../../src/core/heatmap.ts","../../src/core/logger.ts","../../src/tracker.ts","../../src/react/context.tsx","../../src/react/hooks.ts"],"sourcesContent":["import { useEffect, useRef, type ReactNode } from \"react\";\nimport { UserTracker } from \"../tracker\";\nimport type { TrackerConfig } from \"../types\";\nimport { TrackerContext } from \"./context\";\n\nexport interface UserTrackerProviderProps {\n /** Tracker configuration. Captured on first render — changes are ignored. */\n config?: TrackerConfig;\n children: ReactNode;\n}\n\n/**\n * Wraps your application (or a subtree) and provides a `UserTracker` instance\n * via React context.\n *\n * The tracker is created once, initialized on mount, and destroyed on unmount.\n *\n * **Next.js App Router** — mark your layout wrapper as a Client Component:\n * ```tsx\n * 'use client';\n * import { UserTrackerProvider } from 'user-tracker/react';\n * export default function RootLayout({ children }) {\n * return <UserTrackerProvider config={{ endpoint: '/api/events' }}>{children}</UserTrackerProvider>;\n * }\n * ```\n */\nexport function UserTrackerProvider({\n config = {},\n children,\n}: UserTrackerProviderProps) {\n // Create the tracker instance exactly once (lazy ref initialisation).\n const trackerRef = useRef<UserTracker | null>(null);\n if (trackerRef.current === null) {\n trackerRef.current = new UserTracker(config);\n }\n\n useEffect(() => {\n const tracker = trackerRef.current!;\n tracker.init();\n return () => tracker.destroy();\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n return (\n <TrackerContext.Provider value={trackerRef.current}>\n {children}\n </TrackerContext.Provider>\n );\n}\n","const SESSION_STORAGE_KEY = \"__ut_sid__\";\nconst VISITOR_STORAGE_KEY = \"__ut_vid__\";\n\n/** Generate a RFC-4122 v4 UUID using the native crypto API with a fallback. */\nexport function generateSessionId(): string {\n if (\n typeof crypto !== \"undefined\" &&\n typeof crypto.randomUUID === \"function\"\n ) {\n return crypto.randomUUID();\n }\n // Math.random fallback (not cryptographically secure, but sufficient for analytics)\n return \"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx\".replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0;\n const v = c === \"x\" ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n });\n}\n\n/**\n * Retrieve the session ID from sessionStorage, or create and persist a new one.\n * Falls back to an in-memory ID when sessionStorage is unavailable (e.g. SSR).\n */\nexport function getOrCreateSessionId(): string {\n if (typeof sessionStorage === \"undefined\") return generateSessionId();\n try {\n const existing = sessionStorage.getItem(SESSION_STORAGE_KEY);\n if (existing) return existing;\n const id = generateSessionId();\n sessionStorage.setItem(SESSION_STORAGE_KEY, id);\n return id;\n } catch {\n return generateSessionId();\n }\n}\n\n/**\n * Retrieve the visitor ID from localStorage (persists across sessions/tabs),\n * or create and store a new one. Returns a temporary in-memory ID when\n * localStorage is unavailable.\n */\nexport function getOrCreateVisitorId(): string {\n if (typeof localStorage === \"undefined\") return generateSessionId();\n try {\n const existing = localStorage.getItem(VISITOR_STORAGE_KEY);\n if (existing) return existing;\n const id = generateSessionId();\n localStorage.setItem(VISITOR_STORAGE_KEY, id);\n return id;\n } catch {\n return generateSessionId();\n }\n}\n","import type { GeoLocation } from \"../types\";\n\n/**\n * Resolves the visitor's approximate location from their public IP address\n * using the ipapi.co free-tier JSON endpoint (no API-key required, up to\n * 1 000 requests/day on the free plan).\n *\n * Runs silently — returns `null` on any network error, rate-limit, or\n * reserved/private IP so that tracking is never blocked.\n */\nexport async function fetchLocation(): Promise<GeoLocation | null> {\n try {\n const res = await fetch(\"https://ipapi.co/json/\", {\n method: \"GET\",\n headers: { Accept: \"application/json\" },\n });\n if (!res.ok) return null;\n const d = (await res.json()) as Record<string, unknown>;\n // ipapi returns { \"error\": true, \"reason\": \"...\" } for private/reserved IPs\n if (d[\"error\"]) return null;\n return {\n country: typeof d[\"country_code\"] === \"string\" ? d[\"country_code\"] : \"\",\n countryName:\n typeof d[\"country_name\"] === \"string\" ? d[\"country_name\"] : \"\",\n city: typeof d[\"city\"] === \"string\" ? d[\"city\"] : undefined,\n region: typeof d[\"region\"] === \"string\" ? d[\"region\"] : undefined,\n latitude: typeof d[\"latitude\"] === \"number\" ? d[\"latitude\"] : undefined,\n longitude:\n typeof d[\"longitude\"] === \"number\" ? d[\"longitude\"] : undefined,\n };\n } catch {\n return null;\n }\n}\n","import type { TrackerEvent } from \"../types\";\n\ntype EmitFn = (event: TrackerEvent) => void;\n\ninterface NavigationPluginOptions {\n emit: EmitFn;\n sessionId: string;\n}\n\n/**\n * Tracks SPA route changes by monkey-patching history.pushState /\n * history.replaceState and listening to the popstate event.\n *\n * For every navigation it:\n * 1. Emits a `pageview` event.\n * 2. Dispatches the custom DOM event `tracker:navigate` so that other\n * plugins (TimePlugin, HeatmapPlugin) can react without having to\n * duplicate the pushState patching.\n * 3. Detects U-turns: if the user leaves a page in ≤5 s, a `uturn` event\n * is emitted with the time-on-page and destination path.\n *\n * Next.js App Router note:\n * The App Router manages navigation internally; use `usePageView(pathname)`\n * from `user-tracker/react` together with `usePathname()` instead.\n */\nexport class NavigationPlugin {\n private readonly emit: EmitFn;\n private readonly sessionId: string;\n private previousPath = \"\";\n private originalPushState: typeof history.pushState | null = null;\n private originalReplaceState: typeof history.replaceState | null = null;\n private pageEntryTime = 0;\n private static readonly UTURN_THRESHOLD_MS = 5_000;\n\n constructor({ emit, sessionId }: NavigationPluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n }\n\n init(): void {\n // Record the initial page view on load.\n this.recordPageView(window.location.pathname + window.location.search);\n\n window.addEventListener(\"popstate\", this.handlePopState);\n\n // Patch pushState\n this.originalPushState = history.pushState.bind(history);\n const origPush = this.originalPushState;\n history.pushState = (state, title, url): void => {\n origPush(state, title, url);\n this.handleNavigation();\n };\n\n // Patch replaceState\n this.originalReplaceState = history.replaceState.bind(history);\n const origReplace = this.originalReplaceState;\n history.replaceState = (state, title, url): void => {\n origReplace(state, title, url);\n this.handleNavigation();\n };\n }\n\n destroy(): void {\n window.removeEventListener(\"popstate\", this.handlePopState);\n if (this.originalPushState) history.pushState = this.originalPushState;\n if (this.originalReplaceState)\n history.replaceState = this.originalReplaceState;\n }\n\n // Arrow property → always bound to `this`, safe to use as event listener.\n private handlePopState = (): void => {\n this.handleNavigation();\n };\n\n private handleNavigation(): void {\n const newPath = window.location.pathname + window.location.search;\n if (newPath === this.previousPath) return; // hash-only or duplicate call\n\n // U-turn detection: user left the previous page very quickly.\n if (this.previousPath && this.pageEntryTime > 0) {\n const timeOnPage = Date.now() - this.pageEntryTime;\n if (timeOnPage <= NavigationPlugin.UTURN_THRESHOLD_MS) {\n this.emit({\n type: \"uturn\",\n data: {\n fromPath: this.previousPath,\n toPath: newPath,\n timeOnPageMs: timeOnPage,\n timestamp: Date.now(),\n sessionId: this.sessionId,\n },\n });\n }\n }\n\n this.recordPageView(newPath);\n }\n\n private recordPageView(path: string): void {\n this.previousPath = path;\n this.pageEntryTime = Date.now();\n\n this.emit({\n type: \"pageview\",\n data: {\n path,\n title: document.title,\n timestamp: Date.now(),\n sessionId: this.sessionId,\n referrer: document.referrer || undefined,\n },\n });\n\n // Notify other plugins via a custom DOM event (synchronous dispatch).\n window.dispatchEvent(\n new CustomEvent(\"tracker:navigate\", {\n detail: { path, title: document.title },\n }),\n );\n }\n}\n","import type { TrackerEvent } from \"../types\";\n\ntype EmitFn = (event: TrackerEvent) => void;\n\ninterface TimePluginOptions {\n emit: EmitFn;\n sessionId: string;\n}\n\n/**\n * Tracks the time a user spends on each page.\n *\n * - Starts a timer when the page becomes active (init / tab focus).\n * - Stops and emits a `timespent` event when:\n * • The user navigates away (tracker:navigate)\n * • The tab is hidden (visibilitychange)\n * • The page is unloading (beforeunload / pagehide)\n * - Resumes timing when the tab becomes visible again.\n */\nexport class TimePlugin {\n private readonly emit: EmitFn;\n private readonly sessionId: string;\n private currentPath = \"\";\n private startTime = 0;\n private tracking = false;\n\n constructor({ emit, sessionId }: TimePluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n }\n\n init(): void {\n this.currentPath = window.location.pathname + window.location.search;\n this.startTracking();\n\n window.addEventListener(\"tracker:navigate\", this.handleNavigate);\n document.addEventListener(\"visibilitychange\", this.handleVisibilityChange);\n window.addEventListener(\"beforeunload\", this.handleUnload);\n window.addEventListener(\"pagehide\", this.handleUnload);\n }\n\n destroy(): void {\n this.stopTracking();\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n document.removeEventListener(\n \"visibilitychange\",\n this.handleVisibilityChange,\n );\n window.removeEventListener(\"beforeunload\", this.handleUnload);\n window.removeEventListener(\"pagehide\", this.handleUnload);\n }\n\n private startTracking(): void {\n this.startTime = Date.now();\n this.tracking = true;\n }\n\n private stopTracking(): void {\n if (!this.tracking || !this.currentPath) return;\n const duration = Date.now() - this.startTime;\n if (duration < 100) {\n this.tracking = false;\n return; // Ignore sub-100 ms blips (e.g. rapid navigations).\n }\n this.emit({\n type: \"timespent\",\n data: {\n path: this.currentPath,\n duration,\n sessionId: this.sessionId,\n timestamp: Date.now(),\n },\n });\n this.tracking = false;\n }\n\n private handleNavigate = (e: CustomEvent<{ path: string }>): void => {\n this.stopTracking();\n this.currentPath = e.detail.path;\n this.startTracking();\n };\n\n private handleVisibilityChange = (): void => {\n if (document.hidden) {\n this.stopTracking();\n } else {\n this.startTracking();\n }\n };\n\n private handleUnload = (): void => {\n this.stopTracking();\n };\n}\n","/**\n * Returns a function that invokes `fn` at most once every `delay` ms.\n * The first call in a new window is executed immediately.\n */\nexport function throttle<Args extends unknown[]>(\n fn: (...args: Args) => void,\n delay: number,\n): (...args: Args) => void {\n let lastCall = 0;\n return (...args: Args): void => {\n const now = Date.now();\n if (now - lastCall >= delay) {\n lastCall = now;\n fn(...args);\n }\n };\n}\n","import type { TrackerEvent, HeatmapPoint } from \"../types\";\nimport { throttle } from \"../utils/throttle\";\n\ntype EmitFn = (event: TrackerEvent) => void;\n\ninterface HeatmapPluginOptions {\n emit: EmitFn;\n sessionId: string;\n /** Fraction of mousemove / scroll events to sample (0–1). Default: 0.3 */\n sampleRate?: number;\n /** Maximum points stored per page before recording stops. Default: 2000 */\n maxPoints?: number;\n /**\n * If provided, heatmap data will only be collected for paths in this list.\n * An empty array or undefined means all pages are tracked.\n */\n allowedPaths?: string[];\n}\n\n/**\n * Collects mouse-move, click, and scroll positions for heatmap analysis.\n *\n * Also detects rage clicks (≥3 clicks within 1 s in a 50 px radius) and emits\n * a `rageclik` event so the backend can surface problematic UI hotspots.\n *\n * Click events include an optional `target` field containing a human-readable\n * label for the clicked element (aria-label › id › button/link text).\n */\nexport class HeatmapPlugin {\n private readonly emit: EmitFn;\n private readonly sessionId: string;\n private readonly sampleRate: number;\n private readonly maxPoints: number;\n private readonly allowedPaths: Set<string> | null;\n private currentPath = \"\";\n private pointCounts: Record<string, number> = {};\n\n // Rage-click detection state\n private recentClicks: { x: number; y: number; t: number }[] = [];\n private static readonly RAGE_THRESHOLD = 3;\n private static readonly RAGE_WINDOW_MS = 1_000;\n private static readonly RAGE_RADIUS_PX = 50;\n\n private readonly throttledMouseMove: (e: MouseEvent) => void;\n private readonly throttledScroll: () => void;\n\n constructor({\n emit,\n sessionId,\n sampleRate = 0.3,\n maxPoints = 2000,\n allowedPaths,\n }: HeatmapPluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n this.sampleRate = sampleRate;\n this.maxPoints = maxPoints;\n this.allowedPaths =\n allowedPaths && allowedPaths.length > 0 ? new Set(allowedPaths) : null;\n\n this.throttledMouseMove = throttle(this.handleMouseMove, 50);\n this.throttledScroll = throttle(this.handleScroll, 100);\n }\n\n init(): void {\n this.currentPath = window.location.pathname + window.location.search;\n\n document.addEventListener(\"mousemove\", this.throttledMouseMove);\n document.addEventListener(\"click\", this.handleClick);\n document.addEventListener(\"touchend\", this.handleTouchEnd, {\n passive: true,\n });\n window.addEventListener(\"scroll\", this.throttledScroll, { passive: true });\n window.addEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n destroy(): void {\n document.removeEventListener(\"mousemove\", this.throttledMouseMove);\n document.removeEventListener(\"click\", this.handleClick);\n document.removeEventListener(\"touchend\", this.handleTouchEnd);\n window.removeEventListener(\"scroll\", this.throttledScroll);\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n private canRecord(): boolean {\n if (\n this.allowedPaths !== null &&\n !this.allowedPaths.has(this.currentPath)\n ) {\n return false;\n }\n return (this.pointCounts[this.currentPath] ?? 0) < this.maxPoints;\n }\n\n private recordPoint(\n point: Omit<HeatmapPoint, \"path\" | \"timestamp\" | \"sessionId\">,\n ): void {\n if (!this.canRecord()) return;\n this.pointCounts[this.currentPath] =\n (this.pointCounts[this.currentPath] ?? 0) + 1;\n this.emit({\n type: \"heatmap\",\n data: {\n ...point,\n path: this.currentPath,\n timestamp: Date.now(),\n },\n });\n }\n\n /** Extract a human-readable label for the clicked element. */\n private getClickTarget(e: MouseEvent): string | undefined {\n const el = e.target as HTMLElement | null;\n if (!el) return undefined;\n const label =\n el.getAttribute(\"aria-label\") ||\n el.closest(\"[aria-label]\")?.getAttribute(\"aria-label\") ||\n el.getAttribute(\"data-track-label\") ||\n el.getAttribute(\"id\") ||\n (el instanceof HTMLButtonElement || el instanceof HTMLAnchorElement\n ? el.innerText?.trim().slice(0, 60)\n : el.closest(\"button, a\")?.textContent?.trim().slice(0, 60));\n return label || undefined;\n }\n\n /** Detect rage-click bursts and emit a `rageclik` event when found. */\n private checkRageClick(x: number, y: number, path: string): void {\n const now = Date.now();\n this.recentClicks = this.recentClicks.filter(\n (c) => now - c.t < HeatmapPlugin.RAGE_WINDOW_MS,\n );\n this.recentClicks.push({ x, y, t: now });\n\n const nearby = this.recentClicks.filter(\n (c) => Math.hypot(c.x - x, c.y - y) <= HeatmapPlugin.RAGE_RADIUS_PX,\n );\n\n if (nearby.length >= HeatmapPlugin.RAGE_THRESHOLD) {\n this.emit({\n type: \"rageclik\",\n data: {\n path,\n x,\n y,\n count: nearby.length,\n timestamp: now,\n sessionId: this.sessionId,\n },\n });\n // Reset to avoid continuously re-triggering in the same burst.\n this.recentClicks = [];\n }\n }\n\n private handleMouseMove = (e: MouseEvent): void => {\n if (Math.random() > this.sampleRate) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = e.clientY + window.scrollY;\n this.recordPoint({\n x: e.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (e.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"move\",\n });\n };\n\n private handleClick = (e: MouseEvent): void => {\n // Skip if a touchend already recorded this tap (within 500 ms)\n if (Date.now() - this.lastTouchTime < 500) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = e.clientY + window.scrollY;\n const target = this.getClickTarget(e);\n this.recordPoint({\n x: e.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (e.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"click\",\n ...(target ? { target } : {}),\n });\n this.checkRageClick(e.clientX, absY, this.currentPath);\n };\n\n /**\n * Records taps on touch devices.\n * Mobile browsers do fire a synthetic `click` after `touchend`, but some\n * frameworks call `preventDefault()` on touch events which suppresses it.\n * Listening to `touchend` directly ensures taps are always captured.\n * We mark the point with a short-lived flag so the subsequent synthetic\n * `click` event (if it fires) is deduplicated.\n */\n private lastTouchTime = 0;\n\n private handleTouchEnd = (e: TouchEvent): void => {\n const touch = e.changedTouches[0];\n if (!touch) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = touch.clientY + window.scrollY;\n this.lastTouchTime = Date.now();\n this.recordPoint({\n x: touch.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (touch.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"click\",\n });\n this.checkRageClick(touch.clientX, absY, this.currentPath);\n };\n\n private handleScroll = (): void => {\n if (Math.random() > this.sampleRate) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const vw = window.innerWidth;\n const scrollX = window.scrollX;\n const scrollY = window.scrollY;\n const absX = scrollX + vw / 2;\n this.recordPoint({\n x: vw / 2,\n y: scrollY,\n xPct: pageWidth > 0 ? (absX / pageWidth) * 100 : 50,\n yPct: pageHeight > 0 ? (scrollY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"scroll\",\n });\n };\n\n private handleNavigate = (e: CustomEvent<{ path: string }>): void => {\n this.currentPath = e.detail.path;\n };\n}\n","export type LogLevel = \"info\" | \"warn\" | \"error\";\n\nexport interface LogEntry {\n sessionId?: string;\n appId?: string;\n level: LogLevel;\n message: string;\n url?: string;\n stack?: string;\n meta?: Record<string, unknown>;\n timestamp: number;\n}\n\ntype ConsoleFn = (...args: unknown[]) => void;\n\n/**\n * Automatically captures console.info/warn/error output and unhandled errors,\n * then ships them to the backend `/logs/ingest` endpoint.\n *\n * All console methods are restored exactly in `destroy()`.\n */\nexport class LogCapture {\n private readonly endpoint: string;\n private readonly sessionId: string;\n private readonly appId?: string;\n private readonly authHeaders: Record<string, string>;\n\n // Original console methods preserved so we can restore them.\n private origInfo!: ConsoleFn;\n private origWarn!: ConsoleFn;\n private origError!: ConsoleFn;\n\n private prevOnError: OnErrorEventHandler = null;\n private prevOnUnhandledRejection:\n | ((e: PromiseRejectionEvent) => void)\n | null = null;\n\n private initialized = false;\n\n constructor(options: {\n endpoint: string;\n sessionId: string;\n secretKey?: string;\n appId?: string;\n }) {\n // Derive the API base URL by stripping everything from the last path\n // segment that isn't a versioning prefix. The tracker config `endpoint`\n // is the *events* URL (e.g. http://host/api/events), but logs live at\n // http://host/api/logs/ingest, so we walk up until we reach the common\n // base (i.e. remove the final segment).\n try {\n const u = new URL(options.endpoint);\n // Remove the last non-empty path segment (e.g. \"/api/events\" → \"/api\")\n const parts = u.pathname.replace(/\\/$/, \"\").split(\"/\");\n parts.pop();\n u.pathname = parts.join(\"/\") || \"/\";\n this.endpoint = u.toString().replace(/\\/$/, \"\");\n } catch {\n this.endpoint = options.endpoint;\n }\n this.sessionId = options.sessionId;\n this.appId = options.appId;\n this.authHeaders = options.secretKey\n ? { Authorization: `Bearer ${options.secretKey}` }\n : {};\n }\n\n init(): void {\n if (typeof window === \"undefined\" || this.initialized) return;\n\n this.origInfo = console.info.bind(console);\n this.origWarn = console.warn.bind(console);\n this.origError = console.error.bind(console);\n\n console.info = (...args: unknown[]) => {\n this.origInfo(...args);\n this.send(\"info\", this.format(args));\n };\n\n console.warn = (...args: unknown[]) => {\n this.origWarn(...args);\n this.send(\"warn\", this.format(args));\n };\n\n console.error = (...args: unknown[]) => {\n this.origError(...args);\n const [first] = args;\n const stack = first instanceof Error ? first.stack : undefined;\n this.send(\"error\", this.format(args), { stack });\n };\n\n this.prevOnError = window.onerror;\n window.onerror = (msg, src, line, col, err) => {\n this.send(\"error\", String(msg), {\n stack: err?.stack,\n meta: { src, line, col },\n });\n if (typeof this.prevOnError === \"function\") {\n return this.prevOnError(msg, src, line, col, err);\n }\n return false;\n };\n\n this.prevOnUnhandledRejection = (e: PromiseRejectionEvent) => {\n const reason = e.reason;\n const message =\n reason instanceof Error\n ? reason.message\n : String(reason ?? \"Unhandled promise rejection\");\n this.send(\"error\", message, {\n stack: reason instanceof Error ? reason.stack : undefined,\n });\n };\n window.addEventListener(\n \"unhandledrejection\",\n this.prevOnUnhandledRejection,\n );\n\n this.initialized = true;\n }\n\n destroy(): void {\n if (!this.initialized) return;\n console.info = this.origInfo;\n console.warn = this.origWarn;\n console.error = this.origError;\n\n window.onerror = this.prevOnError;\n if (this.prevOnUnhandledRejection) {\n window.removeEventListener(\n \"unhandledrejection\",\n this.prevOnUnhandledRejection,\n );\n }\n this.initialized = false;\n }\n\n /** Manually capture a log entry (e.g. from try/catch). */\n capture(\n level: LogLevel,\n message: string,\n extra?: { stack?: string; meta?: Record<string, unknown> },\n ): void {\n this.send(level, message, extra);\n }\n\n private format(args: unknown[]): string {\n return args\n .map((a) => {\n if (a instanceof Error) return a.message;\n if (typeof a === \"object\") {\n try {\n return JSON.stringify(a);\n } catch {\n return String(a);\n }\n }\n return String(a);\n })\n .join(\" \");\n }\n\n private send(\n level: LogLevel,\n message: string,\n extra?: { stack?: string; meta?: Record<string, unknown> },\n ): void {\n const entry: LogEntry = {\n sessionId: this.sessionId,\n ...(this.appId ? { appId: this.appId } : {}),\n level,\n message,\n url: typeof window !== \"undefined\" ? window.location.href : undefined,\n stack: extra?.stack,\n meta: extra?.meta,\n timestamp: Date.now(),\n };\n const url = `${this.endpoint}/logs/ingest`;\n const body = JSON.stringify(entry);\n\n // Use fetch with keepalive so the request survives page navigation.\n // Errors are logged to the (original, unpatched) console so they are\n // visible in DevTools without creating an infinite log loop.\n void fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...this.authHeaders },\n body,\n keepalive: true,\n }).catch((err: unknown) => {\n // Use the original (pre-patch) error logger to avoid recursion.\n if (this.origError) {\n this.origError(\"[user-tracker] Failed to send log:\", err);\n }\n });\n }\n}\n","import type {\n TrackerConfig,\n TrackerEvent,\n SessionData,\n PageView,\n HeatmapPoint,\n GeoLocation,\n} from \"./types\";\nimport { generateSessionId, getOrCreateVisitorId } from \"./utils/session\";\nimport { fetchLocation } from \"./utils/geo\";\nimport { NavigationPlugin } from \"./core/navigation\";\nimport { TimePlugin } from \"./core/time\";\nimport { HeatmapPlugin } from \"./core/heatmap\";\nimport { LogCapture } from \"./core/logger\";\n\nexport const DEFAULT_ENDPOINT = \"https://api.alphana.ir/api/events\";\n\nconst DEFAULTS = {\n endpoint: DEFAULT_ENDPOINT,\n trackNavigation: true,\n trackTime: true,\n trackHeatmap: true,\n trackLogs: true,\n trackSnapshots: true,\n mouseSampleRate: 0.3,\n maxHeatmapPoints: 2000,\n batchSize: 20,\n flushInterval: 5_000,\n} as const;\n\ntype SubscriberFn = (event: TrackerEvent) => void;\n\n/**\n * Core tracker class. Framework-agnostic — works in any environment that has\n * a browser DOM (React, Next.js Pages Router, Vite, vanilla JS/TS, etc.).\n *\n * Usage:\n * ```ts\n * const tracker = new UserTracker({ endpoint: 'https://my-api.com/events' });\n * tracker.init(); // call once; safe to call in SSR (no-op server-side)\n * ```\n *\n * Destroy when done (e.g. component unmount):\n * ```ts\n * tracker.destroy();\n * ```\n */\ntype ResolvedConfig = Required<\n Pick<\n TrackerConfig,\n | \"endpoint\"\n | \"trackNavigation\"\n | \"trackTime\"\n | \"trackHeatmap\"\n | \"trackLogs\"\n | \"mouseSampleRate\"\n | \"maxHeatmapPoints\"\n | \"batchSize\"\n | \"flushInterval\"\n >\n> &\n TrackerConfig;\n\nexport class UserTracker {\n private readonly cfg: ResolvedConfig;\n private session: SessionData;\n private navigation?: NavigationPlugin;\n private time?: TimePlugin;\n private heatmap?: HeatmapPlugin;\n /** Public so consumers can call logCapture.capture() for manual log entries. */\n logCapture?: LogCapture;\n private initialized = false;\n private readonly subscribers = new Set<SubscriberFn>();\n\n /** In-memory queue of events waiting to be flushed. */\n private queue: TrackerEvent[] = [];\n private flushTimer: ReturnType<typeof setInterval> | null = null;\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private location: GeoLocation | null = null;\n\n constructor(config: TrackerConfig = {}) {\n this.cfg = { ...DEFAULTS, ...config } as ResolvedConfig;\n\n // Validate endpoint URL up front so the error is thrown at construction\n // time rather than silently failing during a network request.\n try {\n new URL(this.cfg.endpoint);\n } catch {\n throw new Error(\n `[alpha-tracker] Invalid endpoint URL: \"${this.cfg.endpoint}\"`,\n );\n }\n\n this.session = {\n id: config.sessionId ?? generateSessionId(),\n visitorId: getOrCreateVisitorId(),\n startedAt: Date.now(),\n pageViews: [],\n timeSpent: {},\n heatmap: {},\n };\n }\n\n // ─── Lifecycle ──────────────────────────────────────────────────────────────\n\n /**\n * Attach event listeners and start tracking.\n * Safe to call during SSR — returns `this` immediately if `window` is\n * undefined so it can be chained: `const tracker = new UserTracker(cfg).init()`.\n */\n init(): this {\n if (typeof window === \"undefined\" || this.initialized) return this;\n\n const emit = this.emit.bind(this);\n const { id: sessionId } = this.session;\n\n if (this.cfg.trackNavigation) {\n this.navigation = new NavigationPlugin({ emit, sessionId });\n this.navigation.init();\n }\n\n if (this.cfg.trackTime) {\n this.time = new TimePlugin({ emit, sessionId });\n this.time.init();\n }\n\n if (this.cfg.trackHeatmap) {\n this.heatmap = new HeatmapPlugin({\n emit,\n sessionId,\n sampleRate: this.cfg.mouseSampleRate,\n maxPoints: this.cfg.maxHeatmapPoints,\n allowedPaths: this.cfg.heatmapPages,\n });\n this.heatmap.init();\n }\n\n if (this.cfg.endpoint) {\n // Flush on a regular interval — even if the batch threshold isn't hit.\n this.flushTimer = setInterval(() => {\n if (this.queue.length > 0) void this.flush();\n }, this.cfg.flushInterval);\n\n // Resolve visitor location from IP in the background.\n void fetchLocation().then((loc) => {\n this.location = loc;\n if (loc) this.session.location = loc;\n });\n\n // Flush remaining queue when the tab is hidden or the page is unloaded.\n window.addEventListener(\"visibilitychange\", this.handleVisibilityChange);\n window.addEventListener(\"pagehide\", this.handlePageHide);\n\n // Periodic keep-alive heartbeat every 30 s so the backend knows the\n // session is still active and doesn't expire it prematurely.\n this.heartbeatTimer = setInterval(() => {\n if (document.visibilityState !== \"hidden\") void this.sendHeartbeat();\n }, 30_000);\n\n // Auto-capture console logs and unhandled errors.\n if (this.cfg.trackLogs) {\n this.logCapture = new LogCapture({\n endpoint: this.cfg.endpoint,\n sessionId: this.session.id,\n secretKey: this.cfg.secretKey,\n appId: this.cfg.appId,\n });\n this.logCapture.init();\n }\n }\n\n this.initialized = true;\n return this;\n }\n\n /** Remove all event listeners, flush remaining queue, and reset state. */\n destroy(): void {\n if (this.flushTimer !== null) {\n clearInterval(this.flushTimer);\n this.flushTimer = null;\n }\n if (this.heartbeatTimer !== null) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n\n if (typeof window !== \"undefined\") {\n window.removeEventListener(\n \"visibilitychange\",\n this.handleVisibilityChange,\n );\n window.removeEventListener(\"pagehide\", this.handlePageHide);\n }\n\n this.navigation?.destroy();\n this.time?.destroy();\n this.heatmap?.destroy();\n this.logCapture?.destroy();\n\n if (typeof window !== \"undefined\") {\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n // Best-effort flush of any remaining queued events.\n if (this.queue.length > 0 && this.cfg.endpoint) {\n this.flushBeacon();\n }\n\n this.initialized = false;\n }\n\n private handleVisibilityChange = (): void => {\n if (document.visibilityState === \"hidden\") {\n if (this.queue.length > 0) this.flushBeacon();\n this.sendDeactivate();\n }\n };\n\n private handlePageHide = (): void => {\n if (this.queue.length > 0) this.flushBeacon();\n this.sendDeactivate();\n };\n\n private handleNavigate = (_e: Event): void => {\n // Navigation hook kept for potential future use.\n };\n\n /**\n * Send a keep-alive heartbeat so the backend knows this session is still\n * active. Called every 30 s while the tab is visible.\n */\n private async sendHeartbeat(): Promise<void> {\n if (!this.cfg.endpoint) return;\n const url = `${this.cfg.endpoint}/heartbeat`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n const body = JSON.stringify({\n sessionId: this.session.id,\n visitorId: this.session.visitorId,\n path: typeof window !== \"undefined\" ? window.location.pathname : \"/\",\n active: true,\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n ...(this.location ? { location: this.location } : {}),\n });\n try {\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body,\n keepalive: true,\n });\n } catch {\n // Silent — heartbeat failure should never surface to the user.\n }\n }\n\n /**\n * Send a synchronous beacon to mark this session as inactive.\n * Called on page unload / tab hidden so the dashboard reflects real-time.\n */\n private sendDeactivate(): void {\n if (!this.cfg.endpoint) return;\n const url = `${this.cfg.endpoint}/heartbeat`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n const body = JSON.stringify({\n sessionId: this.session.id,\n path: typeof window !== \"undefined\" ? window.location.pathname : \"/\",\n active: false,\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n });\n // sendBeacon fires even if the page is being unloaded.\n if (typeof navigator !== \"undefined\" && navigator.sendBeacon) {\n navigator.sendBeacon(url, new Blob([body], { type: \"application/json\" }));\n } else {\n // Fallback for environments without sendBeacon.\n void fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body,\n keepalive: true,\n }).catch(() => undefined);\n }\n }\n\n // ─── Event pipeline ─────────────────────────────────────────────────────────\n\n /** Emit a tracker event. Also used internally by the plugins. */\n emit(event: TrackerEvent): void {\n // 1 – accumulate into session data\n switch (event.type) {\n case \"pageview\":\n this.session.pageViews.push(event.data);\n break;\n\n case \"timespent\": {\n const prev = this.session.timeSpent[event.data.path] ?? 0;\n this.session.timeSpent[event.data.path] = prev + event.data.duration;\n break;\n }\n\n case \"heatmap\": {\n const key = event.data.path;\n if (!this.session.heatmap[key]) this.session.heatmap[key] = [];\n const pts = this.session.heatmap[key];\n if (pts.length < this.cfg.maxHeatmapPoints) pts.push(event.data);\n break;\n }\n\n // rageclik and uturn are forwarded to the backend only; no local state needed.\n case \"rageclik\":\n case \"uturn\":\n break;\n }\n\n // 2 – notify subscribers (used internally by React hooks)\n this.subscribers.forEach((fn) => fn(event));\n\n // 3 – user callback\n this.cfg.onEvent?.(event);\n\n // 4 – enqueue for batched remote sending\n if (this.cfg.endpoint) {\n this.queue.push(event);\n // Auto-flush once the batch size threshold is reached.\n if (this.queue.length >= this.cfg.batchSize) {\n void this.flush();\n }\n }\n }\n\n /**\n * Subscribe to every emitted event. Returns an unsubscribe function.\n *\n * ```ts\n * const unsub = tracker.subscribe(event => console.log(event));\n * // later…\n * unsub();\n * ```\n */\n subscribe(fn: SubscriberFn): () => void {\n this.subscribers.add(fn);\n return () => this.subscribers.delete(fn);\n }\n\n // ─── Manual tracking helpers ────────────────────────────────────────────────\n\n /**\n * Manually record a page view and dispatch `tracker:navigate`.\n * Required for Next.js App Router — call it inside a `useEffect` that\n * depends on `usePathname()`:\n *\n * ```tsx\n * const pathname = usePathname();\n * useEffect(() => { tracker.trackPageView(pathname); }, [pathname]);\n * ```\n */\n trackPageView(path?: string): void {\n const resolvedPath =\n path ??\n (typeof window !== \"undefined\"\n ? window.location.pathname + window.location.search\n : \"/\");\n\n this.emit({\n type: \"pageview\",\n data: {\n path: resolvedPath,\n title: typeof document !== \"undefined\" ? document.title : \"\",\n timestamp: Date.now(),\n sessionId: this.session.id,\n referrer:\n typeof document !== \"undefined\"\n ? document.referrer || undefined\n : undefined,\n },\n });\n\n if (typeof window !== \"undefined\") {\n window.dispatchEvent(\n new CustomEvent(\"tracker:navigate\", {\n detail: { path: resolvedPath, title: document.title },\n }),\n );\n }\n }\n\n // ─── Data accessors ─────────────────────────────────────────────────────────\n\n /** A read-only snapshot of the current session. */\n getSession(): Readonly<SessionData> {\n return this.session;\n }\n\n /** All page views recorded so far. */\n getPageViews(): PageView[] {\n return [...this.session.pageViews];\n }\n\n /** Cumulative milliseconds spent per path. */\n getTimeSpent(): Record<string, number> {\n return { ...this.session.timeSpent };\n }\n\n /** Heatmap points for a specific path. */\n getHeatmapData(path: string): HeatmapPoint[];\n /** Heatmap points for all tracked paths. */\n getHeatmapData(): Record<string, HeatmapPoint[]>;\n getHeatmapData(\n path?: string,\n ): HeatmapPoint[] | Record<string, HeatmapPoint[]> {\n if (path !== undefined) {\n return [...(this.session.heatmap[path] ?? [])];\n }\n return Object.entries(this.session.heatmap).reduce<\n Record<string, HeatmapPoint[]>\n >((acc, [k, v]) => {\n acc[k] = [...v];\n return acc;\n }, {});\n }\n\n // ─── Network ────────────────────────────────────────────────────────────────\n\n /** Drain the queue and POST all pending events to the batch endpoint. */\n private async flush(): Promise<void> {\n if (this.queue.length === 0) return;\n // Splice atomically so new events emitted during the async request don't\n // get lost — they stay in the queue for the next flush.\n const batch = this.queue.splice(0);\n await this.sendBatch(batch);\n }\n\n /**\n * Synchronous best-effort flush via `navigator.sendBeacon`.\n * Used on `pagehide` / `visibilitychange:hidden` where async fetch may be\n * cancelled by the browser before it completes.\n */\n private flushBeacon(): void {\n if (this.queue.length === 0) return;\n const batch = this.queue.splice(0);\n const url = `${this.cfg.endpoint!}/batch`;\n const blob = new Blob([this.buildBatchBody(batch)], {\n type: \"application/json\",\n });\n if (typeof navigator !== \"undefined\" && navigator.sendBeacon) {\n navigator.sendBeacon(url, blob);\n } else {\n // Fallback: fire-and-forget fetch (best effort on platforms without sendBeacon)\n void this.sendBatch(batch);\n }\n }\n\n private buildBatchBody(events: TrackerEvent[]): string {\n return JSON.stringify({\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n visitorId: this.session.visitorId,\n location: this.location ?? undefined,\n events: events.map((e) => ({\n sessionId: this.session.id,\n type: e.type,\n data: e.data,\n })),\n });\n }\n\n private async sendBatch(events: TrackerEvent[]): Promise<void> {\n const url = `${this.cfg.endpoint!}/batch`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n try {\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body: this.buildBatchBody(events),\n keepalive: true,\n });\n } catch {\n // Intentionally silent — analytics must never surface errors to users.\n }\n }\n}\n","import { createContext, useContext } from \"react\";\nimport type { UserTracker } from \"../tracker\";\n\nexport const TrackerContext = createContext<UserTracker | null>(null);\n\n/**\n * Returns the nearest `UserTracker` instance from context.\n * Returns `null` when called outside of a `<UserTrackerProvider>`.\n */\nexport function useTrackerContext(): UserTracker | null {\n return useContext(TrackerContext);\n}\n","import { useEffect, useRef, useState } from \"react\";\nimport type { HeatmapPoint, PageView, TrackerEvent } from \"../types\";\nimport { useTrackerContext } from \"./context\";\n\n// ─── useTracker ───────────────────────────────────────────────────────────────\n\n/**\n * Returns the `UserTracker` instance from the nearest `<UserTrackerProvider>`.\n * Returns `null` when called outside of a provider.\n */\nexport function useTracker() {\n return useTrackerContext();\n}\n\n// ─── usePageView ──────────────────────────────────────────────────────────────\n\n/**\n * Manually records a page view whenever `path` changes.\n *\n * Pass the current pathname — particularly useful with Next.js App Router:\n * ```tsx\n * 'use client';\n * import { usePathname } from 'next/navigation';\n * import { usePageView } from 'user-tracker/react';\n *\n * export function NavigationTracker() {\n * usePageView(usePathname());\n * return null;\n * }\n * ```\n * When no `path` is provided the hook is a no-op (automatic tracking via the\n * NavigationPlugin handles it).\n */\nexport function usePageView(path?: string): void {\n const tracker = useTrackerContext();\n useEffect(() => {\n if (tracker && path !== undefined) {\n tracker.trackPageView(path);\n }\n // Re-fire when the path changes.\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [path]);\n}\n\n// ─── useHeatmapData ───────────────────────────────────────────────────────────\n\n/**\n * Returns a live array of `HeatmapPoint` objects for the given path (defaults\n * to `window.location.pathname`).\n *\n * The state is updated in batches — at most once every `refreshMs` ms — to\n * avoid a re-render on every single mouse move.\n *\n * @param path The page path to query. Defaults to the current pathname.\n * @param refreshMs Minimum interval between state updates. Default: 500.\n */\nexport function useHeatmapData(path?: string, refreshMs = 500): HeatmapPoint[] {\n const tracker = useTrackerContext();\n const [data, setData] = useState<HeatmapPoint[]>([]);\n const pendingRef = useRef(false);\n\n useEffect(() => {\n if (!tracker) return;\n\n const targetPath =\n path ?? (typeof window !== \"undefined\" ? window.location.pathname : \"/\");\n\n const refresh = (): void => {\n setData(tracker.getHeatmapData(targetPath) as HeatmapPoint[]);\n pendingRef.current = false;\n };\n\n // Initial read.\n refresh();\n\n // Re-read after each new heatmap point, debounced by refreshMs.\n const unsub = tracker.subscribe((event: TrackerEvent) => {\n if (event.type === \"heatmap\" && event.data.path === targetPath) {\n if (!pendingRef.current) {\n pendingRef.current = true;\n setTimeout(refresh, refreshMs);\n }\n }\n });\n\n return unsub;\n // refreshMs is intentionally excluded — changing it after mount has no effect.\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [tracker, path]);\n\n return data;\n}\n\n// ─── usePageViews ─────────────────────────────────────────────────────────────\n\n/**\n * Returns a live array of all page views recorded in the current session.\n */\nexport function usePageViews(): PageView[] {\n const tracker = useTrackerContext();\n const [views, setViews] = useState<PageView[]>([]);\n\n useEffect(() => {\n if (!tracker) return;\n\n setViews(tracker.getPageViews());\n\n const unsub = tracker.subscribe((event: TrackerEvent) => {\n if (event.type === \"pageview\") {\n setViews(tracker.getPageViews());\n }\n });\n\n return unsub;\n }, [tracker]);\n\n return views;\n}\n\n// ─── useTimeSpent ─────────────────────────────────────────────────────────────\n\n/**\n * Returns a live record of cumulative milliseconds spent per path.\n */\nexport function useTimeSpent(): Record<string, number> {\n const tracker = useTrackerContext();\n const [time, setTime] = useState<Record<string, number>>({});\n\n useEffect(() => {\n if (!tracker) return;\n\n setTime(tracker.getTimeSpent());\n\n const unsub = tracker.subscribe((event: TrackerEvent) => {\n if (event.type === \"timespent\") {\n setTime(tracker.getTimeSpent());\n }\n });\n\n return unsub;\n }, [tracker]);\n\n return time;\n}\n"],"mappings":"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","names":["useEffect","useRef","VISITOR_STORAGE_KEY","generateSessionId","c","r","getOrCreateVisitorId","generateSessionId","existing","VISITOR_STORAGE_KEY","id","e","fetchLocation","res","d","e","_NavigationPlugin","emit","sessionId","origPush","state","title","url","origReplace","newPath","timeOnPage","path","NavigationPlugin","TimePlugin","emit","sessionId","e","duration","throttle","fn","delay","lastCall","args","now","_HeatmapPlugin","emit","sessionId","sampleRate","maxPoints","allowedPaths","e","pageWidth","pageHeight","absY","target","touch","vw","scrollX","scrollY","absX","throttle","_a","point","_b","_c","_d","el","x","y","path","now","c","nearby","HeatmapPlugin","LogCapture","options","u","parts","args","first","stack","msg","src","line","col","err","e","reason","message","level","extra","a","entry","url","body","DEFAULT_ENDPOINT","DEFAULTS","UserTracker","config","_e","_a","e","generateSessionId","getOrCreateVisitorId","emit","sessionId","NavigationPlugin","TimePlugin","HeatmapPlugin","fetchLocation","loc","LogCapture","_b","_c","_d","url","authHeaders","body","event","prev","key","pts","fn","path","resolvedPath","acc","k","v","batch","blob","events","createContext","useContext","TrackerContext","useTrackerContext","jsx","UserTrackerProvider","config","children","trackerRef","useRef","UserTracker","useEffect","tracker","TrackerContext","useEffect","useRef","useState","useTracker","useTrackerContext","usePageView","path","tracker","useEffect","useHeatmapData","refreshMs","data","setData","useState","pendingRef","useRef","targetPath","refresh","event","usePageViews","views","setViews","useTimeSpent","time","setTime"]}
|