alphana-sdk 0.2.2 → 0.3.2

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/index.d.mts CHANGED
@@ -97,6 +97,27 @@ interface HeatmapPoint {
97
97
  type: "move" | "click" | "scroll";
98
98
  path: string;
99
99
  timestamp: number;
100
+ /** Text/label of the clicked element — only present on click events. */
101
+ target?: string;
102
+ }
103
+ interface RageClick {
104
+ path: string;
105
+ x: number;
106
+ y: number;
107
+ /** How many rapid clicks were detected in the burst. */
108
+ count: number;
109
+ timestamp: number;
110
+ sessionId: string;
111
+ }
112
+ interface UTurn {
113
+ /** Page the user left quickly. */
114
+ fromPath: string;
115
+ /** Page the user navigated to. */
116
+ toPath: string;
117
+ /** Milliseconds spent on fromPath before leaving. */
118
+ timeOnPageMs: number;
119
+ timestamp: number;
120
+ sessionId: string;
100
121
  }
101
122
  type TrackerEvent = {
102
123
  type: "pageview";
@@ -107,9 +128,17 @@ type TrackerEvent = {
107
128
  } | {
108
129
  type: "heatmap";
109
130
  data: HeatmapPoint;
131
+ } | {
132
+ type: "rageclik";
133
+ data: RageClick;
134
+ } | {
135
+ type: "uturn";
136
+ data: UTurn;
110
137
  };
111
138
  interface SessionData {
112
139
  id: string;
140
+ /** Persistent visitor ID stored in localStorage — survives across sessions. */
141
+ visitorId: string;
113
142
  startedAt: number;
114
143
  pageViews: PageView[];
115
144
  /** Cumulative milliseconds per path */
package/dist/index.d.ts CHANGED
@@ -97,6 +97,27 @@ interface HeatmapPoint {
97
97
  type: "move" | "click" | "scroll";
98
98
  path: string;
99
99
  timestamp: number;
100
+ /** Text/label of the clicked element — only present on click events. */
101
+ target?: string;
102
+ }
103
+ interface RageClick {
104
+ path: string;
105
+ x: number;
106
+ y: number;
107
+ /** How many rapid clicks were detected in the burst. */
108
+ count: number;
109
+ timestamp: number;
110
+ sessionId: string;
111
+ }
112
+ interface UTurn {
113
+ /** Page the user left quickly. */
114
+ fromPath: string;
115
+ /** Page the user navigated to. */
116
+ toPath: string;
117
+ /** Milliseconds spent on fromPath before leaving. */
118
+ timeOnPageMs: number;
119
+ timestamp: number;
120
+ sessionId: string;
100
121
  }
101
122
  type TrackerEvent = {
102
123
  type: "pageview";
@@ -107,9 +128,17 @@ type TrackerEvent = {
107
128
  } | {
108
129
  type: "heatmap";
109
130
  data: HeatmapPoint;
131
+ } | {
132
+ type: "rageclik";
133
+ data: RageClick;
134
+ } | {
135
+ type: "uturn";
136
+ data: UTurn;
110
137
  };
111
138
  interface SessionData {
112
139
  id: string;
140
+ /** Persistent visitor ID stored in localStorage — survives across sessions. */
141
+ visitorId: string;
113
142
  startedAt: number;
114
143
  pageViews: PageView[];
115
144
  /** Cumulative milliseconds per path */
package/dist/index.js CHANGED
@@ -1,3 +1,3 @@
1
- 'use strict';var B=require('html2canvas');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var B__default=/*#__PURE__*/_interopDefault(B);function R(){return typeof crypto!="undefined"&&typeof crypto.randomUUID=="function"?crypto.randomUUID():"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g,o=>{let t=Math.random()*16|0;return (o==="x"?t:t&3|8).toString(16)})}async function L(){try{let o=await fetch("https://ipapi.co/json/",{method:"GET",headers:{Accept:"application/json"}});if(!o.ok)return null;let t=await o.json();return t.error?null:{country:typeof t.country_code=="string"?t.country_code:"",countryName:typeof t.country_name=="string"?t.country_name:"",city:typeof t.city=="string"?t.city:void 0,region:typeof t.region=="string"?t.region:void 0,latitude:typeof t.latitude=="number"?t.latitude:void 0,longitude:typeof t.longitude=="number"?t.longitude:void 0}}catch(o){return null}}var w=class{constructor({emit:t,sessionId:e}){this.previousPath="";this.originalPushState=null;this.originalReplaceState=null;this.handlePopState=()=>{this.handleNavigation();};this.emit=t,this.sessionId=e;}init(){this.recordPageView(window.location.pathname+window.location.search),window.addEventListener("popstate",this.handlePopState),this.originalPushState=history.pushState.bind(history);let t=this.originalPushState;history.pushState=(i,n,s)=>{t(i,n,s),this.handleNavigation();},this.originalReplaceState=history.replaceState.bind(history);let e=this.originalReplaceState;history.replaceState=(i,n,s)=>{e(i,n,s),this.handleNavigation();};}destroy(){window.removeEventListener("popstate",this.handlePopState),this.originalPushState&&(history.pushState=this.originalPushState),this.originalReplaceState&&(history.replaceState=this.originalReplaceState);}handleNavigation(){let t=window.location.pathname+window.location.search;t!==this.previousPath&&this.recordPageView(t);}recordPageView(t){this.previousPath=t,this.emit({type:"pageview",data:{path:t,title:document.title,timestamp:Date.now(),sessionId:this.sessionId,referrer:document.referrer||void 0}}),window.dispatchEvent(new CustomEvent("tracker:navigate",{detail:{path:t,title:document.title}}));}};var P=class{constructor({emit:t,sessionId:e}){this.currentPath="";this.startTime=0;this.tracking=false;this.handleNavigate=t=>{this.stopTracking(),this.currentPath=t.detail.path,this.startTracking();};this.handleVisibilityChange=()=>{document.hidden?this.stopTracking():this.startTracking();};this.handleUnload=()=>{this.stopTracking();};this.emit=t,this.sessionId=e;}init(){this.currentPath=window.location.pathname+window.location.search,this.startTracking(),window.addEventListener("tracker:navigate",this.handleNavigate),document.addEventListener("visibilitychange",this.handleVisibilityChange),window.addEventListener("beforeunload",this.handleUnload),window.addEventListener("pagehide",this.handleUnload);}destroy(){this.stopTracking(),window.removeEventListener("tracker:navigate",this.handleNavigate),document.removeEventListener("visibilitychange",this.handleVisibilityChange),window.removeEventListener("beforeunload",this.handleUnload),window.removeEventListener("pagehide",this.handleUnload);}startTracking(){this.startTime=Date.now(),this.tracking=true;}stopTracking(){if(!this.tracking||!this.currentPath)return;let t=Date.now()-this.startTime;if(t<100){this.tracking=false;return}this.emit({type:"timespent",data:{path:this.currentPath,duration:t,sessionId:this.sessionId,timestamp:Date.now()}}),this.tracking=false;}};function S(o,t){let e=0;return (...i)=>{let n=Date.now();n-e>=t&&(e=n,o(...i));}}var E=class{constructor({emit:t,sessionId:e,sampleRate:i=.3,maxPoints:n=2e3}){this.currentPath="";this.pointCounts={};this.handleMouseMove=t=>{if(Math.random()>this.sampleRate)return;let e=document.documentElement.scrollWidth,i=document.documentElement.scrollHeight,n=t.clientY+window.scrollY;this.recordPoint({x:t.clientX,y:n,xPct:e>0?t.clientX/e*100:0,yPct:i>0?n/i*100:0,type:"move"});};this.handleClick=t=>{let e=document.documentElement.scrollWidth,i=document.documentElement.scrollHeight,n=t.clientY+window.scrollY;this.recordPoint({x:t.clientX,y:n,xPct:e>0?t.clientX/e*100:0,yPct:i>0?n/i*100:0,type:"click"});};this.handleScroll=()=>{if(Math.random()>this.sampleRate)return;let t=document.documentElement.scrollWidth,e=document.documentElement.scrollHeight,i=window.innerWidth,n=window.scrollX,s=window.scrollY,r=n+i/2;this.recordPoint({x:i/2,y:s,xPct:t>0?r/t*100:50,yPct:e>0?s/e*100:0,type:"scroll"});};this.handleNavigate=t=>{this.currentPath=t.detail.path;};this.emit=t,this.sessionId=e,this.sampleRate=i,this.maxPoints=n,this.throttledMouseMove=S(this.handleMouseMove,50),this.throttledScroll=S(this.handleScroll,100);}init(){this.currentPath=window.location.pathname+window.location.search,document.addEventListener("mousemove",this.throttledMouseMove),document.addEventListener("click",this.handleClick),window.addEventListener("scroll",this.throttledScroll,{passive:true}),window.addEventListener("tracker:navigate",this.handleNavigate);}destroy(){document.removeEventListener("mousemove",this.throttledMouseMove),document.removeEventListener("click",this.handleClick),window.removeEventListener("scroll",this.throttledScroll),window.removeEventListener("tracker:navigate",this.handleNavigate);}canRecord(){var t;return ((t=this.pointCounts[this.currentPath])!=null?t:0)<this.maxPoints}recordPoint(t){var e;this.canRecord()&&(this.pointCounts[this.currentPath]=((e=this.pointCounts[this.currentPath])!=null?e:0)+1,this.emit({type:"heatmap",data:{...t,path:this.currentPath,timestamp:Date.now()}}));}};var v=class{constructor(t){this.prevOnError=null;this.prevOnUnhandledRejection=null;this.initialized=false;try{let e=new URL(t.endpoint),i=e.pathname.replace(/\/$/,"").split("/");i.pop(),e.pathname=i.join("/")||"/",this.endpoint=e.toString().replace(/\/$/,"");}catch(e){this.endpoint=t.endpoint;}this.sessionId=t.sessionId,this.appId=t.appId,this.authHeaders=t.secretKey?{Authorization:`Bearer ${t.secretKey}`}:{};}init(){typeof window=="undefined"||this.initialized||(this.origInfo=console.info.bind(console),this.origWarn=console.warn.bind(console),this.origError=console.error.bind(console),console.info=(...t)=>{this.origInfo(...t),this.send("info",this.format(t));},console.warn=(...t)=>{this.origWarn(...t),this.send("warn",this.format(t));},console.error=(...t)=>{this.origError(...t);let[e]=t,i=e instanceof Error?e.stack:void 0;this.send("error",this.format(t),{stack:i});},this.prevOnError=window.onerror,window.onerror=(t,e,i,n,s)=>(this.send("error",String(t),{stack:s==null?void 0:s.stack,meta:{src:e,line:i,col:n}}),typeof this.prevOnError=="function"?this.prevOnError(t,e,i,n,s):false),this.prevOnUnhandledRejection=t=>{let e=t.reason,i=e instanceof Error?e.message:String(e!=null?e:"Unhandled promise rejection");this.send("error",i,{stack:e instanceof Error?e.stack:void 0});},window.addEventListener("unhandledrejection",this.prevOnUnhandledRejection),this.initialized=true);}destroy(){this.initialized&&(console.info=this.origInfo,console.warn=this.origWarn,console.error=this.origError,window.onerror=this.prevOnError,this.prevOnUnhandledRejection&&window.removeEventListener("unhandledrejection",this.prevOnUnhandledRejection),this.initialized=false);}capture(t,e,i){this.send(t,e,i);}format(t){return t.map(e=>{if(e instanceof Error)return e.message;if(typeof e=="object")try{return JSON.stringify(e)}catch(i){return String(e)}return String(e)}).join(" ")}send(t,e,i){let n={sessionId:this.sessionId,...this.appId?{appId:this.appId}:{},level:t,message:e,url:typeof window!="undefined"?window.location.href:void 0,stack:i==null?void 0:i.stack,meta:i==null?void 0:i.meta,timestamp:Date.now()},s=`${this.endpoint}/logs/ingest`,r=JSON.stringify(n);fetch(s,{method:"POST",headers:{"Content-Type":"application/json",...this.authHeaders},body:r,keepalive:true}).catch(h=>{this.origError&&this.origError("[user-tracker] Failed to send log:",h);});}};var U=300*1e3,C="__ut_snap_ts__",b=class{constructor(t){this.lastSentPerPath={};var e;this.snapshotUrl=t.endpoint.replace(/\/events$/,"/snapshots"),this.appId=t.appId,this.secretKey=t.secretKey,this.intervalMs=(e=t.intervalMs)!=null?e:U;try{let i=localStorage.getItem(C);i&&(this.lastSentPerPath=JSON.parse(i));}catch(i){}}async capture(t){var i;if(typeof window=="undefined")return;let e=(i=this.lastSentPerPath[t])!=null?i:0;if(!(Date.now()-e<this.intervalMs))try{let n=document.documentElement,s=getComputedStyle(n).backgroundColor||getComputedStyle(document.body).backgroundColor||"#ffffff",r=await B__default.default(n,{allowTaint:!0,useCORS:!0,logging:!1,scale:window.devicePixelRatio||1,width:n.scrollWidth,height:n.scrollHeight,windowWidth:window.innerWidth,windowHeight:window.innerHeight,scrollX:0,scrollY:0,x:0,y:0,backgroundColor:s,foreignObjectRendering:!1,removeContainer:!0}),h=await new Promise(d=>r.toBlob(d,"image/png"));if(!h)return;let a=new FormData;a.append("screenshot",h,"screenshot.png"),a.append("path",t),a.append("width",String(n.scrollWidth)),a.append("height",String(n.scrollHeight)),this.appId&&a.append("appId",this.appId);let p={};this.secretKey&&(p.Authorization=`Bearer ${this.secretKey}`),await fetch(this.snapshotUrl,{method:"POST",headers:p,body:a}),this.lastSentPerPath[t]=Date.now();try{localStorage.setItem(C,JSON.stringify(this.lastSentPerPath));}catch(d){}}catch(n){}}destroy(){}};var H="https://api.alphana.ir/api/events",_={endpoint:H,trackNavigation:true,trackTime:true,trackHeatmap:true,trackLogs:true,trackSnapshots:true,mouseSampleRate:.3,maxHeatmapPoints:2e3,batchSize:20,flushInterval:5e3},k=class{constructor(t={}){this.initialized=false;this.subscribers=new Set;this.queue=[];this.flushTimer=null;this.heartbeatTimer=null;this.location=null;this.handleVisibilityChange=()=>{document.visibilityState==="hidden"&&(this.queue.length>0&&this.flushBeacon(),this.sendDeactivate());};this.handlePageHide=()=>{this.queue.length>0&&this.flushBeacon(),this.sendDeactivate();};this.handleNavigate=t=>{var i;let e=(i=t.detail)==null?void 0:i.path;e&&this.snapshot&&this.snapshot.capture(e);};var e;this.cfg={..._,...t};try{new URL(this.cfg.endpoint);}catch(i){throw new Error(`[alpha-tracker] Invalid endpoint URL: "${this.cfg.endpoint}"`)}this.session={id:(e=t.sessionId)!=null?e:R(),startedAt:Date.now(),pageViews:[],timeSpent:{},heatmap:{}};}init(){if(typeof window=="undefined"||this.initialized)return this;let t=this.emit.bind(this),{id:e}=this.session;return this.cfg.trackNavigation&&(this.navigation=new w({emit:t,sessionId:e}),this.navigation.init()),this.cfg.trackTime&&(this.time=new P({emit:t,sessionId:e}),this.time.init()),this.cfg.trackHeatmap&&(this.heatmap=new E({emit:t,sessionId:e,sampleRate:this.cfg.mouseSampleRate,maxPoints:this.cfg.maxHeatmapPoints}),this.heatmap.init()),this.cfg.endpoint&&(this.flushTimer=setInterval(()=>{this.queue.length>0&&this.flush();},this.cfg.flushInterval),L().then(i=>{this.location=i,i&&(this.session.location=i);}),window.addEventListener("visibilitychange",this.handleVisibilityChange),window.addEventListener("pagehide",this.handlePageHide),this.heartbeatTimer=setInterval(()=>{document.visibilityState!=="hidden"&&this.sendHeartbeat();},3e4),this.cfg.trackLogs&&(this.logCapture=new v({endpoint:this.cfg.endpoint,sessionId:this.session.id,secretKey:this.cfg.secretKey,appId:this.cfg.appId}),this.logCapture.init()),this.cfg.trackSnapshots!==false&&(this.snapshot=new b({endpoint:this.cfg.endpoint,appId:this.cfg.appId,secretKey:this.cfg.secretKey,intervalMs:this.cfg.snapshotIntervalMs}),this.snapshot.capture(window.location.pathname),window.addEventListener("tracker:navigate",this.handleNavigate))),this.initialized=true,this}destroy(){var t,e,i,n,s;this.flushTimer!==null&&(clearInterval(this.flushTimer),this.flushTimer=null),this.heartbeatTimer!==null&&(clearInterval(this.heartbeatTimer),this.heartbeatTimer=null),typeof window!="undefined"&&(window.removeEventListener("visibilitychange",this.handleVisibilityChange),window.removeEventListener("pagehide",this.handlePageHide)),(t=this.navigation)==null||t.destroy(),(e=this.time)==null||e.destroy(),(i=this.heatmap)==null||i.destroy(),(n=this.logCapture)==null||n.destroy(),(s=this.snapshot)==null||s.destroy(),typeof window!="undefined"&&window.removeEventListener("tracker:navigate",this.handleNavigate),this.queue.length>0&&this.cfg.endpoint&&this.flushBeacon(),this.initialized=false;}async sendHeartbeat(){if(!this.cfg.endpoint)return;let t=`${this.cfg.endpoint}/heartbeat`,e=this.cfg.secretKey?{Authorization:`Bearer ${this.cfg.secretKey}`}:{},i=JSON.stringify({sessionId:this.session.id,path:typeof window!="undefined"?window.location.pathname:"/",active:true,...this.cfg.appId?{appId:this.cfg.appId}:{},...this.location?{location:this.location}:{}});try{await fetch(t,{method:"POST",headers:{"Content-Type":"application/json",...e},body:i,keepalive:!0});}catch(n){}}sendDeactivate(){if(!this.cfg.endpoint)return;let t=`${this.cfg.endpoint}/heartbeat`,e=this.cfg.secretKey?{Authorization:`Bearer ${this.cfg.secretKey}`}:{},i=JSON.stringify({sessionId:this.session.id,path:typeof window!="undefined"?window.location.pathname:"/",active:false,...this.cfg.appId?{appId:this.cfg.appId}:{}});typeof navigator!="undefined"&&navigator.sendBeacon?navigator.sendBeacon(t,new Blob([i],{type:"application/json"})):fetch(t,{method:"POST",headers:{"Content-Type":"application/json",...e},body:i,keepalive:true}).catch(()=>{});}emit(t){var e,i,n;switch(t.type){case "pageview":this.session.pageViews.push(t.data);break;case "timespent":{let s=(e=this.session.timeSpent[t.data.path])!=null?e:0;this.session.timeSpent[t.data.path]=s+t.data.duration;break}case "heatmap":{let s=t.data.path;this.session.heatmap[s]||(this.session.heatmap[s]=[]);let r=this.session.heatmap[s];r.length<this.cfg.maxHeatmapPoints&&r.push(t.data);break}}this.subscribers.forEach(s=>s(t)),(n=(i=this.cfg).onEvent)==null||n.call(i,t),this.cfg.endpoint&&(this.queue.push(t),this.queue.length>=this.cfg.batchSize&&this.flush());}subscribe(t){return this.subscribers.add(t),()=>this.subscribers.delete(t)}trackPageView(t){let e=t!=null?t:typeof window!="undefined"?window.location.pathname+window.location.search:"/";this.emit({type:"pageview",data:{path:e,title:typeof document!="undefined"?document.title:"",timestamp:Date.now(),sessionId:this.session.id,referrer:typeof document!="undefined"&&document.referrer||void 0}}),typeof window!="undefined"&&window.dispatchEvent(new CustomEvent("tracker:navigate",{detail:{path:e,title:document.title}}));}getSession(){return this.session}getPageViews(){return [...this.session.pageViews]}getTimeSpent(){return {...this.session.timeSpent}}getHeatmapData(t){var e;return t!==void 0?[...(e=this.session.heatmap[t])!=null?e:[]]:Object.entries(this.session.heatmap).reduce((i,[n,s])=>(i[n]=[...s],i),{})}async flush(){if(this.queue.length===0)return;let t=this.queue.splice(0);await this.sendBatch(t);}flushBeacon(){if(this.queue.length===0)return;let t=this.queue.splice(0),e=`${this.cfg.endpoint}/batch`,i=new Blob([this.buildBatchBody(t)],{type:"application/json"});typeof navigator!="undefined"&&navigator.sendBeacon?navigator.sendBeacon(e,i):this.sendBatch(t);}buildBatchBody(t){var e;return JSON.stringify({...this.cfg.appId?{appId:this.cfg.appId}:{},location:(e=this.location)!=null?e:void 0,events:t.map(i=>({sessionId:this.session.id,type:i.type,data:i.data}))})}async sendBatch(t){let e=`${this.cfg.endpoint}/batch`,i=this.cfg.secretKey?{Authorization:`Bearer ${this.cfg.secretKey}`}:{};try{await fetch(e,{method:"POST",headers:{"Content-Type":"application/json",...i},body:this.buildBatchBody(t),keepalive:!0});}catch(n){}}};function j(){let o=[[0,0,255],[0,255,255],[0,255,0],[255,255,0],[255,128,0],[255,0,0]],t=[],e=51;for(let i=0;i<o.length-1;i++){let[n,s,r]=o[i],[h,a,p]=o[i+1];for(let d=0;d<e;d++){let g=d/e;t.push([Math.round(n+(h-n)*g),Math.round(s+(a-s)*g),Math.round(r+(p-r)*g)]);}}return t}var O=j();function F(o,t,e={}){let{radius:i=25,maxOpacity:n=.85,minOpacity:s=0}=e,r=o.getContext("2d");if(!r)return;let{width:h,height:a}=o;if(r.clearRect(0,0,h,a),t.length===0)return;let p=document.createElement("canvas");p.width=h,p.height=a;let d=p.getContext("2d");if(!d)return;for(let c of t){let l=c.xPct/100*h,u=c.yPct/100*a,y=c.type==="click"?i*1.6:i,m=d.createRadialGradient(l,u,0,l,u,y);m.addColorStop(0,"rgba(0,0,0,0.15)"),m.addColorStop(1,"rgba(0,0,0,0)"),d.fillStyle=m,d.beginPath(),d.arc(l,u,y,0,Math.PI*2),d.fill();}let g=d.getImageData(0,0,h,a),I=r.createImageData(h,a),x=g.data,f=I.data,T=O.length-1;for(let c=0;c<x.length;c+=4){let l=x[c+3];if(l===0)continue;let u=l/255,y=Math.min(Math.floor(u*T),T),[m,M,D]=O[y],N=s+u*(n-s);f[c]=m,f[c+1]=M,f[c+2]=D,f[c+3]=Math.round(N*255);}r.putImageData(I,0,0);}
2
- exports.DEFAULT_ENDPOINT=H;exports.LogCapture=v;exports.UserTracker=k;exports.renderHeatmap=F;//# sourceMappingURL=index.js.map
1
+ 'use strict';var j=require('html2canvas');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var j__default=/*#__PURE__*/_interopDefault(j);var H="__ut_vid__";function f(){return typeof crypto!="undefined"&&typeof crypto.randomUUID=="function"?crypto.randomUUID():"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g,r=>{let t=Math.random()*16|0;return (r==="x"?t:t&3|8).toString(16)})}function O(){if(typeof localStorage=="undefined")return f();try{let r=localStorage.getItem(H);if(r)return r;let t=f();return localStorage.setItem(H,t),t}catch(r){return f()}}async function D(){try{let r=await fetch("https://ipapi.co/json/",{method:"GET",headers:{Accept:"application/json"}});if(!r.ok)return null;let t=await r.json();return t.error?null:{country:typeof t.country_code=="string"?t.country_code:"",countryName:typeof t.country_name=="string"?t.country_name:"",city:typeof t.city=="string"?t.city:void 0,region:typeof t.region=="string"?t.region:void 0,latitude:typeof t.latitude=="number"?t.latitude:void 0,longitude:typeof t.longitude=="number"?t.longitude:void 0}}catch(r){return null}}var P=class P{constructor({emit:t,sessionId:e}){this.previousPath="";this.originalPushState=null;this.originalReplaceState=null;this.pageEntryTime=0;this.handlePopState=()=>{this.handleNavigation();};this.emit=t,this.sessionId=e;}init(){this.recordPageView(window.location.pathname+window.location.search),window.addEventListener("popstate",this.handlePopState),this.originalPushState=history.pushState.bind(history);let t=this.originalPushState;history.pushState=(i,n,s)=>{t(i,n,s),this.handleNavigation();},this.originalReplaceState=history.replaceState.bind(history);let e=this.originalReplaceState;history.replaceState=(i,n,s)=>{e(i,n,s),this.handleNavigation();};}destroy(){window.removeEventListener("popstate",this.handlePopState),this.originalPushState&&(history.pushState=this.originalPushState),this.originalReplaceState&&(history.replaceState=this.originalReplaceState);}handleNavigation(){let t=window.location.pathname+window.location.search;if(t!==this.previousPath){if(this.previousPath&&this.pageEntryTime>0){let e=Date.now()-this.pageEntryTime;e<=P.UTURN_THRESHOLD_MS&&this.emit({type:"uturn",data:{fromPath:this.previousPath,toPath:t,timeOnPageMs:e,timestamp:Date.now(),sessionId:this.sessionId}});}this.recordPageView(t);}}recordPageView(t){this.previousPath=t,this.pageEntryTime=Date.now(),this.emit({type:"pageview",data:{path:t,title:document.title,timestamp:Date.now(),sessionId:this.sessionId,referrer:document.referrer||void 0}}),window.dispatchEvent(new CustomEvent("tracker:navigate",{detail:{path:t,title:document.title}}));}};P.UTURN_THRESHOLD_MS=5e3;var b=P;var k=class{constructor({emit:t,sessionId:e}){this.currentPath="";this.startTime=0;this.tracking=false;this.handleNavigate=t=>{this.stopTracking(),this.currentPath=t.detail.path,this.startTracking();};this.handleVisibilityChange=()=>{document.hidden?this.stopTracking():this.startTracking();};this.handleUnload=()=>{this.stopTracking();};this.emit=t,this.sessionId=e;}init(){this.currentPath=window.location.pathname+window.location.search,this.startTracking(),window.addEventListener("tracker:navigate",this.handleNavigate),document.addEventListener("visibilitychange",this.handleVisibilityChange),window.addEventListener("beforeunload",this.handleUnload),window.addEventListener("pagehide",this.handleUnload);}destroy(){this.stopTracking(),window.removeEventListener("tracker:navigate",this.handleNavigate),document.removeEventListener("visibilitychange",this.handleVisibilityChange),window.removeEventListener("beforeunload",this.handleUnload),window.removeEventListener("pagehide",this.handleUnload);}startTracking(){this.startTime=Date.now(),this.tracking=true;}stopTracking(){if(!this.tracking||!this.currentPath)return;let t=Date.now()-this.startTime;if(t<100){this.tracking=false;return}this.emit({type:"timespent",data:{path:this.currentPath,duration:t,sessionId:this.sessionId,timestamp:Date.now()}}),this.tracking=false;}};function T(r,t){let e=0;return (...i)=>{let n=Date.now();n-e>=t&&(e=n,r(...i));}}var p=class p{constructor({emit:t,sessionId:e,sampleRate:i=.3,maxPoints:n=2e3}){this.currentPath="";this.pointCounts={};this.recentClicks=[];this.handleMouseMove=t=>{if(Math.random()>this.sampleRate)return;let e=document.documentElement.scrollWidth,i=document.documentElement.scrollHeight,n=t.clientY+window.scrollY;this.recordPoint({x:t.clientX,y:n,xPct:e>0?t.clientX/e*100:0,yPct:i>0?n/i*100:0,type:"move"});};this.handleClick=t=>{let e=document.documentElement.scrollWidth,i=document.documentElement.scrollHeight,n=t.clientY+window.scrollY,s=this.getClickTarget(t);this.recordPoint({x:t.clientX,y:n,xPct:e>0?t.clientX/e*100:0,yPct:i>0?n/i*100:0,type:"click",...s?{target:s}:{}}),this.checkRageClick(t.clientX,n,this.currentPath);};this.handleScroll=()=>{if(Math.random()>this.sampleRate)return;let t=document.documentElement.scrollWidth,e=document.documentElement.scrollHeight,i=window.innerWidth,n=window.scrollX,s=window.scrollY,o=n+i/2;this.recordPoint({x:i/2,y:s,xPct:t>0?o/t*100:50,yPct:e>0?s/e*100:0,type:"scroll"});};this.handleNavigate=t=>{this.currentPath=t.detail.path;};this.emit=t,this.sessionId=e,this.sampleRate=i,this.maxPoints=n,this.throttledMouseMove=T(this.handleMouseMove,50),this.throttledScroll=T(this.handleScroll,100);}init(){this.currentPath=window.location.pathname+window.location.search,document.addEventListener("mousemove",this.throttledMouseMove),document.addEventListener("click",this.handleClick),window.addEventListener("scroll",this.throttledScroll,{passive:true}),window.addEventListener("tracker:navigate",this.handleNavigate);}destroy(){document.removeEventListener("mousemove",this.throttledMouseMove),document.removeEventListener("click",this.handleClick),window.removeEventListener("scroll",this.throttledScroll),window.removeEventListener("tracker:navigate",this.handleNavigate);}canRecord(){var t;return ((t=this.pointCounts[this.currentPath])!=null?t:0)<this.maxPoints}recordPoint(t){var e;this.canRecord()&&(this.pointCounts[this.currentPath]=((e=this.pointCounts[this.currentPath])!=null?e:0)+1,this.emit({type:"heatmap",data:{...t,path:this.currentPath,timestamp:Date.now()}}));}getClickTarget(t){var n,s,o,a;let e=t.target;return e&&(e.getAttribute("aria-label")||((n=e.closest("[aria-label]"))==null?void 0:n.getAttribute("aria-label"))||e.getAttribute("data-track-label")||e.getAttribute("id")||(e instanceof HTMLButtonElement||e instanceof HTMLAnchorElement?(s=e.innerText)==null?void 0:s.trim().slice(0,60):(a=(o=e.closest("button, a"))==null?void 0:o.textContent)==null?void 0:a.trim().slice(0,60)))||void 0}checkRageClick(t,e,i){let n=Date.now();this.recentClicks=this.recentClicks.filter(o=>n-o.t<p.RAGE_WINDOW_MS),this.recentClicks.push({x:t,y:e,t:n});let s=this.recentClicks.filter(o=>Math.hypot(o.x-t,o.y-e)<=p.RAGE_RADIUS_PX);s.length>=p.RAGE_THRESHOLD&&(this.emit({type:"rageclik",data:{path:i,x:t,y:e,count:s.length,timestamp:n,sessionId:this.sessionId}}),this.recentClicks=[]);}};p.RAGE_THRESHOLD=3,p.RAGE_WINDOW_MS=1e3,p.RAGE_RADIUS_PX=50;var S=p;var y=class{constructor(t){this.prevOnError=null;this.prevOnUnhandledRejection=null;this.initialized=false;try{let e=new URL(t.endpoint),i=e.pathname.replace(/\/$/,"").split("/");i.pop(),e.pathname=i.join("/")||"/",this.endpoint=e.toString().replace(/\/$/,"");}catch(e){this.endpoint=t.endpoint;}this.sessionId=t.sessionId,this.appId=t.appId,this.authHeaders=t.secretKey?{Authorization:`Bearer ${t.secretKey}`}:{};}init(){typeof window=="undefined"||this.initialized||(this.origInfo=console.info.bind(console),this.origWarn=console.warn.bind(console),this.origError=console.error.bind(console),console.info=(...t)=>{this.origInfo(...t),this.send("info",this.format(t));},console.warn=(...t)=>{this.origWarn(...t),this.send("warn",this.format(t));},console.error=(...t)=>{this.origError(...t);let[e]=t,i=e instanceof Error?e.stack:void 0;this.send("error",this.format(t),{stack:i});},this.prevOnError=window.onerror,window.onerror=(t,e,i,n,s)=>(this.send("error",String(t),{stack:s==null?void 0:s.stack,meta:{src:e,line:i,col:n}}),typeof this.prevOnError=="function"?this.prevOnError(t,e,i,n,s):false),this.prevOnUnhandledRejection=t=>{let e=t.reason,i=e instanceof Error?e.message:String(e!=null?e:"Unhandled promise rejection");this.send("error",i,{stack:e instanceof Error?e.stack:void 0});},window.addEventListener("unhandledrejection",this.prevOnUnhandledRejection),this.initialized=true);}destroy(){this.initialized&&(console.info=this.origInfo,console.warn=this.origWarn,console.error=this.origError,window.onerror=this.prevOnError,this.prevOnUnhandledRejection&&window.removeEventListener("unhandledrejection",this.prevOnUnhandledRejection),this.initialized=false);}capture(t,e,i){this.send(t,e,i);}format(t){return t.map(e=>{if(e instanceof Error)return e.message;if(typeof e=="object")try{return JSON.stringify(e)}catch(i){return String(e)}return String(e)}).join(" ")}send(t,e,i){let n={sessionId:this.sessionId,...this.appId?{appId:this.appId}:{},level:t,message:e,url:typeof window!="undefined"?window.location.href:void 0,stack:i==null?void 0:i.stack,meta:i==null?void 0:i.meta,timestamp:Date.now()},s=`${this.endpoint}/logs/ingest`,o=JSON.stringify(n);fetch(s,{method:"POST",headers:{"Content-Type":"application/json",...this.authHeaders},body:o,keepalive:true}).catch(a=>{this.origError&&this.origError("[user-tracker] Failed to send log:",a);});}};var F=300*1e3,M="__ut_snap_ts__",I=class{constructor(t){this.lastSentPerPath={};var e;this.snapshotUrl=t.endpoint.replace(/\/events$/,"/snapshots"),this.appId=t.appId,this.secretKey=t.secretKey,this.intervalMs=(e=t.intervalMs)!=null?e:F;try{let i=localStorage.getItem(M);i&&(this.lastSentPerPath=JSON.parse(i));}catch(i){}}async capture(t){var i;if(typeof window=="undefined")return;let e=(i=this.lastSentPerPath[t])!=null?i:0;if(!(Date.now()-e<this.intervalMs))try{let n=document.documentElement,s=getComputedStyle(n).backgroundColor||getComputedStyle(document.body).backgroundColor||"#ffffff",o=await j__default.default(n,{allowTaint:!0,useCORS:!0,logging:!1,scale:window.devicePixelRatio||1,width:n.scrollWidth,height:n.scrollHeight,windowWidth:window.innerWidth,windowHeight:window.innerHeight,scrollX:0,scrollY:0,x:0,y:0,backgroundColor:s,foreignObjectRendering:!1,removeContainer:!0}),a=await new Promise(c=>o.toBlob(c,"image/png"));if(!a)return;let d=new FormData;d.append("screenshot",a,"screenshot.png"),d.append("path",t),d.append("width",String(n.scrollWidth)),d.append("height",String(n.scrollHeight)),this.appId&&d.append("appId",this.appId);let l={};this.secretKey&&(l.Authorization=`Bearer ${this.secretKey}`),await fetch(this.snapshotUrl,{method:"POST",headers:l,body:d}),this.lastSentPerPath[t]=Date.now();try{localStorage.setItem(M,JSON.stringify(this.lastSentPerPath));}catch(c){}}catch(n){}}destroy(){}};var _="https://api.alphana.ir/api/events",K={endpoint:_,trackNavigation:true,trackTime:true,trackHeatmap:true,trackLogs:true,trackSnapshots:true,mouseSampleRate:.3,maxHeatmapPoints:2e3,batchSize:20,flushInterval:5e3},R=class{constructor(t={}){this.initialized=false;this.subscribers=new Set;this.queue=[];this.flushTimer=null;this.heartbeatTimer=null;this.location=null;this.handleVisibilityChange=()=>{document.visibilityState==="hidden"&&(this.queue.length>0&&this.flushBeacon(),this.sendDeactivate());};this.handlePageHide=()=>{this.queue.length>0&&this.flushBeacon(),this.sendDeactivate();};this.handleNavigate=t=>{var i;let e=(i=t.detail)==null?void 0:i.path;e&&this.snapshot&&this.snapshot.capture(e);};var e;this.cfg={...K,...t};try{new URL(this.cfg.endpoint);}catch(i){throw new Error(`[alpha-tracker] Invalid endpoint URL: "${this.cfg.endpoint}"`)}this.session={id:(e=t.sessionId)!=null?e:f(),visitorId:O(),startedAt:Date.now(),pageViews:[],timeSpent:{},heatmap:{}};}init(){if(typeof window=="undefined"||this.initialized)return this;let t=this.emit.bind(this),{id:e}=this.session;return this.cfg.trackNavigation&&(this.navigation=new b({emit:t,sessionId:e}),this.navigation.init()),this.cfg.trackTime&&(this.time=new k({emit:t,sessionId:e}),this.time.init()),this.cfg.trackHeatmap&&(this.heatmap=new S({emit:t,sessionId:e,sampleRate:this.cfg.mouseSampleRate,maxPoints:this.cfg.maxHeatmapPoints}),this.heatmap.init()),this.cfg.endpoint&&(this.flushTimer=setInterval(()=>{this.queue.length>0&&this.flush();},this.cfg.flushInterval),D().then(i=>{this.location=i,i&&(this.session.location=i);}),window.addEventListener("visibilitychange",this.handleVisibilityChange),window.addEventListener("pagehide",this.handlePageHide),this.heartbeatTimer=setInterval(()=>{document.visibilityState!=="hidden"&&this.sendHeartbeat();},3e4),this.cfg.trackLogs&&(this.logCapture=new y({endpoint:this.cfg.endpoint,sessionId:this.session.id,secretKey:this.cfg.secretKey,appId:this.cfg.appId}),this.logCapture.init()),this.cfg.trackSnapshots!==false&&(this.snapshot=new I({endpoint:this.cfg.endpoint,appId:this.cfg.appId,secretKey:this.cfg.secretKey,intervalMs:this.cfg.snapshotIntervalMs}),this.snapshot.capture(window.location.pathname),window.addEventListener("tracker:navigate",this.handleNavigate))),this.initialized=true,this}destroy(){var t,e,i,n,s;this.flushTimer!==null&&(clearInterval(this.flushTimer),this.flushTimer=null),this.heartbeatTimer!==null&&(clearInterval(this.heartbeatTimer),this.heartbeatTimer=null),typeof window!="undefined"&&(window.removeEventListener("visibilitychange",this.handleVisibilityChange),window.removeEventListener("pagehide",this.handlePageHide)),(t=this.navigation)==null||t.destroy(),(e=this.time)==null||e.destroy(),(i=this.heatmap)==null||i.destroy(),(n=this.logCapture)==null||n.destroy(),(s=this.snapshot)==null||s.destroy(),typeof window!="undefined"&&window.removeEventListener("tracker:navigate",this.handleNavigate),this.queue.length>0&&this.cfg.endpoint&&this.flushBeacon(),this.initialized=false;}async sendHeartbeat(){if(!this.cfg.endpoint)return;let t=`${this.cfg.endpoint}/heartbeat`,e=this.cfg.secretKey?{Authorization:`Bearer ${this.cfg.secretKey}`}:{},i=JSON.stringify({sessionId:this.session.id,visitorId:this.session.visitorId,path:typeof window!="undefined"?window.location.pathname:"/",active:true,...this.cfg.appId?{appId:this.cfg.appId}:{},...this.location?{location:this.location}:{}});try{await fetch(t,{method:"POST",headers:{"Content-Type":"application/json",...e},body:i,keepalive:!0});}catch(n){}}sendDeactivate(){if(!this.cfg.endpoint)return;let t=`${this.cfg.endpoint}/heartbeat`,e=this.cfg.secretKey?{Authorization:`Bearer ${this.cfg.secretKey}`}:{},i=JSON.stringify({sessionId:this.session.id,path:typeof window!="undefined"?window.location.pathname:"/",active:false,...this.cfg.appId?{appId:this.cfg.appId}:{}});typeof navigator!="undefined"&&navigator.sendBeacon?navigator.sendBeacon(t,new Blob([i],{type:"application/json"})):fetch(t,{method:"POST",headers:{"Content-Type":"application/json",...e},body:i,keepalive:true}).catch(()=>{});}emit(t){var e,i,n;switch(t.type){case "pageview":this.session.pageViews.push(t.data);break;case "timespent":{let s=(e=this.session.timeSpent[t.data.path])!=null?e:0;this.session.timeSpent[t.data.path]=s+t.data.duration;break}case "heatmap":{let s=t.data.path;this.session.heatmap[s]||(this.session.heatmap[s]=[]);let o=this.session.heatmap[s];o.length<this.cfg.maxHeatmapPoints&&o.push(t.data);break}}this.subscribers.forEach(s=>s(t)),(n=(i=this.cfg).onEvent)==null||n.call(i,t),this.cfg.endpoint&&(this.queue.push(t),this.queue.length>=this.cfg.batchSize&&this.flush());}subscribe(t){return this.subscribers.add(t),()=>this.subscribers.delete(t)}trackPageView(t){let e=t!=null?t:typeof window!="undefined"?window.location.pathname+window.location.search:"/";this.emit({type:"pageview",data:{path:e,title:typeof document!="undefined"?document.title:"",timestamp:Date.now(),sessionId:this.session.id,referrer:typeof document!="undefined"&&document.referrer||void 0}}),typeof window!="undefined"&&window.dispatchEvent(new CustomEvent("tracker:navigate",{detail:{path:e,title:document.title}}));}getSession(){return this.session}getPageViews(){return [...this.session.pageViews]}getTimeSpent(){return {...this.session.timeSpent}}getHeatmapData(t){var e;return t!==void 0?[...(e=this.session.heatmap[t])!=null?e:[]]:Object.entries(this.session.heatmap).reduce((i,[n,s])=>(i[n]=[...s],i),{})}async flush(){if(this.queue.length===0)return;let t=this.queue.splice(0);await this.sendBatch(t);}flushBeacon(){if(this.queue.length===0)return;let t=this.queue.splice(0),e=`${this.cfg.endpoint}/batch`,i=new Blob([this.buildBatchBody(t)],{type:"application/json"});typeof navigator!="undefined"&&navigator.sendBeacon?navigator.sendBeacon(e,i):this.sendBatch(t);}buildBatchBody(t){var e;return JSON.stringify({...this.cfg.appId?{appId:this.cfg.appId}:{},visitorId:this.session.visitorId,location:(e=this.location)!=null?e:void 0,events:t.map(i=>({sessionId:this.session.id,type:i.type,data:i.data}))})}async sendBatch(t){let e=`${this.cfg.endpoint}/batch`,i=this.cfg.secretKey?{Authorization:`Bearer ${this.cfg.secretKey}`}:{};try{await fetch(e,{method:"POST",headers:{"Content-Type":"application/json",...i},body:this.buildBatchBody(t),keepalive:!0});}catch(n){}}};function V(){let r=[[0,0,255],[0,255,255],[0,255,0],[255,255,0],[255,128,0],[255,0,0]],t=[],e=51;for(let i=0;i<r.length-1;i++){let[n,s,o]=r[i],[a,d,l]=r[i+1];for(let c=0;c<e;c++){let m=c/e;t.push([Math.round(n+(a-n)*m),Math.round(s+(d-s)*m),Math.round(o+(l-o)*m)]);}}return t}var N=V();function G(r,t,e={}){let{radius:i=25,maxOpacity:n=.85,minOpacity:s=0}=e,o=r.getContext("2d");if(!o)return;let{width:a,height:d}=r;if(o.clearRect(0,0,a,d),t.length===0)return;let l=document.createElement("canvas");l.width=a,l.height=d;let c=l.getContext("2d");if(!c)return;for(let h of t){let u=h.xPct/100*a,g=h.yPct/100*d,E=h.type==="click"?i*1.6:i,v=c.createRadialGradient(u,g,0,u,g,E);v.addColorStop(0,"rgba(0,0,0,0.15)"),v.addColorStop(1,"rgba(0,0,0,0)"),c.fillStyle=v,c.beginPath(),c.arc(u,g,E,0,Math.PI*2),c.fill();}let m=c.getImageData(0,0,a,d),x=o.createImageData(a,d),C=m.data,w=x.data,L=N.length-1;for(let h=0;h<C.length;h+=4){let u=C[h+3];if(u===0)continue;let g=u/255,E=Math.min(Math.floor(g*L),L),[v,A,B]=N[E],U=s+g*(n-s);w[h]=v,w[h+1]=A,w[h+2]=B,w[h+3]=Math.round(U*255);}o.putImageData(x,0,0);}
2
+ exports.DEFAULT_ENDPOINT=_;exports.LogCapture=y;exports.UserTracker=R;exports.renderHeatmap=G;//# sourceMappingURL=index.js.map
3
3
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../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/heatmap-renderer.ts"],"names":["generateSessionId","c","r","fetchLocation","res","d","e","NavigationPlugin","emit","sessionId","origPush","state","title","url","origReplace","path","TimePlugin","duration","throttle","fn","delay","lastCall","args","now","HeatmapPlugin","sampleRate","maxPoints","pageWidth","pageHeight","absY","vw","scrollX","scrollY","absX","_a","point","LogCapture","options","u","parts","first","stack","msg","src","line","col","err","reason","message","level","extra","a","entry","body","DEFAULT_INTERVAL_MS","STORAGE_KEY","SnapshotPlugin","cfg","stored","last","root","pageBg","canvas","html2canvas","blob","resolve","form","headers","DEFAULT_ENDPOINT","DEFAULTS","UserTracker","config","loc","_b","_c","_d","_e","authHeaders","event","prev","key","pts","resolvedPath","acc","k","v","batch","events","buildPalette","stops","palette","stepsPerSegment","s","fr","fg","fb","tr","tg","tb","i","t","COLOR_PALETTE","renderHeatmap","points","radius","maxOpacity","minOpacity","ctx","width","height","shadow","sCtx","pt","x","y","grad","density","out","dst","lastIdx","alpha","ratio","colorIdx","g","b","opacity"],"mappings":"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","file":"index.js","sourcesContent":["const SESSION_STORAGE_KEY = \"__ut_sid__\";\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","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 *\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\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 path = window.location.pathname + window.location.search;\n if (path === this.previousPath) return; // hash-only or duplicate call\n this.recordPageView(path);\n }\n\n private recordPageView(path: string): void {\n this.previousPath = path;\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\n/**\n * Collects mouse-move, click, and scroll positions for heatmap analysis.\n *\n * Coordinates are stored both as absolute pixels and as percentages of the\n * full page dimensions so data stays meaningful across different screen sizes.\n *\n * Mouse moves and scroll events are throttled (50 ms) and further reduced by\n * the configurable `sampleRate`. Clicks are never sampled — each one is always\n * recorded (up to `maxPoints`).\n *\n * Scroll events record the scroll bar position (top edge of the visible\n * viewport), so dots in the heatmap align with the scroll indicator position\n * on the page snapshot.\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 currentPath = \"\";\n private pointCounts: Record<string, number> = {};\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 }: HeatmapPluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n this.sampleRate = sampleRate;\n this.maxPoints = maxPoints;\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 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 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 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 });\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\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 // 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\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 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 } 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 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 });\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 });\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 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\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 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 type { HeatmapPoint, HeatmapRenderOptions } from \"./types\";\n\n// ─── Color palette ────────────────────────────────────────────────────────────\n// Built once; maps 0 (cool) → 255 (hot) to an RGB color.\n\ntype RGB = [number, number, number];\n\nfunction buildPalette(): RGB[] {\n const stops: RGB[] = [\n [0, 0, 255], // blue\n [0, 255, 255], // cyan\n [0, 255, 0], // green\n [255, 255, 0], // yellow\n [255, 128, 0], // orange\n [255, 0, 0], // red\n ];\n\n const palette: RGB[] = [];\n const stepsPerSegment = 51; // ≈ 255 total entries\n\n for (let s = 0; s < stops.length - 1; s++) {\n const [fr, fg, fb] = stops[s];\n const [tr, tg, tb] = stops[s + 1];\n for (let i = 0; i < stepsPerSegment; i++) {\n const t = i / stepsPerSegment;\n palette.push([\n Math.round(fr + (tr - fr) * t),\n Math.round(fg + (tg - fg) * t),\n Math.round(fb + (tb - fb) * t),\n ]);\n }\n }\n\n return palette;\n}\n\nconst COLOR_PALETTE = buildPalette();\n\n// ─── Public API ───────────────────────────────────────────────────────────────\n\n/**\n * Renders `points` onto `canvas` as a color heatmap.\n *\n * Algorithm:\n * 1. Draw each point as a soft radial gradient on an off-screen canvas,\n * accumulating \"heat\" in the alpha channel.\n * 2. Map each pixel's accumulated alpha value through the color palette\n * (blue → cyan → green → yellow → orange → red) and write to the\n * destination canvas.\n *\n * Coordinates in `HeatmapPoint` are percentages (0–100) of page dimensions,\n * which are scaled to the canvas size at render time — making it resolution\n * independent.\n *\n * @param canvas Target canvas element (will NOT be resized automatically).\n * @param points Array of heatmap points to render.\n * @param options Visual tuning options.\n */\nexport function renderHeatmap(\n canvas: HTMLCanvasElement,\n points: HeatmapPoint[],\n options: HeatmapRenderOptions = {},\n): void {\n const { radius = 25, maxOpacity = 0.85, minOpacity = 0 } = options;\n\n const ctx = canvas.getContext(\"2d\");\n if (!ctx) return;\n\n const { width, height } = canvas;\n ctx.clearRect(0, 0, width, height);\n\n if (points.length === 0) return;\n\n // ── Step 1: draw density layer on off-screen canvas ──\n const shadow = document.createElement(\"canvas\");\n shadow.width = width;\n shadow.height = height;\n const sCtx = shadow.getContext(\"2d\");\n if (!sCtx) return;\n\n for (const pt of points) {\n const x = (pt.xPct / 100) * width;\n const y = (pt.yPct / 100) * height;\n // Clicks get a larger radius to make them visually distinct.\n const r = pt.type === \"click\" ? radius * 1.6 : radius;\n\n const grad = sCtx.createRadialGradient(x, y, 0, x, y, r);\n grad.addColorStop(0, \"rgba(0,0,0,0.15)\");\n grad.addColorStop(1, \"rgba(0,0,0,0)\");\n\n sCtx.fillStyle = grad;\n sCtx.beginPath();\n sCtx.arc(x, y, r, 0, Math.PI * 2);\n sCtx.fill();\n }\n\n // ── Step 2: colorize density layer ──\n const density = sCtx.getImageData(0, 0, width, height);\n const out = ctx.createImageData(width, height);\n const src = density.data;\n const dst = out.data;\n const lastIdx = COLOR_PALETTE.length - 1;\n\n for (let i = 0; i < src.length; i += 4) {\n const alpha = src[i + 3];\n if (alpha === 0) continue;\n\n const ratio = alpha / 255;\n const colorIdx = Math.min(Math.floor(ratio * lastIdx), lastIdx);\n const [r, g, b] = COLOR_PALETTE[colorIdx];\n const opacity = minOpacity + ratio * (maxOpacity - minOpacity);\n\n dst[i] = r;\n dst[i + 1] = g;\n dst[i + 2] = b;\n dst[i + 3] = Math.round(opacity * 255);\n }\n\n ctx.putImageData(out, 0, 0);\n}\n"]}
1
+ {"version":3,"sources":["../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/heatmap-renderer.ts"],"names":["VISITOR_STORAGE_KEY","generateSessionId","c","r","getOrCreateVisitorId","existing","id","e","fetchLocation","res","d","_NavigationPlugin","emit","sessionId","origPush","state","title","url","origReplace","newPath","timeOnPage","path","NavigationPlugin","TimePlugin","duration","throttle","fn","delay","lastCall","args","now","_HeatmapPlugin","sampleRate","maxPoints","pageWidth","pageHeight","absY","target","vw","scrollX","scrollY","absX","_a","point","_b","_c","_d","el","x","y","nearby","HeatmapPlugin","LogCapture","options","u","parts","first","stack","msg","src","line","col","err","reason","message","level","extra","a","entry","body","DEFAULT_INTERVAL_MS","STORAGE_KEY","SnapshotPlugin","cfg","stored","last","root","pageBg","canvas","html2canvas","blob","resolve","form","headers","DEFAULT_ENDPOINT","DEFAULTS","UserTracker","config","loc","_e","authHeaders","event","prev","key","pts","resolvedPath","acc","k","v","batch","events","buildPalette","stops","palette","stepsPerSegment","s","fr","fg","fb","tr","tg","tb","i","t","COLOR_PALETTE","renderHeatmap","points","radius","maxOpacity","minOpacity","ctx","width","height","shadow","sCtx","pt","grad","density","out","dst","lastIdx","alpha","ratio","colorIdx","g","b","opacity"],"mappings":"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","file":"index.js","sourcesContent":["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\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 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 }: HeatmapPluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n this.sampleRate = sampleRate;\n this.maxPoints = maxPoints;\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 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\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 // 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\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 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 });\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 });\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 type { HeatmapPoint, HeatmapRenderOptions } from \"./types\";\n\n// ─── Color palette ────────────────────────────────────────────────────────────\n// Built once; maps 0 (cool) → 255 (hot) to an RGB color.\n\ntype RGB = [number, number, number];\n\nfunction buildPalette(): RGB[] {\n const stops: RGB[] = [\n [0, 0, 255], // blue\n [0, 255, 255], // cyan\n [0, 255, 0], // green\n [255, 255, 0], // yellow\n [255, 128, 0], // orange\n [255, 0, 0], // red\n ];\n\n const palette: RGB[] = [];\n const stepsPerSegment = 51; // ≈ 255 total entries\n\n for (let s = 0; s < stops.length - 1; s++) {\n const [fr, fg, fb] = stops[s];\n const [tr, tg, tb] = stops[s + 1];\n for (let i = 0; i < stepsPerSegment; i++) {\n const t = i / stepsPerSegment;\n palette.push([\n Math.round(fr + (tr - fr) * t),\n Math.round(fg + (tg - fg) * t),\n Math.round(fb + (tb - fb) * t),\n ]);\n }\n }\n\n return palette;\n}\n\nconst COLOR_PALETTE = buildPalette();\n\n// ─── Public API ───────────────────────────────────────────────────────────────\n\n/**\n * Renders `points` onto `canvas` as a color heatmap.\n *\n * Algorithm:\n * 1. Draw each point as a soft radial gradient on an off-screen canvas,\n * accumulating \"heat\" in the alpha channel.\n * 2. Map each pixel's accumulated alpha value through the color palette\n * (blue → cyan → green → yellow → orange → red) and write to the\n * destination canvas.\n *\n * Coordinates in `HeatmapPoint` are percentages (0–100) of page dimensions,\n * which are scaled to the canvas size at render time — making it resolution\n * independent.\n *\n * @param canvas Target canvas element (will NOT be resized automatically).\n * @param points Array of heatmap points to render.\n * @param options Visual tuning options.\n */\nexport function renderHeatmap(\n canvas: HTMLCanvasElement,\n points: HeatmapPoint[],\n options: HeatmapRenderOptions = {},\n): void {\n const { radius = 25, maxOpacity = 0.85, minOpacity = 0 } = options;\n\n const ctx = canvas.getContext(\"2d\");\n if (!ctx) return;\n\n const { width, height } = canvas;\n ctx.clearRect(0, 0, width, height);\n\n if (points.length === 0) return;\n\n // ── Step 1: draw density layer on off-screen canvas ──\n const shadow = document.createElement(\"canvas\");\n shadow.width = width;\n shadow.height = height;\n const sCtx = shadow.getContext(\"2d\");\n if (!sCtx) return;\n\n for (const pt of points) {\n const x = (pt.xPct / 100) * width;\n const y = (pt.yPct / 100) * height;\n // Clicks get a larger radius to make them visually distinct.\n const r = pt.type === \"click\" ? radius * 1.6 : radius;\n\n const grad = sCtx.createRadialGradient(x, y, 0, x, y, r);\n grad.addColorStop(0, \"rgba(0,0,0,0.15)\");\n grad.addColorStop(1, \"rgba(0,0,0,0)\");\n\n sCtx.fillStyle = grad;\n sCtx.beginPath();\n sCtx.arc(x, y, r, 0, Math.PI * 2);\n sCtx.fill();\n }\n\n // ── Step 2: colorize density layer ──\n const density = sCtx.getImageData(0, 0, width, height);\n const out = ctx.createImageData(width, height);\n const src = density.data;\n const dst = out.data;\n const lastIdx = COLOR_PALETTE.length - 1;\n\n for (let i = 0; i < src.length; i += 4) {\n const alpha = src[i + 3];\n if (alpha === 0) continue;\n\n const ratio = alpha / 255;\n const colorIdx = Math.min(Math.floor(ratio * lastIdx), lastIdx);\n const [r, g, b] = COLOR_PALETTE[colorIdx];\n const opacity = minOpacity + ratio * (maxOpacity - minOpacity);\n\n dst[i] = r;\n dst[i + 1] = g;\n dst[i + 2] = b;\n dst[i + 3] = Math.round(opacity * 255);\n }\n\n ctx.putImageData(out, 0, 0);\n}\n"]}