@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.
package/dist/experience-sdk.d.ts
CHANGED
|
@@ -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
|
|
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,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,""").replace(/'/g,"'")}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: #
|
|
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
|
-
|
|
37
|
-
|
|
38
|
-
|
|
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
|
|
49
|
-
|
|
50
|
-
|
|
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:
|
|
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: #
|
|
92
|
+
background: #f3f4f6;
|
|
88
93
|
color: #374151;
|
|
89
|
-
border: 1px solid #
|
|
94
|
+
border: 1px solid #e5e7eb;
|
|
90
95
|
}
|
|
91
96
|
|
|
92
97
|
.xp-banner__button--secondary:hover {
|
|
93
|
-
background: #
|
|
98
|
+
background: #e5e7eb;
|
|
94
99
|
}
|
|
95
100
|
|
|
96
101
|
.xp-banner__button--link {
|
|
97
102
|
background: transparent;
|
|
98
103
|
color: #2563eb;
|
|
99
|
-
padding:
|
|
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:
|
|
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: #
|
|
112
|
-
font-size:
|
|
116
|
+
color: #9ca3af;
|
|
117
|
+
font-size: 20px;
|
|
113
118
|
line-height: 1;
|
|
114
119
|
cursor: pointer;
|
|
115
|
-
padding:
|
|
120
|
+
padding: 4px;
|
|
116
121
|
margin: 0;
|
|
117
|
-
|
|
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
|
-
|
|
133
|
+
color: #111827;
|
|
134
|
+
background: #f3f4f6;
|
|
124
135
|
}
|
|
125
136
|
|
|
126
137
|
@media (max-width: 640px) {
|
|
127
138
|
.xp-banner__container {
|
|
128
|
-
flex-
|
|
129
|
-
|
|
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
|
-
|
|
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: #
|
|
146
|
-
color: #
|
|
147
|
-
border-bottom-color: #
|
|
168
|
+
background: #111827;
|
|
169
|
+
color: #f9fafb;
|
|
170
|
+
border-bottom-color: #1f2937;
|
|
148
171
|
}
|
|
149
172
|
|
|
150
173
|
.xp-banner--bottom {
|
|
151
|
-
border-top-color: #
|
|
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: #
|
|
164
|
-
color: #
|
|
165
|
-
border-color: #
|
|
190
|
+
background: #1f2937;
|
|
191
|
+
color: #f9fafb;
|
|
192
|
+
border-color: #374151;
|
|
166
193
|
}
|
|
167
194
|
|
|
168
195
|
.xp-banner__button--secondary:hover {
|
|
169
|
-
background: #
|
|
196
|
+
background: #374151;
|
|
170
197
|
}
|
|
171
198
|
|
|
172
199
|
.xp-banner__button--link {
|
|
173
|
-
color: #
|
|
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: #
|
|
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="×",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="×",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
|