@prosdevlab/experience-sdk 0.1.4 → 0.2.0

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.
@@ -110,6 +110,59 @@ interface Context {
110
110
  timestamp?: number;
111
111
  /** Custom context properties */
112
112
  custom?: Record<string, any>;
113
+ /** Trigger state (for display condition plugins) */
114
+ triggers?: TriggerState;
115
+ }
116
+ /**
117
+ * Trigger State
118
+ *
119
+ * Tracks which trigger-based display conditions have fired.
120
+ * Used by plugins like exitIntent, scrollDepth, pageVisits, timeDelay.
121
+ */
122
+ interface TriggerState {
123
+ /** Exit intent trigger state */
124
+ exitIntent?: {
125
+ /** Whether the trigger has fired */
126
+ triggered: boolean;
127
+ /** When the trigger fired (unix timestamp) */
128
+ timestamp?: number;
129
+ /** Additional trigger-specific data */
130
+ lastY?: number;
131
+ previousY?: number;
132
+ velocity?: number;
133
+ timeOnPage?: number;
134
+ };
135
+ /** Scroll depth trigger state */
136
+ scrollDepth?: {
137
+ triggered: boolean;
138
+ timestamp?: number;
139
+ /** Current scroll percentage (0-100) */
140
+ percent?: number;
141
+ };
142
+ /** Page visits trigger state */
143
+ pageVisits?: {
144
+ triggered: boolean;
145
+ timestamp?: number;
146
+ /** Total visit count */
147
+ count?: number;
148
+ /** Whether this is the first visit */
149
+ firstVisit?: boolean;
150
+ };
151
+ /** Time delay trigger state */
152
+ timeDelay?: {
153
+ triggered: boolean;
154
+ timestamp?: number;
155
+ /** Total elapsed time (ms, includes paused time) */
156
+ elapsed?: number;
157
+ /** Active elapsed time (ms, excludes paused time) */
158
+ activeElapsed?: number;
159
+ /** Whether timer was paused */
160
+ wasPaused?: boolean;
161
+ /** Number of visibility changes */
162
+ visibilityChanges?: number;
163
+ };
164
+ /** Extensible for future triggers */
165
+ [key: string]: any;
113
166
  }
114
167
  /**
115
168
  * User Context
@@ -223,7 +276,13 @@ declare class ExperienceRuntime {
223
276
  private decisions;
224
277
  private initialized;
225
278
  private destroyed;
279
+ private triggerContext;
226
280
  constructor(config?: ExperienceConfig);
281
+ /**
282
+ * Setup listeners for trigger:* events
283
+ * This enables event-driven display conditions
284
+ */
285
+ private setupTriggerListeners;
227
286
  /**
228
287
  * Initialize the runtime
229
288
  */
