alphana-sdk 0.4.5 → 0.4.6

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.
@@ -1 +1 @@
1
- "use strict";var AlphanaSDK=(()=>{var I=Object.defineProperty;var j=Object.getOwnPropertyDescriptor;var F=Object.getOwnPropertyNames;var V=Object.prototype.hasOwnProperty;var G=(o,t)=>{for(var e in t)I(o,e,{get:t[e],enumerable:!0})},W=(o,t,e,i)=>{if(t&&typeof t=="object"||typeof t=="function")for(let n of F(t))!V.call(o,n)&&n!==e&&I(o,n,{get:()=>t[n],enumerable:!(i=j(t,n))||i.enumerable});return o};var z=o=>W(I({},"__esModule",{value:!0}),o);var Y={};G(Y,{DEFAULT_ENDPOINT:()=>x,LogCapture:()=>g,UserTracker:()=>T,renderHeatmap:()=>_});var O="__ut_vid__";function y(){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)})}function D(){if(typeof localStorage=="undefined")return y();try{let o=localStorage.getItem(O);if(o)return o;let t=y();return localStorage.setItem(O,t),t}catch(o){return y()}}async function M(){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 b=class b{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<=b.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}}))}};b.UTURN_THRESHOLD_MS=5e3;var k=b;var P=class{constructor({emit:t,sessionId:e}){this.currentPath="";this.startTime=0;this.tracking=!1;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=!0}stopTracking(){if(!this.tracking||!this.currentPath)return;let t=Date.now()-this.startTime;if(t<100){this.tracking=!1;return}this.emit({type:"timespent",data:{path:this.currentPath,duration:t,sessionId:this.sessionId,timestamp:Date.now()}}),this.tracking=!1}};function R(o,t){let e=0;return(...i)=>{let n=Date.now();n-e>=t&&(e=n,o(...i))}}var h=class h{constructor({emit:t,sessionId:e,sampleRate:i=.3,maxPoints:n=2e3,allowedPaths:s}){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,pw:e,ph:i,type:"move"})};this.handleClick=t=>{if(Date.now()-this.lastTouchTime<500)return;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,pw:e,ph:i,type:"click",...s?{target:s}:{}}),this.checkRageClick(t.clientX,n,this.currentPath)};this.lastTouchTime=0;this.handleTouchEnd=t=>{let e=t.changedTouches[0];if(!e)return;let i=document.documentElement.scrollWidth,n=document.documentElement.scrollHeight,s=e.clientY+window.scrollY;this.lastTouchTime=Date.now(),this.recordPoint({x:e.clientX,y:s,xPct:i>0?e.clientX/i*100:0,yPct:n>0?s/n*100:0,pw:i,ph:n,type:"click"}),this.checkRageClick(e.clientX,s,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,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,pw:t,ph:e,type:"scroll"})};this.handleNavigate=t=>{this.currentPath=t.detail.path};this.emit=t,this.sessionId=e,this.sampleRate=i,this.maxPoints=n,this.allowedPaths=s&&s.length>0?new Set(s):null,this.throttledMouseMove=R(this.handleMouseMove,50),this.throttledScroll=R(this.handleScroll,100)}init(){this.currentPath=window.location.pathname+window.location.search,document.addEventListener("mousemove",this.throttledMouseMove),document.addEventListener("click",this.handleClick),document.addEventListener("touchend",this.handleTouchEnd,{passive:!0}),window.addEventListener("scroll",this.throttledScroll,{passive:!0}),window.addEventListener("tracker:navigate",this.handleNavigate)}destroy(){document.removeEventListener("mousemove",this.throttledMouseMove),document.removeEventListener("click",this.handleClick),document.removeEventListener("touchend",this.handleTouchEnd),window.removeEventListener("scroll",this.throttledScroll),window.removeEventListener("tracker:navigate",this.handleNavigate)}canRecord(){var t;return this.allowedPaths!==null&&!this.allowedPaths.has(this.currentPath)?!1:((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,r,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=(r=e.closest("button, a"))==null?void 0:r.textContent)==null?void 0:a.trim().slice(0,60)))||void 0}checkRageClick(t,e,i){let n=Date.now();this.recentClicks=this.recentClicks.filter(r=>n-r.t<h.RAGE_WINDOW_MS),this.recentClicks.push({x:t,y:e,t:n});let s=this.recentClicks.filter(r=>Math.hypot(r.x-t,r.y-e)<=h.RAGE_RADIUS_PX);s.length>=h.RAGE_THRESHOLD&&(this.emit({type:"rageclik",data:{path:i,x:t,y:e,count:s.length,timestamp:n,sessionId:this.sessionId}}),this.recentClicks=[])}};h.RAGE_THRESHOLD=3,h.RAGE_WINDOW_MS=1e3,h.RAGE_RADIUS_PX=50;var S=h;var g=class{constructor(t){this.prevOnError=null;this.prevOnUnhandledRejection=null;this.initialized=!1;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):!1),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=!0)}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=!1)}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:!0}).catch(a=>{this.origError&&this.origError("[user-tracker] Failed to send log:",a)})}};var x="https://api.alphana.ir/api/events",K={endpoint:x,trackNavigation:!0,trackTime:!0,trackHeatmap:!0,trackLogs:!0,trackSnapshots:!0,mouseSampleRate:.3,maxHeatmapPoints:2e3,batchSize:20,flushInterval:5e3},T=class{constructor(t={}){this.initialized=!1;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 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:y(),visitorId:D(),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 k({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 S({emit:t,sessionId:e,sampleRate:this.cfg.mouseSampleRate,maxPoints:this.cfg.maxHeatmapPoints,allowedPaths:this.cfg.heatmapPages}),this.heatmap.init()),this.cfg.endpoint&&(this.flushTimer=setInterval(()=>{this.queue.length>0&&this.flush()},this.cfg.flushInterval),M().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 g({endpoint:this.cfg.endpoint,sessionId:this.session.id,secretKey:this.cfg.secretKey,appId:this.cfg.appId}),this.logCapture.init())),this.initialized=!0,this}destroy(){var t,e,i,n;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(),typeof window!="undefined"&&window.removeEventListener("tracker:navigate",this.handleNavigate),this.queue.length>0&&this.cfg.endpoint&&this.flushBeacon(),this.initialized=!1}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:!0,...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:!1,...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:!0}).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}case"rageclik":case"uturn":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 X(){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],[a,l,m]=o[i+1];for(let d=0;d<e;d++){let v=d/e;t.push([Math.round(n+(a-n)*v),Math.round(s+(l-s)*v),Math.round(r+(m-r)*v)])}}return t}var N=X();function _(o,t,e={}){let{radius:i=25,maxOpacity:n=.85,minOpacity:s=0}=e,r=o.getContext("2d");if(!r)return;let{width:a,height:l}=o;if(r.clearRect(0,0,a,l),t.length===0)return;let m=document.createElement("canvas");m.width=a,m.height=l;let d=m.getContext("2d");if(!d)return;for(let c of t){let p=c.xPct/100*a,u=c.yPct/100*l,E=c.type==="click"?i*1.6:i,f=d.createRadialGradient(p,u,0,p,u,E);f.addColorStop(0,"rgba(0,0,0,0.15)"),f.addColorStop(1,"rgba(0,0,0,0)"),d.fillStyle=f,d.beginPath(),d.arc(p,u,E,0,Math.PI*2),d.fill()}let v=d.getImageData(0,0,a,l),L=r.createImageData(a,l),C=v.data,w=L.data,H=N.length-1;for(let c=0;c<C.length;c+=4){let p=C[c+3];if(p===0)continue;let u=p/255,E=Math.min(Math.floor(u*H),H),[f,A,B]=N[E],U=s+u*(n-s);w[c]=f,w[c+1]=A,w[c+2]=B,w[c+3]=Math.round(U*255)}r.putImageData(L,0,0)}return z(Y);})();
1
+ "use strict";var AlphanaSDK=(()=>{var I=Object.defineProperty;var j=Object.getOwnPropertyDescriptor;var F=Object.getOwnPropertyNames;var V=Object.prototype.hasOwnProperty;var G=(o,t)=>{for(var e in t)I(o,e,{get:t[e],enumerable:!0})},W=(o,t,e,i)=>{if(t&&typeof t=="object"||typeof t=="function")for(let n of F(t))!V.call(o,n)&&n!==e&&I(o,n,{get:()=>t[n],enumerable:!(i=j(t,n))||i.enumerable});return o};var z=o=>W(I({},"__esModule",{value:!0}),o);var Y={};G(Y,{DEFAULT_ENDPOINT:()=>x,LogCapture:()=>g,UserTracker:()=>T,renderHeatmap:()=>_});var O="__ut_vid__";function y(){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)})}function D(){if(typeof localStorage=="undefined")return y();try{let o=localStorage.getItem(O);if(o)return o;let t=y();return localStorage.setItem(O,t),t}catch(o){return y()}}async function M(){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 b=class b{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<=b.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}}))}};b.UTURN_THRESHOLD_MS=5e3;var k=b;var P=class{constructor({emit:t,sessionId:e}){this.currentPath="";this.startTime=0;this.tracking=!1;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=!0}stopTracking(){if(!this.tracking||!this.currentPath)return;let t=Date.now()-this.startTime;if(t<100){this.tracking=!1;return}this.emit({type:"timespent",data:{path:this.currentPath,duration:t,sessionId:this.sessionId,timestamp:Date.now()}}),this.tracking=!1}};function R(o,t){let e=0;return(...i)=>{let n=Date.now();n-e>=t&&(e=n,o(...i))}}var h=class h{constructor({emit:t,sessionId:e,sampleRate:i=.3,maxPoints:n=2e3,allowedPaths:s}){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,pw:e,ph:i,type:"move"})};this.handleClick=t=>{if(Date.now()-this.lastTouchTime<500)return;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,pw:e,ph:i,type:"click",...s?{target:s}:{}}),this.checkRageClick(t.clientX,n,this.currentPath)};this.lastTouchTime=0;this.handleTouchEnd=t=>{let e=t.changedTouches[0];if(!e)return;let i=document.documentElement.scrollWidth,n=document.documentElement.scrollHeight,s=e.clientY+window.scrollY;this.lastTouchTime=Date.now(),this.recordPoint({x:e.clientX,y:s,xPct:i>0?e.clientX/i*100:0,yPct:n>0?s/n*100:0,pw:i,ph:n,type:"click"}),this.checkRageClick(e.clientX,s,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,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,pw:t,ph:e,type:"scroll"})};this.handleNavigate=t=>{this.currentPath=t.detail.path};this.emit=t,this.sessionId=e,this.sampleRate=i,this.maxPoints=n,this.allowedPaths=s&&s.length>0?new Set(s):null,this.throttledMouseMove=R(this.handleMouseMove,50),this.throttledScroll=R(this.handleScroll,100)}init(){this.currentPath=window.location.pathname+window.location.search,document.addEventListener("mousemove",this.throttledMouseMove),document.addEventListener("click",this.handleClick),document.addEventListener("touchend",this.handleTouchEnd,{passive:!0}),window.addEventListener("scroll",this.throttledScroll,{passive:!0}),window.addEventListener("tracker:navigate",this.handleNavigate)}destroy(){document.removeEventListener("mousemove",this.throttledMouseMove),document.removeEventListener("click",this.handleClick),document.removeEventListener("touchend",this.handleTouchEnd),window.removeEventListener("scroll",this.throttledScroll),window.removeEventListener("tracker:navigate",this.handleNavigate)}canRecord(){var t;return this.allowedPaths!==null&&!this.allowedPaths.has(this.currentPath)?!1:((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,r,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=(r=e.closest("button, a"))==null?void 0:r.textContent)==null?void 0:a.trim().slice(0,60)))||void 0}checkRageClick(t,e,i){let n=Date.now();this.recentClicks=this.recentClicks.filter(r=>n-r.t<h.RAGE_WINDOW_MS),this.recentClicks.push({x:t,y:e,t:n});let s=this.recentClicks.filter(r=>Math.hypot(r.x-t,r.y-e)<=h.RAGE_RADIUS_PX);s.length>=h.RAGE_THRESHOLD&&(this.emit({type:"rageclik",data:{path:i,x:t,y:e,count:s.length,timestamp:n,sessionId:this.sessionId}}),this.recentClicks=[])}};h.RAGE_THRESHOLD=3,h.RAGE_WINDOW_MS=1e3,h.RAGE_RADIUS_PX=50;var S=h;var g=class{constructor(t){this.prevOnError=null;this.prevOnUnhandledRejection=null;this.initialized=!1;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):!1),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=!0)}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=!1)}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:!0}).catch(a=>{this.origError&&this.origError("[user-tracker] Failed to send log:",a)})}};var x="https://api.alphana.ir/api/events",K={endpoint:x,trackNavigation:!0,trackTime:!0,trackHeatmap:!0,trackLogs:!0,mouseSampleRate:.3,maxHeatmapPoints:2e3,batchSize:20,flushInterval:5e3},T=class{constructor(t={}){this.initialized=!1;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 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:y(),visitorId:D(),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 k({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 S({emit:t,sessionId:e,sampleRate:this.cfg.mouseSampleRate,maxPoints:this.cfg.maxHeatmapPoints,allowedPaths:this.cfg.heatmapPages}),this.heatmap.init()),this.cfg.endpoint&&(this.flushTimer=setInterval(()=>{this.queue.length>0&&this.flush()},this.cfg.flushInterval),M().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 g({endpoint:this.cfg.endpoint,sessionId:this.session.id,secretKey:this.cfg.secretKey,appId:this.cfg.appId}),this.logCapture.init())),this.initialized=!0,this}destroy(){var t,e,i,n;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(),typeof window!="undefined"&&window.removeEventListener("tracker:navigate",this.handleNavigate),this.queue.length>0&&this.cfg.endpoint&&this.flushBeacon(),this.initialized=!1}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:!0,...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:!1,...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:!0}).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}case"rageclik":case"uturn":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 X(){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],[a,l,m]=o[i+1];for(let d=0;d<e;d++){let v=d/e;t.push([Math.round(n+(a-n)*v),Math.round(s+(l-s)*v),Math.round(r+(m-r)*v)])}}return t}var N=X();function _(o,t,e={}){let{radius:i=25,maxOpacity:n=.85,minOpacity:s=0}=e,r=o.getContext("2d");if(!r)return;let{width:a,height:l}=o;if(r.clearRect(0,0,a,l),t.length===0)return;let m=document.createElement("canvas");m.width=a,m.height=l;let d=m.getContext("2d");if(!d)return;for(let c of t){let p=c.xPct/100*a,u=c.yPct/100*l,E=c.type==="click"?i*1.6:i,f=d.createRadialGradient(p,u,0,p,u,E);f.addColorStop(0,"rgba(0,0,0,0.15)"),f.addColorStop(1,"rgba(0,0,0,0)"),d.fillStyle=f,d.beginPath(),d.arc(p,u,E,0,Math.PI*2),d.fill()}let v=d.getImageData(0,0,a,l),L=r.createImageData(a,l),C=v.data,w=L.data,H=N.length-1;for(let c=0;c<C.length;c+=4){let p=C[c+3];if(p===0)continue;let u=p/255,E=Math.min(Math.floor(u*H),H),[f,A,B]=N[E],U=s+u*(n-s);w[c]=f,w[c+1]=A,w[c+2]=B,w[c+3]=Math.round(U*255)}r.putImageData(L,0,0)}return z(Y);})();
package/dist/index.js CHANGED
@@ -1,3 +1,3 @@
1
- 'use strict';var C="__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 H(){if(typeof localStorage=="undefined")return f();try{let r=localStorage.getItem(C);if(r)return r;let t=f();return localStorage.setItem(C,t),t}catch(r){return f()}}async function O(){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 b=class b{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<=b.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}}));}};b.UTURN_THRESHOLD_MS=5e3;var k=b;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 T(r,t){let e=0;return (...i)=>{let n=Date.now();n-e>=t&&(e=n,r(...i));}}var h=class h{constructor({emit:t,sessionId:e,sampleRate:i=.3,maxPoints:n=2e3,allowedPaths:s}){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,pw:e,ph:i,type:"move"});};this.handleClick=t=>{if(Date.now()-this.lastTouchTime<500)return;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,pw:e,ph:i,type:"click",...s?{target:s}:{}}),this.checkRageClick(t.clientX,n,this.currentPath);};this.lastTouchTime=0;this.handleTouchEnd=t=>{let e=t.changedTouches[0];if(!e)return;let i=document.documentElement.scrollWidth,n=document.documentElement.scrollHeight,s=e.clientY+window.scrollY;this.lastTouchTime=Date.now(),this.recordPoint({x:e.clientX,y:s,xPct:i>0?e.clientX/i*100:0,yPct:n>0?s/n*100:0,pw:i,ph:n,type:"click"}),this.checkRageClick(e.clientX,s,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,pw:t,ph:e,type:"scroll"});};this.handleNavigate=t=>{this.currentPath=t.detail.path;};this.emit=t,this.sessionId=e,this.sampleRate=i,this.maxPoints=n,this.allowedPaths=s&&s.length>0?new Set(s):null,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),document.addEventListener("touchend",this.handleTouchEnd,{passive:true}),window.addEventListener("scroll",this.throttledScroll,{passive:true}),window.addEventListener("tracker:navigate",this.handleNavigate);}destroy(){document.removeEventListener("mousemove",this.throttledMouseMove),document.removeEventListener("click",this.handleClick),document.removeEventListener("touchend",this.handleTouchEnd),window.removeEventListener("scroll",this.throttledScroll),window.removeEventListener("tracker:navigate",this.handleNavigate);}canRecord(){var t;return this.allowedPaths!==null&&!this.allowedPaths.has(this.currentPath)?false:((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<h.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)<=h.RAGE_RADIUS_PX);s.length>=h.RAGE_THRESHOLD&&(this.emit({type:"rageclik",data:{path:i,x:t,y:e,count:s.length,timestamp:n,sessionId:this.sessionId}}),this.recentClicks=[]);}};h.RAGE_THRESHOLD=3,h.RAGE_WINDOW_MS=1e3,h.RAGE_RADIUS_PX=50;var S=h;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 D="https://api.alphana.ir/api/events",B={endpoint:D,trackNavigation:true,trackTime:true,trackHeatmap:true,trackLogs:true,trackSnapshots:true,mouseSampleRate:.3,maxHeatmapPoints:2e3,batchSize:20,flushInterval:5e3},I=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 e;this.cfg={...B,...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:H(),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 k({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 S({emit:t,sessionId:e,sampleRate:this.cfg.mouseSampleRate,maxPoints:this.cfg.maxHeatmapPoints,allowedPaths:this.cfg.heatmapPages}),this.heatmap.init()),this.cfg.endpoint&&(this.flushTimer=setInterval(()=>{this.queue.length>0&&this.flush();},this.cfg.flushInterval),O().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.initialized=true,this}destroy(){var t,e,i,n;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(),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 U(){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,l,g]=r[i+1];for(let d=0;d<e;d++){let m=d/e;t.push([Math.round(n+(a-n)*m),Math.round(s+(l-s)*m),Math.round(o+(g-o)*m)]);}}return t}var M=U();function j(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:l}=r;if(o.clearRect(0,0,a,l),t.length===0)return;let g=document.createElement("canvas");g.width=a,g.height=l;let d=g.getContext("2d");if(!d)return;for(let c of t){let p=c.xPct/100*a,u=c.yPct/100*l,E=c.type==="click"?i*1.6:i,v=d.createRadialGradient(p,u,0,p,u,E);v.addColorStop(0,"rgba(0,0,0,0.15)"),v.addColorStop(1,"rgba(0,0,0,0)"),d.fillStyle=v,d.beginPath(),d.arc(p,u,E,0,Math.PI*2),d.fill();}let m=d.getImageData(0,0,a,l),R=o.createImageData(a,l),x=m.data,w=R.data,L=M.length-1;for(let c=0;c<x.length;c+=4){let p=x[c+3];if(p===0)continue;let u=p/255,E=Math.min(Math.floor(u*L),L),[v,N,_]=M[E],A=s+u*(n-s);w[c]=v,w[c+1]=N,w[c+2]=_,w[c+3]=Math.round(A*255);}o.putImageData(R,0,0);}
1
+ 'use strict';var C="__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 H(){if(typeof localStorage=="undefined")return f();try{let r=localStorage.getItem(C);if(r)return r;let t=f();return localStorage.setItem(C,t),t}catch(r){return f()}}async function O(){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 b=class b{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<=b.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}}));}};b.UTURN_THRESHOLD_MS=5e3;var k=b;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 T(r,t){let e=0;return (...i)=>{let n=Date.now();n-e>=t&&(e=n,r(...i));}}var h=class h{constructor({emit:t,sessionId:e,sampleRate:i=.3,maxPoints:n=2e3,allowedPaths:s}){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,pw:e,ph:i,type:"move"});};this.handleClick=t=>{if(Date.now()-this.lastTouchTime<500)return;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,pw:e,ph:i,type:"click",...s?{target:s}:{}}),this.checkRageClick(t.clientX,n,this.currentPath);};this.lastTouchTime=0;this.handleTouchEnd=t=>{let e=t.changedTouches[0];if(!e)return;let i=document.documentElement.scrollWidth,n=document.documentElement.scrollHeight,s=e.clientY+window.scrollY;this.lastTouchTime=Date.now(),this.recordPoint({x:e.clientX,y:s,xPct:i>0?e.clientX/i*100:0,yPct:n>0?s/n*100:0,pw:i,ph:n,type:"click"}),this.checkRageClick(e.clientX,s,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,pw:t,ph:e,type:"scroll"});};this.handleNavigate=t=>{this.currentPath=t.detail.path;};this.emit=t,this.sessionId=e,this.sampleRate=i,this.maxPoints=n,this.allowedPaths=s&&s.length>0?new Set(s):null,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),document.addEventListener("touchend",this.handleTouchEnd,{passive:true}),window.addEventListener("scroll",this.throttledScroll,{passive:true}),window.addEventListener("tracker:navigate",this.handleNavigate);}destroy(){document.removeEventListener("mousemove",this.throttledMouseMove),document.removeEventListener("click",this.handleClick),document.removeEventListener("touchend",this.handleTouchEnd),window.removeEventListener("scroll",this.throttledScroll),window.removeEventListener("tracker:navigate",this.handleNavigate);}canRecord(){var t;return this.allowedPaths!==null&&!this.allowedPaths.has(this.currentPath)?false:((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<h.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)<=h.RAGE_RADIUS_PX);s.length>=h.RAGE_THRESHOLD&&(this.emit({type:"rageclik",data:{path:i,x:t,y:e,count:s.length,timestamp:n,sessionId:this.sessionId}}),this.recentClicks=[]);}};h.RAGE_THRESHOLD=3,h.RAGE_WINDOW_MS=1e3,h.RAGE_RADIUS_PX=50;var S=h;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 D="https://api.alphana.ir/api/events",B={endpoint:D,trackNavigation:true,trackTime:true,trackHeatmap:true,trackLogs:true,mouseSampleRate:.3,maxHeatmapPoints:2e3,batchSize:20,flushInterval:5e3},I=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 e;this.cfg={...B,...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:H(),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 k({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 S({emit:t,sessionId:e,sampleRate:this.cfg.mouseSampleRate,maxPoints:this.cfg.maxHeatmapPoints,allowedPaths:this.cfg.heatmapPages}),this.heatmap.init()),this.cfg.endpoint&&(this.flushTimer=setInterval(()=>{this.queue.length>0&&this.flush();},this.cfg.flushInterval),O().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.initialized=true,this}destroy(){var t,e,i,n;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(),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 U(){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,l,g]=r[i+1];for(let d=0;d<e;d++){let m=d/e;t.push([Math.round(n+(a-n)*m),Math.round(s+(l-s)*m),Math.round(o+(g-o)*m)]);}}return t}var M=U();function j(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:l}=r;if(o.clearRect(0,0,a,l),t.length===0)return;let g=document.createElement("canvas");g.width=a,g.height=l;let d=g.getContext("2d");if(!d)return;for(let c of t){let p=c.xPct/100*a,u=c.yPct/100*l,E=c.type==="click"?i*1.6:i,v=d.createRadialGradient(p,u,0,p,u,E);v.addColorStop(0,"rgba(0,0,0,0.15)"),v.addColorStop(1,"rgba(0,0,0,0)"),d.fillStyle=v,d.beginPath(),d.arc(p,u,E,0,Math.PI*2),d.fill();}let m=d.getImageData(0,0,a,l),R=o.createImageData(a,l),x=m.data,w=R.data,L=M.length-1;for(let c=0;c<x.length;c+=4){let p=x[c+3];if(p===0)continue;let u=p/255,E=Math.min(Math.floor(u*L),L),[v,N,_]=M[E],A=s+u*(n-s);w[c]=v,w[c+1]=N,w[c+2]=_,w[c+3]=Math.round(A*255);}o.putImageData(R,0,0);}
2
2
  exports.DEFAULT_ENDPOINT=D;exports.LogCapture=y;exports.UserTracker=I;exports.renderHeatmap=j;//# 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/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","allowedPaths","pageWidth","pageHeight","absY","target","touch","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_ENDPOINT","DEFAULTS","UserTracker","config","_e","loc","authHeaders","event","prev","key","pts","resolvedPath","acc","k","v","batch","blob","events","buildPalette","stops","palette","stepsPerSegment","s","fr","fg","fb","tr","tg","tb","i","t","COLOR_PALETTE","renderHeatmap","canvas","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 * If provided, heatmap data will only be collected for paths in this list.\n * An empty array or undefined means all pages are tracked.\n */\n allowedPaths?: string[];\n}\n\n/**\n * Collects mouse-move, click, and scroll positions for heatmap analysis.\n *\n * Also detects rage clicks (≥3 clicks within 1 s in a 50 px radius) and emits\n * a `rageclik` event so the backend can surface problematic UI hotspots.\n *\n * Click events include an optional `target` field containing a human-readable\n * label for the clicked element (aria-label › id › button/link text).\n */\nexport class HeatmapPlugin {\n private readonly emit: EmitFn;\n private readonly sessionId: string;\n private readonly sampleRate: number;\n private readonly maxPoints: number;\n private readonly allowedPaths: Set<string> | null;\n private currentPath = \"\";\n private pointCounts: Record<string, number> = {};\n\n // Rage-click detection state\n private recentClicks: { x: number; y: number; t: number }[] = [];\n private static readonly RAGE_THRESHOLD = 3;\n private static readonly RAGE_WINDOW_MS = 1_000;\n private static readonly RAGE_RADIUS_PX = 50;\n\n private readonly throttledMouseMove: (e: MouseEvent) => void;\n private readonly throttledScroll: () => void;\n\n constructor({\n emit,\n sessionId,\n sampleRate = 0.3,\n maxPoints = 2000,\n allowedPaths,\n }: HeatmapPluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n this.sampleRate = sampleRate;\n this.maxPoints = maxPoints;\n this.allowedPaths =\n allowedPaths && allowedPaths.length > 0 ? new Set(allowedPaths) : null;\n\n this.throttledMouseMove = throttle(this.handleMouseMove, 50);\n this.throttledScroll = throttle(this.handleScroll, 100);\n }\n\n init(): void {\n this.currentPath = window.location.pathname + window.location.search;\n\n document.addEventListener(\"mousemove\", this.throttledMouseMove);\n document.addEventListener(\"click\", this.handleClick);\n document.addEventListener(\"touchend\", this.handleTouchEnd, {\n passive: true,\n });\n window.addEventListener(\"scroll\", this.throttledScroll, { passive: true });\n window.addEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n destroy(): void {\n document.removeEventListener(\"mousemove\", this.throttledMouseMove);\n document.removeEventListener(\"click\", this.handleClick);\n document.removeEventListener(\"touchend\", this.handleTouchEnd);\n window.removeEventListener(\"scroll\", this.throttledScroll);\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n private canRecord(): boolean {\n if (\n this.allowedPaths !== null &&\n !this.allowedPaths.has(this.currentPath)\n ) {\n return false;\n }\n return (this.pointCounts[this.currentPath] ?? 0) < this.maxPoints;\n }\n\n private recordPoint(\n point: Omit<HeatmapPoint, \"path\" | \"timestamp\" | \"sessionId\">,\n ): void {\n if (!this.canRecord()) return;\n this.pointCounts[this.currentPath] =\n (this.pointCounts[this.currentPath] ?? 0) + 1;\n this.emit({\n type: \"heatmap\",\n data: {\n ...point,\n path: this.currentPath,\n timestamp: Date.now(),\n },\n });\n }\n\n /** Extract a human-readable label for the clicked element. */\n private getClickTarget(e: MouseEvent): string | undefined {\n const el = e.target as HTMLElement | null;\n if (!el) return undefined;\n const label =\n el.getAttribute(\"aria-label\") ||\n el.closest(\"[aria-label]\")?.getAttribute(\"aria-label\") ||\n el.getAttribute(\"data-track-label\") ||\n el.getAttribute(\"id\") ||\n (el instanceof HTMLButtonElement || el instanceof HTMLAnchorElement\n ? el.innerText?.trim().slice(0, 60)\n : el.closest(\"button, a\")?.textContent?.trim().slice(0, 60));\n return label || undefined;\n }\n\n /** Detect rage-click bursts and emit a `rageclik` event when found. */\n private checkRageClick(x: number, y: number, path: string): void {\n const now = Date.now();\n this.recentClicks = this.recentClicks.filter(\n (c) => now - c.t < HeatmapPlugin.RAGE_WINDOW_MS,\n );\n this.recentClicks.push({ x, y, t: now });\n\n const nearby = this.recentClicks.filter(\n (c) => Math.hypot(c.x - x, c.y - y) <= HeatmapPlugin.RAGE_RADIUS_PX,\n );\n\n if (nearby.length >= HeatmapPlugin.RAGE_THRESHOLD) {\n this.emit({\n type: \"rageclik\",\n data: {\n path,\n x,\n y,\n count: nearby.length,\n timestamp: now,\n sessionId: this.sessionId,\n },\n });\n // Reset to avoid continuously re-triggering in the same burst.\n this.recentClicks = [];\n }\n }\n\n private handleMouseMove = (e: MouseEvent): void => {\n if (Math.random() > this.sampleRate) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = e.clientY + window.scrollY;\n this.recordPoint({\n x: e.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (e.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"move\",\n });\n };\n\n private handleClick = (e: MouseEvent): void => {\n // Skip if a touchend already recorded this tap (within 500 ms)\n if (Date.now() - this.lastTouchTime < 500) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = e.clientY + window.scrollY;\n const target = this.getClickTarget(e);\n this.recordPoint({\n x: e.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (e.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"click\",\n ...(target ? { target } : {}),\n });\n this.checkRageClick(e.clientX, absY, this.currentPath);\n };\n\n /**\n * Records taps on touch devices.\n * Mobile browsers do fire a synthetic `click` after `touchend`, but some\n * frameworks call `preventDefault()` on touch events which suppresses it.\n * Listening to `touchend` directly ensures taps are always captured.\n * We mark the point with a short-lived flag so the subsequent synthetic\n * `click` event (if it fires) is deduplicated.\n */\n private lastTouchTime = 0;\n\n private handleTouchEnd = (e: TouchEvent): void => {\n const touch = e.changedTouches[0];\n if (!touch) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = touch.clientY + window.scrollY;\n this.lastTouchTime = Date.now();\n this.recordPoint({\n x: touch.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (touch.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"click\",\n });\n this.checkRageClick(touch.clientX, absY, this.currentPath);\n };\n\n private handleScroll = (): void => {\n if (Math.random() > this.sampleRate) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const vw = window.innerWidth;\n const scrollX = window.scrollX;\n const scrollY = window.scrollY;\n const absX = scrollX + vw / 2;\n this.recordPoint({\n x: vw / 2,\n y: scrollY,\n xPct: pageWidth > 0 ? (absX / pageWidth) * 100 : 50,\n yPct: pageHeight > 0 ? (scrollY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"scroll\",\n });\n };\n\n private handleNavigate = (e: CustomEvent<{ path: string }>): void => {\n this.currentPath = e.detail.path;\n };\n}\n","export type LogLevel = \"info\" | \"warn\" | \"error\";\n\nexport interface LogEntry {\n sessionId?: string;\n appId?: string;\n level: LogLevel;\n message: string;\n url?: string;\n stack?: string;\n meta?: Record<string, unknown>;\n timestamp: number;\n}\n\ntype ConsoleFn = (...args: unknown[]) => void;\n\n/**\n * Automatically captures console.info/warn/error output and unhandled errors,\n * then ships them to the backend `/logs/ingest` endpoint.\n *\n * All console methods are restored exactly in `destroy()`.\n */\nexport class LogCapture {\n private readonly endpoint: string;\n private readonly sessionId: string;\n private readonly appId?: string;\n private readonly authHeaders: Record<string, string>;\n\n // Original console methods preserved so we can restore them.\n private origInfo!: ConsoleFn;\n private origWarn!: ConsoleFn;\n private origError!: ConsoleFn;\n\n private prevOnError: OnErrorEventHandler = null;\n private prevOnUnhandledRejection:\n | ((e: PromiseRejectionEvent) => void)\n | null = null;\n\n private initialized = false;\n\n constructor(options: {\n endpoint: string;\n sessionId: string;\n secretKey?: string;\n appId?: string;\n }) {\n // Derive the API base URL by stripping everything from the last path\n // segment that isn't a versioning prefix. The tracker config `endpoint`\n // is the *events* URL (e.g. http://host/api/events), but logs live at\n // http://host/api/logs/ingest, so we walk up until we reach the common\n // base (i.e. remove the final segment).\n try {\n const u = new URL(options.endpoint);\n // Remove the last non-empty path segment (e.g. \"/api/events\" → \"/api\")\n const parts = u.pathname.replace(/\\/$/, \"\").split(\"/\");\n parts.pop();\n u.pathname = parts.join(\"/\") || \"/\";\n this.endpoint = u.toString().replace(/\\/$/, \"\");\n } catch {\n this.endpoint = options.endpoint;\n }\n this.sessionId = options.sessionId;\n this.appId = options.appId;\n this.authHeaders = options.secretKey\n ? { Authorization: `Bearer ${options.secretKey}` }\n : {};\n }\n\n init(): void {\n if (typeof window === \"undefined\" || this.initialized) return;\n\n this.origInfo = console.info.bind(console);\n this.origWarn = console.warn.bind(console);\n this.origError = console.error.bind(console);\n\n console.info = (...args: unknown[]) => {\n this.origInfo(...args);\n this.send(\"info\", this.format(args));\n };\n\n console.warn = (...args: unknown[]) => {\n this.origWarn(...args);\n this.send(\"warn\", this.format(args));\n };\n\n console.error = (...args: unknown[]) => {\n this.origError(...args);\n const [first] = args;\n const stack = first instanceof Error ? first.stack : undefined;\n this.send(\"error\", this.format(args), { stack });\n };\n\n this.prevOnError = window.onerror;\n window.onerror = (msg, src, line, col, err) => {\n this.send(\"error\", String(msg), {\n stack: err?.stack,\n meta: { src, line, col },\n });\n if (typeof this.prevOnError === \"function\") {\n return this.prevOnError(msg, src, line, col, err);\n }\n return false;\n };\n\n this.prevOnUnhandledRejection = (e: PromiseRejectionEvent) => {\n const reason = e.reason;\n const message =\n reason instanceof Error\n ? reason.message\n : String(reason ?? \"Unhandled promise rejection\");\n this.send(\"error\", message, {\n stack: reason instanceof Error ? reason.stack : undefined,\n });\n };\n window.addEventListener(\n \"unhandledrejection\",\n this.prevOnUnhandledRejection,\n );\n\n this.initialized = true;\n }\n\n destroy(): void {\n if (!this.initialized) return;\n console.info = this.origInfo;\n console.warn = this.origWarn;\n console.error = this.origError;\n\n window.onerror = this.prevOnError;\n if (this.prevOnUnhandledRejection) {\n window.removeEventListener(\n \"unhandledrejection\",\n this.prevOnUnhandledRejection,\n );\n }\n this.initialized = false;\n }\n\n /** Manually capture a log entry (e.g. from try/catch). */\n capture(\n level: LogLevel,\n message: string,\n extra?: { stack?: string; meta?: Record<string, unknown> },\n ): void {\n this.send(level, message, extra);\n }\n\n private format(args: unknown[]): string {\n return args\n .map((a) => {\n if (a instanceof Error) return a.message;\n if (typeof a === \"object\") {\n try {\n return JSON.stringify(a);\n } catch {\n return String(a);\n }\n }\n return String(a);\n })\n .join(\" \");\n }\n\n private send(\n level: LogLevel,\n message: string,\n extra?: { stack?: string; meta?: Record<string, unknown> },\n ): void {\n const entry: LogEntry = {\n sessionId: this.sessionId,\n ...(this.appId ? { appId: this.appId } : {}),\n level,\n message,\n url: typeof window !== \"undefined\" ? window.location.href : undefined,\n stack: extra?.stack,\n meta: extra?.meta,\n timestamp: Date.now(),\n };\n const url = `${this.endpoint}/logs/ingest`;\n const body = JSON.stringify(entry);\n\n // Use fetch with keepalive so the request survives page navigation.\n // Errors are logged to the (original, unpatched) console so they are\n // visible in DevTools without creating an infinite log loop.\n void fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...this.authHeaders },\n body,\n keepalive: true,\n }).catch((err: unknown) => {\n // Use the original (pre-patch) error logger to avoid recursion.\n if (this.origError) {\n this.origError(\"[user-tracker] Failed to send log:\", err);\n }\n });\n }\n}\n","import type {\n TrackerConfig,\n TrackerEvent,\n SessionData,\n PageView,\n HeatmapPoint,\n GeoLocation,\n} from \"./types\";\nimport { generateSessionId, getOrCreateVisitorId } from \"./utils/session\";\nimport { fetchLocation } from \"./utils/geo\";\nimport { NavigationPlugin } from \"./core/navigation\";\nimport { TimePlugin } from \"./core/time\";\nimport { HeatmapPlugin } from \"./core/heatmap\";\nimport { LogCapture } from \"./core/logger\";\n\nexport const DEFAULT_ENDPOINT = \"https://api.alphana.ir/api/events\";\n\nconst DEFAULTS = {\n endpoint: DEFAULT_ENDPOINT,\n trackNavigation: true,\n trackTime: true,\n trackHeatmap: true,\n trackLogs: true,\n trackSnapshots: true,\n mouseSampleRate: 0.3,\n maxHeatmapPoints: 2000,\n batchSize: 20,\n flushInterval: 5_000,\n} as const;\n\ntype SubscriberFn = (event: TrackerEvent) => void;\n\n/**\n * Core tracker class. Framework-agnostic — works in any environment that has\n * a browser DOM (React, Next.js Pages Router, Vite, vanilla JS/TS, etc.).\n *\n * Usage:\n * ```ts\n * const tracker = new UserTracker({ endpoint: 'https://my-api.com/events' });\n * tracker.init(); // call once; safe to call in SSR (no-op server-side)\n * ```\n *\n * Destroy when done (e.g. component unmount):\n * ```ts\n * tracker.destroy();\n * ```\n */\ntype ResolvedConfig = Required<\n Pick<\n TrackerConfig,\n | \"endpoint\"\n | \"trackNavigation\"\n | \"trackTime\"\n | \"trackHeatmap\"\n | \"trackLogs\"\n | \"mouseSampleRate\"\n | \"maxHeatmapPoints\"\n | \"batchSize\"\n | \"flushInterval\"\n >\n> &\n TrackerConfig;\n\nexport class UserTracker {\n private readonly cfg: ResolvedConfig;\n private session: SessionData;\n private navigation?: NavigationPlugin;\n private time?: TimePlugin;\n private heatmap?: HeatmapPlugin;\n /** Public so consumers can call logCapture.capture() for manual log entries. */\n logCapture?: LogCapture;\n private initialized = false;\n private readonly subscribers = new Set<SubscriberFn>();\n\n /** In-memory queue of events waiting to be flushed. */\n private queue: TrackerEvent[] = [];\n private flushTimer: ReturnType<typeof setInterval> | null = null;\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private location: GeoLocation | null = null;\n\n constructor(config: TrackerConfig = {}) {\n this.cfg = { ...DEFAULTS, ...config } as ResolvedConfig;\n\n // Validate endpoint URL up front so the error is thrown at construction\n // time rather than silently failing during a network request.\n try {\n new URL(this.cfg.endpoint);\n } catch {\n throw new Error(\n `[alpha-tracker] Invalid endpoint URL: \"${this.cfg.endpoint}\"`,\n );\n }\n\n this.session = {\n id: config.sessionId ?? generateSessionId(),\n visitorId: getOrCreateVisitorId(),\n startedAt: Date.now(),\n pageViews: [],\n timeSpent: {},\n heatmap: {},\n };\n }\n\n // ─── Lifecycle ──────────────────────────────────────────────────────────────\n\n /**\n * Attach event listeners and start tracking.\n * Safe to call during SSR — returns `this` immediately if `window` is\n * undefined so it can be chained: `const tracker = new UserTracker(cfg).init()`.\n */\n init(): this {\n if (typeof window === \"undefined\" || this.initialized) return this;\n\n const emit = this.emit.bind(this);\n const { id: sessionId } = this.session;\n\n if (this.cfg.trackNavigation) {\n this.navigation = new NavigationPlugin({ emit, sessionId });\n this.navigation.init();\n }\n\n if (this.cfg.trackTime) {\n this.time = new TimePlugin({ emit, sessionId });\n this.time.init();\n }\n\n if (this.cfg.trackHeatmap) {\n this.heatmap = new HeatmapPlugin({\n emit,\n sessionId,\n sampleRate: this.cfg.mouseSampleRate,\n maxPoints: this.cfg.maxHeatmapPoints,\n allowedPaths: this.cfg.heatmapPages,\n });\n this.heatmap.init();\n }\n\n if (this.cfg.endpoint) {\n // Flush on a regular interval — even if the batch threshold isn't hit.\n this.flushTimer = setInterval(() => {\n if (this.queue.length > 0) void this.flush();\n }, this.cfg.flushInterval);\n\n // Resolve visitor location from IP in the background.\n void fetchLocation().then((loc) => {\n this.location = loc;\n if (loc) this.session.location = loc;\n });\n\n // Flush remaining queue when the tab is hidden or the page is unloaded.\n window.addEventListener(\"visibilitychange\", this.handleVisibilityChange);\n window.addEventListener(\"pagehide\", this.handlePageHide);\n\n // Periodic keep-alive heartbeat every 30 s so the backend knows the\n // session is still active and doesn't expire it prematurely.\n this.heartbeatTimer = setInterval(() => {\n if (document.visibilityState !== \"hidden\") void this.sendHeartbeat();\n }, 30_000);\n\n // Auto-capture console logs and unhandled errors.\n if (this.cfg.trackLogs) {\n this.logCapture = new LogCapture({\n endpoint: this.cfg.endpoint,\n sessionId: this.session.id,\n secretKey: this.cfg.secretKey,\n appId: this.cfg.appId,\n });\n this.logCapture.init();\n }\n }\n\n this.initialized = true;\n return this;\n }\n\n /** Remove all event listeners, flush remaining queue, and reset state. */\n destroy(): void {\n if (this.flushTimer !== null) {\n clearInterval(this.flushTimer);\n this.flushTimer = null;\n }\n if (this.heartbeatTimer !== null) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n\n if (typeof window !== \"undefined\") {\n window.removeEventListener(\n \"visibilitychange\",\n this.handleVisibilityChange,\n );\n window.removeEventListener(\"pagehide\", this.handlePageHide);\n }\n\n this.navigation?.destroy();\n this.time?.destroy();\n this.heatmap?.destroy();\n this.logCapture?.destroy();\n\n if (typeof window !== \"undefined\") {\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n // Best-effort flush of any remaining queued events.\n if (this.queue.length > 0 && this.cfg.endpoint) {\n this.flushBeacon();\n }\n\n this.initialized = false;\n }\n\n private handleVisibilityChange = (): void => {\n if (document.visibilityState === \"hidden\") {\n if (this.queue.length > 0) this.flushBeacon();\n this.sendDeactivate();\n }\n };\n\n private handlePageHide = (): void => {\n if (this.queue.length > 0) this.flushBeacon();\n this.sendDeactivate();\n };\n\n private handleNavigate = (_e: Event): void => {\n // Navigation hook kept for potential future use.\n };\n\n /**\n * Send a keep-alive heartbeat so the backend knows this session is still\n * active. Called every 30 s while the tab is visible.\n */\n private async sendHeartbeat(): Promise<void> {\n if (!this.cfg.endpoint) return;\n const url = `${this.cfg.endpoint}/heartbeat`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n const body = JSON.stringify({\n sessionId: this.session.id,\n visitorId: this.session.visitorId,\n path: typeof window !== \"undefined\" ? window.location.pathname : \"/\",\n active: true,\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n ...(this.location ? { location: this.location } : {}),\n });\n try {\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body,\n keepalive: true,\n });\n } catch {\n // Silent — heartbeat failure should never surface to the user.\n }\n }\n\n /**\n * Send a synchronous beacon to mark this session as inactive.\n * Called on page unload / tab hidden so the dashboard reflects real-time.\n */\n private sendDeactivate(): void {\n if (!this.cfg.endpoint) return;\n const url = `${this.cfg.endpoint}/heartbeat`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n const body = JSON.stringify({\n sessionId: this.session.id,\n path: typeof window !== \"undefined\" ? window.location.pathname : \"/\",\n active: false,\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n });\n // sendBeacon fires even if the page is being unloaded.\n if (typeof navigator !== \"undefined\" && navigator.sendBeacon) {\n navigator.sendBeacon(url, new Blob([body], { type: \"application/json\" }));\n } else {\n // Fallback for environments without sendBeacon.\n void fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body,\n keepalive: true,\n }).catch(() => undefined);\n }\n }\n\n // ─── Event pipeline ─────────────────────────────────────────────────────────\n\n /** Emit a tracker event. Also used internally by the plugins. */\n emit(event: TrackerEvent): void {\n // 1 – accumulate into session data\n switch (event.type) {\n case \"pageview\":\n this.session.pageViews.push(event.data);\n break;\n\n case \"timespent\": {\n const prev = this.session.timeSpent[event.data.path] ?? 0;\n this.session.timeSpent[event.data.path] = prev + event.data.duration;\n break;\n }\n\n case \"heatmap\": {\n const key = event.data.path;\n if (!this.session.heatmap[key]) this.session.heatmap[key] = [];\n const pts = this.session.heatmap[key];\n if (pts.length < this.cfg.maxHeatmapPoints) pts.push(event.data);\n break;\n }\n\n // rageclik and uturn are forwarded to the backend only; no local state needed.\n case \"rageclik\":\n case \"uturn\":\n break;\n }\n\n // 2 – notify subscribers (used internally by React hooks)\n this.subscribers.forEach((fn) => fn(event));\n\n // 3 – user callback\n this.cfg.onEvent?.(event);\n\n // 4 – enqueue for batched remote sending\n if (this.cfg.endpoint) {\n this.queue.push(event);\n // Auto-flush once the batch size threshold is reached.\n if (this.queue.length >= this.cfg.batchSize) {\n void this.flush();\n }\n }\n }\n\n /**\n * Subscribe to every emitted event. Returns an unsubscribe function.\n *\n * ```ts\n * const unsub = tracker.subscribe(event => console.log(event));\n * // later…\n * unsub();\n * ```\n */\n subscribe(fn: SubscriberFn): () => void {\n this.subscribers.add(fn);\n return () => this.subscribers.delete(fn);\n }\n\n // ─── Manual tracking helpers ────────────────────────────────────────────────\n\n /**\n * Manually record a page view and dispatch `tracker:navigate`.\n * Required for Next.js App Router — call it inside a `useEffect` that\n * depends on `usePathname()`:\n *\n * ```tsx\n * const pathname = usePathname();\n * useEffect(() => { tracker.trackPageView(pathname); }, [pathname]);\n * ```\n */\n trackPageView(path?: string): void {\n const resolvedPath =\n path ??\n (typeof window !== \"undefined\"\n ? window.location.pathname + window.location.search\n : \"/\");\n\n this.emit({\n type: \"pageview\",\n data: {\n path: resolvedPath,\n title: typeof document !== \"undefined\" ? document.title : \"\",\n timestamp: Date.now(),\n sessionId: this.session.id,\n referrer:\n typeof document !== \"undefined\"\n ? document.referrer || undefined\n : undefined,\n },\n });\n\n if (typeof window !== \"undefined\") {\n window.dispatchEvent(\n new CustomEvent(\"tracker:navigate\", {\n detail: { path: resolvedPath, title: document.title },\n }),\n );\n }\n }\n\n // ─── Data accessors ─────────────────────────────────────────────────────────\n\n /** A read-only snapshot of the current session. */\n getSession(): Readonly<SessionData> {\n return this.session;\n }\n\n /** All page views recorded so far. */\n getPageViews(): PageView[] {\n return [...this.session.pageViews];\n }\n\n /** Cumulative milliseconds spent per path. */\n getTimeSpent(): Record<string, number> {\n return { ...this.session.timeSpent };\n }\n\n /** Heatmap points for a specific path. */\n getHeatmapData(path: string): HeatmapPoint[];\n /** Heatmap points for all tracked paths. */\n getHeatmapData(): Record<string, HeatmapPoint[]>;\n getHeatmapData(\n path?: string,\n ): HeatmapPoint[] | Record<string, HeatmapPoint[]> {\n if (path !== undefined) {\n return [...(this.session.heatmap[path] ?? [])];\n }\n return Object.entries(this.session.heatmap).reduce<\n Record<string, HeatmapPoint[]>\n >((acc, [k, v]) => {\n acc[k] = [...v];\n return acc;\n }, {});\n }\n\n // ─── Network ────────────────────────────────────────────────────────────────\n\n /** Drain the queue and POST all pending events to the batch endpoint. */\n private async flush(): Promise<void> {\n if (this.queue.length === 0) return;\n // Splice atomically so new events emitted during the async request don't\n // get lost — they stay in the queue for the next flush.\n const batch = this.queue.splice(0);\n await this.sendBatch(batch);\n }\n\n /**\n * Synchronous best-effort flush via `navigator.sendBeacon`.\n * Used on `pagehide` / `visibilitychange:hidden` where async fetch may be\n * cancelled by the browser before it completes.\n */\n private flushBeacon(): void {\n if (this.queue.length === 0) return;\n const batch = this.queue.splice(0);\n const url = `${this.cfg.endpoint!}/batch`;\n const blob = new Blob([this.buildBatchBody(batch)], {\n type: \"application/json\",\n });\n if (typeof navigator !== \"undefined\" && navigator.sendBeacon) {\n navigator.sendBeacon(url, blob);\n } else {\n // Fallback: fire-and-forget fetch (best effort on platforms without sendBeacon)\n void this.sendBatch(batch);\n }\n }\n\n private buildBatchBody(events: TrackerEvent[]): string {\n return JSON.stringify({\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n visitorId: this.session.visitorId,\n location: this.location ?? undefined,\n events: events.map((e) => ({\n sessionId: this.session.id,\n type: e.type,\n data: e.data,\n })),\n });\n }\n\n private async sendBatch(events: TrackerEvent[]): Promise<void> {\n const url = `${this.cfg.endpoint!}/batch`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n try {\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body: this.buildBatchBody(events),\n keepalive: true,\n });\n } catch {\n // Intentionally silent — analytics must never surface errors to users.\n }\n }\n}\n","import 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/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","allowedPaths","pageWidth","pageHeight","absY","target","touch","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_ENDPOINT","DEFAULTS","UserTracker","config","_e","loc","authHeaders","event","prev","key","pts","resolvedPath","acc","k","v","batch","blob","events","buildPalette","stops","palette","stepsPerSegment","s","fr","fg","fb","tr","tg","tb","i","t","COLOR_PALETTE","renderHeatmap","canvas","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 * If provided, heatmap data will only be collected for paths in this list.\n * An empty array or undefined means all pages are tracked.\n */\n allowedPaths?: string[];\n}\n\n/**\n * Collects mouse-move, click, and scroll positions for heatmap analysis.\n *\n * Also detects rage clicks (≥3 clicks within 1 s in a 50 px radius) and emits\n * a `rageclik` event so the backend can surface problematic UI hotspots.\n *\n * Click events include an optional `target` field containing a human-readable\n * label for the clicked element (aria-label › id › button/link text).\n */\nexport class HeatmapPlugin {\n private readonly emit: EmitFn;\n private readonly sessionId: string;\n private readonly sampleRate: number;\n private readonly maxPoints: number;\n private readonly allowedPaths: Set<string> | null;\n private currentPath = \"\";\n private pointCounts: Record<string, number> = {};\n\n // Rage-click detection state\n private recentClicks: { x: number; y: number; t: number }[] = [];\n private static readonly RAGE_THRESHOLD = 3;\n private static readonly RAGE_WINDOW_MS = 1_000;\n private static readonly RAGE_RADIUS_PX = 50;\n\n private readonly throttledMouseMove: (e: MouseEvent) => void;\n private readonly throttledScroll: () => void;\n\n constructor({\n emit,\n sessionId,\n sampleRate = 0.3,\n maxPoints = 2000,\n allowedPaths,\n }: HeatmapPluginOptions) {\n this.emit = emit;\n this.sessionId = sessionId;\n this.sampleRate = sampleRate;\n this.maxPoints = maxPoints;\n this.allowedPaths =\n allowedPaths && allowedPaths.length > 0 ? new Set(allowedPaths) : null;\n\n this.throttledMouseMove = throttle(this.handleMouseMove, 50);\n this.throttledScroll = throttle(this.handleScroll, 100);\n }\n\n init(): void {\n this.currentPath = window.location.pathname + window.location.search;\n\n document.addEventListener(\"mousemove\", this.throttledMouseMove);\n document.addEventListener(\"click\", this.handleClick);\n document.addEventListener(\"touchend\", this.handleTouchEnd, {\n passive: true,\n });\n window.addEventListener(\"scroll\", this.throttledScroll, { passive: true });\n window.addEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n destroy(): void {\n document.removeEventListener(\"mousemove\", this.throttledMouseMove);\n document.removeEventListener(\"click\", this.handleClick);\n document.removeEventListener(\"touchend\", this.handleTouchEnd);\n window.removeEventListener(\"scroll\", this.throttledScroll);\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n private canRecord(): boolean {\n if (\n this.allowedPaths !== null &&\n !this.allowedPaths.has(this.currentPath)\n ) {\n return false;\n }\n return (this.pointCounts[this.currentPath] ?? 0) < this.maxPoints;\n }\n\n private recordPoint(\n point: Omit<HeatmapPoint, \"path\" | \"timestamp\" | \"sessionId\">,\n ): void {\n if (!this.canRecord()) return;\n this.pointCounts[this.currentPath] =\n (this.pointCounts[this.currentPath] ?? 0) + 1;\n this.emit({\n type: \"heatmap\",\n data: {\n ...point,\n path: this.currentPath,\n timestamp: Date.now(),\n },\n });\n }\n\n /** Extract a human-readable label for the clicked element. */\n private getClickTarget(e: MouseEvent): string | undefined {\n const el = e.target as HTMLElement | null;\n if (!el) return undefined;\n const label =\n el.getAttribute(\"aria-label\") ||\n el.closest(\"[aria-label]\")?.getAttribute(\"aria-label\") ||\n el.getAttribute(\"data-track-label\") ||\n el.getAttribute(\"id\") ||\n (el instanceof HTMLButtonElement || el instanceof HTMLAnchorElement\n ? el.innerText?.trim().slice(0, 60)\n : el.closest(\"button, a\")?.textContent?.trim().slice(0, 60));\n return label || undefined;\n }\n\n /** Detect rage-click bursts and emit a `rageclik` event when found. */\n private checkRageClick(x: number, y: number, path: string): void {\n const now = Date.now();\n this.recentClicks = this.recentClicks.filter(\n (c) => now - c.t < HeatmapPlugin.RAGE_WINDOW_MS,\n );\n this.recentClicks.push({ x, y, t: now });\n\n const nearby = this.recentClicks.filter(\n (c) => Math.hypot(c.x - x, c.y - y) <= HeatmapPlugin.RAGE_RADIUS_PX,\n );\n\n if (nearby.length >= HeatmapPlugin.RAGE_THRESHOLD) {\n this.emit({\n type: \"rageclik\",\n data: {\n path,\n x,\n y,\n count: nearby.length,\n timestamp: now,\n sessionId: this.sessionId,\n },\n });\n // Reset to avoid continuously re-triggering in the same burst.\n this.recentClicks = [];\n }\n }\n\n private handleMouseMove = (e: MouseEvent): void => {\n if (Math.random() > this.sampleRate) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = e.clientY + window.scrollY;\n this.recordPoint({\n x: e.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (e.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"move\",\n });\n };\n\n private handleClick = (e: MouseEvent): void => {\n // Skip if a touchend already recorded this tap (within 500 ms)\n if (Date.now() - this.lastTouchTime < 500) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = e.clientY + window.scrollY;\n const target = this.getClickTarget(e);\n this.recordPoint({\n x: e.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (e.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"click\",\n ...(target ? { target } : {}),\n });\n this.checkRageClick(e.clientX, absY, this.currentPath);\n };\n\n /**\n * Records taps on touch devices.\n * Mobile browsers do fire a synthetic `click` after `touchend`, but some\n * frameworks call `preventDefault()` on touch events which suppresses it.\n * Listening to `touchend` directly ensures taps are always captured.\n * We mark the point with a short-lived flag so the subsequent synthetic\n * `click` event (if it fires) is deduplicated.\n */\n private lastTouchTime = 0;\n\n private handleTouchEnd = (e: TouchEvent): void => {\n const touch = e.changedTouches[0];\n if (!touch) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const absY = touch.clientY + window.scrollY;\n this.lastTouchTime = Date.now();\n this.recordPoint({\n x: touch.clientX,\n y: absY,\n xPct: pageWidth > 0 ? (touch.clientX / pageWidth) * 100 : 0,\n yPct: pageHeight > 0 ? (absY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"click\",\n });\n this.checkRageClick(touch.clientX, absY, this.currentPath);\n };\n\n private handleScroll = (): void => {\n if (Math.random() > this.sampleRate) return;\n const pageWidth = document.documentElement.scrollWidth;\n const pageHeight = document.documentElement.scrollHeight;\n const vw = window.innerWidth;\n const scrollX = window.scrollX;\n const scrollY = window.scrollY;\n const absX = scrollX + vw / 2;\n this.recordPoint({\n x: vw / 2,\n y: scrollY,\n xPct: pageWidth > 0 ? (absX / pageWidth) * 100 : 50,\n yPct: pageHeight > 0 ? (scrollY / pageHeight) * 100 : 0,\n pw: pageWidth,\n ph: pageHeight,\n type: \"scroll\",\n });\n };\n\n private handleNavigate = (e: CustomEvent<{ path: string }>): void => {\n this.currentPath = e.detail.path;\n };\n}\n","export type LogLevel = \"info\" | \"warn\" | \"error\";\n\nexport interface LogEntry {\n sessionId?: string;\n appId?: string;\n level: LogLevel;\n message: string;\n url?: string;\n stack?: string;\n meta?: Record<string, unknown>;\n timestamp: number;\n}\n\ntype ConsoleFn = (...args: unknown[]) => void;\n\n/**\n * Automatically captures console.info/warn/error output and unhandled errors,\n * then ships them to the backend `/logs/ingest` endpoint.\n *\n * All console methods are restored exactly in `destroy()`.\n */\nexport class LogCapture {\n private readonly endpoint: string;\n private readonly sessionId: string;\n private readonly appId?: string;\n private readonly authHeaders: Record<string, string>;\n\n // Original console methods preserved so we can restore them.\n private origInfo!: ConsoleFn;\n private origWarn!: ConsoleFn;\n private origError!: ConsoleFn;\n\n private prevOnError: OnErrorEventHandler = null;\n private prevOnUnhandledRejection:\n | ((e: PromiseRejectionEvent) => void)\n | null = null;\n\n private initialized = false;\n\n constructor(options: {\n endpoint: string;\n sessionId: string;\n secretKey?: string;\n appId?: string;\n }) {\n // Derive the API base URL by stripping everything from the last path\n // segment that isn't a versioning prefix. The tracker config `endpoint`\n // is the *events* URL (e.g. http://host/api/events), but logs live at\n // http://host/api/logs/ingest, so we walk up until we reach the common\n // base (i.e. remove the final segment).\n try {\n const u = new URL(options.endpoint);\n // Remove the last non-empty path segment (e.g. \"/api/events\" → \"/api\")\n const parts = u.pathname.replace(/\\/$/, \"\").split(\"/\");\n parts.pop();\n u.pathname = parts.join(\"/\") || \"/\";\n this.endpoint = u.toString().replace(/\\/$/, \"\");\n } catch {\n this.endpoint = options.endpoint;\n }\n this.sessionId = options.sessionId;\n this.appId = options.appId;\n this.authHeaders = options.secretKey\n ? { Authorization: `Bearer ${options.secretKey}` }\n : {};\n }\n\n init(): void {\n if (typeof window === \"undefined\" || this.initialized) return;\n\n this.origInfo = console.info.bind(console);\n this.origWarn = console.warn.bind(console);\n this.origError = console.error.bind(console);\n\n console.info = (...args: unknown[]) => {\n this.origInfo(...args);\n this.send(\"info\", this.format(args));\n };\n\n console.warn = (...args: unknown[]) => {\n this.origWarn(...args);\n this.send(\"warn\", this.format(args));\n };\n\n console.error = (...args: unknown[]) => {\n this.origError(...args);\n const [first] = args;\n const stack = first instanceof Error ? first.stack : undefined;\n this.send(\"error\", this.format(args), { stack });\n };\n\n this.prevOnError = window.onerror;\n window.onerror = (msg, src, line, col, err) => {\n this.send(\"error\", String(msg), {\n stack: err?.stack,\n meta: { src, line, col },\n });\n if (typeof this.prevOnError === \"function\") {\n return this.prevOnError(msg, src, line, col, err);\n }\n return false;\n };\n\n this.prevOnUnhandledRejection = (e: PromiseRejectionEvent) => {\n const reason = e.reason;\n const message =\n reason instanceof Error\n ? reason.message\n : String(reason ?? \"Unhandled promise rejection\");\n this.send(\"error\", message, {\n stack: reason instanceof Error ? reason.stack : undefined,\n });\n };\n window.addEventListener(\n \"unhandledrejection\",\n this.prevOnUnhandledRejection,\n );\n\n this.initialized = true;\n }\n\n destroy(): void {\n if (!this.initialized) return;\n console.info = this.origInfo;\n console.warn = this.origWarn;\n console.error = this.origError;\n\n window.onerror = this.prevOnError;\n if (this.prevOnUnhandledRejection) {\n window.removeEventListener(\n \"unhandledrejection\",\n this.prevOnUnhandledRejection,\n );\n }\n this.initialized = false;\n }\n\n /** Manually capture a log entry (e.g. from try/catch). */\n capture(\n level: LogLevel,\n message: string,\n extra?: { stack?: string; meta?: Record<string, unknown> },\n ): void {\n this.send(level, message, extra);\n }\n\n private format(args: unknown[]): string {\n return args\n .map((a) => {\n if (a instanceof Error) return a.message;\n if (typeof a === \"object\") {\n try {\n return JSON.stringify(a);\n } catch {\n return String(a);\n }\n }\n return String(a);\n })\n .join(\" \");\n }\n\n private send(\n level: LogLevel,\n message: string,\n extra?: { stack?: string; meta?: Record<string, unknown> },\n ): void {\n const entry: LogEntry = {\n sessionId: this.sessionId,\n ...(this.appId ? { appId: this.appId } : {}),\n level,\n message,\n url: typeof window !== \"undefined\" ? window.location.href : undefined,\n stack: extra?.stack,\n meta: extra?.meta,\n timestamp: Date.now(),\n };\n const url = `${this.endpoint}/logs/ingest`;\n const body = JSON.stringify(entry);\n\n // Use fetch with keepalive so the request survives page navigation.\n // Errors are logged to the (original, unpatched) console so they are\n // visible in DevTools without creating an infinite log loop.\n void fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...this.authHeaders },\n body,\n keepalive: true,\n }).catch((err: unknown) => {\n // Use the original (pre-patch) error logger to avoid recursion.\n if (this.origError) {\n this.origError(\"[user-tracker] Failed to send log:\", err);\n }\n });\n }\n}\n","import type {\n TrackerConfig,\n TrackerEvent,\n SessionData,\n PageView,\n HeatmapPoint,\n GeoLocation,\n} from \"./types\";\nimport { generateSessionId, getOrCreateVisitorId } from \"./utils/session\";\nimport { fetchLocation } from \"./utils/geo\";\nimport { NavigationPlugin } from \"./core/navigation\";\nimport { TimePlugin } from \"./core/time\";\nimport { HeatmapPlugin } from \"./core/heatmap\";\nimport { LogCapture } from \"./core/logger\";\n\nexport const DEFAULT_ENDPOINT = \"https://api.alphana.ir/api/events\";\n\nconst DEFAULTS = {\n endpoint: DEFAULT_ENDPOINT,\n trackNavigation: true,\n trackTime: true,\n trackHeatmap: true,\n trackLogs: true,\n mouseSampleRate: 0.3,\n maxHeatmapPoints: 2000,\n batchSize: 20,\n flushInterval: 5_000,\n} as const;\n\ntype SubscriberFn = (event: TrackerEvent) => void;\n\n/**\n * Core tracker class. Framework-agnostic — works in any environment that has\n * a browser DOM (React, Next.js Pages Router, Vite, vanilla JS/TS, etc.).\n *\n * Usage:\n * ```ts\n * const tracker = new UserTracker({ endpoint: 'https://my-api.com/events' });\n * tracker.init(); // call once; safe to call in SSR (no-op server-side)\n * ```\n *\n * Destroy when done (e.g. component unmount):\n * ```ts\n * tracker.destroy();\n * ```\n */\ntype ResolvedConfig = Required<\n Pick<\n TrackerConfig,\n | \"endpoint\"\n | \"trackNavigation\"\n | \"trackTime\"\n | \"trackHeatmap\"\n | \"trackLogs\"\n | \"mouseSampleRate\"\n | \"maxHeatmapPoints\"\n | \"batchSize\"\n | \"flushInterval\"\n >\n> &\n TrackerConfig;\n\nexport class UserTracker {\n private readonly cfg: ResolvedConfig;\n private session: SessionData;\n private navigation?: NavigationPlugin;\n private time?: TimePlugin;\n private heatmap?: HeatmapPlugin;\n /** Public so consumers can call logCapture.capture() for manual log entries. */\n logCapture?: LogCapture;\n private initialized = false;\n private readonly subscribers = new Set<SubscriberFn>();\n\n /** In-memory queue of events waiting to be flushed. */\n private queue: TrackerEvent[] = [];\n private flushTimer: ReturnType<typeof setInterval> | null = null;\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private location: GeoLocation | null = null;\n\n constructor(config: TrackerConfig = {}) {\n this.cfg = { ...DEFAULTS, ...config } as ResolvedConfig;\n\n // Validate endpoint URL up front so the error is thrown at construction\n // time rather than silently failing during a network request.\n try {\n new URL(this.cfg.endpoint);\n } catch {\n throw new Error(\n `[alpha-tracker] Invalid endpoint URL: \"${this.cfg.endpoint}\"`,\n );\n }\n\n this.session = {\n id: config.sessionId ?? generateSessionId(),\n visitorId: getOrCreateVisitorId(),\n startedAt: Date.now(),\n pageViews: [],\n timeSpent: {},\n heatmap: {},\n };\n }\n\n // ─── Lifecycle ──────────────────────────────────────────────────────────────\n\n /**\n * Attach event listeners and start tracking.\n * Safe to call during SSR — returns `this` immediately if `window` is\n * undefined so it can be chained: `const tracker = new UserTracker(cfg).init()`.\n */\n init(): this {\n if (typeof window === \"undefined\" || this.initialized) return this;\n\n const emit = this.emit.bind(this);\n const { id: sessionId } = this.session;\n\n if (this.cfg.trackNavigation) {\n this.navigation = new NavigationPlugin({ emit, sessionId });\n this.navigation.init();\n }\n\n if (this.cfg.trackTime) {\n this.time = new TimePlugin({ emit, sessionId });\n this.time.init();\n }\n\n if (this.cfg.trackHeatmap) {\n this.heatmap = new HeatmapPlugin({\n emit,\n sessionId,\n sampleRate: this.cfg.mouseSampleRate,\n maxPoints: this.cfg.maxHeatmapPoints,\n allowedPaths: this.cfg.heatmapPages,\n });\n this.heatmap.init();\n }\n\n if (this.cfg.endpoint) {\n // Flush on a regular interval — even if the batch threshold isn't hit.\n this.flushTimer = setInterval(() => {\n if (this.queue.length > 0) void this.flush();\n }, this.cfg.flushInterval);\n\n // Resolve visitor location from IP in the background.\n void fetchLocation().then((loc) => {\n this.location = loc;\n if (loc) this.session.location = loc;\n });\n\n // Flush remaining queue when the tab is hidden or the page is unloaded.\n window.addEventListener(\"visibilitychange\", this.handleVisibilityChange);\n window.addEventListener(\"pagehide\", this.handlePageHide);\n\n // Periodic keep-alive heartbeat every 30 s so the backend knows the\n // session is still active and doesn't expire it prematurely.\n this.heartbeatTimer = setInterval(() => {\n if (document.visibilityState !== \"hidden\") void this.sendHeartbeat();\n }, 30_000);\n\n // Auto-capture console logs and unhandled errors.\n if (this.cfg.trackLogs) {\n this.logCapture = new LogCapture({\n endpoint: this.cfg.endpoint,\n sessionId: this.session.id,\n secretKey: this.cfg.secretKey,\n appId: this.cfg.appId,\n });\n this.logCapture.init();\n }\n }\n\n this.initialized = true;\n return this;\n }\n\n /** Remove all event listeners, flush remaining queue, and reset state. */\n destroy(): void {\n if (this.flushTimer !== null) {\n clearInterval(this.flushTimer);\n this.flushTimer = null;\n }\n if (this.heartbeatTimer !== null) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n\n if (typeof window !== \"undefined\") {\n window.removeEventListener(\n \"visibilitychange\",\n this.handleVisibilityChange,\n );\n window.removeEventListener(\"pagehide\", this.handlePageHide);\n }\n\n this.navigation?.destroy();\n this.time?.destroy();\n this.heatmap?.destroy();\n this.logCapture?.destroy();\n\n if (typeof window !== \"undefined\") {\n window.removeEventListener(\"tracker:navigate\", this.handleNavigate);\n }\n\n // Best-effort flush of any remaining queued events.\n if (this.queue.length > 0 && this.cfg.endpoint) {\n this.flushBeacon();\n }\n\n this.initialized = false;\n }\n\n private handleVisibilityChange = (): void => {\n if (document.visibilityState === \"hidden\") {\n if (this.queue.length > 0) this.flushBeacon();\n this.sendDeactivate();\n }\n };\n\n private handlePageHide = (): void => {\n if (this.queue.length > 0) this.flushBeacon();\n this.sendDeactivate();\n };\n\n private handleNavigate = (_e: Event): void => {\n // Navigation hook kept for potential future use.\n };\n\n /**\n * Send a keep-alive heartbeat so the backend knows this session is still\n * active. Called every 30 s while the tab is visible.\n */\n private async sendHeartbeat(): Promise<void> {\n if (!this.cfg.endpoint) return;\n const url = `${this.cfg.endpoint}/heartbeat`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n const body = JSON.stringify({\n sessionId: this.session.id,\n visitorId: this.session.visitorId,\n path: typeof window !== \"undefined\" ? window.location.pathname : \"/\",\n active: true,\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n ...(this.location ? { location: this.location } : {}),\n });\n try {\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body,\n keepalive: true,\n });\n } catch {\n // Silent — heartbeat failure should never surface to the user.\n }\n }\n\n /**\n * Send a synchronous beacon to mark this session as inactive.\n * Called on page unload / tab hidden so the dashboard reflects real-time.\n */\n private sendDeactivate(): void {\n if (!this.cfg.endpoint) return;\n const url = `${this.cfg.endpoint}/heartbeat`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n const body = JSON.stringify({\n sessionId: this.session.id,\n path: typeof window !== \"undefined\" ? window.location.pathname : \"/\",\n active: false,\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n });\n // sendBeacon fires even if the page is being unloaded.\n if (typeof navigator !== \"undefined\" && navigator.sendBeacon) {\n navigator.sendBeacon(url, new Blob([body], { type: \"application/json\" }));\n } else {\n // Fallback for environments without sendBeacon.\n void fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body,\n keepalive: true,\n }).catch(() => undefined);\n }\n }\n\n // ─── Event pipeline ─────────────────────────────────────────────────────────\n\n /** Emit a tracker event. Also used internally by the plugins. */\n emit(event: TrackerEvent): void {\n // 1 – accumulate into session data\n switch (event.type) {\n case \"pageview\":\n this.session.pageViews.push(event.data);\n break;\n\n case \"timespent\": {\n const prev = this.session.timeSpent[event.data.path] ?? 0;\n this.session.timeSpent[event.data.path] = prev + event.data.duration;\n break;\n }\n\n case \"heatmap\": {\n const key = event.data.path;\n if (!this.session.heatmap[key]) this.session.heatmap[key] = [];\n const pts = this.session.heatmap[key];\n if (pts.length < this.cfg.maxHeatmapPoints) pts.push(event.data);\n break;\n }\n\n // rageclik and uturn are forwarded to the backend only; no local state needed.\n case \"rageclik\":\n case \"uturn\":\n break;\n }\n\n // 2 – notify subscribers (used internally by React hooks)\n this.subscribers.forEach((fn) => fn(event));\n\n // 3 – user callback\n this.cfg.onEvent?.(event);\n\n // 4 – enqueue for batched remote sending\n if (this.cfg.endpoint) {\n this.queue.push(event);\n // Auto-flush once the batch size threshold is reached.\n if (this.queue.length >= this.cfg.batchSize) {\n void this.flush();\n }\n }\n }\n\n /**\n * Subscribe to every emitted event. Returns an unsubscribe function.\n *\n * ```ts\n * const unsub = tracker.subscribe(event => console.log(event));\n * // later…\n * unsub();\n * ```\n */\n subscribe(fn: SubscriberFn): () => void {\n this.subscribers.add(fn);\n return () => this.subscribers.delete(fn);\n }\n\n // ─── Manual tracking helpers ────────────────────────────────────────────────\n\n /**\n * Manually record a page view and dispatch `tracker:navigate`.\n * Required for Next.js App Router — call it inside a `useEffect` that\n * depends on `usePathname()`:\n *\n * ```tsx\n * const pathname = usePathname();\n * useEffect(() => { tracker.trackPageView(pathname); }, [pathname]);\n * ```\n */\n trackPageView(path?: string): void {\n const resolvedPath =\n path ??\n (typeof window !== \"undefined\"\n ? window.location.pathname + window.location.search\n : \"/\");\n\n this.emit({\n type: \"pageview\",\n data: {\n path: resolvedPath,\n title: typeof document !== \"undefined\" ? document.title : \"\",\n timestamp: Date.now(),\n sessionId: this.session.id,\n referrer:\n typeof document !== \"undefined\"\n ? document.referrer || undefined\n : undefined,\n },\n });\n\n if (typeof window !== \"undefined\") {\n window.dispatchEvent(\n new CustomEvent(\"tracker:navigate\", {\n detail: { path: resolvedPath, title: document.title },\n }),\n );\n }\n }\n\n // ─── Data accessors ─────────────────────────────────────────────────────────\n\n /** A read-only snapshot of the current session. */\n getSession(): Readonly<SessionData> {\n return this.session;\n }\n\n /** All page views recorded so far. */\n getPageViews(): PageView[] {\n return [...this.session.pageViews];\n }\n\n /** Cumulative milliseconds spent per path. */\n getTimeSpent(): Record<string, number> {\n return { ...this.session.timeSpent };\n }\n\n /** Heatmap points for a specific path. */\n getHeatmapData(path: string): HeatmapPoint[];\n /** Heatmap points for all tracked paths. */\n getHeatmapData(): Record<string, HeatmapPoint[]>;\n getHeatmapData(\n path?: string,\n ): HeatmapPoint[] | Record<string, HeatmapPoint[]> {\n if (path !== undefined) {\n return [...(this.session.heatmap[path] ?? [])];\n }\n return Object.entries(this.session.heatmap).reduce<\n Record<string, HeatmapPoint[]>\n >((acc, [k, v]) => {\n acc[k] = [...v];\n return acc;\n }, {});\n }\n\n // ─── Network ────────────────────────────────────────────────────────────────\n\n /** Drain the queue and POST all pending events to the batch endpoint. */\n private async flush(): Promise<void> {\n if (this.queue.length === 0) return;\n // Splice atomically so new events emitted during the async request don't\n // get lost — they stay in the queue for the next flush.\n const batch = this.queue.splice(0);\n await this.sendBatch(batch);\n }\n\n /**\n * Synchronous best-effort flush via `navigator.sendBeacon`.\n * Used on `pagehide` / `visibilitychange:hidden` where async fetch may be\n * cancelled by the browser before it completes.\n */\n private flushBeacon(): void {\n if (this.queue.length === 0) return;\n const batch = this.queue.splice(0);\n const url = `${this.cfg.endpoint!}/batch`;\n const blob = new Blob([this.buildBatchBody(batch)], {\n type: \"application/json\",\n });\n if (typeof navigator !== \"undefined\" && navigator.sendBeacon) {\n navigator.sendBeacon(url, blob);\n } else {\n // Fallback: fire-and-forget fetch (best effort on platforms without sendBeacon)\n void this.sendBatch(batch);\n }\n }\n\n private buildBatchBody(events: TrackerEvent[]): string {\n return JSON.stringify({\n ...(this.cfg.appId ? { appId: this.cfg.appId } : {}),\n visitorId: this.session.visitorId,\n location: this.location ?? undefined,\n events: events.map((e) => ({\n sessionId: this.session.id,\n type: e.type,\n data: e.data,\n })),\n });\n }\n\n private async sendBatch(events: TrackerEvent[]): Promise<void> {\n const url = `${this.cfg.endpoint!}/batch`;\n const authHeaders: Record<string, string> = this.cfg.secretKey\n ? { Authorization: `Bearer ${this.cfg.secretKey}` }\n : {};\n try {\n await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\", ...authHeaders },\n body: this.buildBatchBody(events),\n keepalive: true,\n });\n } catch {\n // Intentionally silent — analytics must never surface errors to users.\n }\n }\n}\n","import 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"]}