@behindthescenes/cart 0.0.19 → 0.0.22

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- var U="https://api.bts.it.com/v2/website/cart",_="bts-cart",A="default",Q=2;function E(){try{return globalThis.window??null}catch{return null}}function T(){try{return globalThis.localStorage??E()?.localStorage??null}catch{return null}}var x=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],K=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function L(B){if(!B)return{utm:{},clickIds:{}};try{let F=JSON.parse(B);if(!F||typeof F!=="object"||Array.isArray(F))return{utm:{},clickIds:{}};let J=F,G=J.utm&&typeof J.utm==="object"&&!Array.isArray(J.utm)?Object.fromEntries(Object.entries(J.utm).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{},Z=J.clickIds&&typeof J.clickIds==="object"&&!Array.isArray(J.clickIds)?Object.fromEntries(Object.entries(J.clickIds).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{};return{utm:G,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function S(B,F){let J=new URL(B);J.searchParams.set("bts_site",F);let G=T(),Z=G?.getItem("bts_analytics_jt")?.trim();if(Z)J.searchParams.set("bts_jt",Z);let{utm:X,clickIds:M}=L(G?.getItem("bts_analytics_attr")??null);for(let N of x){let V=X[N];if(V)J.searchParams.set(N,V)}for(let N of K){let V=M[N];if(V)J.searchParams.set(N,V)}return J.toString()}class $ extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,J,G,Z,X,M){super(F);this.name="BTSCartError",this.code=B,this.status=J,this.details=G,this.available=Z,this.sold_out=X,this.quantity_exceeded=M}}function b(){return T()??void 0}function w(B){return(B??U).replace(/\/$/,"")}function f(B){if(!Number.isFinite(B??1))return 1;return Math.max(1,Math.floor(B??1))}function R(B){return{items:B.map((F)=>({...F}))}}function z(B){let F=B?.trim();return F&&F.length>0?F:void 0}function g(B){return B.accessTierId??B.productId??B.lineItemId}function p(B){if(B.discountEligibleTierOrProductIds?.length)return[...new Set(B.discountEligibleTierOrProductIds.filter(Boolean))];return[...new Set([B.accessTierId,B.productId,B.lineItemId].filter((F)=>Boolean(F)))]}function C(B){let F=new Set;for(let J of B)for(let G of[J.lineItemId,J.accessTierId,J.productId])if(G)F.add(G);return B.map((J)=>{if(!J.discount)return J;let G=p(J);if(G.length===0||!G.some((Z)=>F.has(Z))){let Z={...J};return delete Z.discount,delete Z.discountEligibleTierOrProductIds,Z}return J})}function I(B,F){let J=new Headers(B);if(!F)return J;return new Headers(F).forEach((G,Z)=>J.set(Z,G)),J}function P(B){let F=B.checkoutLinkId.trim();try{let J=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,G=new URL(J),Z=G.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),M=X>=0?Z[X+1]:Z.at(-1);if(M)return{checkoutLinkId:M,discountCode:B.discountCode??G.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function h(B){let F=B?.details;if(!Array.isArray(F))return;let J=F.flatMap((G)=>{if(!G||typeof G!=="object")return[];let Z=typeof G.lineItemId==="string"?G.lineItemId:"",X=typeof G.discountCode==="string"?G.discountCode:"",M=typeof G.reason==="string"?G.reason:"";if(!Z||!X||!M)return[];return[{lineItemId:Z,...typeof G.productId==="string"?{productId:G.productId}:{},...typeof G.accessTierId==="string"?{accessTierId:G.accessTierId}:{},discountCode:X,...typeof G.discountId==="string"?{discountId:G.discountId}:{},reason:M}]});return J.length>0?J:void 0}class O{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=w(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??_,this.storage=B.storage??b(),this.checkoutLinks=B.checkoutLinks??{},this.defaultCheckoutLinkKey=B.defaultCheckoutLinkKey??Object.keys(this.checkoutLinks)[0]??A,this.requestHeaders=B.requestHeaders,this.rehydrate()}static init(B){return new O(B)}getItems(){return R(this.items).items}getState(){return R(this.items)}getItemCount(){return this.items.reduce((B,F)=>B+F.quantity,0)}getSubtotal(){return this.items.reduce((B,F)=>B+(F.fullPrice??F.unitPrice)*F.quantity,0)}getTotal(){return this.items.reduce((B,F)=>B+F.unitPrice*F.quantity,0)}subscribe(B){return this.listeners.add(B),B(this.getState()),()=>{this.listeners.delete(B)}}destroy(){this.listeners.clear()}rehydrate(){if(!this.persist||!this.storage)return;try{let B=this.storage.getItem(this.storageKey);if(!B)return;let F=JSON.parse(B);if(Array.isArray(F)||F.version!==Q){this.storage.removeItem(this.storageKey);return}let J=F.items;if(!Array.isArray(J))return;this.items=C(J.map((G)=>this.normalizeItem(G)).filter((G)=>Boolean(G))),this.emit()}catch(B){this.log("Failed to hydrate cart",B)}}addItem(B){let F=this.normalizeItem(B);if(!F)return;let J=this.items.find((G)=>G.id===F.id);if(J){this.setQuantity(J.id,J.quantity+F.quantity);return}this.items=[...this.items,F],this.commit()}removeItem(B){this.items=this.items.filter((F)=>F.id!==B),this.commit()}setQuantity(B,F){let J=Math.floor(F);this.items=J<=0?this.items.filter((G)=>G.id!==B):this.items.map((G)=>G.id===B?{...G,quantity:J}:G),this.commit()}increment(B,F=1){let J=this.items.find((G)=>G.id===B);if(J)this.setQuantity(B,J.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let J=this.items.find((G)=>G.id===B);if(J)this.setQuantity(B,J.quantity-Math.max(1,Math.floor(F)))}clear(){if(this.items=[],this.persist&&this.storage)try{this.storage.removeItem(this.storageKey)}catch(B){this.log("Failed to clear cart storage",B)}this.emit()}removeDiscountForLineItem(B){this.items=this.items.map((F)=>{if(F.lineItemId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.accessTierId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.productId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};return F})}async checkIfItemsAvailable(B){let F=[];for(let J of B){let G=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,J.lineItemId,J.discountId,J.quantity);if(G instanceof $)throw new $(G.code,G.message);F.push(G)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,J,G){let Z=this.checkoutLinks[B];if(!Z)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let X=P(Z),M={siteKey:this.siteKey,checkoutLinkId:X.checkoutLinkId,mode:Z.mode,lineItemId:F,discountId:J,quantity:G},N=await this.post("/check-availability",M);if(typeof N.ok!=="boolean")throw new $("request_failed","Availability check response was invalid.");return N}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let J=P(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:J.checkoutLinkId,mode:F.mode,...J.discountCode?{discountCode:J.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new $("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof $)if(X.code==="discounts_invalid")throw this.reconcileInvalidDiscounts(X.details),new $(X.code,X.message,X.status,X.details);else throw new $(X.code,X.message,X.status);throw new $("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let J=P(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:J.checkoutLinkId,mode:F.mode,...J.discountCode?{discountCode:J.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new $("request_failed","Cart response was invalid.");return S(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof $)if(X.code==="discounts_invalid")throw this.reconcileInvalidDiscounts(X.details),new $(X.code,X.message,X.status,X.details);else throw new $(X.code,X.message,X.status);throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),J=E();if(J)J.location.assign(F);return F}catch(F){if(F instanceof $)if(F.code==="discounts_invalid")throw new $(F.code,F.message,F.status,F.details);else throw new $(F.code,F.message,F.status);throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let J=this.items.filter((G)=>G.checkoutLinkKey===B).map((G)=>({lineItemId:g(G),quantity:G.quantity,...G.discount?{discount:G.discount}:{}}));if(J.length===0)throw new $("cart_empty","Your cart is empty.");return J}normalizeItem(B){let F=z(B.accessTierId),J=z(B.productId),G=z(B.lineItemId)??F??J;if(!B.id||!G||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:G,...F?{accessTierId:F}:{},...J?{productId:J}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,quantity:f(B.quantity)}}reconcileInvalidDiscounts(B){if(!B?.length)return;let F=this.items.map((G)=>{if(!B.some((M)=>{if(M.lineItemId===G.lineItemId)return!0;if(M.productId&&M.productId===G.productId)return!0;return Boolean(M.accessTierId&&M.accessTierId===G.accessTierId)}))return G;let X={...G,unitPrice:G.fullPrice??G.unitPrice};return delete X.discount,delete X.discountEligibleTierOrProductIds,X});if(!F.some((G,Z)=>G!==this.items[Z]))return;this.items=F,this.commit()}async post(B,F){let J=`${this.endpoint}${B}`,G=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:G,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:J},M=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,N=I(Z,M),V=await fetch(J,{method:"POST",headers:N,body:G}),Y=await V.json().catch(()=>null);if(!V.ok){let W=typeof Y?.code==="string"&&Y.code.length>0?Y.code:"request_failed",v=typeof Y?.message==="string"&&Y.message.length>0?Y.message:"Cart request failed";throw new $(W,v,V.status,h(Y))}if(!Y||typeof Y!=="object")throw new $("request_failed","Cart response was invalid.",V.status);return Y}commit(){if(this.items=C(this.items),this.persist&&this.storage)try{let B={version:Q,items:this.items};this.storage.setItem(this.storageKey,JSON.stringify(B))}catch(B){this.log("Failed to persist cart",B)}this.emit()}emit(){let B=this.getState();for(let F of this.listeners)F(B)}log(...B){if(this.debug)console.log("[@behindthescenes/cart]",...B)}}function H(B){return O.init(B)}function q(B){return typeof B==="object"&&B!==null&&!Array.isArray(B)}function j(B){let F=E();if(!F)return;let[J,G,Z]=B;if(typeof J!=="string")return;if(J==="config"){if(!q(G))return;F.btsCart?.destroy?.(),F.btsCart=H(G);return}let X=F.btsCart;if(!X)return;if(J==="add"&&q(G)){X.addItem(G);return}if(J==="remove"&&typeof G==="string"){X.removeItem(G);return}if(J==="quantity"&&typeof G==="string"&&typeof Z==="number"){X.setQuantity(G,Z);return}if(J==="clear"){X.clear();return}if(J==="checkout")X.checkout(typeof G==="string"?G:void 0)}var D=E();if(D){let B=Array.isArray(D.btsCartDataLayer)?[...D.btsCartDataLayer]:[];D.btsCartDataLayer=Array.isArray(D.btsCartDataLayer)?D.btsCartDataLayer:[],D.BTSCart={BTSCart:O,createBTSCart:H},D.createBTSCart=H;for(let F of B)j(Array.from(F));D.btsCartCommand=(...F)=>{D.btsCartDataLayer?.push(F),j(F)}}export{H as createBTSCart,O as BTSCart};
1
+ var Q="https://api.bts.it.com/v2/website/cart",S="bts-cart",q="default",T=2;function D(){try{return globalThis.window??null}catch{return null}}function O(){try{return globalThis.localStorage??D()?.localStorage??null}catch{return null}}var x=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],w=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function b(B){if(!B)return{utm:{},clickIds:{}};try{let F=JSON.parse(B);if(!F||typeof F!=="object"||Array.isArray(F))return{utm:{},clickIds:{}};let N=F,G=N.utm&&typeof N.utm==="object"&&!Array.isArray(N.utm)?Object.fromEntries(Object.entries(N.utm).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{},Z=N.clickIds&&typeof N.clickIds==="object"&&!Array.isArray(N.clickIds)?Object.fromEntries(Object.entries(N.clickIds).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{};return{utm:G,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function C(B,F){let N=new URL(B);N.searchParams.set("bts_site",F);let G=O(),Z=G?.getItem("bts_analytics_jt")?.trim();if(Z)N.searchParams.set("bts_jt",Z);let{utm:X,clickIds:J}=b(G?.getItem("bts_analytics_attr")??null);for(let Y of x){let V=X[Y];if(V)N.searchParams.set(Y,V)}for(let Y of w){let V=J[Y];if(V)N.searchParams.set(Y,V)}return N.toString()}class $ extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,N,G,Z,X,J){super(F);this.name="BTSCartError",this.code=B,this.status=N,this.details=G,this.available=Z,this.sold_out=X,this.quantity_exceeded=J}}function f(){return O()??void 0}function p(B){return(B??Q).replace(/\/$/,"")}function g(B){if(!Number.isFinite(B??1))return 1;return Math.max(1,Math.floor(B??1))}function R(B){return{items:B.map((F)=>({...F}))}}function z(B){let F=B?.trim();return F&&F.length>0?F:void 0}function I(B){return B.accessTierId??B.productId??B.lineItemId}function h(B){if(B.discountEligibleTierOrProductIds?.length)return[...new Set(B.discountEligibleTierOrProductIds.filter(Boolean))];return[...new Set([B.accessTierId,B.productId,B.lineItemId].filter((F)=>Boolean(F)))]}function W(B){let F=new Set;for(let N of B)for(let G of[N.lineItemId,N.accessTierId,N.productId])if(G)F.add(G);return B.map((N)=>{if(!N.discount)return N;let G=h(N);if(G.length===0||!G.some((Z)=>F.has(Z))){let Z={...N};return delete Z.discount,delete Z.discountEligibleTierOrProductIds,Z}return N})}function y(B,F){let N=new Headers(B);if(!F)return N;return new Headers(F).forEach((G,Z)=>N.set(Z,G)),N}function U(B){let F=B.checkoutLinkId.trim();try{let N=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,G=new URL(N),Z=G.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),J=X>=0?Z[X+1]:Z.at(-1);if(J)return{checkoutLinkId:J,discountCode:B.discountCode??G.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function k(B){let F=B?.details;if(!Array.isArray(F))return;let N=F.flatMap((G)=>{if(!G||typeof G!=="object")return[];let Z=typeof G.lineItemId==="string"?G.lineItemId:"",X=typeof G.reason==="string"?G.reason:"";if(!Z||!X)return[];let J=typeof G.discountCode==="string"?G.discountCode:void 0,Y=typeof G.requestedQuantity==="number"?G.requestedQuantity:void 0,V=typeof G.adjustedQuantity==="number"?G.adjustedQuantity:void 0;if(!J&&!(Y!==void 0||V!==void 0||X==="line_item_sold_out"||X==="quantity_exceeded"||X==="product_unavailable"))return[];return[{lineItemId:Z,...typeof G.productId==="string"?{productId:G.productId}:{},...typeof G.accessTierId==="string"?{accessTierId:G.accessTierId}:{},...J?{discountCode:J}:{},...typeof G.discountId==="string"?{discountId:G.discountId}:{},reason:X,...Y!==void 0?{requestedQuantity:Y}:{},...V!==void 0?{adjustedQuantity:V}:{}}]});return N.length>0?N:void 0}var u=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function _(B){return u.has(B)}function A(B,F){if(F.lineItemId===B.lineItemId)return!0;if(F.productId&&F.productId===B.productId)return!0;return Boolean(F.accessTierId&&F.accessTierId===B.accessTierId)}class H{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=p(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??S,this.storage=B.storage??f(),this.checkoutLinks=B.checkoutLinks??{},this.defaultCheckoutLinkKey=B.defaultCheckoutLinkKey??Object.keys(this.checkoutLinks)[0]??q,this.requestHeaders=B.requestHeaders,this.rehydrate()}static init(B){return new H(B)}getItems(){return R(this.items).items}getState(){return R(this.items)}getItemCount(){return this.items.reduce((B,F)=>B+F.quantity,0)}getSubtotal(){return this.items.reduce((B,F)=>B+(F.fullPrice??F.unitPrice)*F.quantity,0)}getTotal(){return this.items.reduce((B,F)=>B+F.unitPrice*F.quantity,0)}subscribe(B){return this.listeners.add(B),B(this.getState()),()=>{this.listeners.delete(B)}}destroy(){this.listeners.clear()}rehydrate(){if(!this.persist||!this.storage)return;try{let B=this.storage.getItem(this.storageKey);if(!B)return;let F=JSON.parse(B);if(Array.isArray(F)||F.version!==T){this.storage.removeItem(this.storageKey);return}let N=F.items;if(!Array.isArray(N))return;this.items=W(N.map((G)=>this.normalizeItem(G)).filter((G)=>Boolean(G))),this.emit()}catch(B){this.log("Failed to hydrate cart",B)}}addItem(B){let F=this.normalizeItem(B);if(!F)return;let N=this.items.find((G)=>G.id===F.id);if(N){this.setQuantity(N.id,N.quantity+F.quantity);return}this.items=[...this.items,F],this.commit()}removeItem(B){this.items=this.items.filter((F)=>F.id!==B),this.commit()}setQuantity(B,F){let N=Math.floor(F);this.items=N<=0?this.items.filter((G)=>G.id!==B):this.items.map((G)=>G.id===B?{...G,quantity:N}:G),this.commit()}increment(B,F=1){let N=this.items.find((G)=>G.id===B);if(N)this.setQuantity(B,N.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let N=this.items.find((G)=>G.id===B);if(N)this.setQuantity(B,N.quantity-Math.max(1,Math.floor(F)))}clear(){if(this.items=[],this.persist&&this.storage)try{this.storage.removeItem(this.storageKey)}catch(B){this.log("Failed to clear cart storage",B)}this.emit()}removeDiscountForLineItem(B){this.items=this.items.map((F)=>{if(F.lineItemId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.accessTierId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.productId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};return F})}async checkIfItemsAvailable(B){let F=[];for(let N of B){let G=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,N.lineItemId,N.discountId);if(G instanceof $)throw new $(G.code,G.message);F.push(G)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,N){let G=this.checkoutLinks[B];if(!G)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let Z=U(G),X={siteKey:this.siteKey,checkoutLinkId:Z.checkoutLinkId,mode:G.mode,lineItemId:F,discountId:N},J=await this.post("/check-availability",X);if(typeof J.ok!=="boolean")throw new $("request_failed","Availability check response was invalid.");return J}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=U(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:F.mode,...N.discountCode?{discountCode:N.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new $("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof $){if(_(X.code))throw this.reconcileCartAdjustments(X.details),new $(X.code,X.message,X.status,X.details);throw new $(X.code,X.message,X.status)}throw new $("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=U(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:F.mode,...N.discountCode?{discountCode:N.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new $("request_failed","Cart response was invalid.");return C(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof $){if(_(X.code))throw this.reconcileCartAdjustments(X.details),new $(X.code,X.message,X.status,X.details);throw new $(X.code,X.message,X.status)}throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),N=D();if(N)N.location.assign(F);return F}catch(F){if(F instanceof $){if(_(F.code))throw this.reconcileCartAdjustments(F.details),new $(F.code,F.message,F.status,F.details);throw new $(F.code,F.message,F.status)}throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let N=this.items.filter((G)=>G.checkoutLinkKey===B).map((G)=>({lineItemId:I(G),quantity:G.quantity,...G.discount?{discount:G.discount}:{}}));if(N.length===0)throw new $("cart_empty","Your cart is empty.");return N}normalizeItem(B){let F=z(B.accessTierId),N=z(B.productId),G=z(B.lineItemId)??F??N;if(!B.id||!G||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:G,...F?{accessTierId:F}:{},...N?{productId:N}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,quantity:g(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],N=!1;for(let G of B){if(G.discountCode){F=F.map((X)=>{if(!A(X,G))return X;let J={...X,unitPrice:X.fullPrice??X.unitPrice};return delete J.discount,delete J.discountEligibleTierOrProductIds,N=!0,J});continue}if(G.reason==="line_item_sold_out"||G.reason==="product_unavailable"||G.adjustedQuantity===0){let X=F.length;if(F=F.filter((J)=>!A(J,G)),F.length!==X)N=!0;continue}if(G.reason==="quantity_exceeded"&&typeof G.adjustedQuantity==="number"&&G.adjustedQuantity>0)F=F.map((X)=>{if(!A(X,G))return X;return N=!0,{...X,quantity:G.adjustedQuantity}})}if(!N)return;this.items=F,this.commit()}async post(B,F){let N=`${this.endpoint}${B}`,G=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:G,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:N},J=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,Y=y(Z,J),V=await fetch(N,{method:"POST",headers:Y,body:G}),E=await V.json().catch(()=>null);if(!V.ok){let K=typeof E?.code==="string"&&E.code.length>0?E.code:"request_failed",L=typeof E?.message==="string"&&E.message.length>0?E.message:"Cart request failed";throw new $(K,L,V.status,k(E))}if(!E||typeof E!=="object")throw new $("request_failed","Cart response was invalid.",V.status);return E}commit(){if(this.items=W(this.items),this.persist&&this.storage)try{let B={version:T,items:this.items};this.storage.setItem(this.storageKey,JSON.stringify(B))}catch(B){this.log("Failed to persist cart",B)}this.emit()}emit(){let B=this.getState();for(let F of this.listeners)F(B)}log(...B){if(this.debug)console.log("[@behindthescenes/cart]",...B)}}function P(B){return H.init(B)}function j(B){return typeof B==="object"&&B!==null&&!Array.isArray(B)}function v(B){let F=D();if(!F)return;let[N,G,Z]=B;if(typeof N!=="string")return;if(N==="config"){if(!j(G))return;F.btsCart?.destroy?.(),F.btsCart=P(G);return}let X=F.btsCart;if(!X)return;if(N==="add"&&j(G)){X.addItem(G);return}if(N==="remove"&&typeof G==="string"){X.removeItem(G);return}if(N==="quantity"&&typeof G==="string"&&typeof Z==="number"){X.setQuantity(G,Z);return}if(N==="clear"){X.clear();return}if(N==="checkout")X.checkout(typeof G==="string"?G:void 0)}var M=D();if(M){let B=Array.isArray(M.btsCartDataLayer)?[...M.btsCartDataLayer]:[];M.btsCartDataLayer=Array.isArray(M.btsCartDataLayer)?M.btsCartDataLayer:[],M.BTSCart={BTSCart:H,createBTSCart:P},M.createBTSCart=P;for(let F of B)v(Array.from(F));M.btsCartCommand=(...F)=>{M.btsCartDataLayer?.push(F),v(F)}}export{P as createBTSCart,H as BTSCart};
package/dist/cjs/index.js CHANGED
@@ -1 +1 @@
1
- var{defineProperty:P,getOwnPropertyNames:w,getOwnPropertyDescriptor:v}=Object,b=Object.prototype.hasOwnProperty;function x(B){return this[B]}var K=(B)=>{var F=(U??=new WeakMap).get(B),G;if(F)return F;if(F=P({},"__esModule",{value:!0}),B&&typeof B==="object"||typeof B==="function"){for(var J of w(B))if(!b.call(F,J))P(F,J,{get:x.bind(B,J),enumerable:!(G=v(B,J))||G.enumerable})}return U.set(B,F),F},U;var f=(B)=>B;function L(B,F){this[B]=f.bind(null,F)}var g=(B,F)=>{for(var G in F)P(B,G,{get:F[G],enumerable:!0,configurable:!0,set:L.bind(F,G)})};var a={};g(a,{resolveLineItemIdForCheckout:()=>S,createBTSCart:()=>d,BTSCartError:()=>$,BTSCart:()=>O});module.exports=K(a);var T="https://api.bts.it.com/v2/website/cart",W="bts-cart",_="default",D=2;function H(){try{return globalThis.window??null}catch{return null}}function Y(){try{return globalThis.localStorage??H()?.localStorage??null}catch{return null}}var p=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],I=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function h(B){if(!B)return{utm:{},clickIds:{}};try{let F=JSON.parse(B);if(!F||typeof F!=="object"||Array.isArray(F))return{utm:{},clickIds:{}};let G=F,J=G.utm&&typeof G.utm==="object"&&!Array.isArray(G.utm)?Object.fromEntries(Object.entries(G.utm).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{},Z=G.clickIds&&typeof G.clickIds==="object"&&!Array.isArray(G.clickIds)?Object.fromEntries(Object.entries(G.clickIds).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{};return{utm:J,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function R(B,F){let G=new URL(B);G.searchParams.set("bts_site",F);let J=Y(),Z=J?.getItem("bts_analytics_jt")?.trim();if(Z)G.searchParams.set("bts_jt",Z);let{utm:X,clickIds:E}=h(J?.getItem("bts_analytics_attr")??null);for(let M of p){let N=X[M];if(N)G.searchParams.set(M,N)}for(let M of I){let N=E[M];if(N)G.searchParams.set(M,N)}return G.toString()}class $ extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,G,J,Z,X,E){super(F);this.name="BTSCartError",this.code=B,this.status=G,this.details=J,this.available=Z,this.sold_out=X,this.quantity_exceeded=E}}function y(){return Y()??void 0}function k(B){return(B??T).replace(/\/$/,"")}function c(B){if(!Number.isFinite(B??1))return 1;return Math.max(1,Math.floor(B??1))}function A(B){return{items:B.map((F)=>({...F}))}}function Q(B){let F=B?.trim();return F&&F.length>0?F:void 0}function S(B){return B.accessTierId??B.productId??B.lineItemId}function u(B){if(B.discountEligibleTierOrProductIds?.length)return[...new Set(B.discountEligibleTierOrProductIds.filter(Boolean))];return[...new Set([B.accessTierId,B.productId,B.lineItemId].filter((F)=>Boolean(F)))]}function q(B){let F=new Set;for(let G of B)for(let J of[G.lineItemId,G.accessTierId,G.productId])if(J)F.add(J);return B.map((G)=>{if(!G.discount)return G;let J=u(G);if(J.length===0||!J.some((Z)=>F.has(Z))){let Z={...G};return delete Z.discount,delete Z.discountEligibleTierOrProductIds,Z}return G})}function l(B,F){let G=new Headers(B);if(!F)return G;return new Headers(F).forEach((J,Z)=>G.set(Z,J)),G}function z(B){let F=B.checkoutLinkId.trim();try{let G=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,J=new URL(G),Z=J.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),E=X>=0?Z[X+1]:Z.at(-1);if(E)return{checkoutLinkId:E,discountCode:B.discountCode??J.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function s(B){let F=B?.details;if(!Array.isArray(F))return;let G=F.flatMap((J)=>{if(!J||typeof J!=="object")return[];let Z=typeof J.lineItemId==="string"?J.lineItemId:"",X=typeof J.discountCode==="string"?J.discountCode:"",E=typeof J.reason==="string"?J.reason:"";if(!Z||!X||!E)return[];return[{lineItemId:Z,...typeof J.productId==="string"?{productId:J.productId}:{},...typeof J.accessTierId==="string"?{accessTierId:J.accessTierId}:{},discountCode:X,...typeof J.discountId==="string"?{discountId:J.discountId}:{},reason:E}]});return G.length>0?G:void 0}class O{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=k(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??W,this.storage=B.storage??y(),this.checkoutLinks=B.checkoutLinks??{},this.defaultCheckoutLinkKey=B.defaultCheckoutLinkKey??Object.keys(this.checkoutLinks)[0]??_,this.requestHeaders=B.requestHeaders,this.rehydrate()}static init(B){return new O(B)}getItems(){return A(this.items).items}getState(){return A(this.items)}getItemCount(){return this.items.reduce((B,F)=>B+F.quantity,0)}getSubtotal(){return this.items.reduce((B,F)=>B+(F.fullPrice??F.unitPrice)*F.quantity,0)}getTotal(){return this.items.reduce((B,F)=>B+F.unitPrice*F.quantity,0)}subscribe(B){return this.listeners.add(B),B(this.getState()),()=>{this.listeners.delete(B)}}destroy(){this.listeners.clear()}rehydrate(){if(!this.persist||!this.storage)return;try{let B=this.storage.getItem(this.storageKey);if(!B)return;let F=JSON.parse(B);if(Array.isArray(F)||F.version!==D){this.storage.removeItem(this.storageKey);return}let G=F.items;if(!Array.isArray(G))return;this.items=q(G.map((J)=>this.normalizeItem(J)).filter((J)=>Boolean(J))),this.emit()}catch(B){this.log("Failed to hydrate cart",B)}}addItem(B){let F=this.normalizeItem(B);if(!F)return;let G=this.items.find((J)=>J.id===F.id);if(G){this.setQuantity(G.id,G.quantity+F.quantity);return}this.items=[...this.items,F],this.commit()}removeItem(B){this.items=this.items.filter((F)=>F.id!==B),this.commit()}setQuantity(B,F){let G=Math.floor(F);this.items=G<=0?this.items.filter((J)=>J.id!==B):this.items.map((J)=>J.id===B?{...J,quantity:G}:J),this.commit()}increment(B,F=1){let G=this.items.find((J)=>J.id===B);if(G)this.setQuantity(B,G.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let G=this.items.find((J)=>J.id===B);if(G)this.setQuantity(B,G.quantity-Math.max(1,Math.floor(F)))}clear(){if(this.items=[],this.persist&&this.storage)try{this.storage.removeItem(this.storageKey)}catch(B){this.log("Failed to clear cart storage",B)}this.emit()}removeDiscountForLineItem(B){this.items=this.items.map((F)=>{if(F.lineItemId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.accessTierId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.productId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};return F})}async checkIfItemsAvailable(B){let F=[];for(let G of B){let J=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,G.lineItemId,G.discountId,G.quantity);if(J instanceof $)throw new $(J.code,J.message);F.push(J)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,G,J){let Z=this.checkoutLinks[B];if(!Z)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let X=z(Z),E={siteKey:this.siteKey,checkoutLinkId:X.checkoutLinkId,mode:Z.mode,lineItemId:F,discountId:G,quantity:J},M=await this.post("/check-availability",E);if(typeof M.ok!=="boolean")throw new $("request_failed","Availability check response was invalid.");return M}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=z(F),J=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...J.length>0?{items:J}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new $("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof $)if(X.code==="discounts_invalid")throw this.reconcileInvalidDiscounts(X.details),new $(X.code,X.message,X.status,X.details);else throw new $(X.code,X.message,X.status);throw new $("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=z(F),J=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...J.length>0?{items:J}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new $("request_failed","Cart response was invalid.");return R(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof $)if(X.code==="discounts_invalid")throw this.reconcileInvalidDiscounts(X.details),new $(X.code,X.message,X.status,X.details);else throw new $(X.code,X.message,X.status);throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),G=H();if(G)G.location.assign(F);return F}catch(F){if(F instanceof $)if(F.code==="discounts_invalid")throw new $(F.code,F.message,F.status,F.details);else throw new $(F.code,F.message,F.status);throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let G=this.items.filter((J)=>J.checkoutLinkKey===B).map((J)=>({lineItemId:S(J),quantity:J.quantity,...J.discount?{discount:J.discount}:{}}));if(G.length===0)throw new $("cart_empty","Your cart is empty.");return G}normalizeItem(B){let F=Q(B.accessTierId),G=Q(B.productId),J=Q(B.lineItemId)??F??G;if(!B.id||!J||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:J,...F?{accessTierId:F}:{},...G?{productId:G}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,quantity:c(B.quantity)}}reconcileInvalidDiscounts(B){if(!B?.length)return;let F=this.items.map((J)=>{if(!B.some((E)=>{if(E.lineItemId===J.lineItemId)return!0;if(E.productId&&E.productId===J.productId)return!0;return Boolean(E.accessTierId&&E.accessTierId===J.accessTierId)}))return J;let X={...J,unitPrice:J.fullPrice??J.unitPrice};return delete X.discount,delete X.discountEligibleTierOrProductIds,X});if(!F.some((J,Z)=>J!==this.items[Z]))return;this.items=F,this.commit()}async post(B,F){let G=`${this.endpoint}${B}`,J=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:J,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:G},E=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,M=l(Z,E),N=await fetch(G,{method:"POST",headers:M,body:J}),V=await N.json().catch(()=>null);if(!N.ok){let C=typeof V?.code==="string"&&V.code.length>0?V.code:"request_failed",j=typeof V?.message==="string"&&V.message.length>0?V.message:"Cart request failed";throw new $(C,j,N.status,s(V))}if(!V||typeof V!=="object")throw new $("request_failed","Cart response was invalid.",N.status);return V}commit(){if(this.items=q(this.items),this.persist&&this.storage)try{let B={version:D,items:this.items};this.storage.setItem(this.storageKey,JSON.stringify(B))}catch(B){this.log("Failed to persist cart",B)}this.emit()}emit(){let B=this.getState();for(let F of this.listeners)F(B)}log(...B){if(this.debug)console.log("[@behindthescenes/cart]",...B)}}function d(B){return O.init(B)}
1
+ var{defineProperty:H,getOwnPropertyNames:b,getOwnPropertyDescriptor:K}=Object,f=Object.prototype.hasOwnProperty;function x(B){return this[B]}var L=(B)=>{var F=(_??=new WeakMap).get(B),N;if(F)return F;if(F=H({},"__esModule",{value:!0}),B&&typeof B==="object"||typeof B==="function"){for(var G of b(B))if(!f.call(F,G))H(F,G,{get:x.bind(B,G),enumerable:!(N=K(B,G))||N.enumerable})}return _.set(B,F),F},_;var g=(B)=>B;function p(B,F){this[B]=g.bind(null,F)}var I=(B,F)=>{for(var N in F)H(B,N,{get:F[N],enumerable:!0,configurable:!0,set:p.bind(F,N)})};var r={};I(r,{resolveLineItemIdForCheckout:()=>w,createBTSCart:()=>a,BTSCartError:()=>$,BTSCart:()=>M});module.exports=L(r);var Q="https://api.bts.it.com/v2/website/cart",A="bts-cart",R="default",O=2;function z(){try{return globalThis.window??null}catch{return null}}function Y(){try{return globalThis.localStorage??z()?.localStorage??null}catch{return null}}var h=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],y=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function k(B){if(!B)return{utm:{},clickIds:{}};try{let F=JSON.parse(B);if(!F||typeof F!=="object"||Array.isArray(F))return{utm:{},clickIds:{}};let N=F,G=N.utm&&typeof N.utm==="object"&&!Array.isArray(N.utm)?Object.fromEntries(Object.entries(N.utm).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{},Z=N.clickIds&&typeof N.clickIds==="object"&&!Array.isArray(N.clickIds)?Object.fromEntries(Object.entries(N.clickIds).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{};return{utm:G,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function q(B,F){let N=new URL(B);N.searchParams.set("bts_site",F);let G=Y(),Z=G?.getItem("bts_analytics_jt")?.trim();if(Z)N.searchParams.set("bts_jt",Z);let{utm:X,clickIds:E}=k(G?.getItem("bts_analytics_attr")??null);for(let P of h){let J=X[P];if(J)N.searchParams.set(P,J)}for(let P of y){let J=E[P];if(J)N.searchParams.set(P,J)}return N.toString()}class $ extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,N,G,Z,X,E){super(F);this.name="BTSCartError",this.code=B,this.status=N,this.details=G,this.available=Z,this.sold_out=X,this.quantity_exceeded=E}}function c(){return Y()??void 0}function u(B){return(B??Q).replace(/\/$/,"")}function l(B){if(!Number.isFinite(B??1))return 1;return Math.max(1,Math.floor(B??1))}function S(B){return{items:B.map((F)=>({...F}))}}function D(B){let F=B?.trim();return F&&F.length>0?F:void 0}function w(B){return B.accessTierId??B.productId??B.lineItemId}function s(B){if(B.discountEligibleTierOrProductIds?.length)return[...new Set(B.discountEligibleTierOrProductIds.filter(Boolean))];return[...new Set([B.accessTierId,B.productId,B.lineItemId].filter((F)=>Boolean(F)))]}function C(B){let F=new Set;for(let N of B)for(let G of[N.lineItemId,N.accessTierId,N.productId])if(G)F.add(G);return B.map((N)=>{if(!N.discount)return N;let G=s(N);if(G.length===0||!G.some((Z)=>F.has(Z))){let Z={...N};return delete Z.discount,delete Z.discountEligibleTierOrProductIds,Z}return N})}function o(B,F){let N=new Headers(B);if(!F)return N;return new Headers(F).forEach((G,Z)=>N.set(Z,G)),N}function U(B){let F=B.checkoutLinkId.trim();try{let N=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,G=new URL(N),Z=G.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),E=X>=0?Z[X+1]:Z.at(-1);if(E)return{checkoutLinkId:E,discountCode:B.discountCode??G.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function d(B){let F=B?.details;if(!Array.isArray(F))return;let N=F.flatMap((G)=>{if(!G||typeof G!=="object")return[];let Z=typeof G.lineItemId==="string"?G.lineItemId:"",X=typeof G.reason==="string"?G.reason:"";if(!Z||!X)return[];let E=typeof G.discountCode==="string"?G.discountCode:void 0,P=typeof G.requestedQuantity==="number"?G.requestedQuantity:void 0,J=typeof G.adjustedQuantity==="number"?G.adjustedQuantity:void 0;if(!E&&!(P!==void 0||J!==void 0||X==="line_item_sold_out"||X==="quantity_exceeded"||X==="product_unavailable"))return[];return[{lineItemId:Z,...typeof G.productId==="string"?{productId:G.productId}:{},...typeof G.accessTierId==="string"?{accessTierId:G.accessTierId}:{},...E?{discountCode:E}:{},...typeof G.discountId==="string"?{discountId:G.discountId}:{},reason:X,...P!==void 0?{requestedQuantity:P}:{},...J!==void 0?{adjustedQuantity:J}:{}}]});return N.length>0?N:void 0}var m=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function W(B){return m.has(B)}function T(B,F){if(F.lineItemId===B.lineItemId)return!0;if(F.productId&&F.productId===B.productId)return!0;return Boolean(F.accessTierId&&F.accessTierId===B.accessTierId)}class M{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=u(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??A,this.storage=B.storage??c(),this.checkoutLinks=B.checkoutLinks??{},this.defaultCheckoutLinkKey=B.defaultCheckoutLinkKey??Object.keys(this.checkoutLinks)[0]??R,this.requestHeaders=B.requestHeaders,this.rehydrate()}static init(B){return new M(B)}getItems(){return S(this.items).items}getState(){return S(this.items)}getItemCount(){return this.items.reduce((B,F)=>B+F.quantity,0)}getSubtotal(){return this.items.reduce((B,F)=>B+(F.fullPrice??F.unitPrice)*F.quantity,0)}getTotal(){return this.items.reduce((B,F)=>B+F.unitPrice*F.quantity,0)}subscribe(B){return this.listeners.add(B),B(this.getState()),()=>{this.listeners.delete(B)}}destroy(){this.listeners.clear()}rehydrate(){if(!this.persist||!this.storage)return;try{let B=this.storage.getItem(this.storageKey);if(!B)return;let F=JSON.parse(B);if(Array.isArray(F)||F.version!==O){this.storage.removeItem(this.storageKey);return}let N=F.items;if(!Array.isArray(N))return;this.items=C(N.map((G)=>this.normalizeItem(G)).filter((G)=>Boolean(G))),this.emit()}catch(B){this.log("Failed to hydrate cart",B)}}addItem(B){let F=this.normalizeItem(B);if(!F)return;let N=this.items.find((G)=>G.id===F.id);if(N){this.setQuantity(N.id,N.quantity+F.quantity);return}this.items=[...this.items,F],this.commit()}removeItem(B){this.items=this.items.filter((F)=>F.id!==B),this.commit()}setQuantity(B,F){let N=Math.floor(F);this.items=N<=0?this.items.filter((G)=>G.id!==B):this.items.map((G)=>G.id===B?{...G,quantity:N}:G),this.commit()}increment(B,F=1){let N=this.items.find((G)=>G.id===B);if(N)this.setQuantity(B,N.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let N=this.items.find((G)=>G.id===B);if(N)this.setQuantity(B,N.quantity-Math.max(1,Math.floor(F)))}clear(){if(this.items=[],this.persist&&this.storage)try{this.storage.removeItem(this.storageKey)}catch(B){this.log("Failed to clear cart storage",B)}this.emit()}removeDiscountForLineItem(B){this.items=this.items.map((F)=>{if(F.lineItemId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.accessTierId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.productId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};return F})}async checkIfItemsAvailable(B){let F=[];for(let N of B){let G=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,N.lineItemId,N.discountId);if(G instanceof $)throw new $(G.code,G.message);F.push(G)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,N){let G=this.checkoutLinks[B];if(!G)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let Z=U(G),X={siteKey:this.siteKey,checkoutLinkId:Z.checkoutLinkId,mode:G.mode,lineItemId:F,discountId:N},E=await this.post("/check-availability",X);if(typeof E.ok!=="boolean")throw new $("request_failed","Availability check response was invalid.");return E}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=U(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:F.mode,...N.discountCode?{discountCode:N.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new $("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof $){if(W(X.code))throw this.reconcileCartAdjustments(X.details),new $(X.code,X.message,X.status,X.details);throw new $(X.code,X.message,X.status)}throw new $("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=U(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:F.mode,...N.discountCode?{discountCode:N.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new $("request_failed","Cart response was invalid.");return q(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof $){if(W(X.code))throw this.reconcileCartAdjustments(X.details),new $(X.code,X.message,X.status,X.details);throw new $(X.code,X.message,X.status)}throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),N=z();if(N)N.location.assign(F);return F}catch(F){if(F instanceof $){if(W(F.code))throw this.reconcileCartAdjustments(F.details),new $(F.code,F.message,F.status,F.details);throw new $(F.code,F.message,F.status)}throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let N=this.items.filter((G)=>G.checkoutLinkKey===B).map((G)=>({lineItemId:w(G),quantity:G.quantity,...G.discount?{discount:G.discount}:{}}));if(N.length===0)throw new $("cart_empty","Your cart is empty.");return N}normalizeItem(B){let F=D(B.accessTierId),N=D(B.productId),G=D(B.lineItemId)??F??N;if(!B.id||!G||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:G,...F?{accessTierId:F}:{},...N?{productId:N}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,quantity:l(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],N=!1;for(let G of B){if(G.discountCode){F=F.map((X)=>{if(!T(X,G))return X;let E={...X,unitPrice:X.fullPrice??X.unitPrice};return delete E.discount,delete E.discountEligibleTierOrProductIds,N=!0,E});continue}if(G.reason==="line_item_sold_out"||G.reason==="product_unavailable"||G.adjustedQuantity===0){let X=F.length;if(F=F.filter((E)=>!T(E,G)),F.length!==X)N=!0;continue}if(G.reason==="quantity_exceeded"&&typeof G.adjustedQuantity==="number"&&G.adjustedQuantity>0)F=F.map((X)=>{if(!T(X,G))return X;return N=!0,{...X,quantity:G.adjustedQuantity}})}if(!N)return;this.items=F,this.commit()}async post(B,F){let N=`${this.endpoint}${B}`,G=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:G,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:N},E=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,P=o(Z,E),J=await fetch(N,{method:"POST",headers:P,body:G}),V=await J.json().catch(()=>null);if(!J.ok){let j=typeof V?.code==="string"&&V.code.length>0?V.code:"request_failed",v=typeof V?.message==="string"&&V.message.length>0?V.message:"Cart request failed";throw new $(j,v,J.status,d(V))}if(!V||typeof V!=="object")throw new $("request_failed","Cart response was invalid.",J.status);return V}commit(){if(this.items=C(this.items),this.persist&&this.storage)try{let B={version:O,items:this.items};this.storage.setItem(this.storageKey,JSON.stringify(B))}catch(B){this.log("Failed to persist cart",B)}this.emit()}emit(){let B=this.getState();for(let F of this.listeners)F(B)}log(...B){if(this.debug)console.log("[@behindthescenes/cart]",...B)}}function a(B){return M.init(B)}
package/dist/esm/index.js CHANGED
@@ -1 +1 @@
1
- var z="https://api.bts.it.com/v2/website/cart",U="bts-cart",T="default",O=2;function P(){try{return globalThis.window??null}catch{return null}}function Y(){try{return globalThis.localStorage??P()?.localStorage??null}catch{return null}}var S=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],C=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function j(B){if(!B)return{utm:{},clickIds:{}};try{let F=JSON.parse(B);if(!F||typeof F!=="object"||Array.isArray(F))return{utm:{},clickIds:{}};let J=F,G=J.utm&&typeof J.utm==="object"&&!Array.isArray(J.utm)?Object.fromEntries(Object.entries(J.utm).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{},Z=J.clickIds&&typeof J.clickIds==="object"&&!Array.isArray(J.clickIds)?Object.fromEntries(Object.entries(J.clickIds).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{};return{utm:G,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function W(B,F){let J=new URL(B);J.searchParams.set("bts_site",F);let G=Y(),Z=G?.getItem("bts_analytics_jt")?.trim();if(Z)J.searchParams.set("bts_jt",Z);let{utm:X,clickIds:E}=j(G?.getItem("bts_analytics_attr")??null);for(let M of S){let N=X[M];if(N)J.searchParams.set(M,N)}for(let M of C){let N=E[M];if(N)J.searchParams.set(M,N)}return J.toString()}class $ extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,J,G,Z,X,E){super(F);this.name="BTSCartError",this.code=B,this.status=J,this.details=G,this.available=Z,this.sold_out=X,this.quantity_exceeded=E}}function w(){return Y()??void 0}function v(B){return(B??z).replace(/\/$/,"")}function b(B){if(!Number.isFinite(B??1))return 1;return Math.max(1,Math.floor(B??1))}function _(B){return{items:B.map((F)=>({...F}))}}function D(B){let F=B?.trim();return F&&F.length>0?F:void 0}function x(B){return B.accessTierId??B.productId??B.lineItemId}function K(B){if(B.discountEligibleTierOrProductIds?.length)return[...new Set(B.discountEligibleTierOrProductIds.filter(Boolean))];return[...new Set([B.accessTierId,B.productId,B.lineItemId].filter((F)=>Boolean(F)))]}function R(B){let F=new Set;for(let J of B)for(let G of[J.lineItemId,J.accessTierId,J.productId])if(G)F.add(G);return B.map((J)=>{if(!J.discount)return J;let G=K(J);if(G.length===0||!G.some((Z)=>F.has(Z))){let Z={...J};return delete Z.discount,delete Z.discountEligibleTierOrProductIds,Z}return J})}function f(B,F){let J=new Headers(B);if(!F)return J;return new Headers(F).forEach((G,Z)=>J.set(Z,G)),J}function H(B){let F=B.checkoutLinkId.trim();try{let J=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,G=new URL(J),Z=G.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),E=X>=0?Z[X+1]:Z.at(-1);if(E)return{checkoutLinkId:E,discountCode:B.discountCode??G.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function L(B){let F=B?.details;if(!Array.isArray(F))return;let J=F.flatMap((G)=>{if(!G||typeof G!=="object")return[];let Z=typeof G.lineItemId==="string"?G.lineItemId:"",X=typeof G.discountCode==="string"?G.discountCode:"",E=typeof G.reason==="string"?G.reason:"";if(!Z||!X||!E)return[];return[{lineItemId:Z,...typeof G.productId==="string"?{productId:G.productId}:{},...typeof G.accessTierId==="string"?{accessTierId:G.accessTierId}:{},discountCode:X,...typeof G.discountId==="string"?{discountId:G.discountId}:{},reason:E}]});return J.length>0?J:void 0}class Q{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=v(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??U,this.storage=B.storage??w(),this.checkoutLinks=B.checkoutLinks??{},this.defaultCheckoutLinkKey=B.defaultCheckoutLinkKey??Object.keys(this.checkoutLinks)[0]??T,this.requestHeaders=B.requestHeaders,this.rehydrate()}static init(B){return new Q(B)}getItems(){return _(this.items).items}getState(){return _(this.items)}getItemCount(){return this.items.reduce((B,F)=>B+F.quantity,0)}getSubtotal(){return this.items.reduce((B,F)=>B+(F.fullPrice??F.unitPrice)*F.quantity,0)}getTotal(){return this.items.reduce((B,F)=>B+F.unitPrice*F.quantity,0)}subscribe(B){return this.listeners.add(B),B(this.getState()),()=>{this.listeners.delete(B)}}destroy(){this.listeners.clear()}rehydrate(){if(!this.persist||!this.storage)return;try{let B=this.storage.getItem(this.storageKey);if(!B)return;let F=JSON.parse(B);if(Array.isArray(F)||F.version!==O){this.storage.removeItem(this.storageKey);return}let J=F.items;if(!Array.isArray(J))return;this.items=R(J.map((G)=>this.normalizeItem(G)).filter((G)=>Boolean(G))),this.emit()}catch(B){this.log("Failed to hydrate cart",B)}}addItem(B){let F=this.normalizeItem(B);if(!F)return;let J=this.items.find((G)=>G.id===F.id);if(J){this.setQuantity(J.id,J.quantity+F.quantity);return}this.items=[...this.items,F],this.commit()}removeItem(B){this.items=this.items.filter((F)=>F.id!==B),this.commit()}setQuantity(B,F){let J=Math.floor(F);this.items=J<=0?this.items.filter((G)=>G.id!==B):this.items.map((G)=>G.id===B?{...G,quantity:J}:G),this.commit()}increment(B,F=1){let J=this.items.find((G)=>G.id===B);if(J)this.setQuantity(B,J.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let J=this.items.find((G)=>G.id===B);if(J)this.setQuantity(B,J.quantity-Math.max(1,Math.floor(F)))}clear(){if(this.items=[],this.persist&&this.storage)try{this.storage.removeItem(this.storageKey)}catch(B){this.log("Failed to clear cart storage",B)}this.emit()}removeDiscountForLineItem(B){this.items=this.items.map((F)=>{if(F.lineItemId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.accessTierId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.productId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};return F})}async checkIfItemsAvailable(B){let F=[];for(let J of B){let G=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,J.lineItemId,J.discountId,J.quantity);if(G instanceof $)throw new $(G.code,G.message);F.push(G)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,J,G){let Z=this.checkoutLinks[B];if(!Z)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let X=H(Z),E={siteKey:this.siteKey,checkoutLinkId:X.checkoutLinkId,mode:Z.mode,lineItemId:F,discountId:J,quantity:G},M=await this.post("/check-availability",E);if(typeof M.ok!=="boolean")throw new $("request_failed","Availability check response was invalid.");return M}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let J=H(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:J.checkoutLinkId,mode:F.mode,...J.discountCode?{discountCode:J.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new $("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof $)if(X.code==="discounts_invalid")throw this.reconcileInvalidDiscounts(X.details),new $(X.code,X.message,X.status,X.details);else throw new $(X.code,X.message,X.status);throw new $("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let J=H(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:J.checkoutLinkId,mode:F.mode,...J.discountCode?{discountCode:J.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new $("request_failed","Cart response was invalid.");return W(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof $)if(X.code==="discounts_invalid")throw this.reconcileInvalidDiscounts(X.details),new $(X.code,X.message,X.status,X.details);else throw new $(X.code,X.message,X.status);throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),J=P();if(J)J.location.assign(F);return F}catch(F){if(F instanceof $)if(F.code==="discounts_invalid")throw new $(F.code,F.message,F.status,F.details);else throw new $(F.code,F.message,F.status);throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let J=this.items.filter((G)=>G.checkoutLinkKey===B).map((G)=>({lineItemId:x(G),quantity:G.quantity,...G.discount?{discount:G.discount}:{}}));if(J.length===0)throw new $("cart_empty","Your cart is empty.");return J}normalizeItem(B){let F=D(B.accessTierId),J=D(B.productId),G=D(B.lineItemId)??F??J;if(!B.id||!G||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:G,...F?{accessTierId:F}:{},...J?{productId:J}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,quantity:b(B.quantity)}}reconcileInvalidDiscounts(B){if(!B?.length)return;let F=this.items.map((G)=>{if(!B.some((E)=>{if(E.lineItemId===G.lineItemId)return!0;if(E.productId&&E.productId===G.productId)return!0;return Boolean(E.accessTierId&&E.accessTierId===G.accessTierId)}))return G;let X={...G,unitPrice:G.fullPrice??G.unitPrice};return delete X.discount,delete X.discountEligibleTierOrProductIds,X});if(!F.some((G,Z)=>G!==this.items[Z]))return;this.items=F,this.commit()}async post(B,F){let J=`${this.endpoint}${B}`,G=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:G,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:J},E=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,M=f(Z,E),N=await fetch(J,{method:"POST",headers:M,body:G}),V=await N.json().catch(()=>null);if(!N.ok){let A=typeof V?.code==="string"&&V.code.length>0?V.code:"request_failed",q=typeof V?.message==="string"&&V.message.length>0?V.message:"Cart request failed";throw new $(A,q,N.status,L(V))}if(!V||typeof V!=="object")throw new $("request_failed","Cart response was invalid.",N.status);return V}commit(){if(this.items=R(this.items),this.persist&&this.storage)try{let B={version:O,items:this.items};this.storage.setItem(this.storageKey,JSON.stringify(B))}catch(B){this.log("Failed to persist cart",B)}this.emit()}emit(){let B=this.getState();for(let F of this.listeners)F(B)}log(...B){if(this.debug)console.log("[@behindthescenes/cart]",...B)}}function a(B){return Q.init(B)}export{x as resolveLineItemIdForCheckout,a as createBTSCart,$ as BTSCartError,Q as BTSCart};
1
+ var T="https://api.bts.it.com/v2/website/cart",_="bts-cart",Q="default",M=2;function H(){try{return globalThis.window??null}catch{return null}}function Y(){try{return globalThis.localStorage??H()?.localStorage??null}catch{return null}}var w=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],j=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function v(B){if(!B)return{utm:{},clickIds:{}};try{let F=JSON.parse(B);if(!F||typeof F!=="object"||Array.isArray(F))return{utm:{},clickIds:{}};let N=F,G=N.utm&&typeof N.utm==="object"&&!Array.isArray(N.utm)?Object.fromEntries(Object.entries(N.utm).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{},Z=N.clickIds&&typeof N.clickIds==="object"&&!Array.isArray(N.clickIds)?Object.fromEntries(Object.entries(N.clickIds).filter((X)=>typeof X[1]==="string"&&X[1].length>0)):{};return{utm:G,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function A(B,F){let N=new URL(B);N.searchParams.set("bts_site",F);let G=Y(),Z=G?.getItem("bts_analytics_jt")?.trim();if(Z)N.searchParams.set("bts_jt",Z);let{utm:X,clickIds:E}=v(G?.getItem("bts_analytics_attr")??null);for(let P of w){let J=X[P];if(J)N.searchParams.set(P,J)}for(let P of j){let J=E[P];if(J)N.searchParams.set(P,J)}return N.toString()}class $ extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,N,G,Z,X,E){super(F);this.name="BTSCartError",this.code=B,this.status=N,this.details=G,this.available=Z,this.sold_out=X,this.quantity_exceeded=E}}function b(){return Y()??void 0}function K(B){return(B??T).replace(/\/$/,"")}function f(B){if(!Number.isFinite(B??1))return 1;return Math.max(1,Math.floor(B??1))}function R(B){return{items:B.map((F)=>({...F}))}}function O(B){let F=B?.trim();return F&&F.length>0?F:void 0}function x(B){return B.accessTierId??B.productId??B.lineItemId}function L(B){if(B.discountEligibleTierOrProductIds?.length)return[...new Set(B.discountEligibleTierOrProductIds.filter(Boolean))];return[...new Set([B.accessTierId,B.productId,B.lineItemId].filter((F)=>Boolean(F)))]}function q(B){let F=new Set;for(let N of B)for(let G of[N.lineItemId,N.accessTierId,N.productId])if(G)F.add(G);return B.map((N)=>{if(!N.discount)return N;let G=L(N);if(G.length===0||!G.some((Z)=>F.has(Z))){let Z={...N};return delete Z.discount,delete Z.discountEligibleTierOrProductIds,Z}return N})}function g(B,F){let N=new Headers(B);if(!F)return N;return new Headers(F).forEach((G,Z)=>N.set(Z,G)),N}function z(B){let F=B.checkoutLinkId.trim();try{let N=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,G=new URL(N),Z=G.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),E=X>=0?Z[X+1]:Z.at(-1);if(E)return{checkoutLinkId:E,discountCode:B.discountCode??G.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function p(B){let F=B?.details;if(!Array.isArray(F))return;let N=F.flatMap((G)=>{if(!G||typeof G!=="object")return[];let Z=typeof G.lineItemId==="string"?G.lineItemId:"",X=typeof G.reason==="string"?G.reason:"";if(!Z||!X)return[];let E=typeof G.discountCode==="string"?G.discountCode:void 0,P=typeof G.requestedQuantity==="number"?G.requestedQuantity:void 0,J=typeof G.adjustedQuantity==="number"?G.adjustedQuantity:void 0;if(!E&&!(P!==void 0||J!==void 0||X==="line_item_sold_out"||X==="quantity_exceeded"||X==="product_unavailable"))return[];return[{lineItemId:Z,...typeof G.productId==="string"?{productId:G.productId}:{},...typeof G.accessTierId==="string"?{accessTierId:G.accessTierId}:{},...E?{discountCode:E}:{},...typeof G.discountId==="string"?{discountId:G.discountId}:{},reason:X,...P!==void 0?{requestedQuantity:P}:{},...J!==void 0?{adjustedQuantity:J}:{}}]});return N.length>0?N:void 0}var I=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function D(B){return I.has(B)}function U(B,F){if(F.lineItemId===B.lineItemId)return!0;if(F.productId&&F.productId===B.productId)return!0;return Boolean(F.accessTierId&&F.accessTierId===B.accessTierId)}class W{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=K(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??_,this.storage=B.storage??b(),this.checkoutLinks=B.checkoutLinks??{},this.defaultCheckoutLinkKey=B.defaultCheckoutLinkKey??Object.keys(this.checkoutLinks)[0]??Q,this.requestHeaders=B.requestHeaders,this.rehydrate()}static init(B){return new W(B)}getItems(){return R(this.items).items}getState(){return R(this.items)}getItemCount(){return this.items.reduce((B,F)=>B+F.quantity,0)}getSubtotal(){return this.items.reduce((B,F)=>B+(F.fullPrice??F.unitPrice)*F.quantity,0)}getTotal(){return this.items.reduce((B,F)=>B+F.unitPrice*F.quantity,0)}subscribe(B){return this.listeners.add(B),B(this.getState()),()=>{this.listeners.delete(B)}}destroy(){this.listeners.clear()}rehydrate(){if(!this.persist||!this.storage)return;try{let B=this.storage.getItem(this.storageKey);if(!B)return;let F=JSON.parse(B);if(Array.isArray(F)||F.version!==M){this.storage.removeItem(this.storageKey);return}let N=F.items;if(!Array.isArray(N))return;this.items=q(N.map((G)=>this.normalizeItem(G)).filter((G)=>Boolean(G))),this.emit()}catch(B){this.log("Failed to hydrate cart",B)}}addItem(B){let F=this.normalizeItem(B);if(!F)return;let N=this.items.find((G)=>G.id===F.id);if(N){this.setQuantity(N.id,N.quantity+F.quantity);return}this.items=[...this.items,F],this.commit()}removeItem(B){this.items=this.items.filter((F)=>F.id!==B),this.commit()}setQuantity(B,F){let N=Math.floor(F);this.items=N<=0?this.items.filter((G)=>G.id!==B):this.items.map((G)=>G.id===B?{...G,quantity:N}:G),this.commit()}increment(B,F=1){let N=this.items.find((G)=>G.id===B);if(N)this.setQuantity(B,N.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let N=this.items.find((G)=>G.id===B);if(N)this.setQuantity(B,N.quantity-Math.max(1,Math.floor(F)))}clear(){if(this.items=[],this.persist&&this.storage)try{this.storage.removeItem(this.storageKey)}catch(B){this.log("Failed to clear cart storage",B)}this.emit()}removeDiscountForLineItem(B){this.items=this.items.map((F)=>{if(F.lineItemId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.accessTierId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};if(F.productId===B&&F.discount)return{...F,unitPrice:F.fullPrice??F.unitPrice};return F})}async checkIfItemsAvailable(B){let F=[];for(let N of B){let G=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,N.lineItemId,N.discountId);if(G instanceof $)throw new $(G.code,G.message);F.push(G)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,N){let G=this.checkoutLinks[B];if(!G)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let Z=z(G),X={siteKey:this.siteKey,checkoutLinkId:Z.checkoutLinkId,mode:G.mode,lineItemId:F,discountId:N},E=await this.post("/check-availability",X);if(typeof E.ok!=="boolean")throw new $("request_failed","Availability check response was invalid.");return E}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=z(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:F.mode,...N.discountCode?{discountCode:N.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new $("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof $){if(D(X.code))throw this.reconcileCartAdjustments(X.details),new $(X.code,X.message,X.status,X.details);throw new $(X.code,X.message,X.status)}throw new $("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new $("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=z(F),G=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:F.mode,...N.discountCode?{discountCode:N.discountCode}:{},...G.length>0?{items:G}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new $("request_failed","Cart response was invalid.");return A(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof $){if(D(X.code))throw this.reconcileCartAdjustments(X.details),new $(X.code,X.message,X.status,X.details);throw new $(X.code,X.message,X.status)}throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),N=H();if(N)N.location.assign(F);return F}catch(F){if(F instanceof $){if(D(F.code))throw this.reconcileCartAdjustments(F.details),new $(F.code,F.message,F.status,F.details);throw new $(F.code,F.message,F.status)}throw new $("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let N=this.items.filter((G)=>G.checkoutLinkKey===B).map((G)=>({lineItemId:x(G),quantity:G.quantity,...G.discount?{discount:G.discount}:{}}));if(N.length===0)throw new $("cart_empty","Your cart is empty.");return N}normalizeItem(B){let F=O(B.accessTierId),N=O(B.productId),G=O(B.lineItemId)??F??N;if(!B.id||!G||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:G,...F?{accessTierId:F}:{},...N?{productId:N}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,quantity:f(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],N=!1;for(let G of B){if(G.discountCode){F=F.map((X)=>{if(!U(X,G))return X;let E={...X,unitPrice:X.fullPrice??X.unitPrice};return delete E.discount,delete E.discountEligibleTierOrProductIds,N=!0,E});continue}if(G.reason==="line_item_sold_out"||G.reason==="product_unavailable"||G.adjustedQuantity===0){let X=F.length;if(F=F.filter((E)=>!U(E,G)),F.length!==X)N=!0;continue}if(G.reason==="quantity_exceeded"&&typeof G.adjustedQuantity==="number"&&G.adjustedQuantity>0)F=F.map((X)=>{if(!U(X,G))return X;return N=!0,{...X,quantity:G.adjustedQuantity}})}if(!N)return;this.items=F,this.commit()}async post(B,F){let N=`${this.endpoint}${B}`,G=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:G,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:N},E=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,P=g(Z,E),J=await fetch(N,{method:"POST",headers:P,body:G}),V=await J.json().catch(()=>null);if(!J.ok){let S=typeof V?.code==="string"&&V.code.length>0?V.code:"request_failed",C=typeof V?.message==="string"&&V.message.length>0?V.message:"Cart request failed";throw new $(S,C,J.status,p(V))}if(!V||typeof V!=="object")throw new $("request_failed","Cart response was invalid.",J.status);return V}commit(){if(this.items=q(this.items),this.persist&&this.storage)try{let B={version:M,items:this.items};this.storage.setItem(this.storageKey,JSON.stringify(B))}catch(B){this.log("Failed to persist cart",B)}this.emit()}emit(){let B=this.getState();for(let F of this.listeners)F(B)}log(...B){if(this.debug)console.log("[@behindthescenes/cart]",...B)}}function r(B){return W.init(B)}export{x as resolveLineItemIdForCheckout,r as createBTSCart,$ as BTSCartError,W as BTSCart};
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@behindthescenes/cart",
3
- "version": "0.0.19",
4
- "generatedAt": "2026-05-22T04:14:22.992Z",
3
+ "version": "0.0.22",
4
+ "generatedAt": "2026-05-22T04:36:12.139Z",
5
5
  "artifacts": {
6
6
  "browser": "browser/browser.js",
7
7
  "cjs": "cjs/index.js",
@@ -55,7 +55,7 @@ export declare class BTSCart {
55
55
  discountId: string;
56
56
  quantity: number;
57
57
  }[]): Promise<BTSCartAvailabilityCheckResponse[]>;
58
- checkIfItemAvailable(checkoutLinkKey: string | undefined, lineItemId: string, discountId: string, quantity: number): Promise<{
58
+ checkIfItemAvailable(checkoutLinkKey: string | undefined, lineItemId: string, discountId: string): Promise<{
59
59
  ok: boolean;
60
60
  available: boolean;
61
61
  sold_out: boolean;
@@ -73,8 +73,8 @@ export declare class BTSCart {
73
73
  private itemsForCheckout;
74
74
  /** Normalize partial external item input into the persisted cart line shape. */
75
75
  private normalizeItem;
76
- /** Remove invalid discounts from affected lines and restore the display / checkout price to full price. */
77
- private reconcileInvalidDiscounts;
76
+ /** Sync local cart state after checkout validation adjusts discounts or quantities. */
77
+ private reconcileCartAdjustments;
78
78
  /** POST a cart request with optional caller-supplied headers and normalized API errors. */
79
79
  private post;
80
80
  /** Recompute derived cart state, persist it when enabled, and notify subscribers. */
@@ -92,7 +92,6 @@ export type BTSCartAvailabilityCheckRequest = {
92
92
  mode: BTSCartMode;
93
93
  lineItemId: string;
94
94
  discountId: string;
95
- quantity: number;
96
95
  };
97
96
  export type BTSCartAvailabilityCheckResponse = {
98
97
  ok: boolean;
@@ -100,7 +99,7 @@ export type BTSCartAvailabilityCheckResponse = {
100
99
  sold_out: boolean;
101
100
  quantity_exceeded: boolean;
102
101
  };
103
- export type BTSCartErrorCode = "unknown_site" | "domain_not_verified" | "analytics_disabled" | "checkout_link_not_found" | "checkout_link_inactive" | "checkout_link_expired" | "checkout_link_sold_out" | "checkout_link_mode_mismatch" | "cart_empty" | "line_item_not_found" | "invalid_quantity" | "line_item_sold_out" | "quantity_exceeded" | "product_unavailable" | "discounts_invalid" | "request_failed" | "discounts_invalid";
102
+ export type BTSCartErrorCode = "unknown_site" | "domain_not_verified" | "analytics_disabled" | "checkout_link_not_found" | "checkout_link_inactive" | "checkout_link_expired" | "checkout_link_sold_out" | "checkout_link_mode_mismatch" | "cart_empty" | "line_item_not_found" | "invalid_quantity" | "line_item_sold_out" | "quantity_exceeded" | "product_unavailable" | "discounts_invalid" | "quantity_invalid" | "cart_adjusted" | "request_failed";
104
103
  export declare class BTSCartError extends Error {
105
104
  code: BTSCartErrorCode;
106
105
  status?: number;
@@ -111,9 +110,11 @@ export declare class BTSCartError extends Error {
111
110
  lineItemId: string;
112
111
  productId?: string;
113
112
  accessTierId?: string;
114
- discountCode: string;
113
+ discountCode?: string;
115
114
  discountId?: string;
116
115
  reason: string;
116
+ requestedQuantity?: number;
117
+ adjustedQuantity?: number;
117
118
  }[];
118
119
  constructor(code: BTSCartErrorCode, message: string, status?: number, details?: BTSCartError["details"], available?: boolean, sold_out?: boolean, quantity_exceeded?: boolean);
119
120
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@behindthescenes/cart",
3
- "version": "0.0.19",
3
+ "version": "0.0.22",
4
4
  "description": "Browser cart SDK for BTS external-site checkout links.",
5
5
  "license": "UNLICENSED",
6
6
  "type": "module",