@@ -1,19 +1,15 @@
1
- var experiences=(function(exports){'use strict';var q=class{storage=new Map;get(e){return this.storage.get(e)??null}set(e,t,n){this.storage.set(e,t);}remove(e){this.storage.delete(e);}clear(){this.storage.clear();}isSupported(){return true}},X=class{fallback=null;defaultOptions;constructor(e={}){this.defaultOptions={path:"/",sameSite:"lax",...e};}get(e){if(!this.isSupported())return this.getFallback().get(e);try{let t=`${encodeURIComponent(e)}=`,n=document.cookie.split(";");for(let r of n)if(r=r.trim(),r.startsWith(t))return decodeURIComponent(r.substring(t.length));return null}catch(t){return console.warn("Cookie get failed:",t),this.getFallback().get(e)}}set(e,t,n){if(!this.isSupported()){this.getFallback().set(e,t);return}try{let r={...this.defaultOptions,...n},i=[`${encodeURIComponent(e)}=${encodeURIComponent(t)}`];if(r.ttl){let m=new Date;m.setTime(m.getTime()+r.ttl*1e3),i.push(`expires=${m.toUTCString()}`);}r.path&&i.push(`path=${r.path}`),r.domain&&i.push(`domain=${r.domain}`),r.secure&&i.push("secure"),r.sameSite&&i.push(`samesite=${r.sameSite}`),document.cookie=i.join("; ");}catch(r){console.warn("Cookie set failed, falling back to memory:",r),this.getFallback().set(e,t);}}remove(e){if(!this.isSupported()){this.getFallback().remove(e);return}try{let t=this.defaultOptions,n=[`${encodeURIComponent(e)}=`,"expires=Thu, 01 Jan 1970 00:00:00 UTC"];t.path&&n.push(`path=${t.path}`),t.domain&&n.push(`domain=${t.domain}`),document.cookie=n.join("; ");}catch(t){console.warn("Cookie remove failed:",t),this.getFallback().remove(e);}}clear(){if(!this.isSupported()){this.getFallback().clear();return}try{let e=document.cookie.split(";");for(let t of e){t=t.trim();let n=t.indexOf("="),r=n>-1?t.substring(0,n):t;this.remove(decodeURIComponent(r));}}catch(e){console.warn("Cookie clear failed:",e),this.getFallback().clear();}}isSupported(){try{if(typeof document>"u"||!document.cookie)return !1;let e="__cookie_test__";document.cookie=`${e}=test; path=/`;let t=document.cookie.indexOf(e)!==-1;return document.cookie=`${e}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/`,t}catch{return false}}getFallback(){return this.fallback||(this.fallback=new q),this.fallback}},Y=class{fallback=null;get(e){if(!this.isSupported())return this.getFallback().get(e);try{return localStorage.getItem(e)}catch(t){return console.warn("localStorage.getItem failed:",t),this.getFallback().get(e)}}set(e,t,n){if(!this.isSupported()){this.getFallback().set(e,t);return}try{localStorage.setItem(e,t);}catch(r){console.warn("localStorage.setItem failed, falling back to memory:",r),this.getFallback().set(e,t);}}remove(e){if(!this.isSupported()){this.getFallback().remove(e);return}try{localStorage.removeItem(e);}catch(t){console.warn("localStorage.removeItem failed:",t),this.getFallback().remove(e);}}clear(){if(!this.isSupported()){this.getFallback().clear();return}try{localStorage.clear();}catch(e){console.warn("localStorage.clear failed:",e),this.getFallback().clear();}}isSupported(){try{let e="__storage_test__";return localStorage.setItem(e,"test"),localStorage.removeItem(e),!0}catch{return false}}getFallback(){return this.fallback||(this.fallback=new q),this.fallback}},Z=class{fallback=null;get(e){if(!this.isSupported())return this.getFallback().get(e);try{return sessionStorage.getItem(e)}catch(t){return console.warn("sessionStorage.getItem failed:",t),this.getFallback().get(e)}}set(e,t,n){if(!this.isSupported()){this.getFallback().set(e,t);return}try{sessionStorage.setItem(e,t);}catch(r){console.warn("sessionStorage.setItem failed, falling back to memory:",r),this.getFallback().set(e,t);}}remove(e){if(!this.isSupported()){this.getFallback().remove(e);return}try{sessionStorage.removeItem(e);}catch(t){console.warn("sessionStorage.removeItem failed:",t),this.getFallback().remove(e);}}clear(){if(!this.isSupported()){this.getFallback().clear();return}try{sessionStorage.clear();}catch(e){console.warn("sessionStorage.clear failed:",e),this.getFallback().clear();}}isSupported(){try{let e="__storage_test__";return sessionStorage.setItem(e,"test"),sessionStorage.removeItem(e),!0}catch{return false}}getFallback(){return this.fallback||(this.fallback=new q),this.fallback}},_=(e,t,n)=>{e.ns("storage"),e.defaults({storage:{backend:"localStorage",namespace:"",path:"/",sameSite:"lax"}});let r=()=>n.get("storage.backend")??"localStorage",i=()=>n.get("storage.namespace")??"",m=()=>n.get("storage.ttl"),g=()=>({domain:n.get("storage.domain"),path:n.get("storage.path")??"/",secure:n.get("storage.secure"),sameSite:n.get("storage.sameSite")??"lax"}),o={};function u(s){if(!o[s])switch(s){case "localStorage":o[s]=new Y;break;case "sessionStorage":o[s]=new Z;break;case "cookie":o[s]=new X(g());break;case "memory":o[s]=new q;break}return o[s]}function d(s,a){let f=a??i();return f?`${f}:${s}`:s}function c(s){return s.expires?Date.now()>s.expires:false}function h(s,a,f){let y=f?.backend??r(),x=u(y),S=d(s,f?.namespace),p={value:a},w=f?.ttl??m();w&&(p.expires=Date.now()+w*1e3);let C=JSON.stringify(p);x.set(S,C,f),e.emit("storage:set",{key:s,value:a,backend:y});}function k(s,a){let f=a?.backend??r(),y=u(f),x=d(s,a?.namespace),S=y.get(x);if(!S)return null;try{let p=JSON.parse(S);return c(p)?(y.remove(x),e.emit("storage:expired",{key:s,backend:f}),null):(e.emit("storage:get",{key:s,backend:f}),p.value)}catch(p){return console.warn("Failed to parse stored value:",p),y.remove(x),null}}function v(s,a){let f=a?.backend??r(),y=u(f),x=d(s,a?.namespace);y.remove(x),e.emit("storage:remove",{key:s,backend:f});}function l(s){let a=s?.backend??r(),f=u(a),y=s?.namespace;if(!y){f.clear(),e.emit("storage:clear",{backend:a});return}if(a==="localStorage"||a==="sessionStorage"){let x=a==="localStorage"?localStorage:sessionStorage,S=`${y}:`,p=[];for(let w=0;w<x.length;w++){let C=x.key(w);C?.startsWith(S)&&p.push(C);}for(let w of p)f.remove(w);}else if(a==="cookie"){let x=`${y}:`,S=document.cookie.split(";");for(let p of S){let w=p.trim(),C=w.indexOf("=");if(C===-1)continue;let T=w.substring(0,C),F=decodeURIComponent(T);F.startsWith(x)&&f.remove(F);}}else f.clear();e.emit("storage:clear",{backend:a,namespace:y});}function b(s){let a=s??r();return u(a).isSupported()}e.expose({storage:{set:h,get:k,remove:v,clear:l,isSupported:b}}),t.on("sdk:ready",()=>{let s=r();u(s).isSupported()||console.warn(`Storage backend '${s}' is not supported, falling back to memory storage`);});};var ee=["strong","em","a","br","span","b","i","p"],te={a:["href","class","style","title"],span:["class","style"],p:["class","style"]};function L(e){if(!e||typeof e!="string")return "";let t=document.createElement("div");t.innerHTML=e;function n(i){if(i.nodeType===Node.TEXT_NODE)return ne(i.textContent||"");if(i.nodeType===Node.ELEMENT_NODE){let m=i,g=m.tagName.toLowerCase();if(!g||g.includes(" ")||!ee.includes(g))return "";let o=te[g]||[],u=[];for(let h of o){let k=m.getAttribute(h);if(k!==null)if(h==="href"){let v=re(k);v&&u.push(`href="${j(v)}"`);}else u.push(`${h}="${j(k)}"`);}let d=u.length>0?" "+u.join(" "):"",c="";for(let h of Array.from(m.childNodes))c+=n(h);return g==="br"?`<br${d} />`:`<${g}${d}>${c}</${g}>`}return ""}let r="";for(let i of Array.from(t.childNodes))r+=n(i);return r}function ne(e){let t=document.createElement("div");return t.textContent=e,t.innerHTML}function j(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#39;")}function re(e){if(!e||typeof e!="string")return "";let t;try{t=decodeURIComponent(e);}catch{t=e;}let n=t.trim().toLowerCase();return n.startsWith("javascript:")||n.startsWith("data:")||e.toLowerCase().trim().startsWith("javascript:")||e.toLowerCase().trim().startsWith("data:")?"":n.startsWith("http://")||n.startsWith("https://")||n.startsWith("mailto:")||n.startsWith("tel:")||n.startsWith("/")||n.startsWith("#")||n.startsWith("?")||!n.includes(":")?e:""}var D=(e,t,n)=>{e.ns("banner"),e.defaults({banner:{position:"top",dismissable:true,zIndex:1e4}});let r=new Map;function i(){let d="xp-banner-styles";if(document.getElementById(d))return;let c=document.createElement("style");c.id=d,c.textContent=`
1
+ var experiences=(function(exports){'use strict';var O=class{storage=new Map;get(e){return this.storage.get(e)??null}set(e,t,n){this.storage.set(e,t);}remove(e){this.storage.delete(e);}clear(){this.storage.clear();}isSupported(){return true}},ue=class{fallback=null;defaultOptions;constructor(e={}){this.defaultOptions={path:"/",sameSite:"lax",...e};}get(e){if(!this.isSupported())return this.getFallback().get(e);try{let t=`${encodeURIComponent(e)}=`,n=document.cookie.split(";");for(let r of n)if(r=r.trim(),r.startsWith(t))return decodeURIComponent(r.substring(t.length));return null}catch(t){return console.warn("Cookie get failed:",t),this.getFallback().get(e)}}set(e,t,n){if(!this.isSupported()){this.getFallback().set(e,t);return}try{let r={...this.defaultOptions,...n},s=[`${encodeURIComponent(e)}=${encodeURIComponent(t)}`];if(r.ttl){let l=new Date;l.setTime(l.getTime()+r.ttl*1e3),s.push(`expires=${l.toUTCString()}`);}r.path&&s.push(`path=${r.path}`),r.domain&&s.push(`domain=${r.domain}`),r.secure&&s.push("secure"),r.sameSite&&s.push(`samesite=${r.sameSite}`),document.cookie=s.join("; ");}catch(r){console.warn("Cookie set failed, falling back to memory:",r),this.getFallback().set(e,t);}}remove(e){if(!this.isSupported()){this.getFallback().remove(e);return}try{let t=this.defaultOptions,n=[`${encodeURIComponent(e)}=`,"expires=Thu, 01 Jan 1970 00:00:00 UTC"];t.path&&n.push(`path=${t.path}`),t.domain&&n.push(`domain=${t.domain}`),document.cookie=n.join("; ");}catch(t){console.warn("Cookie remove failed:",t),this.getFallback().remove(e);}}clear(){if(!this.isSupported()){this.getFallback().clear();return}try{let e=document.cookie.split(";");for(let t of e){t=t.trim();let n=t.indexOf("="),r=n>-1?t.substring(0,n):t;this.remove(decodeURIComponent(r));}}catch(e){console.warn("Cookie clear failed:",e),this.getFallback().clear();}}isSupported(){try{if(typeof document>"u"||!document.cookie)return !1;let e="__cookie_test__";document.cookie=`${e}=test; path=/`;let t=document.cookie.indexOf(e)!==-1;return document.cookie=`${e}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/`,t}catch{return false}}getFallback(){return this.fallback||(this.fallback=new O),this.fallback}},de=class{fallback=null;get(e){if(!this.isSupported())return this.getFallback().get(e);try{return localStorage.getItem(e)}catch(t){return console.warn("localStorage.getItem failed:",t),this.getFallback().get(e)}}set(e,t,n){if(!this.isSupported()){this.getFallback().set(e,t);return}try{localStorage.setItem(e,t);}catch(r){console.warn("localStorage.setItem failed, falling back to memory:",r),this.getFallback().set(e,t);}}remove(e){if(!this.isSupported()){this.getFallback().remove(e);return}try{localStorage.removeItem(e);}catch(t){console.warn("localStorage.removeItem failed:",t),this.getFallback().remove(e);}}clear(){if(!this.isSupported()){this.getFallback().clear();return}try{localStorage.clear();}catch(e){console.warn("localStorage.clear failed:",e),this.getFallback().clear();}}isSupported(){try{let e="__storage_test__";return localStorage.setItem(e,"test"),localStorage.removeItem(e),!0}catch{return false}}getFallback(){return this.fallback||(this.fallback=new O),this.fallback}},fe=class{fallback=null;get(e){if(!this.isSupported())return this.getFallback().get(e);try{return sessionStorage.getItem(e)}catch(t){return console.warn("sessionStorage.getItem failed:",t),this.getFallback().get(e)}}set(e,t,n){if(!this.isSupported()){this.getFallback().set(e,t);return}try{sessionStorage.setItem(e,t);}catch(r){console.warn("sessionStorage.setItem failed, falling back to memory:",r),this.getFallback().set(e,t);}}remove(e){if(!this.isSupported()){this.getFallback().remove(e);return}try{sessionStorage.removeItem(e);}catch(t){console.warn("sessionStorage.removeItem failed:",t),this.getFallback().remove(e);}}clear(){if(!this.isSupported()){this.getFallback().clear();return}try{sessionStorage.clear();}catch(e){console.warn("sessionStorage.clear failed:",e),this.getFallback().clear();}}isSupported(){try{let e="__storage_test__";return sessionStorage.setItem(e,"test"),sessionStorage.removeItem(e),!0}catch{return false}}getFallback(){return this.fallback||(this.fallback=new O),this.fallback}},P=(e,t,n)=>{e.ns("storage"),e.defaults({storage:{backend:"localStorage",namespace:"",path:"/",sameSite:"lax"}});let r=()=>n.get("storage.backend")??"localStorage",s=()=>n.get("storage.namespace")??"",l=()=>n.get("storage.ttl"),f=()=>({domain:n.get("storage.domain"),path:n.get("storage.path")??"/",secure:n.get("storage.secure"),sameSite:n.get("storage.sameSite")??"lax"}),i={};function a(o){if(!i[o])switch(o){case "localStorage":i[o]=new de;break;case "sessionStorage":i[o]=new fe;break;case "cookie":i[o]=new ue(f());break;case "memory":i[o]=new O;break}return i[o]}function b(o,u){let p=u??s();return p?`${p}:${o}`:o}function v(o){return o.expires?Date.now()>o.expires:false}function d(o,u,p){let h=p?.backend??r(),w=a(h),k=b(o,p?.namespace),S={value:u},x=p?.ttl??l();x&&(S.expires=Date.now()+x*1e3);let T=JSON.stringify(S);w.set(k,T,p),e.emit("storage:set",{key:o,value:u,backend:h});}function c(o,u){let p=u?.backend??r(),h=a(p),w=b(o,u?.namespace),k=h.get(w);if(!k)return null;try{let S=JSON.parse(k);return v(S)?(h.remove(w),e.emit("storage:expired",{key:o,backend:p}),null):(e.emit("storage:get",{key:o,backend:p}),S.value)}catch(S){return console.warn("Failed to parse stored value:",S),h.remove(w),null}}function y(o,u){let p=u?.backend??r(),h=a(p),w=b(o,u?.namespace);h.remove(w),e.emit("storage:remove",{key:o,backend:p});}function m(o){let u=o?.backend??r(),p=a(u),h=o?.namespace;if(!h){p.clear(),e.emit("storage:clear",{backend:u});return}if(u==="localStorage"||u==="sessionStorage"){let w=u==="localStorage"?localStorage:sessionStorage,k=`${h}:`,S=[];for(let x=0;x<w.length;x++){let T=w.key(x);T?.startsWith(k)&&S.push(T);}for(let x of S)p.remove(x);}else if(u==="cookie"){let w=`${h}:`,k=document.cookie.split(";");for(let S of k){let x=S.trim(),T=x.indexOf("=");if(T===-1)continue;let E=x.substring(0,T),C=decodeURIComponent(E);C.startsWith(w)&&p.remove(C);}}else p.clear();e.emit("storage:clear",{backend:u,namespace:h});}function g(o){let u=o??r();return a(u).isSupported()}e.expose({storage:{set:d,get:c,remove:y,clear:m,isSupported:g}}),t.on("sdk:ready",()=>{let o=r();a(o).isSupported()||console.warn(`Storage backend '${o}' is not supported, falling back to memory storage`);});};var pe=["strong","em","a","br","span","b","i","p"],me={a:["href","class","style","title"],span:["class","style"],p:["class","style"]};function W(e){if(!e||typeof e!="string")return "";let t=document.createElement("div");t.innerHTML=e;function n(s){if(s.nodeType===Node.TEXT_NODE)return ge(s.textContent||"");if(s.nodeType===Node.ELEMENT_NODE){let l=s,f=l.tagName.toLowerCase();if(!f||f.includes(" ")||!pe.includes(f))return "";let i=me[f]||[],a=[];for(let d of i){let c=l.getAttribute(d);if(c!==null)if(d==="href"){let y=he(c);y&&a.push(`href="${J(y)}"`);}else a.push(`${d}="${J(c)}"`);}let b=a.length>0?` ${a.join(" ")}`:"",v="";for(let d of Array.from(l.childNodes))v+=n(d);return f==="br"?`<br${b} />`:`<${f}${b}>${v}</${f}>`}return ""}let r="";for(let s of Array.from(t.childNodes))r+=n(s);return r}function ge(e){let t=document.createElement("div");return t.textContent=e,t.innerHTML}function J(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#39;")}function he(e){if(!e||typeof e!="string")return "";let t;try{t=decodeURIComponent(e);}catch{t=e;}let n=t.trim().toLowerCase();return n.startsWith("javascript:")||n.startsWith("data:")||e.toLowerCase().trim().startsWith("javascript:")||e.toLowerCase().trim().startsWith("data:")?"":n.startsWith("http://")||n.startsWith("https://")||n.startsWith("mailto:")||n.startsWith("tel:")||n.startsWith("/")||n.startsWith("#")||n.startsWith("?")||!n.includes(":")?e:""}var A=(e,t,n)=>{e.ns("banner"),e.defaults({banner:{position:"top",dismissable:true,zIndex:1e4}});let r=new Map;function s(){let d="xp-banner-styles";if(document.getElementById(d))return;let c=document.createElement("style");c.id=d,c.textContent=`
2
2
  .xp-banner {
3
3
  position: fixed;
4
4
  left: 0;
5
5
  right: 0;
6
6
  width: 100%;
7
- padding: 16px 20px;
8
7
  font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
9
8
  font-size: 14px;
10
9
  line-height: 1.5;
11
- display: flex;
12
- align-items: center;
13
- justify-content: space-between;
14
10
  box-sizing: border-box;
15
11
  z-index: 10000;
16
- background: #f9fafb;
12
+ background: #ffffff;
17
13
  color: #111827;
18
14
  border-bottom: 1px solid #e5e7eb;
19
15
  box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.05);
@@ -33,33 +29,38 @@ var experiences=(function(exports){'use strict';var q=class{storage=new Map;get(
33
29
  .xp-banner__container {
34
30
  display: flex;
35
31
  align-items: center;
36
- justify-content: space-between;
37
- gap: 20px;
38
- width: 100%;
32
+ gap: 16px;
33
+ max-width: 1280px;
34
+ margin: 0 auto;
35
+ padding: 14px 24px;
39
36
  }
40
37
 
41
38
  .xp-banner__content {
42
39
  flex: 1;
43
40
  min-width: 0;
41
+ display: flex;
42
+ flex-direction: column;
43
+ gap: 4px;
44
44
  }
45
45
 
46
46
  .xp-banner__title {
47
47
  font-weight: 600;
48
- margin-bottom: 4px;
49
- margin-top: 0;
50
- font-size: 14px;
48
+ margin: 0;
49
+ font-size: 15px;
50
+ line-height: 1.4;
51
51
  }
52
52
 
53
53
  .xp-banner__message {
54
54
  margin: 0;
55
55
  font-size: 14px;
56
+ line-height: 1.5;
57
+ color: #6b7280;
56
58
  }
57
59
 
58
60
  .xp-banner__buttons {
59
61
  display: flex;
60
62
  align-items: center;
61
- gap: 12px;
62
- flex-wrap: wrap;
63
+ gap: 8px;
63
64
  flex-shrink: 0;
64
65
  }
65
66
 
@@ -72,6 +73,10 @@ var experiences=(function(exports){'use strict';var q=class{storage=new Map;get(
72
73
  cursor: pointer;
73
74
  transition: all 0.2s;
74
75
  text-decoration: none;
76
+ display: inline-flex;
77
+ align-items: center;
78
+ justify-content: center;
79
+ white-space: nowrap;
75
80
  }
76
81
 
77
82
  .xp-banner__button--primary {
@@ -84,71 +89,93 @@ var experiences=(function(exports){'use strict';var q=class{storage=new Map;get(
84
89
  }
85
90
 
86
91
  .xp-banner__button--secondary {
87
- background: #ffffff;
92
+ background: #f3f4f6;
88
93
  color: #374151;
89
- border: 1px solid #d1d5db;
94
+ border: 1px solid #e5e7eb;
90
95
  }
91
96
 
92
97
  .xp-banner__button--secondary:hover {
93
- background: #f9fafb;
98
+ background: #e5e7eb;
94
99
  }
95
100
 
96
101
  .xp-banner__button--link {
97
102
  background: transparent;
98
103
  color: #2563eb;
99
- padding: 4px 8px;
104
+ padding: 6px 12px;
100
105
  font-weight: 400;
101
- text-decoration: underline;
102
106
  }
103
107
 
104
108
  .xp-banner__button--link:hover {
105
- background: rgba(0, 0, 0, 0.05);
109
+ background: #f3f4f6;
110
+ text-decoration: underline;
106
111
  }
107
112
 
108
113
  .xp-banner__close {
109
114
  background: transparent;
110
115
  border: none;
111
- color: #6b7280;
112
- font-size: 24px;
116
+ color: #9ca3af;
117
+ font-size: 20px;
113
118
  line-height: 1;
114
119
  cursor: pointer;
115
- padding: 0;
120
+ padding: 4px;
116
121
  margin: 0;
117
- opacity: 0.7;
118
- transition: opacity 0.2s;
122
+ transition: color 0.2s;
119
123
  flex-shrink: 0;
124
+ width: 28px;
125
+ height: 28px;
126
+ display: flex;
127
+ align-items: center;
128
+ justify-content: center;
129
+ border-radius: 4px;
120
130
  }
121
131
 
122
132
  .xp-banner__close:hover {
123
- opacity: 1;
133
+ color: #111827;
134
+ background: #f3f4f6;
124
135
  }
125
136
 
126
137
  @media (max-width: 640px) {
127
138
  .xp-banner__container {
128
- flex-direction: column;
129
- align-items: stretch;
139
+ flex-wrap: wrap;
140
+ padding: 14px 16px;
141
+ position: relative;
142
+ }
143
+
144
+ .xp-banner__content {
145
+ flex: 1 1 100%;
146
+ padding-right: 32px;
130
147
  }
131
148
 
132
149
  .xp-banner__buttons {
150
+ flex: 1 1 auto;
133
151
  width: 100%;
134
- flex-direction: column;
135
152
  }
136
153
 
137
154
  .xp-banner__button {
138
- width: 100%;
155
+ flex: 1;
156
+ }
157
+
158
+ .xp-banner__close {
159
+ position: absolute;
160
+ top: 12px;
161
+ right: 12px;
139
162
  }
140
163
  }
141
164
 
142
165
  /* Dark mode support */
143
166
  @media (prefers-color-scheme: dark) {
144
167
  .xp-banner {
145
- background: #1f2937;
146
- color: #f3f4f6;
147
- border-bottom-color: #374151;
168
+ background: #111827;
169
+ color: #f9fafb;
170
+ border-bottom-color: #1f2937;
148
171
  }
149
172
 
150
173
  .xp-banner--bottom {
151
- border-top-color: #374151;
174
+ border-top-color: #1f2937;
175
+ }
176
+
177
+ .xp-banner__message {
178
+ color: #9ca3af;
152
179
  }
153
180
 
154
181
  .xp-banner__button--primary {
@@ -160,27 +187,31 @@ var experiences=(function(exports){'use strict';var q=class{storage=new Map;get(
160
187
  }
161
188
 
162
189
  .xp-banner__button--secondary {
163
- background: #374151;
164
- color: #f3f4f6;
165
- border-color: #4b5563;
190
+ background: #1f2937;
191
+ color: #f9fafb;
192
+ border-color: #374151;
166
193
  }
167
194
 
168
195
  .xp-banner__button--secondary:hover {
169
- background: #4b5563;
196
+ background: #374151;
170
197
  }
171
198
 
172
199
  .xp-banner__button--link {
173
- color: #93c5fd;
200
+ color: #60a5fa;
201
+ }
202
+
203
+ .xp-banner__button--link:hover {
204
+ background: #1f2937;
174
205
  }
175
206
 
176
207
  .xp-banner__close {
177
- color: #9ca3af;
208
+ color: #6b7280;
209
+ }
210
+
211
+ .xp-banner__close:hover {
212
+ color: #f9fafb;
213
+ background: #1f2937;
178
214
  }
179
215
  }
180
- `,document.head.appendChild(c);}function m(d){let c=d.content,h=c.position??n.get("banner.position")??"top",k=c.dismissable??n.get("banner.dismissable")??true,v=n.get("banner.zIndex")??1e4;i();let l=document.createElement("div");l.setAttribute("data-experience-id",d.id);let b=["xp-banner",`xp-banner--${h}`];c.className&&b.push(c.className),l.className=b.join(" "),c.style&&Object.assign(l.style,c.style),v!==1e4&&(l.style.zIndex=String(v));let s=document.createElement("div");s.className="xp-banner__container",l.appendChild(s);let a=document.createElement("div");if(a.className="xp-banner__content",c.title){let p=document.createElement("h3");p.className="xp-banner__title",p.innerHTML=L(c.title),a.appendChild(p);}let f=document.createElement("p");f.className="xp-banner__message",f.innerHTML=L(c.message),a.appendChild(f),s.appendChild(a),l.appendChild(a);let y=document.createElement("div");y.style.cssText=`
181
- display: flex;
182
- align-items: center;
183
- gap: 12px;
184
- flex-wrap: wrap;
185
- `;let x=document.createElement("div");x.className="xp-banner__buttons";function S(p){let w=document.createElement("button");w.textContent=p.text;let C=p.variant||"primary",T=["xp-banner__button",`xp-banner__button--${C}`];return p.className&&T.push(p.className),w.className=T.join(" "),p.style&&Object.assign(w.style,p.style),w.addEventListener("click",()=>{t.emit("experiences:action",{experienceId:d.id,type:"banner",action:p.action,url:p.url,metadata:p.metadata,variant:C,timestamp:Date.now()}),p.url&&(window.location.href=p.url);}),w}if(c.buttons&&c.buttons.length>0&&c.buttons.forEach(p=>{let w=S(p);x.appendChild(w);}),k){let p=document.createElement("button");p.className="xp-banner__close",p.innerHTML="&times;",p.setAttribute("aria-label","Close banner"),p.addEventListener("click",()=>{o(d.id),t.emit("experiences:dismissed",{experienceId:d.id,type:"banner"});}),x.appendChild(p);}return s.appendChild(x),l}function g(d){if(r.has(d.id)||typeof document>"u")return;let c=m(d);document.body.appendChild(c),r.set(d.id,c),t.emit("experiences:shown",{experienceId:d.id,type:"banner",timestamp:Date.now()});}function o(d){if(d){let c=r.get(d);c?.parentNode&&c.parentNode.removeChild(c),r.delete(d);}else for(let[c,h]of r.entries())h?.parentNode&&h.parentNode.removeChild(h),r.delete(c);}function u(){return r.size>0}e.expose({banner:{show:g,remove:o,isShowing:u}}),t.on("experiences:evaluated",d=>{let c=Array.isArray(d)?d:[d];for(let h of c){let k=h,v=k.decision,l=k.experience;l?.type==="banner"&&(v?.show?g(l):l.id&&r.has(l.id)&&o(l.id));}}),t.on("sdk:destroy",()=>{o();});},O=(e,t,n)=>{e.ns("debug"),e.defaults({debug:{enabled:false,console:false,window:true}});let r=()=>n.get("debug.enabled")??false,i=()=>n.get("debug.console")??false,m=()=>n.get("debug.window")??true,g=(o,u)=>{if(!r())return;let c={timestamp:new Date().toISOString(),message:o,data:u};if(i()&&console.log(`[experiences] ${o}`,u||""),m()&&typeof window<"u"){let h=new CustomEvent("experience-sdk:debug",{detail:c});window.dispatchEvent(h);}};e.expose({debug:{log:g,isEnabled:r}}),r()&&(t.on("experiences:ready",()=>{r()&&g("SDK initialized and ready");}),t.on("experiences:registered",o=>{r()&&g("Experience registered",o);}),t.on("experiences:evaluated",o=>{r()&&g("Experience evaluated",o);}));},R=(e,t,n)=>{e.ns("frequency"),e.defaults({frequency:{enabled:true,namespace:"experiences:frequency"}});let r=new Map;t.storage||t.use(_);let i=()=>n.get("frequency.enabled")??true,m=()=>n.get("frequency.namespace")??"experiences:frequency",g=l=>l==="session"?sessionStorage:localStorage,o=l=>`${m()}:${l}`,u=(l,b)=>{let s=g(b),a=o(l),f=s.getItem(a);if(!f)return {count:0,lastImpression:0,impressions:[],per:b};try{return JSON.parse(f)}catch{return {count:0,lastImpression:0,impressions:[],per:b}}},d=(l,b)=>{let s=b.per||"session",a=g(s),f=o(l);a.setItem(f,JSON.stringify(b));},c=l=>{switch(l){case "session":return Number.POSITIVE_INFINITY;case "day":return 1440*60*1e3;case "week":return 10080*60*1e3}},h=(l,b="session")=>i()?u(l,b).count:0,k=(l,b,s)=>{if(!i())return false;let a=u(l,s),f=c(s),y=Date.now();return s==="session"?a.count>=b:a.impressions.filter(S=>y-S<f).length>=b},v=(l,b="session")=>{if(!i())return;let s=u(l,b),a=Date.now();s.count+=1,s.lastImpression=a,s.impressions.push(a),s.per=b;let f=a-10080*60*1e3;s.impressions=s.impressions.filter(y=>y>f),d(l,s),t.emit("experiences:impression-recorded",{experienceId:l,count:s.count,timestamp:a});};e.expose({frequency:{getImpressionCount:h,hasReachedCap:k,recordImpression:v,_registerExperience:(l,b)=>{r.set(l,b);}}}),i()&&t.on("experiences:evaluated",l=>{let b=Array.isArray(l)?l:[l];for(let s of b){let a=s.decision;if(a?.show&&a.experienceId){let f=r.get(a.experienceId)||"session";if(!r.has(a.experienceId)){let y=a.trace.find(x=>x.step==="check-frequency-cap");y?.input&&typeof y.input=="object"&&"per"in y.input&&(f=y.input.per,r.set(a.experienceId,f));}v(a.experienceId,f);}}});};function N(e,t){let n={...e};for(let r in t){if(!Object.hasOwn(t,r))continue;let i=t[r],m=n[r];if(m===void 0){n[r]=i;continue}U(m)&&U(i)&&(n[r]=N(m,i));}return n}function U(e){return e===null||typeof e!="object"?false:Object.prototype.toString.call(e)==="[object Object]"}var se=class{data={};required=new Set;constructor(e={}){this.data=JSON.parse(JSON.stringify(e));}defaults(e){this.data=N(this.data,e);}merge(e){this.data=N(e,this.data);}get(e){let t=e.split("."),n=this.data;for(let r of t){if(n==null)return;n=n[r];}return n}set(e,t){let n=e.split("."),r=n.pop();if(!r)return;let i=this.data;for(let m of n)(i[m]==null||typeof i[m]!="object")&&(i[m]={}),i=i[m];i[r]=t;}markRequired(e){e!=null&&this.required.add(e);}isRequired(e){return this.required.has(e)}getAll(){return JSON.parse(JSON.stringify(this.data))}},oe=class{subscriptions=[];on(e,t){if(typeof t!="function")throw new TypeError("handler must be a function");let n={pattern:e,compiledPattern:this.compilePattern(e),handler:t};return this.subscriptions.push(n),()=>this.off(e,t)}off(e,t){this.subscriptions=this.subscriptions.filter(n=>!(n.pattern===e&&n.handler===t));}emit(e,...t){for(let n of this.subscriptions)if(n.compiledPattern.test(e))try{n.handler(...t);}catch(r){console.error(`Error in event handler for "${e}":`,r);}}removeAllListeners(){this.subscriptions=[];}compilePattern(e){let n=this.escapeRegExp(e).replace(/\\\*/g,"(.*?)");return new RegExp(`^${n}$`)}escapeRegExp(e){return e.replace(/[\\^$*+?.()|[\]{}]/g,"\\$&")}},ie=class{sdk;constructor(e){this.sdk=e;}expose(e){Object.assign(this.sdk,e);}},ae=class{name="";ns(e){if(this.name)throw new Error(`Namespace already set to "${this.name}". Cannot reassign to "${e}".`);this.name=e;}},z=class{configInstance;emitter;plugins=new Map;isInitialized=false;constructor(e={}){this.emitter=new oe,this.configInstance=new se(e);}use(e){let t=new ae,n=new ie(this),r={ns:t.ns.bind(t),defaults:this.configInstance.defaults.bind(this.configInstance),on:this.emitter.on.bind(this.emitter),off:this.emitter.off.bind(this.emitter),emit:this.emitter.emit.bind(this.emitter),expose:n.expose.bind(n),mustEnable:()=>{t.name&&this.configInstance.markRequired(t.name);}};return e(r,this,this.configInstance),t.name&&this.plugins.set(t.name,{namespace:t,plugin:r}),this}async init(e){if(this.isInitialized){console.warn("SDK already initialized");return}e&&this.configInstance.merge(e),this.emitter.emit("sdk:init");let t=Array.from(this.plugins.keys()).filter(n=>this.configInstance.isRequired(n));for(let n of t)if(!this.plugins.has(n))throw new Error(`Required plugin "${n}" is not registered`);this.isInitialized=true,this.emitter.emit("sdk:ready");}async destroy(){this.emitter.emit("sdk:destroy"),this.plugins.clear(),this.emitter.removeAllListeners(),this.isInitialized=false;}get(e){return this.configInstance.get(e)}set(e,t){this.configInstance.set(e,t);}on(e,t){return this.emitter.on(e,t)}off(e,t){this.emitter.off(e,t);}emit(e,...t){this.emitter.emit(e,...t);}getAll(){return this.configInstance.getAll()}isReady(){return this.isInitialized}};var E=class{sdk;experiences=new Map;decisions=[];initialized=false;destroyed=false;constructor(t={}){this.sdk=new z({name:"experience-sdk",...t}),this.sdk.use(_),this.sdk.use(O),this.sdk.use(R),this.sdk.use(D);}async init(t){if(this.initialized){console.warn("[experiences] Already initialized");return}this.destroyed&&(this.sdk=new z({name:"experience-sdk",...t}),this.sdk.use(_),this.sdk.use(O),this.sdk.use(R),this.sdk.use(D),this.destroyed=false),t&&Object.entries(t).forEach(([n,r])=>{this.sdk.set(n,r);}),await this.sdk.init(),this.initialized=true,this.sdk.emit("experiences:ready");}register(t,n){let r={id:t,...n};this.experiences.set(t,r),r.frequency&&this.sdk.frequency?._registerExperience&&this.sdk.frequency._registerExperience(t,r.frequency.per),this.sdk.emit("experiences:registered",{id:t,experience:r});}evaluate(t){let n=Date.now(),r=P(t),i,m=[],g=[];for(let[,u]of this.experiences){let d=A(u,r);if(m.push(...d.reasons),g.push(...d.trace),d.matched){if(u.frequency&&this.sdk.frequency){let c=Date.now(),h=this.sdk.frequency.hasReachedCap(u.id,u.frequency.max,u.frequency.per);if(g.push({step:"check-frequency-cap",timestamp:c,duration:Date.now()-c,input:u.frequency,output:h,passed:!h}),h){let v=this.sdk.frequency.getImpressionCount(u.id,u.frequency.per);m.push(`Frequency cap reached (${v}/${u.frequency.max} this ${u.frequency.per})`);continue}let k=this.sdk.frequency.getImpressionCount(u.id,u.frequency.per);m.push(`Frequency cap not reached (${k}/${u.frequency.max} this ${u.frequency.per})`);}i=u;break}}let o={show:!!i,experienceId:i?.id,reasons:m,trace:g,context:r,metadata:{evaluatedAt:Date.now(),totalDuration:Date.now()-n,experiencesEvaluated:this.experiences.size}};return this.decisions.push(o),this.sdk.emit("experiences:evaluated",{decision:o,experience:i}),o}evaluateAll(t){let n=P(t),r=Array.from(this.experiences.values()).sort((o,u)=>{let d=o.priority??0;return (u.priority??0)-d}),i=[];for(let o of r){let u=Date.now(),d=A(o,n),c=d.matched,h=[...d.reasons],k=[...d.trace];if(c&&o.frequency&&this.sdk.frequency){let l=Date.now(),b=this.sdk.frequency.hasReachedCap(o.id,o.frequency.max,o.frequency.per);if(k.push({step:"check-frequency-cap",timestamp:l,duration:Date.now()-l,input:o.frequency,output:b,passed:!b}),b){let s=this.sdk.frequency.getImpressionCount(o.id,o.frequency.per);h.push(`Frequency cap reached (${s}/${o.frequency.max} this ${o.frequency.per})`),c=false;}else {let s=this.sdk.frequency.getImpressionCount(o.id,o.frequency.per);h.push(`Frequency cap not reached (${s}/${o.frequency.max} this ${o.frequency.per})`);}}let v={show:c,experienceId:o.id,reasons:h,trace:k,context:n,metadata:{evaluatedAt:Date.now(),totalDuration:Date.now()-u,experiencesEvaluated:1}};i.push(v),this.decisions.push(v);}let m=i.filter(o=>o.show),g=m.map(o=>o.experienceId&&this.experiences.get(o.experienceId)).filter(o=>o!==void 0);return this.sdk.emit("experiences:evaluated",m.map((o,u)=>({decision:o,experience:g[u]}))),i}explain(t){let n=this.experiences.get(t);if(!n)return null;let r=P(),i=A(n,r);return {show:i.matched,experienceId:t,reasons:i.reasons,trace:i.trace,context:r,metadata:{evaluatedAt:Date.now(),totalDuration:0,experiencesEvaluated:1}}}getState(){return {initialized:this.initialized,experiences:new Map(this.experiences),decisions:[...this.decisions],config:this.sdk?this.sdk.getAll():{}}}on(t,n){return this.sdk.on(t,n)}async destroy(){this.sdk&&await this.sdk.destroy(),this.destroyed=true,this.experiences.clear(),this.decisions=[],this.initialized=false;}};function P(e){return {url:e?.url??(typeof window<"u"?window.location.href:""),timestamp:Date.now(),user:e?.user,custom:e?.custom}}function A(e,t){let n=[],r=[],i=true;if(e.targeting.url){let m=Date.now(),g=M(e.targeting.url,t.url);r.push({step:"evaluate-url-rule",timestamp:m,duration:Date.now()-m,input:{rule:e.targeting.url,url:t.url},output:g,passed:g}),g?n.push("URL matches targeting rule"):(n.push("URL does not match targeting rule"),i=false);}return {matched:i,reasons:n,trace:r}}function M(e,t=""){return e.equals!==void 0?t===e.equals:e.contains!==void 0?t.includes(e.contains):e.matches!==void 0?e.matches.test(t):true}function $(e){return new E(e)}var I=$();async function B(e){return I.init(e)}function W(e,t){I.register(e,t);}function K(e){return I.evaluate(e)}function H(e){return I.evaluateAll(e)}function V(e){return I.explain(e)}function J(){return I.getState()}function Q(e,t){return I.on(e,t)}async function G(){return I.destroy()}var ce={createInstance:$,init:B,register:W,evaluate:K,evaluateAll:H,explain:V,getState:J,on:Q,destroy:G};typeof window<"u"&&(window.experiences=ce);exports.ExperienceRuntime=E;exports.bannerPlugin=D;exports.buildContext=P;exports.createInstance=$;exports.debugPlugin=O;exports.destroy=G;exports.evaluate=K;exports.evaluateAll=H;exports.evaluateExperience=A;exports.evaluateUrlRule=M;exports.explain=V;exports.frequencyPlugin=R;exports.getState=J;exports.init=B;exports.on=Q;exports.register=W;return exports;})({});//# sourceMappingURL=experience-sdk.global.js.map
216
+ `,document.head.appendChild(c);}function l(d){let c=d.content,y=c.position??n.get("banner.position")??"top",m=c.dismissable??n.get("banner.dismissable")??true,g=n.get("banner.zIndex")??1e4;s();let o=document.createElement("div");o.setAttribute("data-experience-id",d.id);let u=["xp-banner",`xp-banner--${y}`];c.className&&u.push(c.className),o.className=u.join(" "),c.style&&Object.assign(o.style,c.style),g!==1e4&&(o.style.zIndex=String(g));let p=document.createElement("div");p.className="xp-banner__container",o.appendChild(p);let h=document.createElement("div");if(h.className="xp-banner__content",c.title){let x=document.createElement("h3");x.className="xp-banner__title",x.innerHTML=W(c.title),h.appendChild(x);}let w=document.createElement("p");w.className="xp-banner__message",w.innerHTML=W(c.message),h.appendChild(w),p.appendChild(h);let k=document.createElement("div");k.className="xp-banner__buttons";function S(x){let T=document.createElement("button");T.textContent=x.text;let E=x.variant||"primary",C=["xp-banner__button",`xp-banner__button--${E}`];return x.className&&C.push(x.className),T.className=C.join(" "),x.style&&Object.assign(T.style,x.style),T.addEventListener("click",()=>{t.emit("experiences:action",{experienceId:d.id,type:"banner",action:x.action,url:x.url,metadata:x.metadata,variant:E,timestamp:Date.now()}),x.url&&(window.location.href=x.url);}),T}if(c.buttons&&c.buttons.length>0&&c.buttons.forEach(x=>{let T=S(x);k.appendChild(T);}),m){let x=document.createElement("button");x.className="xp-banner__close",x.innerHTML="&times;",x.setAttribute("aria-label","Close banner"),x.addEventListener("click",()=>{b(d.id),t.emit("experiences:dismissed",{experienceId:d.id,type:"banner"});}),k.appendChild(x);}return p.appendChild(k),o}function f(d,c){let y=n.get("banner.pushDown");if(!y||c!=="top")return;let m=document.querySelector(y);if(!m||!(m instanceof HTMLElement))return;let g=d.offsetHeight;m.style.transition="margin-top 0.3s ease",m.style.marginTop=`${g}px`;}function i(){let d=n.get("banner.pushDown");if(!d)return;let c=document.querySelector(d);!c||!(c instanceof HTMLElement)||(c.style.transition="margin-top 0.3s ease",c.style.marginTop="0");}function a(d){if(r.has(d.id)||typeof document>"u")return;let c=l(d);document.body.appendChild(c),r.set(d.id,c);let m=d.content.position??n.get("banner.position")??"top";f(c,m),t.emit("experiences:shown",{experienceId:d.id,type:"banner",timestamp:Date.now()});}function b(d){if(d){let c=r.get(d);c?.parentNode&&c.parentNode.removeChild(c),r.delete(d),r.size===0&&i();}else {for(let[c,y]of r.entries())y?.parentNode&&y.parentNode.removeChild(y),r.delete(c);i();}}function v(){return r.size>0}e.expose({banner:{show:a,remove:b,isShowing:v}}),t.on("experiences:evaluated",d=>{let c=Array.isArray(d)?d:[d];for(let y of c){let m=y,g=m.decision,o=m.experience;o?.type==="banner"&&(g?.show?a(o):o.id&&r.has(o.id)&&b(o.id));}}),t.on("sdk:destroy",()=>{b();});},R=(e,t,n)=>{e.ns("debug"),e.defaults({debug:{enabled:false,console:false,window:true}});let r=()=>n.get("debug.enabled")??false,s=()=>n.get("debug.console")??false,l=()=>n.get("debug.window")??true,f=(i,a)=>{if(!r())return;let v={timestamp:new Date().toISOString(),message:i,data:a};if(s()&&console.log(`[experiences] ${i}`,a||""),l()&&typeof window<"u"){let d=new CustomEvent("experience-sdk:debug",{detail:v});window.dispatchEvent(d);}};e.expose({debug:{log:f,isEnabled:r}}),r()&&(t.on("experiences:ready",()=>{r()&&f("SDK initialized and ready");}),t.on("experiences:registered",i=>{r()&&f("Experience registered",i);}),t.on("experiences:evaluated",i=>{r()&&f("Experience evaluated",i);}));};function ye(e){return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(e)}function be(e,t,n){return n-e>=t}function xe(e,t,n){let r=[...e,t];return r.length>n?r.slice(1):r}function we(e,t){return Math.abs(e-t)}function ve(e,t,n){if(e.length<2)return {shouldTrigger:false,lastY:0,previousY:0,velocity:0};if(n&&n.nodeName!=="HTML")return {shouldTrigger:false,lastY:0,previousY:0,velocity:0};let r=e[e.length-1].y,s=e[e.length-2].y,l=we(r,s),f=r<s,i=r-l<=t;return {shouldTrigger:f&&i,lastY:r,previousY:s,velocity:l}}function ke(e,t,n,r,s){return {timestamp:s,lastY:e,previousY:t,velocity:n,timeOnPage:s-r}}var V=(e,t,n)=>{e.ns("experiences.exitIntent"),e.defaults({exitIntent:{sensitivity:50,minTimeOnPage:2e3,delay:0,positionHistorySize:30,disableOnMobile:true}});let r=n.get("exitIntent");if(!r)return;let s=[],l=false,f=Date.now(),i=null,a=null;function b(){return r?.disableOnMobile?ye(navigator.userAgent):false}function v(g){let o={x:g.clientX,y:g.clientY},u=r?.positionHistorySize??30;s=xe(s,o,u);}function d(g){if(l)return;let o=r?.minTimeOnPage??2e3;if(!be(f,o,Date.now()))return;let u=r?.sensitivity??50,p=g.relatedTarget||g.toElement,h=ve(s,u,p);if(h.shouldTrigger){l=true;let w=ke(h.lastY,h.previousY,h.velocity,f,Date.now()),k=r?.delay??0;k>0?setTimeout(()=>{t.emit("trigger:exitIntent",w);},k):t.emit("trigger:exitIntent",w);try{sessionStorage.setItem("xp:exitIntent:triggered",Date.now().toString());}catch{}c();}}function c(){i&&(document.removeEventListener("mousemove",i),i=null),a&&(document.removeEventListener("mouseout",a),a=null);}function y(){if(!b()){try{if(sessionStorage.getItem("xp:exitIntent:triggered")){l=!0;return}}catch{}i=v,a=d,document.addEventListener("mousemove",i),document.addEventListener("mouseout",a);}}e.expose({exitIntent:{isTriggered:()=>l,reset:()=>{l=false,s=[];try{sessionStorage.removeItem("xp:exitIntent:triggered");}catch{}c(),y();},getPositions:()=>[...s]}}),y();let m=()=>{c();};t.on("destroy",m);},L=(e,t,n)=>{e.ns("frequency"),e.defaults({frequency:{enabled:true,namespace:"experiences:frequency"}});let r=new Map;t.storage||t.use(P);let s=()=>n.get("frequency.enabled")??true,l=()=>n.get("frequency.namespace")??"experiences:frequency",f=m=>m==="session"?sessionStorage:localStorage,i=m=>`${l()}:${m}`,a=(m,g)=>{let o=f(g),u=i(m),p=o.getItem(u);if(!p)return {count:0,lastImpression:0,impressions:[],per:g};try{return JSON.parse(p)}catch{return {count:0,lastImpression:0,impressions:[],per:g}}},b=(m,g)=>{let o=g.per||"session",u=f(o),p=i(m);u.setItem(p,JSON.stringify(g));},v=m=>{switch(m){case "session":return Number.POSITIVE_INFINITY;case "day":return 1440*60*1e3;case "week":return 10080*60*1e3}},d=(m,g="session")=>s()?a(m,g).count:0,c=(m,g,o)=>{if(!s())return false;let u=a(m,o),p=v(o),h=Date.now();return o==="session"?u.count>=g:u.impressions.filter(k=>h-k<p).length>=g},y=(m,g="session")=>{if(!s())return;let o=a(m,g),u=Date.now();o.count+=1,o.lastImpression=u,o.impressions.push(u),o.per=g;let p=u-10080*60*1e3;o.impressions=o.impressions.filter(h=>h>p),b(m,o),t.emit("experiences:impression-recorded",{experienceId:m,count:o.count,timestamp:u});};e.expose({frequency:{getImpressionCount:d,hasReachedCap:c,recordImpression:y,_registerExperience:(m,g)=>{r.set(m,g);}}}),s()&&t.on("experiences:evaluated",m=>{let g=Array.isArray(m)?m:[m];for(let o of g){let u=o.decision;if(u?.show&&u.experienceId){let p=r.get(u.experienceId)||"session";if(!r.has(u.experienceId)){let h=u.trace.find(w=>w.step==="check-frequency-cap");h?.input&&typeof h.input=="object"&&"per"in h.input&&(p=h.input.per,r.set(u.experienceId,p));}y(u.experienceId,p);}}});};function Se(){return typeof navigator>"u"?false:navigator.doNotTrack==="1"||navigator.msDoNotTrack==="1"||window.doNotTrack==="1"}function Q(e,t,n,r,s,l){return {isFirstVisit:e,totalVisits:t,sessionVisits:n,firstVisitTime:r,lastVisitTime:s,timestamp:l}}var $=(e,t,n)=>{e.ns("pageVisits"),e.defaults({pageVisits:{enabled:true,respectDNT:true,sessionKey:"pageVisits:session",totalKey:"pageVisits:total",ttl:void 0,autoIncrement:true}}),t.storage||(console.warn("[PageVisits] Storage plugin not found, auto-loading..."),t.use(P));let r=t,s=0,l=0,f,i,a=false,b=false;function v(){let o=n.get("pageVisits.sessionKey")??"pageVisits:session",u=n.get("pageVisits.totalKey")??"pageVisits:total";s=r.storage.get(o,{backend:"sessionStorage"})??0;let h=r.storage.get(u,{backend:"localStorage"});h?(l=h.count??0,f=h.first,i=h.last,a=false):(l=0,f=void 0,i=void 0,a=true);}function d(){let o=n.get("pageVisits.sessionKey")??"pageVisits:session",u=n.get("pageVisits.totalKey")??"pageVisits:total",p=n.get("pageVisits.ttl");r.storage.set(o,s,{backend:"sessionStorage"});let h={count:l,first:f??Date.now(),last:i??Date.now()};r.storage.set(u,h,{backend:"localStorage",...p&&{ttl:p}});}function c(){b||(v(),b=true),s+=1,l+=1;let o=Date.now();a&&(f=o),i=o,d();let u=Q(a,l,s,f,i,o);e.emit("pageVisits:incremented",u),a&&(a=false);}function y(){let o=n.get("pageVisits.sessionKey")??"pageVisits:session",u=n.get("pageVisits.totalKey")??"pageVisits:total";r.storage.remove(o,{backend:"sessionStorage"}),r.storage.remove(u,{backend:"localStorage"}),s=0,l=0,f=void 0,i=void 0,a=false,b=false,e.emit("pageVisits:reset");}function m(){return Q(a,l,s,f,i,Date.now())}function g(){let o=n.get("pageVisits.enabled")??true,u=n.get("pageVisits.respectDNT")??true,p=n.get("pageVisits.autoIncrement")??true;if(u&&Se()){e.emit("pageVisits:disabled",{reason:"dnt"});return}if(!o){e.emit("pageVisits:disabled",{reason:"config"});return}p&&c();}t.on("sdk:ready",g),e.expose({pageVisits:{getTotalCount:()=>l,getSessionCount:()=>s,isFirstVisit:()=>a,getFirstVisitTime:()=>f,getLastVisitTime:()=>i,increment:c,reset:y,getState:m}});};function Te(){if(typeof window>"u")return "desktop";let e=navigator.userAgent,t=/Android|webOS|iPhone|iPod|BlackBerry|IEMobile|Opera Mini/i.test(e),n=/iPad|Android(?!.*Mobile)/i.test(e),r=window.innerWidth;return r<768?"mobile":r<1024?"tablet":t?"mobile":n?"tablet":"desktop"}function Y(e,t){let n=null,r=0;return function(...l){let f=Date.now(),i=t-(f-r);i<=0||i>t?(n&&(clearTimeout(n),n=null),r=f,e(...l)):n||(n=setTimeout(()=>{r=Date.now(),n=null,e(...l);},i));}}function G(e){if(typeof document>"u")return 0;let t=document.scrollingElement||document.documentElement,n=t.scrollTop,r=t.scrollHeight,s=t.clientHeight;return r<=s?100:Math.min(e?(n+s)/r*100:n/(r-s)*100,100)}function Ce(e,t,n,r,s){let l=Math.min(e/t*50,50),f=Math.min(n/5*30,30),i=Math.min(r/s*20,20);return Math.max(0,100-(l+f+i))}var j=(e,t,n)=>{e.ns("experiences.scrollDepth"),e.defaults({scrollDepth:{thresholds:[25,50,75,100],throttle:100,includeViewportHeight:true,recalculateOnResize:true,trackAdvancedMetrics:false,fastScrollVelocityThreshold:3,disableOnMobile:false}});let r=n.get("scrollDepth");if(!r)return;let s=r,l=Te();if(s.disableOnMobile&&l==="mobile")return;let f=0,i=new Set,a=Date.now(),b=0,v=Date.now(),d=null,c=0,y=0,m=new Map;function g(){let k=G(s.includeViewportHeight??true),S=Date.now(),T=(document.scrollingElement||document.documentElement).scrollTop,E=0;if(s.trackAdvancedMetrics){let C=S-v,_=T-b;E=C>0?Math.abs(_)/C:0;let D=_>0?"down":_<0?"up":d;D&&d&&D!==d&&c++,D==="up"&&C>0&&(y+=C),d=D,b=T,v=S;}f=Math.max(f,k);for(let C of s.thresholds||[])if(k>=C&&!i.has(C)){i.add(C),s.trackAdvancedMetrics&&m.set(C,S-a);let _={triggered:true,timestamp:S,percent:Math.round(k*100)/100,maxPercent:Math.round(f*100)/100,threshold:C,thresholdsCrossed:Array.from(i).sort((D,N)=>D-N),device:l};if(s.trackAdvancedMetrics){let D=s.fastScrollVelocityThreshold||3,N=E>D,le=Ce(E,D,c,y,S-a);_.advanced={timeToThreshold:S-a,velocity:Math.round(E*1e3)/1e3,isFastScrolling:N,directionChanges:c,timeScrollingUp:y,engagementScore:Math.round(le)},c=0;}t.emit("trigger:scrollDepth",_);}}let o=Y(g,s.throttle||100),u=Y(g,s.throttle||100);function p(){typeof window>"u"||typeof document>"u"||(window.addEventListener("scroll",o,{passive:true}),s.recalculateOnResize&&window.addEventListener("resize",u,{passive:true}));}function h(){window.removeEventListener("scroll",o),window.removeEventListener("resize",u);}let w=()=>{h();};return t.on("destroy",w),e.expose({scrollDepth:{getMaxPercent:()=>f,getCurrentPercent:()=>G(s.includeViewportHeight??true),getThresholdsCrossed:()=>Array.from(i).sort((k,S)=>k-S),getDevice:()=>l,getAdvancedMetrics:()=>s.trackAdvancedMetrics?{timeOnPage:Date.now()-a,directionChanges:c,timeScrollingUp:y,thresholdTimes:Object.fromEntries(m)}:null,reset:()=>{f=0,i.clear(),c=0,y=0,m.clear(),d=null;}}}),typeof window<"u"&&setTimeout(p,0),()=>{h(),t.off("destroy",w);}};function F(e,t){return Date.now()-e-t}function X(){return typeof document>"u"?false:document.hidden||false}function Ee(e,t,n,r){let s=Date.now(),l=s-e,f=l-t;return {timestamp:s,elapsed:l,activeElapsed:f,wasPaused:n,visibilityChanges:r}}var U=(e,t,n)=>{e.ns("experiences.timeDelay"),e.defaults({timeDelay:{delay:0,pauseWhenHidden:true}});let r=n.get("timeDelay");if(!r)return;let s=r.delay??0,l=r.pauseWhenHidden??true;if(s<=0)return;let f=Date.now(),i=false,a=false,b=0,v=0,d=0,c=null,y=null;function m(){if(i)return;i=true;let w=Ee(f,b,d>0,d);t.emit("trigger:timeDelay",w),u();}function g(w){c&&clearTimeout(c),c=setTimeout(()=>{m();},w);}function o(){let w=X();if(w&&!a)a=true,v=Date.now(),d++,c&&(clearTimeout(c),c=null);else if(!w&&a){a=false;let k=Date.now()-v;b+=k,d++;let S=F(f,b),x=s-S;x>0?g(x):m();}}function u(){c&&(clearTimeout(c),c=null),y&&typeof document<"u"&&(document.removeEventListener("visibilitychange",y),y=null);}function p(){l&&X()?(a=true,v=Date.now(),d++):g(s),l&&typeof document<"u"&&(y=o,document.addEventListener("visibilitychange",y));}e.expose({timeDelay:{getElapsed:()=>Date.now()-f,getActiveElapsed:()=>{let w=b;return a&&(w+=Date.now()-v),F(f,w)},getRemaining:()=>{if(i)return 0;let w=F(f,b),k=s-w;return Math.max(0,k)},isPaused:()=>a,isTriggered:()=>i,reset:()=>{i=false,a=false,b=0,v=0,d=0,u(),p();}}}),p();let h=()=>{u();};t.on("destroy",h);};function H(e,t){let n={...e};for(let r in t){if(!Object.hasOwn(t,r))continue;let s=t[r],l=n[r];if(l===void 0){n[r]=s;continue}Z(l)&&Z(s)&&(n[r]=H(l,s));}return n}function Z(e){return e===null||typeof e!="object"?false:Object.prototype.toString.call(e)==="[object Object]"}var De=class{data={};required=new Set;constructor(e={}){this.data=JSON.parse(JSON.stringify(e));}defaults(e){this.data=H(this.data,e);}merge(e){this.data=H(e,this.data);}get(e){let t=e.split("."),n=this.data;for(let r of t){if(n==null)return;n=n[r];}return n}set(e,t){let n=e.split("."),r=n.pop();if(!r)return;let s=this.data;for(let l of n)(s[l]==null||typeof s[l]!="object")&&(s[l]={}),s=s[l];s[r]=t;}markRequired(e){e!=null&&this.required.add(e);}isRequired(e){return this.required.has(e)}getAll(){return JSON.parse(JSON.stringify(this.data))}},Ie=class{subscriptions=[];on(e,t){if(typeof t!="function")throw new TypeError("handler must be a function");let n={pattern:e,compiledPattern:this.compilePattern(e),handler:t};return this.subscriptions.push(n),()=>this.off(e,t)}off(e,t){this.subscriptions=this.subscriptions.filter(n=>!(n.pattern===e&&n.handler===t));}emit(e,...t){for(let n of this.subscriptions)if(n.compiledPattern.test(e))try{n.handler(...t);}catch(r){console.error(`Error in event handler for "${e}":`,r);}}removeAllListeners(){this.subscriptions=[];}compilePattern(e){let n=this.escapeRegExp(e).replace(/\\\*/g,"(.*?)");return new RegExp(`^${n}$`)}escapeRegExp(e){return e.replace(/[\\^$*+?.()|[\]{}]/g,"\\$&")}},_e=class{sdk;constructor(e){this.sdk=e;}expose(e){Object.assign(this.sdk,e);}},Pe=class{name="";ns(e){if(this.name)throw new Error(`Namespace already set to "${this.name}". Cannot reassign to "${e}".`);this.name=e;}},K=class{configInstance;emitter;plugins=new Map;isInitialized=false;constructor(e={}){this.emitter=new Ie,this.configInstance=new De(e);}use(e){let t=new Pe,n=new _e(this),r={ns:t.ns.bind(t),defaults:this.configInstance.defaults.bind(this.configInstance),on:this.emitter.on.bind(this.emitter),off:this.emitter.off.bind(this.emitter),emit:this.emitter.emit.bind(this.emitter),expose:n.expose.bind(n),mustEnable:()=>{t.name&&this.configInstance.markRequired(t.name);}};return e(r,this,this.configInstance),t.name&&this.plugins.set(t.name,{namespace:t,plugin:r}),this}async init(e){if(this.isInitialized){console.warn("SDK already initialized");return}e&&this.configInstance.merge(e),this.emitter.emit("sdk:init");let t=Array.from(this.plugins.keys()).filter(n=>this.configInstance.isRequired(n));for(let n of t)if(!this.plugins.has(n))throw new Error(`Required plugin "${n}" is not registered`);this.isInitialized=true,this.emitter.emit("sdk:ready");}async destroy(){this.emitter.emit("sdk:destroy"),this.plugins.clear(),this.emitter.removeAllListeners(),this.isInitialized=false;}get(e){return this.configInstance.get(e)}set(e,t){this.configInstance.set(e,t);}on(e,t){return this.emitter.on(e,t)}off(e,t){this.emitter.off(e,t);}emit(e,...t){this.emitter.emit(e,...t);}getAll(){return this.configInstance.getAll()}isReady(){return this.isInitialized}};var q=class{sdk;experiences=new Map;decisions=[];initialized=false;destroyed=false;triggerContext={triggers:{}};constructor(t={}){this.sdk=new K({name:"experience-sdk",...t}),this.sdk.use(P),this.sdk.use(R),this.sdk.use(L),this.sdk.use(V),this.sdk.use(j),this.sdk.use($),this.sdk.use(U),this.sdk.use(A),this.setupTriggerListeners();}setupTriggerListeners(){this.sdk.on("trigger:*",(t,n)=>{let r=t.replace("trigger:","");this.triggerContext.triggers=this.triggerContext.triggers||{},this.triggerContext.triggers[r]={triggered:true,timestamp:Date.now(),...n},this.evaluate(this.triggerContext);});}async init(t){if(this.initialized){console.warn("[experiences] Already initialized");return}this.destroyed&&(this.sdk=new K({name:"experience-sdk",...t}),this.sdk.use(P),this.sdk.use(R),this.sdk.use(L),this.sdk.use(V),this.sdk.use(j),this.sdk.use($),this.sdk.use(U),this.sdk.use(A),this.destroyed=false),t&&Object.entries(t).forEach(([n,r])=>{this.sdk.set(n,r);}),await this.sdk.init(),this.initialized=true,this.sdk.emit("experiences:ready");}register(t,n){let r={id:t,...n};this.experiences.set(t,r),r.frequency&&this.sdk.frequency?._registerExperience&&this.sdk.frequency._registerExperience(t,r.frequency.per),this.sdk.emit("experiences:registered",{id:t,experience:r});}evaluate(t){let n=Date.now(),r=M(t),s,l=[],f=[];for(let[,a]of this.experiences){let b=z(a,r);if(l.push(...b.reasons),f.push(...b.trace),b.matched){if(a.frequency&&this.sdk.frequency){let v=Date.now(),d=this.sdk.frequency.hasReachedCap(a.id,a.frequency.max,a.frequency.per);if(f.push({step:"check-frequency-cap",timestamp:v,duration:Date.now()-v,input:a.frequency,output:d,passed:!d}),d){let y=this.sdk.frequency.getImpressionCount(a.id,a.frequency.per);l.push(`Frequency cap reached (${y}/${a.frequency.max} this ${a.frequency.per})`);continue}let c=this.sdk.frequency.getImpressionCount(a.id,a.frequency.per);l.push(`Frequency cap not reached (${c}/${a.frequency.max} this ${a.frequency.per})`);}s=a;break}}let i={show:!!s,experienceId:s?.id,reasons:l,trace:f,context:r,metadata:{evaluatedAt:Date.now(),totalDuration:Date.now()-n,experiencesEvaluated:this.experiences.size}};return this.decisions.push(i),this.sdk.emit("experiences:evaluated",{decision:i,experience:s}),i}evaluateAll(t){let n=M(t),r=Array.from(this.experiences.values()).sort((i,a)=>{let b=i.priority??0;return (a.priority??0)-b}),s=[];for(let i of r){let a=Date.now(),b=z(i,n),v=b.matched,d=[...b.reasons],c=[...b.trace];if(v&&i.frequency&&this.sdk.frequency){let m=Date.now(),g=this.sdk.frequency.hasReachedCap(i.id,i.frequency.max,i.frequency.per);if(c.push({step:"check-frequency-cap",timestamp:m,duration:Date.now()-m,input:i.frequency,output:g,passed:!g}),g){let o=this.sdk.frequency.getImpressionCount(i.id,i.frequency.per);d.push(`Frequency cap reached (${o}/${i.frequency.max} this ${i.frequency.per})`),v=false;}else {let o=this.sdk.frequency.getImpressionCount(i.id,i.frequency.per);d.push(`Frequency cap not reached (${o}/${i.frequency.max} this ${i.frequency.per})`);}}let y={show:v,experienceId:i.id,reasons:d,trace:c,context:n,metadata:{evaluatedAt:Date.now(),totalDuration:Date.now()-a,experiencesEvaluated:1}};s.push(y),this.decisions.push(y);}let l=s.filter(i=>i.show),f=l.map(i=>i.experienceId&&this.experiences.get(i.experienceId)).filter(i=>i!==void 0);return this.sdk.emit("experiences:evaluated",l.map((i,a)=>({decision:i,experience:f[a]}))),s}explain(t){let n=this.experiences.get(t);if(!n)return null;let r=M(),s=z(n,r);return {show:s.matched,experienceId:t,reasons:s.reasons,trace:s.trace,context:r,metadata:{evaluatedAt:Date.now(),totalDuration:0,experiencesEvaluated:1}}}getState(){return {initialized:this.initialized,experiences:new Map(this.experiences),decisions:[...this.decisions],config:this.sdk?this.sdk.getAll():{}}}on(t,n){return this.sdk.on(t,n)}async destroy(){this.sdk&&await this.sdk.destroy(),this.destroyed=true,this.experiences.clear(),this.decisions=[],this.initialized=false;}};function M(e){return {url:e?.url??(typeof window<"u"?window.location.href:""),timestamp:Date.now(),user:e?.user,custom:e?.custom,triggers:e?.triggers??{}}}function z(e,t){let n=[],r=[],s=true;if(e.targeting.url){let l=Date.now(),f=ee(e.targeting.url,t.url);r.push({step:"evaluate-url-rule",timestamp:l,duration:Date.now()-l,input:{rule:e.targeting.url,url:t.url},output:f,passed:f}),f?n.push("URL matches targeting rule"):(n.push("URL does not match targeting rule"),s=false);}return {matched:s,reasons:n,trace:r}}function ee(e,t=""){return e.equals!==void 0?t===e.equals:e.contains!==void 0?t.includes(e.contains):e.matches!==void 0?e.matches.test(t):true}function B(e){return new q(e)}var I=B();async function te(e){return I.init(e)}function ne(e,t){I.register(e,t);}function re(e){return I.evaluate(e)}function se(e){return I.evaluateAll(e)}function oe(e){return I.explain(e)}function ie(){return I.getState()}function ae(e,t){return I.on(e,t)}async function ce(){return I.destroy()}var qe={createInstance:B,init:te,register:ne,evaluate:re,evaluateAll:se,explain:oe,getState:ie,on:ae,destroy:ce};typeof window<"u"&&(window.experiences=qe);exports.ExperienceRuntime=q;exports.bannerPlugin=A;exports.buildContext=M;exports.createInstance=B;exports.debugPlugin=R;exports.destroy=ce;exports.evaluate=re;exports.evaluateAll=se;exports.evaluateExperience=z;exports.evaluateUrlRule=ee;exports.explain=oe;exports.frequencyPlugin=L;exports.getState=ie;exports.init=te;exports.on=ae;exports.register=ne;return exports;})({});//# sourceMappingURL=experience-sdk.global.js.map
186
217
  //# sourceMappingURL=experience-sdk.global.js.map