@formbricks/js
Advanced tools
Comparing version 1.6.5 to 2.0.0-beta.0
@@ -1,37 +0,11 @@ | ||
import { FormbricksAPI } from '@formbricks/api'; | ||
import { TJsConfigInput } from '@formbricks/types/js'; | ||
declare const formbricks: any; | ||
export default formbricks; | ||
declare const _default: { | ||
init: (initConfig: { | ||
environmentId: string; | ||
apiHost: string; | ||
errorHandler?: ((args_0: any, ...args_1: unknown[]) => void) | undefined; | ||
userId?: string | undefined; | ||
attributes?: Record<string, string | number> | undefined; | ||
}) => Promise<void>; | ||
setUserId: () => Promise<void>; | ||
setEmail: (email: string) => Promise<void>; | ||
setAttribute: (key: string, value: any) => Promise<void>; | ||
track: (name: string, properties?: any) => Promise<void>; | ||
logout: () => Promise<void>; | ||
reset: () => Promise<void>; | ||
registerRouteChange: () => Promise<void>; | ||
getApi: () => FormbricksAPI; | ||
}; | ||
export default _default; | ||
export { } | ||
declare const formbricks: { | ||
init: (initConfig: TJsConfigInput) => Promise<void>; | ||
setUserId: () => Promise<void>; | ||
setEmail: (email: string) => Promise<void>; | ||
setAttribute: (key: string, value: any) => Promise<void>; | ||
track: (name: string, properties?: any) => Promise<void>; | ||
logout: () => Promise<void>; | ||
reset: () => Promise<void>; | ||
registerRouteChange: () => Promise<void>; | ||
getApi: () => FormbricksAPI; | ||
}; | ||
declare global { | ||
interface Window { | ||
formbricks: any; | ||
} | ||
} | ||
export declare type FormbricksType = typeof formbricks; | ||
export { } |
@@ -1,2 +0,64 @@ | ||
"use strict";var e=Object.defineProperty,t=(t,n,s)=>(((t,n,s)=>{n in t?e(t,n,{enumerable:!0,configurable:!0,writable:!0,value:s}):t[n]=s})(t,"symbol"!=typeof n?n+"":n,s),s);const n=e=>({ok:!1,error:e});async function s(e,t,s,i){const r=new URL(t,e),o=JSON.stringify(i),a=(d=fetch,(...e)=>{try{return{ok:!0,data:d(...e)}}catch(t){return{ok:!1,error:t}}})(r.toString(),{method:s,headers:{"Content-Type":"application/json"},body:o});var d;if(!1===a.ok)return n(a.error);const c=await a.data,{data:u}=await c.json();return c.ok?(e=>({ok:!0,data:e}))(u):n({code:"network_error",message:c.statusText,status:c.status,url:r})}class i{constructor(e,t){this.apiHost=e,this.environmentId=t}async create(e){return s(this.apiHost,`/api/v1/client/${this.environmentId}/actions`,"POST",e)}}class r{constructor(e,t){this.apiHost=e,this.environmentId=t}async create(e){return s(this.apiHost,`/api/v1/client/${this.environmentId}/displays`,"POST",e)}async update(e,t){return s(this.apiHost,`/api/v1/client/${this.environmentId}/displays/${e}`,"PUT",t)}}class o{constructor(e,t){this.apiHost=e,this.environmentId=t}async create(e){return s(this.apiHost,`/api/v1/client/${this.environmentId}/people`,"POST",{environmentId:this.environmentId,userId:e})}async update(e,t){return s(this.apiHost,`/api/v1/client/${this.environmentId}/people/${e}`,"POST",t)}}class a{constructor(e,t){this.apiHost=e,this.environmentId=t}async create(e){return s(this.apiHost,`/api/v1/client/${this.environmentId}/responses`,"POST",e)}async update({responseId:e,finished:t,data:n,ttc:i}){return s(this.apiHost,`/api/v1/client/${this.environmentId}/responses/${e}`,"PUT",{finished:t,data:n,ttc:i})}}class d{constructor(e,t){this.apiHost=e,this.environmentId=t}async uploadFile(e,{allowedFileExtensions:t,surveyId:n}={}){if(!(e instanceof Blob&&e instanceof File))throw new Error("Invalid file type. Expected Blob or File, but received "+typeof e);const s={fileName:e.name,fileType:e.type,allowedFileExtensions:t,surveyId:n},i=await fetch(`${this.apiHost}/api/v1/client/${this.environmentId}/storage`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(s)});if(!i.ok)throw new Error(`Upload failed with status: ${i.status}`);const r=await i.json(),{data:o}=r,{signedUrl:a,fileUrl:d,signingData:c,presignedFields:u,updatedFileName:l}=o;let g={};if(c){const{signature:t,timestamp:s,uuid:i}=c;g={"X-File-Type":e.type,"X-File-Name":encodeURIComponent(l),"X-Survey-ID":n??"","X-Signature":t,"X-Timestamp":String(s),"X-UUID":i}}const p=new FormData;u&&Object.keys(u).forEach((e=>{p.append(e,u[e])})),p.append("file",e);const h=await fetch(a,{method:"POST",...c?{headers:g}:{},body:p});if(!h.ok){if(c){const e=await h.json(),t=new Error(e.message);throw t.name="FileTooLargeError",t}const e=await h.text();if(u&&e&&e.includes("EntityTooLarge")){const e=new Error("File size exceeds the size limit for your plan");throw e.name="FileTooLargeError",e}throw new Error(`Upload failed with status: ${h.status}`)}return d}}class c{constructor(e){const{apiHost:t,environmentId:n}=e;this.response=new a(t,n),this.display=new r(t,n),this.action=new i(t,n),this.people=new o(t,n),this.storage=new d(t,n)}}class u{constructor(e){this.client=new c(e)}}class l{constructor(){this.logLevel="error"}static getInstance(){return l.instance||(l.instance=new l),l.instance}configure(e){e&&void 0!==e.logLevel&&(this.logLevel=e.logLevel)}logger(e,t){if("debug"===t&&"debug"!==this.logLevel)return;const n=`🧱 Formbricks - ${(new Date).toISOString()} [${t.toUpperCase()}] - ${e}`;"error"===t?console.error(n):console.log(n)}debug(e){this.logger(e,"debug")}error(e){this.logger(e,"error")}}const g=e=>({ok:!0,value:e}),p=e=>({ok:!1,error:e});const h=e=>(...t)=>{try{return{ok:!0,value:e(...t)}}catch(n){return{ok:!1,error:n}}},v=l.getInstance(),f=class e{constructor(e){this.customized=!1,e?(this.handleError=e,this.customized=!0):this.handleError=e=>l.getInstance().error(JSON.stringify(e))}static getInstance(){return e.instance||(e.instance=new e),e.instance}static init(t){this.initialized=!0,e.instance=new e(t)}printStatus(){v.debug("Custom error handler: "+(this.customized?"yes":"no"))}handle(e){console.warn("🧱 Formbricks - Global error: ",e),this.handleError(e)}};f.initialized=!1;let y=f;const m="formbricks-js";class w{constructor(){this.config=null;const e=this.loadFromLocalStorage();e.ok&&(this.config=e.value)}static getInstance(){return w.instance||(w.instance=new w),w.instance}update(e){e&&(this.config={...this.config,...e,status:e.status||"success"},this.saveToLocalStorage())}get(){if(!this.config)throw new Error("config is null, maybe the init function was not called?");return this.config}loadFromLocalStorage(){if("undefined"!=typeof window){const e=localStorage.getItem(m);if(e){const t=JSON.parse(e);return t.expiresAt&&new Date(t.expiresAt)<=new Date?p(new Error("Config in local storage has expired")):g(JSON.parse(e))}}return p(new Error("No or invalid config in local storage"))}saveToLocalStorage(){return h((()=>localStorage.setItem(m,JSON.stringify(this.config))))()}resetConfig(){return this.config=null,h((()=>localStorage.removeItem(m)))()}}const I=(e,t)=>{const n=Math.abs(t.getTime()-e.getTime());return Math.floor(n/864e5)},b=()=>window.location.search.includes("formbricksDebug=true"),k=w.getInstance(),S=l.getInstance();let H=null;const C=async(e,t=!1)=>{var n;try{const i=await(async({apiHost:e,environmentId:t,userId:n},s)=>{try{const i=`${e}/api/v1/client/${t}/in-app/sync`,r="?version=1.6.5";let o={};if((s||b())&&(o.cache="no-cache",S.debug("No cache option set for sync")),!n){const e=i+r,t=await fetch(e,o);if(!t.ok){const n=await t.json();return p({code:"network_error",status:t.status,message:"Error syncing with backend",url:e,responseMessage:n.message})}return g((await t.json()).data)}const a=`${i}/${n}${r}`,d=await fetch(a,o);if(!d.ok){const e=await d.json();return p({code:"network_error",status:d.status,message:"Error syncing with backend",url:a,responseMessage:e.message})}const c=await d.json(),{data:u}=c;return g(u)}catch(i){return p(i)}})(e,t);if(!0!==(null==i?void 0:i.ok))throw i.error;let r;try{r=k.get().state}catch(s){}let o={surveys:i.value.surveys,noCodeActionClasses:i.value.noCodeActionClasses,product:i.value.product,attributes:(null==(n=i.value.person)?void 0:n.attributes)||{}};if(e.userId){const e=o.surveys.map((e=>e.name));S.debug("Fetched "+e.length+" surveys during sync: "+e.join(", "))}else{o={...o,displays:(null==r?void 0:r.displays)||[]},o=E(o);const e=o.surveys.map((e=>e.name));S.debug("Fetched "+e.length+" surveys during sync: "+e.join(", "))}k.update({apiHost:e.apiHost,environmentId:e.environmentId,userId:e.userId,state:o,expiresAt:new Date((new Date).getTime()+12e4)})}catch(i){throw S.error(`Error during sync: ${i}`),i}},E=e=>{const{displays:t,product:n}=e;let{surveys:s}=e;if(!t)return e;let i=s.filter((e=>{if("respondMultiple"===e.displayOption)return!0;if("displayOnce"===e.displayOption)return 0===t.filter((t=>t.surveyId===e.id)).length;if("displayMultiple"===e.displayOption)return 0===t.filter((t=>t.surveyId===e.id&&t.responded)).length;throw Error("Invalid displayOption")}));const r=t.length>0?t[t.length-1]:void 0;return i=i.filter((e=>{if(r){if(null!==e.recontactDays){const n=t.filter((t=>t.surveyId===e.id))[0];return!n||I(new Date,new Date(n.createdAt))>=e.recontactDays}return null===n.recontactDays||I(new Date,new Date(r.createdAt))>=n.recontactDays}return!0})),{...e,surveys:i}},A=()=>{"undefined"!=typeof window&&null!==H&&(window.clearInterval(H),H=null)},F=e=>new Promise((t=>setTimeout(t,e)));class ${constructor(e,n){t(this,"queue",[]),t(this,"config"),t(this,"surveyState"),t(this,"isRequestInProgress",!1),t(this,"api"),this.config=e,this.surveyState=n,this.api=new u({apiHost:e.apiHost,environmentId:e.environmentId})}add(e){this.surveyState.accumulateResponse(e),this.config.setSurveyState&&this.config.setSurveyState(this.surveyState),this.queue.push(e),this.processQueue()}async processQueue(){if(this.isRequestInProgress)return;if(0===this.queue.length)return;this.isRequestInProgress=!0;const e=this.queue[0];let t=0;for(;t<this.config.retryAttempts;){if(await this.sendResponse(e)){this.queue.shift();break}console.error("Formbricks: Failed to send response. Retrying...",t),await F(1e3),t++}t>=this.config.retryAttempts?(console.error("Failed to send response after 2 attempts."),this.config.onResponseSendingFailed&&this.config.onResponseSendingFailed(e),this.isRequestInProgress=!1):(e.finished&&this.config.onResponseSendingFinished&&this.config.onResponseSendingFinished(),this.isRequestInProgress=!1,this.processQueue())}async sendResponse(e){try{if(null!==this.surveyState.responseId)await this.api.client.response.update({...e,responseId:this.surveyState.responseId});else{const n=await this.api.client.response.create({...e,surveyId:this.surveyState.surveyId,userId:this.surveyState.userId||null,singleUseId:this.surveyState.singleUseId||null});if(!n.ok)throw new Error("Could not create response");if(this.surveyState.displayId)try{await this.api.client.display.update(this.surveyState.displayId,{responseId:n.data.id})}catch(t){console.error("Failed to update display, proceeding with the response.",t)}this.surveyState.updateResponseId(n.data.id),this.config.setSurveyState&&this.config.setSurveyState(this.surveyState)}return!0}catch(t){return console.error(t),!1}}updateSurveyState(e){this.surveyState=e}}class x{constructor(e,n,s,i){t(this,"responseId",null),t(this,"displayId",null),t(this,"userId",null),t(this,"surveyId"),t(this,"responseAcc",{finished:!1,data:{},ttc:{}}),t(this,"singleUseId"),this.surveyId=e,this.userId=i??null,this.singleUseId=n??null,this.responseId=s??null}setSurveyId(e){this.surveyId=e,this.clear()}copy(){const e=new x(this.surveyId,this.singleUseId??void 0,this.responseId??void 0,this.userId??void 0);return e.responseId=this.responseId,e.responseAcc=this.responseAcc,e}updateResponseId(e){this.responseId=e}updateDisplayId(e){this.displayId=e}updateUserId(e){this.userId=e}accumulateResponse(e){this.responseAcc={finished:e.finished,ttc:e.ttc,data:{...this.responseAcc.data,...e.data}}}isResponseFinished(){return this.responseAcc.finished}clear(){this.responseId=null,this.responseAcc={finished:!1,data:{},ttc:{}}}}const D=x;let O=!1,U=async function(e){if(e.clientY<=0){const e=await De("Exit Intent (Desktop)");if(!0!==e.ok)return p(e.error)}};const P=()=>{O&&(document.removeEventListener("mouseleave",U),O=!1)};let T=!1,R=!1,L=async()=>{const e=window.scrollY,t=window.innerHeight,n=document.documentElement.scrollHeight;if(0===e&&(R=!1),!R&&e/(n-t)>=.5){R=!0;const e=await De("50% Scroll");if(!0!==e.ok)return p(e.error)}};const N=()=>{T&&(window.removeEventListener("scroll",L),T=!1)},z=w.getInstance(),j=l.getInstance(),q=y.getInstance(),M=async()=>{var e;j.debug(`Checking page url: ${window.location.href}`);const{state:t}=z.get(),{noCodeActionClasses:n=[],surveys:s=[]}=t??{},i=n.filter((e=>{const{innerHtml:t,cssSelector:n,pageUrl:s}=e.noCodeConfig||{};return s&&!t&&!n})),r=s.filter((e=>{var t;const{pageUrl:n,cssSelector:s,innerHtml:i}=(null==(t=e.inlineTriggers)?void 0:t.noCodeConfig)||{};return n&&!s&&!i}));if(i.length>0)for(const o of i){if(!(null==(e=o.noCodeConfig)?void 0:e.pageUrl))continue;const{noCodeConfig:{pageUrl:t}}=o,n=Q(window.location.href,t.value,t.rule);if(!0!==n.ok)return p(n.error);if(!1===n.value)continue;const s=await De(o.name);if(!0!==s.ok)return p(s.error)}return r.length>0&&r.forEach((e=>{const{noCodeConfig:t}=e.inlineTriggers??{},{pageUrl:n}=t??{};if(n){const t=Q(window.location.href,n.value,n.rule);if(!0!==t.ok)return p(t.error);if(!1===t.value)return;Se(e)}})),{ok:!0,value:void 0}};let _=!1;const B=()=>M(),J=["hashchange","popstate","pushstate","replacestate","load"],X=()=>{"undefined"!=typeof window&&_&&(J.forEach((e=>window.removeEventListener(e,B))),_=!1)};function Q(e,t,n){switch(n){case"exactMatch":return g(e===t);case"contains":return g(e.includes(t));case"startsWith":return g(e.startsWith(t));case"endsWith":return g(e.endsWith(t));case"notMatch":return g(e!==t);case"notContains":return g(!e.includes(t));default:return p({code:"invalid_match_type",message:"Invalid match type"})}}const W=(e,t)=>{var n,s,i,r,o,a,d,c;const u=null==(s=null==(n=t.noCodeConfig)?void 0:n.innerHtml)?void 0:s.value,l=null==(r=null==(i=t.noCodeConfig)?void 0:i.cssSelector)?void 0:r.value,g=null==(a=null==(o=t.noCodeConfig)?void 0:o.pageUrl)?void 0:a.value,p=null==(c=null==(d=t.noCodeConfig)?void 0:d.pageUrl)?void 0:c.rule;if(!u&&!l&&!g)return!1;if(u&&e.innerHTML!==u)return!1;if(l){const t=l.split(/\s*(?=[.#])/);for(let n of t)if(!e.matches(n))return!1}if(g&&p){const e=Q(window.location.href,g,p);if(!e.ok||!e.value)return!1}return!0};let Y=!1;const G=e=>(e=>{const{state:t}=z.get();if(!t)return;const{noCodeActionClasses:n}=t;if(!n)return;const s=e.target;n.forEach((e=>{W(s,e)&&De(e.name).then((e=>{var t,n,s;n=e=>{},s=e=>{q.handle(e)},!0===(t=e).ok?n(t.value):s(t.error)}))}));const i=t.surveys;i&&0!==i.length&&i.forEach((e=>{const{inlineTriggers:t}=e;t&&W(s,t)&&Se(e)}))})(e),K=()=>{Y&&(document.removeEventListener("click",G),Y=!1)};let V=!1;const Z=()=>{"undefined"!=typeof window&&null===H&&(H=window.setInterval((async()=>{try{if(k.get().expiresAt&&new Date(k.get().expiresAt)>=new Date)return;S.debug("Config has expired. Starting sync."),await C({apiHost:k.get().apiHost,environmentId:k.get().environmentId,userId:k.get().userId})}catch(e){S.error(`Error during expiry check: ${e}`),S.debug("Extending config and try again later.");const t=k.get();k.update(t)}}),3e4)),"undefined"==typeof window||_||(J.forEach((e=>window.addEventListener(e,B))),_=!0),"undefined"==typeof window||Y||(document.addEventListener("click",G),Y=!0),"undefined"==typeof document||O||(document.querySelector("body").addEventListener("mouseleave",U),O=!0),"undefined"==typeof window||T||(window.addEventListener("load",(()=>{window.addEventListener("scroll",L)})),T=!0)},ee=()=>{A(),X(),K(),P(),N(),V&&(window.removeEventListener("beforeunload",(()=>{A(),X(),K(),P(),N()})),V=!1)},te=w.getInstance(),ne=l.getInstance(),se=async()=>(ne.error("'setUserId' is no longer supported. Please set the userId in the init call instead."),{ok:!0,value:void 0}),ie=async(e,t)=>{if(ne.debug("Setting attribute: "+e+" to value: "+t),((e,t)=>te.get().state.attributes[e]===t)(e,t.toString()))return ne.debug("Attribute already set to this value. Skipping update."),{ok:!0,value:void 0};const n=await(async(e,t)=>{const{apiHost:n,environmentId:s,userId:i}=te.get();if(!i)return p({code:"missing_person",message:"Unable to update attribute. User identification deactivated. No userId set."});const r={attributes:{[e]:t}},o=new u({apiHost:n,environmentId:s}),a=await o.client.people.update(i,r);return a.ok?(a.data.changed&&ne.debug("Attribute updated in Formbricks"),{ok:!0,value:void 0}):p({code:"network_error",status:500,message:`Error updating person with userId ${i}`,url:`${te.get().apiHost}/api/v1/client/${s}/people/${i}`,responseMessage:a.error.message})})(e,t.toString());return n.ok?(te.update({environmentId:te.get().environmentId,apiHost:te.get().apiHost,userId:te.get().userId,state:{...te.get().state,attributes:{...te.get().state.attributes,[e]:t.toString()}},expiresAt:te.get().expiresAt}),{ok:!0,value:void 0}):p(n.error)},re=async()=>{pe(),te.resetConfig()},oe=async()=>{ne.debug("Resetting state & getting new state from backend"),He();const e={environmentId:te.get().environmentId,apiHost:te.get().apiHost,userId:te.get().userId};await re();try{return await le(e),{ok:!0,value:void 0}}catch(t){return p(t)}},ae=w.getInstance(),de=l.getInstance();let ce=!1;const ue=e=>{ce=e},le=async e=>{if(b()&&de.configure({logLevel:"debug"}),ce)return de.debug("Already initialized, skipping initialization."),{ok:!0,value:void 0};let t;try{t=ae.get(),de.debug("Found existing configuration.")}catch(s){de.debug("No existing configuration found.")}if("error"===(null==t?void 0:t.status)){if(de.debug("Formbricks was set to an error state."),(null==t?void 0:t.expiresAt)&&new Date(t.expiresAt)>new Date)return de.debug("Error state is not expired, skipping initialization"),{ok:!0,value:void 0};de.debug("Error state is expired. Continue with initialization.")}if(y.getInstance().printStatus(),de.debug("Start initialize"),!e.environmentId)return de.debug("No environmentId provided"),p({code:"missing_field",field:"environmentId"});if(!e.apiHost)return de.debug("No apiHost provided"),p({code:"missing_field",field:"apiHost"});if(de.debug("Adding widget container to DOM"),Ce(),!e.userId&&e.attributes)return de.error("No userId provided but attributes. Cannot update attributes without userId."),p({code:"missing_field",field:"userId"});let n=null;if(e.userId&&e.attributes){const t=await(async(e,t,n,i)=>{var r,o;if(!n)return p({code:"missing_person",message:"Unable to update attribute. User identification deactivated. No userId set."});const a={...i};try{const e=null==(o=null==(r=te.get())?void 0:r.state)?void 0:o.attributes;if(e)for(const[t,n]of Object.entries(e))a[t]===n&&delete a[t]}catch(s){ne.debug("config not set; sending all attributes to backend")}if(0===Object.keys(a).length)return ne.debug("No attributes to update. Skipping update."),g(a);ne.debug("Updating attributes: "+JSON.stringify(a));const d={attributes:a},c=new u({apiHost:e,environmentId:t}),l=await c.client.people.update(n,d);return l.ok?g(a):p({code:"network_error",status:500,message:`Error updating person with userId ${n}`,url:`${e}/api/v1/client/${t}/people/${n}`,responseMessage:l.error.message})})(e.apiHost,e.environmentId,e.userId,e.attributes);if(!0!==t.ok)return p(t.error);n=t.value}if(t&&t.state&&t.environmentId===e.environmentId&&t.apiHost===e.apiHost&&t.userId===e.userId&&t.expiresAt)if(de.debug("Configuration fits init parameters."),t.expiresAt<new Date){de.debug("Configuration expired.");try{await C({apiHost:e.apiHost,environmentId:e.environmentId,userId:e.userId})}catch(s){he()}}else de.debug("Configuration not expired. Extending expiration."),ae.update(t);else{de.debug("No valid configuration found or it has been expired. Resetting config and creating new one."),ae.resetConfig(),de.debug("Syncing.");try{await C({apiHost:e.apiHost,environmentId:e.environmentId,userId:e.userId})}catch(s){ge()}await De("New Session")}return n&&Object.keys(n).length>0&&ae.update({environmentId:ae.get().environmentId,apiHost:ae.get().apiHost,userId:ae.get().userId,state:{...ae.get().state,attributes:{...ae.get().state.attributes,...e.attributes}},expiresAt:ae.get().expiresAt}),de.debug("Adding event listeners"),Z(),V||(window.addEventListener("beforeunload",(()=>{A(),X(),K(),P(),N()})),V=!0),ue(!0),de.debug("Initialized"),M(),{ok:!0,value:void 0}},ge=()=>{const e={status:"error",expiresAt:new Date((new Date).getTime()+6e5)};throw h((()=>localStorage.setItem(m,JSON.stringify(e))))(),new Error("Could not initialize formbricks")},pe=()=>{de.debug("Deinitializing"),Ee(),ke(!1),ee(),ue(!1)},he=()=>{de.debug("Putting formbricks in error state"),ae.update({...ae.get(),status:"error",expiresAt:new Date((new Date).getTime()+6e5)}),pe()},ve="formbricks-web-container",fe=w.getInstance(),ye=l.getInstance(),me=y.getInstance();let we=!1,Ie=e=>{},be=e=>{};const ke=e=>{we=e},Se=async e=>{if(we)return void ye.debug("A survey is already running. Skipping.");ke(!0),e.delay&&ye.debug(`Delaying survey by ${e.delay} seconds.`);const t=fe.get().state.product,n=new D(e.id,null,null,fe.get().userId),s=new $({apiHost:fe.get().apiHost,environmentId:fe.get().environmentId,retryAttempts:2,onResponseSendingFailed:()=>{Ie(!0)},onResponseSendingFinished:()=>{be(!0)}},n),i=e.productOverwrites??{},r=i.brandColor??t.brandColor,o=i.highlightBorderColor??t.highlightBorderColor,a=i.clickOutsideClose??t.clickOutsideClose,d=i.darkOverlay??t.darkOverlay,c=i.placement??t.placement,l=t.inAppSurveyBranding,g=await Ae();setTimeout((()=>{g.renderSurveyModal({survey:e,brandColor:r,isBrandingEnabled:l,clickOutside:a,darkOverlay:d,highlightBorderColor:o,placement:c,getSetIsError:e=>{Ie=e},getSetIsResponseSendingFinished:e=>{be=e},onDisplay:async()=>{const{userId:t}=fe.get();if(!t){const t={createdAt:new Date,surveyId:e.id,responded:!1},n=fe.get().state.displays,s=n?[...n,t]:[t],i=fe.get();let r=E({...i.state,displays:s});fe.update({...i,state:r})}const i=new u({apiHost:fe.get().apiHost,environmentId:fe.get().environmentId}),r=await i.client.display.create({surveyId:e.id,userId:t});if(!r.ok)throw new Error("Could not create display");const{id:o}=r.data;n.updateDisplayId(o),s.updateSurveyState(n)},onResponse:e=>{const{userId:t}=fe.get();if(!t){const e=fe.get().state.displays,t=e&&e[e.length-1];if(!t)throw new Error("No lastDisplay found");if(!t.responded){t.responded=!0;const n=fe.get();let s=E({...n.state,displays:e});fe.update({...n,state:s})}}t&&n.updateUserId(t),s.updateSurveyState(n),s.add({data:e.data,ttc:e.ttc,finished:e.finished})},onClose:He,onFileUpload:async(e,t)=>{const n=new u({apiHost:fe.get().apiHost,environmentId:fe.get().environmentId});return await n.client.storage.uploadFile(e,t)},onRetry:()=>{Ie(!1),s.processQueue()}})}),1e3*e.delay)},He=async()=>{if(Ee(),Ce(),!fe.get().userId){const e=fe.get().state,t=E(e);return fe.update({...fe.get(),state:t}),void ke(!1)}try{await C({apiHost:fe.get().apiHost,environmentId:fe.get().environmentId,userId:fe.get().userId},!0),ke(!1)}catch(e){me.handle(e),he()}},Ce=()=>{const e=document.createElement("div");e.id=ve,document.body.appendChild(e)},Ee=()=>{var e;null==(e=document.getElementById(ve))||e.remove()},Ae=()=>new Promise(((e,t)=>{if(window.formbricksSurveys)e(window.formbricksSurveys);else{const n=document.createElement("script");n.src="https://unpkg.com/@formbricks/surveys@~1.6.3/dist/index.umd.js",n.async=!0,n.onload=()=>e(window.formbricksSurveys),n.onerror=e=>{console.error("Failed to load Formbricks Surveys library:",e),t(e)},document.head.appendChild(n)}})),Fe=l.getInstance(),$e=w.getInstance(),xe=["Exit Intent (Desktop)","50% Scroll"],De=async e=>{var t;const{userId:n,state:{surveys:s=[]}}=$e.get();s.forEach((async t=>{const{inlineTriggers:n}=t,{codeConfig:s}=n??{};e!==(null==s?void 0:s.identifier)||await Se(t)}));const i={environmentId:$e.get().environmentId,userId:n,name:e};if(n&&!xe.includes(e)){Fe.debug(`Sending action "${e}" to backend`);const t=new u({apiHost:$e.get().apiHost,environmentId:$e.get().environmentId}),s=await t.client.action.create({...i,userId:n});if(!s.ok)return p({code:"network_error",message:`Error tracking action ${e}`,status:500,url:`${$e.get().apiHost}/api/v1/client/${$e.get().environmentId}/actions`,responseMessage:s.error.message});b()&&await C({environmentId:$e.get().environmentId,apiHost:$e.get().apiHost,userId:n},!0)}Fe.debug(`Formbricks: Action "${e}" tracked`);const r=null==(t=$e.get().state)?void 0:t.surveys;return r&&r.length>0?await Oe(e,r):Fe.debug("No active surveys to display"),{ok:!0,value:void 0}},Oe=async(e,t)=>{for(const s of t){if(s.displayPercentage){if(!(n=s.displayPercentage,Math.floor(100*Math.random())+1<=n)){Fe.debug("Survey display skipped based on displayPercentage.");continue}}for(const t of s.triggers)if(t===e)return Fe.debug(`Formbricks: survey ${s.id} triggered by action "${e}"`),void(await Se(s))}var n},Ue=e=>async(...t)=>{try{return{ok:!0,data:await e(...t)}}catch(n){return{ok:!1,error:n}}};l.getInstance().debug("Create command queue");const Pe=new class{constructor(){this.queue=[],this.running=!1,this.resolvePromise=null,this.commandPromise=null}add(e=!0,t,...n){this.queue.push({command:t,checkInitialized:e,commandArgs:n}),this.running||(this.commandPromise=new Promise((e=>{this.resolvePromise=e,this.run()})))}async wait(){this.running&&await this.commandPromise}async run(){for(this.running=!0;this.queue.length>0;){const e=y.getInstance(),t=this.queue.shift();if(!t)continue;if(t.checkInitialized){const t=(de.debug("Check if initialized"),ce&&y.initialized?{ok:!0,value:void 0}:p({code:"not_initialized",message:"Formbricks not initialized. Call initialize() first."}));if(t&&!0!==t.ok){e.handle(t.error);continue}}const n=async()=>await(null==t?void 0:t.command.apply(null,null==t?void 0:t.commandArgs)),s=await Ue(n)();s&&(s.ok&&s.data&&!s.data.ok&&e.handle(s.data.error),!0!==s.ok&&e.handle(s.error))}this.running=!1,this.resolvePromise&&(this.resolvePromise(),this.resolvePromise=null,this.commandPromise=null)}},Te=async(e,t)=>{Pe.add(!0,ie,e,t),await Pe.wait()},Re={init:async e=>{y.init(e.errorHandler),Pe.add(!1,le,e),await Pe.wait()},setUserId:async()=>{Pe.add(!0,se),await Pe.wait()},setEmail:async e=>{Te("email",e),await Pe.wait()},setAttribute:Te,track:async(e,t={})=>{Pe.add(!0,De,e,t),await Pe.wait()},logout:async()=>{Pe.add(!0,re),await Pe.wait()},reset:async()=>{Pe.add(!0,oe),await Pe.wait()},registerRouteChange:async()=>{Pe.add(!0,M),await Pe.wait()},getApi:()=>{const e=w.getInstance(),{environmentId:t,apiHost:n}=e.get();if(!t||!n)throw new Error("formbricks.init() must be called before getApi()");return new u({apiHost:n,environmentId:t})}};module.exports=Re; | ||
let sdkLoadingPromise = null; | ||
let isErrorLoadingSdk = false; | ||
async function loadSDK(apiHost) { | ||
if (!window.formbricks) { | ||
const res = await fetch(`${apiHost}/api/packages/js-core`); | ||
if (!res.ok) | ||
throw new Error("Failed to load Formbricks SDK"); | ||
const sdkScript = await res.text(); | ||
const scriptTag = document.createElement("script"); | ||
scriptTag.innerHTML = sdkScript; | ||
document.head.appendChild(scriptTag); | ||
return new Promise((resolve, reject) => { | ||
const checkInterval = setInterval(() => { | ||
if (window.formbricks) { | ||
clearInterval(checkInterval); | ||
resolve(); | ||
} | ||
}, 100); | ||
setTimeout(() => { | ||
clearInterval(checkInterval); | ||
reject(new Error("Formbricks SDK loading timed out")); | ||
}, 1e4); | ||
}); | ||
} | ||
} | ||
const formbricksProxyHandler = { | ||
get(_target, prop, _receiver) { | ||
return async (...args) => { | ||
if (!window.formbricks && !sdkLoadingPromise && !isErrorLoadingSdk) { | ||
const { apiHost } = args[0]; | ||
sdkLoadingPromise = loadSDK(apiHost).catch((error) => { | ||
console.error(`🧱 Formbricks - Error loading SDK: ${error}`); | ||
sdkLoadingPromise = null; | ||
isErrorLoadingSdk = true; | ||
return; | ||
}); | ||
} | ||
if (isErrorLoadingSdk) { | ||
return; | ||
} | ||
if (sdkLoadingPromise) { | ||
await sdkLoadingPromise; | ||
} | ||
if (!window.formbricks) { | ||
throw new Error("Formbricks SDK is not available"); | ||
} | ||
if (typeof window.formbricks[prop] !== "function") { | ||
console.error(`🧱 Formbricks - SDK does not support method ${String(prop)}`); | ||
return; | ||
} | ||
try { | ||
return window.formbricks[prop](...args); | ||
} catch (error) { | ||
console.error(error); | ||
throw error; | ||
} | ||
}; | ||
} | ||
}; | ||
const formbricks = new Proxy({}, formbricksProxyHandler); | ||
export { | ||
formbricks as default | ||
}; | ||
//# sourceMappingURL=index.js.map |
{ | ||
"name": "@formbricks/js", | ||
"license": "MIT", | ||
"version": "1.6.5", | ||
"version": "2.0.0-beta.0", | ||
"description": "Formbricks-js allows you to connect your app to Formbricks, display surveys and trigger events.", | ||
@@ -20,11 +20,11 @@ "homepage": "https://formbricks.com", | ||
], | ||
"type": "module", | ||
"source": "src/index.ts", | ||
"main": "dist/index.js", | ||
"module": "dist/index.mjs", | ||
"main": "dist/index.umd.cjs", | ||
"module": "dist/index.js", | ||
"types": "dist/index.d.ts", | ||
"unpkg": "./dist/index.umd.js", | ||
"exports": { | ||
".": { | ||
"import": "./dist/index.mjs", | ||
"require": "./dist/index.umd.js", | ||
"import": "./dist/index.js", | ||
"require": "./dist/index.umd.cjs", | ||
"types": "./dist/index.d.ts" | ||
@@ -35,19 +35,7 @@ } | ||
"devDependencies": { | ||
"@babel/core": "^7.24.0", | ||
"@babel/preset-env": "^7.24.0", | ||
"@babel/preset-typescript": "^7.23.3", | ||
"@typescript-eslint/eslint-plugin": "^7.2.0", | ||
"@typescript-eslint/parser": "^7.2.0", | ||
"cross-env": "^7.0.3", | ||
"eslint-config-prettier": "^9.1.0", | ||
"eslint-config-turbo": "1.10.12", | ||
"isomorphic-fetch": "^3.0.0", | ||
"terser": "^5.29.1", | ||
"vite": "^5.1.6", | ||
"vite-plugin-dts": "^3.7.3", | ||
"@formbricks/api": "1.6.0", | ||
"@formbricks/lib": "0.0.0", | ||
"@formbricks/surveys": "1.6.3", | ||
"terser": "^5.30.3", | ||
"vite": "^5.2.9", | ||
"vite-plugin-dts": "^3.8.3", | ||
"@formbricks/tsconfig": "1.0.0", | ||
"@formbricks/types": "0.0.0" | ||
"eslint-config-formbricks": "1.0.0" | ||
}, | ||
@@ -54,0 +42,0 @@ "scripts": { |
Sorry, the diff of this file is not supported yet
Deprecated
MaintenanceThe maintainer of the package marked it as deprecated. This could indicate that a single version should not be used, or that the package is no longer maintained and any new vulnerabilities will not be fixed.
Found 1 instance in 1 package
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
No v1
QualityPackage is not semver >=1. This means it is not stable and does not support ^ ranges.
Found 1 instance in 1 package
5
3
Yes
13998
8
78
1
1