@behindthescenes/cart 0.0.31 → 0.0.34

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 _="https://api.bts.it.com/v2/website/cart",W="bts-cart",A="default",Q=2;function P(){try{return globalThis.window??null}catch{return null}}function O(){try{return globalThis.localStorage??P()?.localStorage??null}catch{return null}}var p=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],g=["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 X=F,G=X.utm&&typeof X.utm==="object"&&!Array.isArray(X.utm)?Object.fromEntries(Object.entries(X.utm).filter((Z)=>typeof Z[1]==="string"&&Z[1].length>0)):{},$=X.clickIds&&typeof X.clickIds==="object"&&!Array.isArray(X.clickIds)?Object.fromEntries(Object.entries(X.clickIds).filter((Z)=>typeof Z[1]==="string"&&Z[1].length>0)):{};return{utm:G,clickIds:$}}catch{return{utm:{},clickIds:{}}}}function R(B,F){let X=new URL(B);X.searchParams.set("bts_site",F);let G=O(),$=G?.getItem("bts_analytics_jt")?.trim();if($)X.searchParams.set("bts_jt",$);let{utm:Z,clickIds:N}=h(G?.getItem("bts_analytics_attr")??null);for(let Y of p){let J=Z[Y];if(J)X.searchParams.set(Y,J)}for(let Y of g){let J=N[Y];if(J)X.searchParams.set(Y,J)}return X.toString()}class V extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,X,G,$,Z,N){super(F);this.name="BTSCartError",this.code=B,this.status=X,this.details=G,this.available=$,this.sold_out=Z,this.quantity_exceeded=N}}function y(){return O()??void 0}function k(B){return(B??_).replace(/\/$/,"")}function C(B){if(!Number.isFinite(B??1))return 1;return Math.max(1,Math.floor(B??1))}function j(B){return{items:B.map((F)=>({...F}))}}function S(B){let F=B?.trim();return F&&F.length>0?F:void 0}function v(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 K(B){let F=new Set;for(let X of B)for(let G of[X.lineItemId,X.accessTierId,X.productId])if(G)F.add(G);return B.map((X)=>{if(!X.discount)return X;let G=u(X);if(G.length===0||!G.some(($)=>F.has($))){let $={...X};return delete $.discount,delete $.discountEligibleTierOrProductIds,$}return X})}function c(B,F){let X=new Headers(B);if(!F)return X;return new Headers(F).forEach((G,$)=>X.set($,G)),X}function T(B){let F=B.checkoutLinkId.trim();try{let X=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,G=new URL(X),$=G.pathname.split("/").filter(Boolean),Z=$.indexOf("c"),N=Z>=0?$[Z+1]:$.at(-1);if(N)return{checkoutLinkId:N,discountCode:B.discountCode??G.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function q(B){let F=B?.details;if(!Array.isArray(F))return;let X=F.flatMap((G)=>{if(!G||typeof G!=="object")return[];let $=typeof G.lineItemId==="string"?G.lineItemId:"",Z=typeof G.reason==="string"?G.reason:"";if(!$||!Z)return[];let N=typeof G.discountCode==="string"?G.discountCode:void 0,Y=typeof G.requestedQuantity==="number"?G.requestedQuantity:void 0,J=typeof G.adjustedQuantity==="number"?G.adjustedQuantity:void 0;if(!N&&!(Y!==void 0||J!==void 0||Z==="line_item_sold_out"||Z==="quantity_exceeded"||Z==="product_unavailable"))return[];return[{lineItemId:$,...typeof G.productId==="string"?{productId:G.productId}:{},...typeof G.accessTierId==="string"?{accessTierId:G.accessTierId}:{},...N?{discountCode:N}:{},...typeof G.discountId==="string"?{discountId:G.discountId}:{},reason:Z,...Y!==void 0?{requestedQuantity:Y}:{},...J!==void 0?{adjustedQuantity:J}:{}}]});return X.length>0?X:void 0}var l=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function m(B){return B.reason==="line_item_sold_out"||B.reason==="product_unavailable"||B.adjustedQuantity===0}function w(B){if(B==="invalid_quantities")return"quantity_invalid";if(B==="invalid_discounts")return"discounts_invalid";return B}function E(B){return l.has(B)}function s(B){let F=B.sold_out===!0||B.code==="line_item_sold_out"||B.code==="product_unavailable",X=B.quantity_exceeded===!0||B.code==="quantity_exceeded";return{ok:!1,available:!F,sold_out:F,quantity_exceeded:X}}function d(B){return[...new Set([B.lineItemId,B.accessTierId,B.productId].filter((F)=>Boolean(F)))]}function D(B,F){let X=d(B);return[F.lineItemId,F.accessTierId,F.productId].filter(($)=>Boolean($)).some(($)=>X.includes($))}class U{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]??A,this.requestHeaders=B.requestHeaders,this.rehydrate()}static init(B){return new U(B)}getItems(){return j(this.items).items}getState(){return j(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 X=F.items;if(!Array.isArray(X))return;this.items=K(X.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 X=this.items.find((G)=>G.id===F.id);if(X){this.setQuantity(X.id,X.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 X=Math.floor(F);this.items=X<=0?this.items.filter((G)=>G.id!==B):this.items.map((G)=>G.id===B?{...G,quantity:X}:G),this.commit()}increment(B,F=1){let X=this.items.find((G)=>G.id===B);if(X)this.setQuantity(B,X.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let X=this.items.find((G)=>G.id===B);if(X)this.setQuantity(B,X.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(!D(F,{lineItemId:B,reason:"discount"}))return F;if(!F.discount)return F;let X={...F,unitPrice:F.fullPrice??F.unitPrice};return delete X.discount,delete X.discountEligibleTierOrProductIds,X}),this.commit()}async checkIfItemsAvailable(B){let F=[];for(let X of B){let G=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,X.lineItemId,X.discountId);if(G instanceof V)throw new V(G.code,G.message);F.push(G)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,X){let G=this.checkoutLinks[B];if(!G)throw new V("checkout_link_not_found",`Unknown checkout link key: ${B}`);let $=T(G),Z={siteKey:this.siteKey,checkoutLinkId:$.checkoutLinkId,mode:G.mode,lineItemId:F,...X?{discountId:X}:{}};try{let N=await this.post("/check-availability",Z);if(typeof N.ok!=="boolean")throw new V("request_failed","Availability check response was invalid.");let Y=N;return{...Y,available:Y.ok===!0&&Y.sold_out!==!0,sold_out:Y.sold_out===!0,quantity_exceeded:Y.quantity_exceeded===!0}}catch(N){if(N instanceof V){if(N.details?.length)this.reconcileCartAdjustments(N.details);if(N.code==="line_item_sold_out"||N.code==="product_unavailable"||N.code==="quantity_exceeded")return s(N)}throw N}}async isCatalogLineSoldOut(B=this.defaultCheckoutLinkKey,F,X){let G=this.checkoutLinks[B];if(!G)throw new V("checkout_link_not_found",`Unknown checkout link key: ${B}`);let $=T(G),Z=v(F),N={siteKey:this.siteKey,checkoutLinkId:$.checkoutLinkId,mode:G.mode,items:[{lineItemId:Z,quantity:C(X?.quantity),...X?.discount?{discount:X.discount}:{}}]},Y=await this.post("/validate-discounts",N);if(Y.ok===!0)return!1;return q(Y)?.some(m)??!1}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new V("checkout_link_not_found",`Unknown checkout link key: ${B}`);let X=T(F),G=this.itemsForCheckout(B,F),$={siteKey:this.siteKey,checkoutLinkId:X.checkoutLinkId,mode:F.mode,...X.discountCode?{discountCode:X.discountCode}:{},...G.length>0?{items:G}:{}};try{let Z=await this.post("/validate-discounts",$);if(typeof Z.ok!=="boolean")throw new V("request_failed","Cart response was invalid.");if(Z.ok===!1){let N=w(typeof Z.code==="string"&&Z.code.length>0?Z.code:"request_failed"),Y=typeof Z.message==="string"&&Z.message.length>0?Z.message:"Unable to validate cart.",J=q(Z);if(E(N))throw this.reconcileCartAdjustments(J),new V(N,Y,200,J);throw new V(N,Y,200,J)}return Z}catch(Z){if(Z instanceof V){if(E(Z.code))throw this.reconcileCartAdjustments(Z.details),new V(Z.code,Z.message,Z.status,Z.details);throw new V(Z.code,Z.message,Z.status)}throw new V("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new V("checkout_link_not_found",`Unknown checkout link key: ${B}`);let X=T(F),G=this.itemsForCheckout(B,F),$={siteKey:this.siteKey,checkoutLinkId:X.checkoutLinkId,mode:F.mode,...X.discountCode?{discountCode:X.discountCode}:{},...G.length>0?{items:G}:{}};try{let Z=await this.post("/checkout-url",$);if(Z.ok!==!0||typeof Z.checkoutUrl!=="string"||Z.checkoutUrl.length===0)throw new V("request_failed","Cart response was invalid.");return R(Z.checkoutUrl,this.siteKey)}catch(Z){if(Z instanceof V){if(E(Z.code))throw this.reconcileCartAdjustments(Z.details),new V(Z.code,Z.message,Z.status,Z.details);throw new V(Z.code,Z.message,Z.status)}throw new V("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),X=P();if(X)X.location.assign(F);return F}catch(F){if(F instanceof V){if(E(F.code))throw this.reconcileCartAdjustments(F.details),new V(F.code,F.message,F.status,F.details);throw new V(F.code,F.message,F.status)}throw new V("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let X=this.items.filter((G)=>G.checkoutLinkKey===B).map((G)=>({lineItemId:v(G),quantity:G.quantity,...G.discount?{discount:G.discount}:{}}));if(X.length===0)throw new V("cart_empty","Your cart is empty.");return X}normalizeItem(B){let F=S(B.accessTierId),X=S(B.productId),G=S(B.lineItemId)??F??X;if(!B.id||!G||typeof B.unitPrice!=="number")return null;let $=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:G,...F?{accessTierId:F}:{},...X?{productId:X}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:$,unitPrice:B.unitPrice,quantity:C(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],X=!1;for(let G of B){if(G.discountCode){F=F.map((Z)=>{if(!D(Z,G))return Z;let N={...Z,unitPrice:Z.fullPrice??Z.unitPrice};return delete N.discount,delete N.discountEligibleTierOrProductIds,X=!0,N});continue}if(G.reason==="line_item_sold_out"||G.reason==="product_unavailable"||G.adjustedQuantity===0){let Z=F.length;if(F=F.filter((N)=>!D(N,G)),F.length!==Z)X=!0;continue}if(G.reason==="quantity_exceeded"&&typeof G.adjustedQuantity==="number"&&G.adjustedQuantity>0){let Z=G.adjustedQuantity;F=F.map((N)=>{if(!D(N,G))return N;return X=!0,{...N,quantity:Z}})}}if(!X)return;this.items=F,this.commit()}async post(B,F){let X=`${this.endpoint}${B}`,G=JSON.stringify(F),$={"Content-Type":"application/json"},Z={body:F,bodyText:G,endpoint:this.endpoint,headers:$,path:B,siteKey:this.siteKey,url:X},N=typeof this.requestHeaders==="function"?await this.requestHeaders(Z):this.requestHeaders,Y=c($,N),J=await fetch(X,{method:"POST",headers:Y,body:G}),H=await J.json().catch(()=>null);if(!J.ok){let b=typeof H?.code==="string"&&H.code.length>0?H.code:"request_failed",f=w(b),I=typeof H?.message==="string"&&H.message.length>0?H.message:"Cart request failed";throw new V(f,I,J.status,q(H),typeof H?.available==="boolean"?H.available:void 0,typeof H?.sold_out==="boolean"?H.sold_out:void 0,typeof H?.quantity_exceeded==="boolean"?H.quantity_exceeded:void 0)}if(!H||typeof H!=="object")throw new V("request_failed","Cart response was invalid.",J.status);return H}commit(){if(this.items=K(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 z(B){return U.init(B)}function x(B){return typeof B==="object"&&B!==null&&!Array.isArray(B)}function L(B){let F=P();if(!F)return;let[X,G,$]=B;if(typeof X!=="string")return;if(X==="config"){if(!x(G))return;F.btsCart?.destroy?.(),F.btsCart=z(G);return}let Z=F.btsCart;if(!Z)return;if(X==="add"&&x(G)){Z.addItem(G);return}if(X==="remove"&&typeof G==="string"){Z.removeItem(G);return}if(X==="quantity"&&typeof G==="string"&&typeof $==="number"){Z.setQuantity(G,$);return}if(X==="clear"){Z.clear();return}if(X==="checkout")Z.checkout(typeof G==="string"?G:void 0)}var M=P();if(M){let B=Array.isArray(M.btsCartDataLayer)?[...M.btsCartDataLayer]:[];M.btsCartDataLayer=Array.isArray(M.btsCartDataLayer)?M.btsCartDataLayer:[],M.BTSCart={BTSCart:U,createBTSCart:z},M.createBTSCart=z;for(let F of B)L(Array.from(F));M.btsCartCommand=(...F)=>{M.btsCartDataLayer?.push(F),L(F)}}export{z as createBTSCart,U as BTSCart};
1
+ var S="https://api.bts.it.com/v2/website/cart",Q="bts-cart",q="default",z=2;function O(){try{return globalThis.window??null}catch{return null}}function U(){try{return globalThis.localStorage??O()?.localStorage??null}catch{return null}}var b=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],f=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function x(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,N=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:N,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function C(B,F){let G=new URL(B);G.searchParams.set("bts_site",F);let N=U(),Z=N?.getItem("bts_analytics_jt")?.trim();if(Z)G.searchParams.set("bts_jt",Z);let{utm:X,clickIds:$}=x(N?.getItem("bts_analytics_attr")??null);for(let E of b){let Y=X[E];if(Y)G.searchParams.set(E,Y)}for(let E of f){let Y=$[E];if(Y)G.searchParams.set(E,Y)}return G.toString()}class J extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,G,N,Z,X,$){super(F);this.name="BTSCartError",this.code=B,this.status=G,this.details=N,this.available=Z,this.sold_out=X,this.quantity_exceeded=$}}function I(){return U()??void 0}function g(B){return(B??S).replace(/\/$/,"")}function p(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,...F.discounts?{discounts:F.discounts.map((G)=>({...G}))}:{}}))}}function P(B){let F=B?.trim();return F&&F.length>0?F:void 0}function y(B){if(!Array.isArray(B)||B.length===0)return;let F=B.flatMap((G)=>{if(!G||typeof G!=="object")return[];let N=P(G.id),Z=typeof G.label==="string"?G.label.trim():"";if(!N||Z.length===0)return[];return[{...G,id:N,label:Z,...P(G.token)?{token:P(G.token)}:{}}]});return F.length>0?F:void 0}function W(B,F){if(!B?.length||!F)return B;let G=B.filter((N)=>N.token!==F);return G.length>0?G:void 0}function h(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 j(B){let F=new Set;for(let G of B)for(let N of[G.lineItemId,G.accessTierId,G.productId])if(N)F.add(N);return B.map((G)=>{if(!G.discount)return G;let N=k(G);if(N.length===0||!N.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((N,Z)=>G.set(Z,N)),G}function _(B){let F=B.checkoutLinkId.trim();try{let G=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,N=new URL(G),Z=N.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),$=X>=0?Z[X+1]:Z.at(-1);if($)return{checkoutLinkId:$,discountCode:B.discountCode??N.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function u(B){let F=B?.details;if(!Array.isArray(F))return;let G=F.flatMap((N)=>{if(!N||typeof N!=="object")return[];let Z=typeof N.lineItemId==="string"?N.lineItemId:"",X=typeof N.reason==="string"?N.reason:"";if(!Z||!X)return[];let $=typeof N.discountCode==="string"?N.discountCode:void 0,E=typeof N.requestedQuantity==="number"?N.requestedQuantity:void 0,Y=typeof N.adjustedQuantity==="number"?N.adjustedQuantity:void 0;if(!$&&!(E!==void 0||Y!==void 0||X==="line_item_sold_out"||X==="quantity_exceeded"||X==="product_unavailable"))return[];return[{lineItemId:Z,...typeof N.productId==="string"?{productId:N.productId}:{},...typeof N.accessTierId==="string"?{accessTierId:N.accessTierId}:{},...$?{discountCode:$}:{},...typeof N.discountId==="string"?{discountId:N.discountId}:{},reason:X,...E!==void 0?{requestedQuantity:E}:{},...Y!==void 0?{adjustedQuantity:Y}:{}}]});return G.length>0?G:void 0}var c=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function A(B){return c.has(B)}function m(B){return[...new Set([B.lineItemId,B.accessTierId,B.productId].filter((F)=>Boolean(F)))]}function D(B,F){let G=m(B);return[F.lineItemId,F.accessTierId,F.productId].filter((Z)=>Boolean(Z)).some((Z)=>G.includes(Z))}class M{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=g(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??Q,this.storage=B.storage??I(),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 M(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!==z){this.storage.removeItem(this.storageKey);return}let G=F.items;if(!Array.isArray(G))return;this.items=j(G.map((N)=>this.normalizeItem(N)).filter((N)=>Boolean(N))),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((N)=>N.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((N)=>N.id!==B):this.items.map((N)=>N.id===B?{...N,quantity:G}:N),this.commit()}updateItem(B,F){let G=this.items.find((Z)=>Z.id===B);if(!G)return;let N=this.normalizeItem({...G,...F,id:B,quantity:F.quantity??G.quantity,checkoutLinkKey:F.checkoutLinkKey??G.checkoutLinkKey});if(!N)return;this.items=this.items.map((Z)=>Z.id===B?N:Z),this.commit()}increment(B,F=1){let G=this.items.find((N)=>N.id===B);if(G)this.setQuantity(B,G.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let G=this.items.find((N)=>N.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(!D(F,{lineItemId:B,reason:"discount"}))return F;if(!F.discount)return F;let G={...F,unitPrice:F.fullPrice??F.unitPrice};return G.discounts=W(F.discounts,F.discount),delete G.discount,delete G.discountEligibleTierOrProductIds,G}),this.commit()}async checkIfItemsAvailable(B){let F=[];for(let G of B){let N=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,G.lineItemId,G.discountId);if(N instanceof J)throw new J(N.code,N.message);F.push(N)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,G){let N=this.checkoutLinks[B];if(!N)throw new J("checkout_link_not_found",`Unknown checkout link key: ${B}`);let Z=_(N),X={siteKey:this.siteKey,checkoutLinkId:Z.checkoutLinkId,mode:N.mode,lineItemId:F,...G?{discountId:G}:{}};try{let $=await this.post("/check-availability",X);if(typeof $.ok!=="boolean")throw new J("request_failed","Availability check response was invalid.");return $}catch($){if($ instanceof J){if($.details?.length)this.reconcileCartAdjustments($.details);if($.code==="line_item_sold_out"||$.code==="product_unavailable"||$.code==="quantity_exceeded")return{ok:!1,available:$.available??!1,sold_out:$.sold_out??$.code==="line_item_sold_out",quantity_exceeded:$.quantity_exceeded??($.code==="line_item_sold_out"||$.code==="quantity_exceeded")}}throw $}}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new J("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=_(F),N=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...N.length>0?{items:N}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new J("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof J){if(A(X.code))throw this.reconcileCartAdjustments(X.details),new J(X.code,X.message,X.status,X.details);throw new J(X.code,X.message,X.status)}throw new J("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new J("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=_(F),N=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...N.length>0?{items:N}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new J("request_failed","Cart response was invalid.");return C(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof J){if(A(X.code))throw this.reconcileCartAdjustments(X.details),new J(X.code,X.message,X.status,X.details);throw new J(X.code,X.message,X.status)}throw new J("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),G=O();if(G)G.location.assign(F);return F}catch(F){if(F instanceof J){if(A(F.code))throw this.reconcileCartAdjustments(F.details),new J(F.code,F.message,F.status,F.details);throw new J(F.code,F.message,F.status)}throw new J("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let G=this.items.filter((N)=>N.checkoutLinkKey===B).map((N)=>({lineItemId:h(N),quantity:N.quantity,...N.discount?{discount:N.discount}:{}}));if(G.length===0)throw new J("cart_empty","Your cart is empty.");return G}normalizeItem(B){let F=P(B.accessTierId),G=P(B.productId),N=P(B.lineItemId)??F??G;if(!B.id||!N||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item",X=y(B.discounts);return{...B,id:B.id,lineItemId:N,...F?{accessTierId:F}:{},...G?{productId:G}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,...X?{discounts:X}:{},quantity:p(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],G=!1;for(let N of B){if(N.discountCode){F=F.map((X)=>{if(!D(X,N))return X;let $={...X,unitPrice:X.fullPrice??X.unitPrice};return $.discounts=W(X.discounts,X.discount),delete $.discount,delete $.discountEligibleTierOrProductIds,G=!0,$});continue}if(N.reason==="line_item_sold_out"||N.reason==="product_unavailable"||N.adjustedQuantity===0){let X=F.length;if(F=F.filter(($)=>!D($,N)),F.length!==X)G=!0;continue}if(N.reason==="quantity_exceeded"&&typeof N.adjustedQuantity==="number"&&N.adjustedQuantity>0){let X=N.adjustedQuantity;F=F.map(($)=>{if(!D($,N))return $;return G=!0,{...$,quantity:X}})}}if(!G)return;this.items=F,this.commit()}async post(B,F){let G=`${this.endpoint}${B}`,N=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:N,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:G},$=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,E=l(Z,$),Y=await fetch(G,{method:"POST",headers:E,body:N}),V=await Y.json().catch(()=>null);if(!Y.ok){let L=typeof V?.code==="string"&&V.code.length>0?V.code:"request_failed",w=typeof V?.message==="string"&&V.message.length>0?V.message:"Cart request failed";throw new J(L,w,Y.status,u(V),typeof V?.available==="boolean"?V.available:void 0,typeof V?.sold_out==="boolean"?V.sold_out:void 0,typeof V?.quantity_exceeded==="boolean"?V.quantity_exceeded:void 0)}if(!V||typeof V!=="object")throw new J("request_failed","Cart response was invalid.",Y.status);return V}commit(){if(this.items=j(this.items),this.persist&&this.storage)try{let B={version:z,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 T(B){return M.init(B)}function v(B){return typeof B==="object"&&B!==null&&!Array.isArray(B)}function K(B){let F=O();if(!F)return;let[G,N,Z]=B;if(typeof G!=="string")return;if(G==="config"){if(!v(N))return;F.btsCart?.destroy?.(),F.btsCart=T(N);return}let X=F.btsCart;if(!X)return;if(G==="add"&&v(N)){X.addItem(N);return}if(G==="remove"&&typeof N==="string"){X.removeItem(N);return}if(G==="quantity"&&typeof N==="string"&&typeof Z==="number"){X.setQuantity(N,Z);return}if(G==="clear"){X.clear();return}if(G==="checkout")X.checkout(typeof N==="string"?N:void 0)}var H=O();if(H){let B=Array.isArray(H.btsCartDataLayer)?[...H.btsCartDataLayer]:[];H.btsCartDataLayer=Array.isArray(H.btsCartDataLayer)?H.btsCartDataLayer:[],H.BTSCart={BTSCart:M,createBTSCart:T},H.createBTSCart=T;for(let F of B)K(Array.from(F));H.btsCartCommand=(...F)=>{H.btsCartDataLayer?.push(F),K(F)}}export{T as createBTSCart,M as BTSCart};
package/dist/cjs/index.js CHANGED
@@ -1 +1 @@
1
- var{defineProperty:z,getOwnPropertyNames:p,getOwnPropertyDescriptor:h}=Object,y=Object.prototype.hasOwnProperty;function k(B){return this[B]}var c=(B)=>{var F=(_??=new WeakMap).get(B),G;if(F)return F;if(F=z({},"__esModule",{value:!0}),B&&typeof B==="object"||typeof B==="function"){for(var X of p(B))if(!y.call(F,X))z(F,X,{get:k.bind(B,X),enumerable:!(G=h(B,X))||G.enumerable})}return _.set(B,F),F},_;var u=(B)=>B;function m(B,F){this[B]=u.bind(null,F)}var l=(B,F)=>{for(var G in F)z(B,G,{get:F[G],enumerable:!0,configurable:!0,set:m.bind(F,G)})};var FB={};l(FB,{resolveLineItemIdForCheckout:()=>q,normalizeCartAdjustmentDetails:()=>E,isSoldOutAdjustmentDetail:()=>f,isCartAdjustmentErrorCode:()=>t,createBTSCart:()=>BB,catalogIdsForCartLine:()=>x,BTSCartError:()=>P,BTSCart:()=>O});module.exports=c(FB);var R="https://api.bts.it.com/v2/website/cart",S="bts-cart",A="default",D=2;function T(){try{return globalThis.window??null}catch{return null}}function H(){try{return globalThis.localStorage??T()?.localStorage??null}catch{return null}}var s=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],d=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function a(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,X=G.utm&&typeof G.utm==="object"&&!Array.isArray(G.utm)?Object.fromEntries(Object.entries(G.utm).filter((Z)=>typeof Z[1]==="string"&&Z[1].length>0)):{},N=G.clickIds&&typeof G.clickIds==="object"&&!Array.isArray(G.clickIds)?Object.fromEntries(Object.entries(G.clickIds).filter((Z)=>typeof Z[1]==="string"&&Z[1].length>0)):{};return{utm:X,clickIds:N}}catch{return{utm:{},clickIds:{}}}}function w(B,F){let G=new URL(B);G.searchParams.set("bts_site",F);let X=H(),N=X?.getItem("bts_analytics_jt")?.trim();if(N)G.searchParams.set("bts_jt",N);let{utm:Z,clickIds:$}=a(X?.getItem("bts_analytics_attr")??null);for(let V of s){let Y=Z[V];if(Y)G.searchParams.set(V,Y)}for(let V of d){let Y=$[V];if(Y)G.searchParams.set(V,Y)}return G.toString()}class P extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,G,X,N,Z,$){super(F);this.name="BTSCartError",this.code=B,this.status=G,this.details=X,this.available=N,this.sold_out=Z,this.quantity_exceeded=$}}function o(){return H()??void 0}function n(B){return(B??R).replace(/\/$/,"")}function v(B){if(!Number.isFinite(B??1))return 1;return Math.max(1,Math.floor(B??1))}function C(B){return{items:B.map((F)=>({...F}))}}function Q(B){let F=B?.trim();return F&&F.length>0?F:void 0}function q(B){return B.accessTierId??B.productId??B.lineItemId}function r(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 j(B){let F=new Set;for(let G of B)for(let X of[G.lineItemId,G.accessTierId,G.productId])if(X)F.add(X);return B.map((G)=>{if(!G.discount)return G;let X=r(G);if(X.length===0||!X.some((N)=>F.has(N))){let N={...G};return delete N.discount,delete N.discountEligibleTierOrProductIds,N}return G})}function i(B,F){let G=new Headers(B);if(!F)return G;return new Headers(F).forEach((X,N)=>G.set(N,X)),G}function M(B){let F=B.checkoutLinkId.trim();try{let G=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,X=new URL(G),N=X.pathname.split("/").filter(Boolean),Z=N.indexOf("c"),$=Z>=0?N[Z+1]:N.at(-1);if($)return{checkoutLinkId:$,discountCode:B.discountCode??X.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function E(B){let F=B?.details;if(!Array.isArray(F))return;let G=F.flatMap((X)=>{if(!X||typeof X!=="object")return[];let N=typeof X.lineItemId==="string"?X.lineItemId:"",Z=typeof X.reason==="string"?X.reason:"";if(!N||!Z)return[];let $=typeof X.discountCode==="string"?X.discountCode:void 0,V=typeof X.requestedQuantity==="number"?X.requestedQuantity:void 0,Y=typeof X.adjustedQuantity==="number"?X.adjustedQuantity:void 0;if(!$&&!(V!==void 0||Y!==void 0||Z==="line_item_sold_out"||Z==="quantity_exceeded"||Z==="product_unavailable"))return[];return[{lineItemId:N,...typeof X.productId==="string"?{productId:X.productId}:{},...typeof X.accessTierId==="string"?{accessTierId:X.accessTierId}:{},...$?{discountCode:$}:{},...typeof X.discountId==="string"?{discountId:X.discountId}:{},reason:Z,...V!==void 0?{requestedQuantity:V}:{},...Y!==void 0?{adjustedQuantity:Y}:{}}]});return G.length>0?G:void 0}var K=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function f(B){return B.reason==="line_item_sold_out"||B.reason==="product_unavailable"||B.adjustedQuantity===0}function t(B){return K.has(B)||B==="invalid_quantities"||B==="invalid_discounts"}function b(B){if(B==="invalid_quantities")return"quantity_invalid";if(B==="invalid_discounts")return"discounts_invalid";return B}function W(B){return K.has(B)}function e(B){let F=B.sold_out===!0||B.code==="line_item_sold_out"||B.code==="product_unavailable",G=B.quantity_exceeded===!0||B.code==="quantity_exceeded";return{ok:!1,available:!F,sold_out:F,quantity_exceeded:G}}function x(B){return[...new Set([B.lineItemId,B.accessTierId,B.productId].filter((F)=>Boolean(F)))]}function U(B,F){let G=x(B);return[F.lineItemId,F.accessTierId,F.productId].filter((N)=>Boolean(N)).some((N)=>G.includes(N))}class O{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=n(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??S,this.storage=B.storage??o(),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 C(this.items).items}getState(){return C(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=j(G.map((X)=>this.normalizeItem(X)).filter((X)=>Boolean(X))),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((X)=>X.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((X)=>X.id!==B):this.items.map((X)=>X.id===B?{...X,quantity:G}:X),this.commit()}increment(B,F=1){let G=this.items.find((X)=>X.id===B);if(G)this.setQuantity(B,G.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let G=this.items.find((X)=>X.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(!U(F,{lineItemId:B,reason:"discount"}))return F;if(!F.discount)return F;let G={...F,unitPrice:F.fullPrice??F.unitPrice};return delete G.discount,delete G.discountEligibleTierOrProductIds,G}),this.commit()}async checkIfItemsAvailable(B){let F=[];for(let G of B){let X=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,G.lineItemId,G.discountId);if(X instanceof P)throw new P(X.code,X.message);F.push(X)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,G){let X=this.checkoutLinks[B];if(!X)throw new P("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=M(X),Z={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:X.mode,lineItemId:F,...G?{discountId:G}:{}};try{let $=await this.post("/check-availability",Z);if(typeof $.ok!=="boolean")throw new P("request_failed","Availability check response was invalid.");let V=$;return{...V,available:V.ok===!0&&V.sold_out!==!0,sold_out:V.sold_out===!0,quantity_exceeded:V.quantity_exceeded===!0}}catch($){if($ instanceof P){if($.details?.length)this.reconcileCartAdjustments($.details);if($.code==="line_item_sold_out"||$.code==="product_unavailable"||$.code==="quantity_exceeded")return e($)}throw $}}async isCatalogLineSoldOut(B=this.defaultCheckoutLinkKey,F,G){let X=this.checkoutLinks[B];if(!X)throw new P("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=M(X),Z=q(F),$={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:X.mode,items:[{lineItemId:Z,quantity:v(G?.quantity),...G?.discount?{discount:G.discount}:{}}]},V=await this.post("/validate-discounts",$);if(V.ok===!0)return!1;return E(V)?.some(f)??!1}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new P("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=M(F),X=this.itemsForCheckout(B,F),N={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...X.length>0?{items:X}:{}};try{let Z=await this.post("/validate-discounts",N);if(typeof Z.ok!=="boolean")throw new P("request_failed","Cart response was invalid.");if(Z.ok===!1){let $=b(typeof Z.code==="string"&&Z.code.length>0?Z.code:"request_failed"),V=typeof Z.message==="string"&&Z.message.length>0?Z.message:"Unable to validate cart.",Y=E(Z);if(W($))throw this.reconcileCartAdjustments(Y),new P($,V,200,Y);throw new P($,V,200,Y)}return Z}catch(Z){if(Z instanceof P){if(W(Z.code))throw this.reconcileCartAdjustments(Z.details),new P(Z.code,Z.message,Z.status,Z.details);throw new P(Z.code,Z.message,Z.status)}throw new P("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new P("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=M(F),X=this.itemsForCheckout(B,F),N={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...X.length>0?{items:X}:{}};try{let Z=await this.post("/checkout-url",N);if(Z.ok!==!0||typeof Z.checkoutUrl!=="string"||Z.checkoutUrl.length===0)throw new P("request_failed","Cart response was invalid.");return w(Z.checkoutUrl,this.siteKey)}catch(Z){if(Z instanceof P){if(W(Z.code))throw this.reconcileCartAdjustments(Z.details),new P(Z.code,Z.message,Z.status,Z.details);throw new P(Z.code,Z.message,Z.status)}throw new P("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),G=T();if(G)G.location.assign(F);return F}catch(F){if(F instanceof P){if(W(F.code))throw this.reconcileCartAdjustments(F.details),new P(F.code,F.message,F.status,F.details);throw new P(F.code,F.message,F.status)}throw new P("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let G=this.items.filter((X)=>X.checkoutLinkKey===B).map((X)=>({lineItemId:q(X),quantity:X.quantity,...X.discount?{discount:X.discount}:{}}));if(G.length===0)throw new P("cart_empty","Your cart is empty.");return G}normalizeItem(B){let F=Q(B.accessTierId),G=Q(B.productId),X=Q(B.lineItemId)??F??G;if(!B.id||!X||typeof B.unitPrice!=="number")return null;let N=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:X,...F?{accessTierId:F}:{},...G?{productId:G}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:N,unitPrice:B.unitPrice,quantity:v(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],G=!1;for(let X of B){if(X.discountCode){F=F.map((Z)=>{if(!U(Z,X))return Z;let $={...Z,unitPrice:Z.fullPrice??Z.unitPrice};return delete $.discount,delete $.discountEligibleTierOrProductIds,G=!0,$});continue}if(X.reason==="line_item_sold_out"||X.reason==="product_unavailable"||X.adjustedQuantity===0){let Z=F.length;if(F=F.filter(($)=>!U($,X)),F.length!==Z)G=!0;continue}if(X.reason==="quantity_exceeded"&&typeof X.adjustedQuantity==="number"&&X.adjustedQuantity>0){let Z=X.adjustedQuantity;F=F.map(($)=>{if(!U($,X))return $;return G=!0,{...$,quantity:Z}})}}if(!G)return;this.items=F,this.commit()}async post(B,F){let G=`${this.endpoint}${B}`,X=JSON.stringify(F),N={"Content-Type":"application/json"},Z={body:F,bodyText:X,endpoint:this.endpoint,headers:N,path:B,siteKey:this.siteKey,url:G},$=typeof this.requestHeaders==="function"?await this.requestHeaders(Z):this.requestHeaders,V=i(N,$),Y=await fetch(G,{method:"POST",headers:V,body:X}),J=await Y.json().catch(()=>null);if(!Y.ok){let L=typeof J?.code==="string"&&J.code.length>0?J.code:"request_failed",I=b(L),g=typeof J?.message==="string"&&J.message.length>0?J.message:"Cart request failed";throw new P(I,g,Y.status,E(J),typeof J?.available==="boolean"?J.available:void 0,typeof J?.sold_out==="boolean"?J.sold_out:void 0,typeof J?.quantity_exceeded==="boolean"?J.quantity_exceeded:void 0)}if(!J||typeof J!=="object")throw new P("request_failed","Cart response was invalid.",Y.status);return J}commit(){if(this.items=j(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 BB(B){return O.init(B)}
1
+ var{defineProperty:U,getOwnPropertyNames:f,getOwnPropertyDescriptor:L}=Object,x=Object.prototype.hasOwnProperty;function g(B){return this[B]}var I=(B)=>{var F=(_??=new WeakMap).get(B),G;if(F)return F;if(F=U({},"__esModule",{value:!0}),B&&typeof B==="object"||typeof B==="function"){for(var N of f(B))if(!x.call(F,N))U(F,N,{get:g.bind(B,N),enumerable:!(G=L(B,N))||G.enumerable})}return _.set(B,F),F},_;var p=(B)=>B;function h(B,F){this[B]=p.bind(null,F)}var y=(B,F)=>{for(var G in F)U(B,G,{get:F[G],enumerable:!0,configurable:!0,set:h.bind(F,G)})};var t={};y(t,{resolveLineItemIdForCheckout:()=>v,createBTSCart:()=>i,catalogIdsForCartLine:()=>j,BTSCartError:()=>E,BTSCart:()=>M});module.exports=I(t);var A="https://api.bts.it.com/v2/website/cart",Q="bts-cart",R="default",W=2;function z(){try{return globalThis.window??null}catch{return null}}function H(){try{return globalThis.localStorage??z()?.localStorage??null}catch{return null}}var k=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],l=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function c(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,N=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:N,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function q(B,F){let G=new URL(B);G.searchParams.set("bts_site",F);let N=H(),Z=N?.getItem("bts_analytics_jt")?.trim();if(Z)G.searchParams.set("bts_jt",Z);let{utm:X,clickIds:$}=c(N?.getItem("bts_analytics_attr")??null);for(let V of k){let P=X[V];if(P)G.searchParams.set(V,P)}for(let V of l){let P=$[V];if(P)G.searchParams.set(V,P)}return G.toString()}class E extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,G,N,Z,X,$){super(F);this.name="BTSCartError",this.code=B,this.status=G,this.details=N,this.available=Z,this.sold_out=X,this.quantity_exceeded=$}}function u(){return H()??void 0}function m(B){return(B??A).replace(/\/$/,"")}function a(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,...F.discounts?{discounts:F.discounts.map((G)=>({...G}))}:{}}))}}function Y(B){let F=B?.trim();return F&&F.length>0?F:void 0}function s(B){if(!Array.isArray(B)||B.length===0)return;let F=B.flatMap((G)=>{if(!G||typeof G!=="object")return[];let N=Y(G.id),Z=typeof G.label==="string"?G.label.trim():"";if(!N||Z.length===0)return[];return[{...G,id:N,label:Z,...Y(G.token)?{token:Y(G.token)}:{}}]});return F.length>0?F:void 0}function C(B,F){if(!B?.length||!F)return B;let G=B.filter((N)=>N.token!==F);return G.length>0?G:void 0}function v(B){return B.accessTierId??B.productId??B.lineItemId}function d(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 G of B)for(let N of[G.lineItemId,G.accessTierId,G.productId])if(N)F.add(N);return B.map((G)=>{if(!G.discount)return G;let N=d(G);if(N.length===0||!N.some((Z)=>F.has(Z))){let Z={...G};return delete Z.discount,delete Z.discountEligibleTierOrProductIds,Z}return G})}function o(B,F){let G=new Headers(B);if(!F)return G;return new Headers(F).forEach((N,Z)=>G.set(Z,N)),G}function D(B){let F=B.checkoutLinkId.trim();try{let G=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,N=new URL(G),Z=N.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),$=X>=0?Z[X+1]:Z.at(-1);if($)return{checkoutLinkId:$,discountCode:B.discountCode??N.searchParams.get("discountCode")??void 0}}catch{}return{checkoutLinkId:F,discountCode:B.discountCode}}function r(B){let F=B?.details;if(!Array.isArray(F))return;let G=F.flatMap((N)=>{if(!N||typeof N!=="object")return[];let Z=typeof N.lineItemId==="string"?N.lineItemId:"",X=typeof N.reason==="string"?N.reason:"";if(!Z||!X)return[];let $=typeof N.discountCode==="string"?N.discountCode:void 0,V=typeof N.requestedQuantity==="number"?N.requestedQuantity:void 0,P=typeof N.adjustedQuantity==="number"?N.adjustedQuantity:void 0;if(!$&&!(V!==void 0||P!==void 0||X==="line_item_sold_out"||X==="quantity_exceeded"||X==="product_unavailable"))return[];return[{lineItemId:Z,...typeof N.productId==="string"?{productId:N.productId}:{},...typeof N.accessTierId==="string"?{accessTierId:N.accessTierId}:{},...$?{discountCode:$}:{},...typeof N.discountId==="string"?{discountId:N.discountId}:{},reason:X,...V!==void 0?{requestedQuantity:V}:{},...P!==void 0?{adjustedQuantity:P}:{}}]});return G.length>0?G:void 0}var n=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function T(B){return n.has(B)}function j(B){return[...new Set([B.lineItemId,B.accessTierId,B.productId].filter((F)=>Boolean(F)))]}function O(B,F){let G=j(B);return[F.lineItemId,F.accessTierId,F.productId].filter((Z)=>Boolean(Z)).some((Z)=>G.includes(Z))}class M{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=m(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??Q,this.storage=B.storage??u(),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!==W){this.storage.removeItem(this.storageKey);return}let G=F.items;if(!Array.isArray(G))return;this.items=w(G.map((N)=>this.normalizeItem(N)).filter((N)=>Boolean(N))),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((N)=>N.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((N)=>N.id!==B):this.items.map((N)=>N.id===B?{...N,quantity:G}:N),this.commit()}updateItem(B,F){let G=this.items.find((Z)=>Z.id===B);if(!G)return;let N=this.normalizeItem({...G,...F,id:B,quantity:F.quantity??G.quantity,checkoutLinkKey:F.checkoutLinkKey??G.checkoutLinkKey});if(!N)return;this.items=this.items.map((Z)=>Z.id===B?N:Z),this.commit()}increment(B,F=1){let G=this.items.find((N)=>N.id===B);if(G)this.setQuantity(B,G.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let G=this.items.find((N)=>N.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(!O(F,{lineItemId:B,reason:"discount"}))return F;if(!F.discount)return F;let G={...F,unitPrice:F.fullPrice??F.unitPrice};return G.discounts=C(F.discounts,F.discount),delete G.discount,delete G.discountEligibleTierOrProductIds,G}),this.commit()}async checkIfItemsAvailable(B){let F=[];for(let G of B){let N=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,G.lineItemId,G.discountId);if(N instanceof E)throw new E(N.code,N.message);F.push(N)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,G){let N=this.checkoutLinks[B];if(!N)throw new E("checkout_link_not_found",`Unknown checkout link key: ${B}`);let Z=D(N),X={siteKey:this.siteKey,checkoutLinkId:Z.checkoutLinkId,mode:N.mode,lineItemId:F,...G?{discountId:G}:{}};try{let $=await this.post("/check-availability",X);if(typeof $.ok!=="boolean")throw new E("request_failed","Availability check response was invalid.");return $}catch($){if($ instanceof E){if($.details?.length)this.reconcileCartAdjustments($.details);if($.code==="line_item_sold_out"||$.code==="product_unavailable"||$.code==="quantity_exceeded")return{ok:!1,available:$.available??!1,sold_out:$.sold_out??$.code==="line_item_sold_out",quantity_exceeded:$.quantity_exceeded??($.code==="line_item_sold_out"||$.code==="quantity_exceeded")}}throw $}}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new E("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=D(F),N=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...N.length>0?{items:N}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new E("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof E){if(T(X.code))throw this.reconcileCartAdjustments(X.details),new E(X.code,X.message,X.status,X.details);throw new E(X.code,X.message,X.status)}throw new E("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new E("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=D(F),N=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...N.length>0?{items:N}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new E("request_failed","Cart response was invalid.");return q(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof E){if(T(X.code))throw this.reconcileCartAdjustments(X.details),new E(X.code,X.message,X.status,X.details);throw new E(X.code,X.message,X.status)}throw new E("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),G=z();if(G)G.location.assign(F);return F}catch(F){if(F instanceof E){if(T(F.code))throw this.reconcileCartAdjustments(F.details),new E(F.code,F.message,F.status,F.details);throw new E(F.code,F.message,F.status)}throw new E("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let G=this.items.filter((N)=>N.checkoutLinkKey===B).map((N)=>({lineItemId:v(N),quantity:N.quantity,...N.discount?{discount:N.discount}:{}}));if(G.length===0)throw new E("cart_empty","Your cart is empty.");return G}normalizeItem(B){let F=Y(B.accessTierId),G=Y(B.productId),N=Y(B.lineItemId)??F??G;if(!B.id||!N||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item",X=s(B.discounts);return{...B,id:B.id,lineItemId:N,...F?{accessTierId:F}:{},...G?{productId:G}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,...X?{discounts:X}:{},quantity:a(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],G=!1;for(let N of B){if(N.discountCode){F=F.map((X)=>{if(!O(X,N))return X;let $={...X,unitPrice:X.fullPrice??X.unitPrice};return $.discounts=C(X.discounts,X.discount),delete $.discount,delete $.discountEligibleTierOrProductIds,G=!0,$});continue}if(N.reason==="line_item_sold_out"||N.reason==="product_unavailable"||N.adjustedQuantity===0){let X=F.length;if(F=F.filter(($)=>!O($,N)),F.length!==X)G=!0;continue}if(N.reason==="quantity_exceeded"&&typeof N.adjustedQuantity==="number"&&N.adjustedQuantity>0){let X=N.adjustedQuantity;F=F.map(($)=>{if(!O($,N))return $;return G=!0,{...$,quantity:X}})}}if(!G)return;this.items=F,this.commit()}async post(B,F){let G=`${this.endpoint}${B}`,N=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:N,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:G},$=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,V=o(Z,$),P=await fetch(G,{method:"POST",headers:V,body:N}),J=await P.json().catch(()=>null);if(!P.ok){let b=typeof J?.code==="string"&&J.code.length>0?J.code:"request_failed",K=typeof J?.message==="string"&&J.message.length>0?J.message:"Cart request failed";throw new E(b,K,P.status,r(J),typeof J?.available==="boolean"?J.available:void 0,typeof J?.sold_out==="boolean"?J.sold_out:void 0,typeof J?.quantity_exceeded==="boolean"?J.quantity_exceeded:void 0)}if(!J||typeof J!=="object")throw new E("request_failed","Cart response was invalid.",P.status);return J}commit(){if(this.items=w(this.items),this.persist&&this.storage)try{let B={version:W,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 i(B){return M.init(B)}
package/dist/esm/index.js CHANGED
@@ -1 +1 @@
1
- var Q="https://api.bts.it.com/v2/website/cart",q="bts-cart",_="default",E=2;function O(){try{return globalThis.window??null}catch{return null}}function H(){try{return globalThis.localStorage??O()?.localStorage??null}catch{return null}}var x=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],L=["fbclid","gclid","gbraid","wbraid","ttclid","li_fat_id","msclkid","_fbp","_fbc"];function I(B){if(!B)return{utm:{},clickIds:{}};try{let F=JSON.parse(B);if(!F||typeof F!=="object"||Array.isArray(F))return{utm:{},clickIds:{}};let X=F,G=X.utm&&typeof X.utm==="object"&&!Array.isArray(X.utm)?Object.fromEntries(Object.entries(X.utm).filter((Z)=>typeof Z[1]==="string"&&Z[1].length>0)):{},N=X.clickIds&&typeof X.clickIds==="object"&&!Array.isArray(X.clickIds)?Object.fromEntries(Object.entries(X.clickIds).filter((Z)=>typeof Z[1]==="string"&&Z[1].length>0)):{};return{utm:G,clickIds:N}}catch{return{utm:{},clickIds:{}}}}function R(B,F){let X=new URL(B);X.searchParams.set("bts_site",F);let G=H(),N=G?.getItem("bts_analytics_jt")?.trim();if(N)X.searchParams.set("bts_jt",N);let{utm:Z,clickIds:$}=I(G?.getItem("bts_analytics_attr")??null);for(let V of x){let Y=Z[V];if(Y)X.searchParams.set(V,Y)}for(let V of L){let Y=$[V];if(Y)X.searchParams.set(V,Y)}return X.toString()}class P extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,X,G,N,Z,$){super(F);this.name="BTSCartError",this.code=B,this.status=X,this.details=G,this.available=N,this.sold_out=Z,this.quantity_exceeded=$}}function g(){return H()??void 0}function p(B){return(B??Q).replace(/\/$/,"")}function S(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 z(B){let F=B?.trim();return F&&F.length>0?F:void 0}function w(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 v(B){let F=new Set;for(let X of B)for(let G of[X.lineItemId,X.accessTierId,X.productId])if(G)F.add(G);return B.map((X)=>{if(!X.discount)return X;let G=h(X);if(G.length===0||!G.some((N)=>F.has(N))){let N={...X};return delete N.discount,delete N.discountEligibleTierOrProductIds,N}return X})}function y(B,F){let X=new Headers(B);if(!F)return X;return new Headers(F).forEach((G,N)=>X.set(N,G)),X}function M(B){let F=B.checkoutLinkId.trim();try{let X=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,G=new URL(X),N=G.pathname.split("/").filter(Boolean),Z=N.indexOf("c"),$=Z>=0?N[Z+1]:N.at(-1);if($)return{checkoutLinkId:$,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 X=F.flatMap((G)=>{if(!G||typeof G!=="object")return[];let N=typeof G.lineItemId==="string"?G.lineItemId:"",Z=typeof G.reason==="string"?G.reason:"";if(!N||!Z)return[];let $=typeof G.discountCode==="string"?G.discountCode:void 0,V=typeof G.requestedQuantity==="number"?G.requestedQuantity:void 0,Y=typeof G.adjustedQuantity==="number"?G.adjustedQuantity:void 0;if(!$&&!(V!==void 0||Y!==void 0||Z==="line_item_sold_out"||Z==="quantity_exceeded"||Z==="product_unavailable"))return[];return[{lineItemId:N,...typeof G.productId==="string"?{productId:G.productId}:{},...typeof G.accessTierId==="string"?{accessTierId:G.accessTierId}:{},...$?{discountCode:$}:{},...typeof G.discountId==="string"?{discountId:G.discountId}:{},reason:Z,...V!==void 0?{requestedQuantity:V}:{},...Y!==void 0?{adjustedQuantity:Y}:{}}]});return X.length>0?X:void 0}var j=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function k(B){return B.reason==="line_item_sold_out"||B.reason==="product_unavailable"||B.adjustedQuantity===0}function BB(B){return j.has(B)||B==="invalid_quantities"||B==="invalid_discounts"}function C(B){if(B==="invalid_quantities")return"quantity_invalid";if(B==="invalid_discounts")return"discounts_invalid";return B}function W(B){return j.has(B)}function c(B){let F=B.sold_out===!0||B.code==="line_item_sold_out"||B.code==="product_unavailable",X=B.quantity_exceeded===!0||B.code==="quantity_exceeded";return{ok:!1,available:!F,sold_out:F,quantity_exceeded:X}}function u(B){return[...new Set([B.lineItemId,B.accessTierId,B.productId].filter((F)=>Boolean(F)))]}function U(B,F){let X=u(B);return[F.lineItemId,F.accessTierId,F.productId].filter((N)=>Boolean(N)).some((N)=>X.includes(N))}class T{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??q,this.storage=B.storage??g(),this.checkoutLinks=B.checkoutLinks??{},this.defaultCheckoutLinkKey=B.defaultCheckoutLinkKey??Object.keys(this.checkoutLinks)[0]??_,this.requestHeaders=B.requestHeaders,this.rehydrate()}static init(B){return new T(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!==E){this.storage.removeItem(this.storageKey);return}let X=F.items;if(!Array.isArray(X))return;this.items=v(X.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 X=this.items.find((G)=>G.id===F.id);if(X){this.setQuantity(X.id,X.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 X=Math.floor(F);this.items=X<=0?this.items.filter((G)=>G.id!==B):this.items.map((G)=>G.id===B?{...G,quantity:X}:G),this.commit()}increment(B,F=1){let X=this.items.find((G)=>G.id===B);if(X)this.setQuantity(B,X.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let X=this.items.find((G)=>G.id===B);if(X)this.setQuantity(B,X.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(!U(F,{lineItemId:B,reason:"discount"}))return F;if(!F.discount)return F;let X={...F,unitPrice:F.fullPrice??F.unitPrice};return delete X.discount,delete X.discountEligibleTierOrProductIds,X}),this.commit()}async checkIfItemsAvailable(B){let F=[];for(let X of B){let G=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,X.lineItemId,X.discountId);if(G instanceof P)throw new P(G.code,G.message);F.push(G)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,X){let G=this.checkoutLinks[B];if(!G)throw new P("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=M(G),Z={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:G.mode,lineItemId:F,...X?{discountId:X}:{}};try{let $=await this.post("/check-availability",Z);if(typeof $.ok!=="boolean")throw new P("request_failed","Availability check response was invalid.");let V=$;return{...V,available:V.ok===!0&&V.sold_out!==!0,sold_out:V.sold_out===!0,quantity_exceeded:V.quantity_exceeded===!0}}catch($){if($ instanceof P){if($.details?.length)this.reconcileCartAdjustments($.details);if($.code==="line_item_sold_out"||$.code==="product_unavailable"||$.code==="quantity_exceeded")return c($)}throw $}}async isCatalogLineSoldOut(B=this.defaultCheckoutLinkKey,F,X){let G=this.checkoutLinks[B];if(!G)throw new P("checkout_link_not_found",`Unknown checkout link key: ${B}`);let N=M(G),Z=w(F),$={siteKey:this.siteKey,checkoutLinkId:N.checkoutLinkId,mode:G.mode,items:[{lineItemId:Z,quantity:S(X?.quantity),...X?.discount?{discount:X.discount}:{}}]},V=await this.post("/validate-discounts",$);if(V.ok===!0)return!1;return D(V)?.some(k)??!1}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new P("checkout_link_not_found",`Unknown checkout link key: ${B}`);let X=M(F),G=this.itemsForCheckout(B,F),N={siteKey:this.siteKey,checkoutLinkId:X.checkoutLinkId,mode:F.mode,...X.discountCode?{discountCode:X.discountCode}:{},...G.length>0?{items:G}:{}};try{let Z=await this.post("/validate-discounts",N);if(typeof Z.ok!=="boolean")throw new P("request_failed","Cart response was invalid.");if(Z.ok===!1){let $=C(typeof Z.code==="string"&&Z.code.length>0?Z.code:"request_failed"),V=typeof Z.message==="string"&&Z.message.length>0?Z.message:"Unable to validate cart.",Y=D(Z);if(W($))throw this.reconcileCartAdjustments(Y),new P($,V,200,Y);throw new P($,V,200,Y)}return Z}catch(Z){if(Z instanceof P){if(W(Z.code))throw this.reconcileCartAdjustments(Z.details),new P(Z.code,Z.message,Z.status,Z.details);throw new P(Z.code,Z.message,Z.status)}throw new P("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new P("checkout_link_not_found",`Unknown checkout link key: ${B}`);let X=M(F),G=this.itemsForCheckout(B,F),N={siteKey:this.siteKey,checkoutLinkId:X.checkoutLinkId,mode:F.mode,...X.discountCode?{discountCode:X.discountCode}:{},...G.length>0?{items:G}:{}};try{let Z=await this.post("/checkout-url",N);if(Z.ok!==!0||typeof Z.checkoutUrl!=="string"||Z.checkoutUrl.length===0)throw new P("request_failed","Cart response was invalid.");return R(Z.checkoutUrl,this.siteKey)}catch(Z){if(Z instanceof P){if(W(Z.code))throw this.reconcileCartAdjustments(Z.details),new P(Z.code,Z.message,Z.status,Z.details);throw new P(Z.code,Z.message,Z.status)}throw new P("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),X=O();if(X)X.location.assign(F);return F}catch(F){if(F instanceof P){if(W(F.code))throw this.reconcileCartAdjustments(F.details),new P(F.code,F.message,F.status,F.details);throw new P(F.code,F.message,F.status)}throw new P("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let X=this.items.filter((G)=>G.checkoutLinkKey===B).map((G)=>({lineItemId:w(G),quantity:G.quantity,...G.discount?{discount:G.discount}:{}}));if(X.length===0)throw new P("cart_empty","Your cart is empty.");return X}normalizeItem(B){let F=z(B.accessTierId),X=z(B.productId),G=z(B.lineItemId)??F??X;if(!B.id||!G||typeof B.unitPrice!=="number")return null;let N=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item";return{...B,id:B.id,lineItemId:G,...F?{accessTierId:F}:{},...X?{productId:X}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:N,unitPrice:B.unitPrice,quantity:S(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],X=!1;for(let G of B){if(G.discountCode){F=F.map((Z)=>{if(!U(Z,G))return Z;let $={...Z,unitPrice:Z.fullPrice??Z.unitPrice};return delete $.discount,delete $.discountEligibleTierOrProductIds,X=!0,$});continue}if(G.reason==="line_item_sold_out"||G.reason==="product_unavailable"||G.adjustedQuantity===0){let Z=F.length;if(F=F.filter(($)=>!U($,G)),F.length!==Z)X=!0;continue}if(G.reason==="quantity_exceeded"&&typeof G.adjustedQuantity==="number"&&G.adjustedQuantity>0){let Z=G.adjustedQuantity;F=F.map(($)=>{if(!U($,G))return $;return X=!0,{...$,quantity:Z}})}}if(!X)return;this.items=F,this.commit()}async post(B,F){let X=`${this.endpoint}${B}`,G=JSON.stringify(F),N={"Content-Type":"application/json"},Z={body:F,bodyText:G,endpoint:this.endpoint,headers:N,path:B,siteKey:this.siteKey,url:X},$=typeof this.requestHeaders==="function"?await this.requestHeaders(Z):this.requestHeaders,V=y(N,$),Y=await fetch(X,{method:"POST",headers:V,body:G}),J=await Y.json().catch(()=>null);if(!Y.ok){let b=typeof J?.code==="string"&&J.code.length>0?J.code:"request_failed",K=C(b),f=typeof J?.message==="string"&&J.message.length>0?J.message:"Cart request failed";throw new P(K,f,Y.status,D(J),typeof J?.available==="boolean"?J.available:void 0,typeof J?.sold_out==="boolean"?J.sold_out:void 0,typeof J?.quantity_exceeded==="boolean"?J.quantity_exceeded:void 0)}if(!J||typeof J!=="object")throw new P("request_failed","Cart response was invalid.",Y.status);return J}commit(){if(this.items=v(this.items),this.persist&&this.storage)try{let B={version:E,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 FB(B){return T.init(B)}export{w as resolveLineItemIdForCheckout,D as normalizeCartAdjustmentDetails,k as isSoldOutAdjustmentDetail,BB as isCartAdjustmentErrorCode,FB as createBTSCart,u as catalogIdsForCartLine,P as BTSCartError,T as BTSCart};
1
+ var T="https://api.bts.it.com/v2/website/cart",_="bts-cart",A="default",M=2;function U(){try{return globalThis.window??null}catch{return null}}function H(){try{return globalThis.localStorage??U()?.localStorage??null}catch{return null}}var v=["utm_source","utm_medium","utm_campaign","utm_term","utm_content"],j=["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 G=F,N=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:N,clickIds:Z}}catch{return{utm:{},clickIds:{}}}}function Q(B,F){let G=new URL(B);G.searchParams.set("bts_site",F);let N=H(),Z=N?.getItem("bts_analytics_jt")?.trim();if(Z)G.searchParams.set("bts_jt",Z);let{utm:X,clickIds:$}=b(N?.getItem("bts_analytics_attr")??null);for(let V of v){let P=X[V];if(P)G.searchParams.set(V,P)}for(let V of j){let P=$[V];if(P)G.searchParams.set(V,P)}return G.toString()}class E extends Error{code;status;available;sold_out;quantity_exceeded;details;constructor(B,F,G,N,Z,X,$){super(F);this.name="BTSCartError",this.code=B,this.status=G,this.details=N,this.available=Z,this.sold_out=X,this.quantity_exceeded=$}}function K(){return H()??void 0}function f(B){return(B??T).replace(/\/$/,"")}function L(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,...F.discounts?{discounts:F.discounts.map((G)=>({...G}))}:{}}))}}function Y(B){let F=B?.trim();return F&&F.length>0?F:void 0}function x(B){if(!Array.isArray(B)||B.length===0)return;let F=B.flatMap((G)=>{if(!G||typeof G!=="object")return[];let N=Y(G.id),Z=typeof G.label==="string"?G.label.trim():"";if(!N||Z.length===0)return[];return[{...G,id:N,label:Z,...Y(G.token)?{token:Y(G.token)}:{}}]});return F.length>0?F:void 0}function q(B,F){if(!B?.length||!F)return B;let G=B.filter((N)=>N.token!==F);return G.length>0?G:void 0}function g(B){return B.accessTierId??B.productId??B.lineItemId}function I(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 S(B){let F=new Set;for(let G of B)for(let N of[G.lineItemId,G.accessTierId,G.productId])if(N)F.add(N);return B.map((G)=>{if(!G.discount)return G;let N=I(G);if(N.length===0||!N.some((Z)=>F.has(Z))){let Z={...G};return delete Z.discount,delete Z.discountEligibleTierOrProductIds,Z}return G})}function p(B,F){let G=new Headers(B);if(!F)return G;return new Headers(F).forEach((N,Z)=>G.set(Z,N)),G}function W(B){let F=B.checkoutLinkId.trim();try{let G=F.includes("://")?F:F.startsWith("/")?`https://bts-cart.invalid${F}`:F,N=new URL(G),Z=N.pathname.split("/").filter(Boolean),X=Z.indexOf("c"),$=X>=0?Z[X+1]:Z.at(-1);if($)return{checkoutLinkId:$,discountCode:B.discountCode??N.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 G=F.flatMap((N)=>{if(!N||typeof N!=="object")return[];let Z=typeof N.lineItemId==="string"?N.lineItemId:"",X=typeof N.reason==="string"?N.reason:"";if(!Z||!X)return[];let $=typeof N.discountCode==="string"?N.discountCode:void 0,V=typeof N.requestedQuantity==="number"?N.requestedQuantity:void 0,P=typeof N.adjustedQuantity==="number"?N.adjustedQuantity:void 0;if(!$&&!(V!==void 0||P!==void 0||X==="line_item_sold_out"||X==="quantity_exceeded"||X==="product_unavailable"))return[];return[{lineItemId:Z,...typeof N.productId==="string"?{productId:N.productId}:{},...typeof N.accessTierId==="string"?{accessTierId:N.accessTierId}:{},...$?{discountCode:$}:{},...typeof N.discountId==="string"?{discountId:N.discountId}:{},reason:X,...V!==void 0?{requestedQuantity:V}:{},...P!==void 0?{adjustedQuantity:P}:{}}]});return G.length>0?G:void 0}var y=new Set(["discounts_invalid","quantity_invalid","cart_adjusted"]);function z(B){return y.has(B)}function k(B){return[...new Set([B.lineItemId,B.accessTierId,B.productId].filter((F)=>Boolean(F)))]}function O(B,F){let G=k(B);return[F.lineItemId,F.accessTierId,F.productId].filter((Z)=>Boolean(Z)).some((Z)=>G.includes(Z))}class D{siteKey;endpoint;debug;persist;storageKey;storage;checkoutLinks;defaultCheckoutLinkKey;requestHeaders;items=[];listeners=new Set;constructor(B){this.siteKey=B.siteKey,this.endpoint=f(B.endpoint),this.debug=B.debug??!1,this.persist=B.persist??!0,this.storageKey=B.storageKey??_,this.storage=B.storage??K(),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 D(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 G=F.items;if(!Array.isArray(G))return;this.items=S(G.map((N)=>this.normalizeItem(N)).filter((N)=>Boolean(N))),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((N)=>N.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((N)=>N.id!==B):this.items.map((N)=>N.id===B?{...N,quantity:G}:N),this.commit()}updateItem(B,F){let G=this.items.find((Z)=>Z.id===B);if(!G)return;let N=this.normalizeItem({...G,...F,id:B,quantity:F.quantity??G.quantity,checkoutLinkKey:F.checkoutLinkKey??G.checkoutLinkKey});if(!N)return;this.items=this.items.map((Z)=>Z.id===B?N:Z),this.commit()}increment(B,F=1){let G=this.items.find((N)=>N.id===B);if(G)this.setQuantity(B,G.quantity+Math.max(1,Math.floor(F)))}decrement(B,F=1){let G=this.items.find((N)=>N.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(!O(F,{lineItemId:B,reason:"discount"}))return F;if(!F.discount)return F;let G={...F,unitPrice:F.fullPrice??F.unitPrice};return G.discounts=q(F.discounts,F.discount),delete G.discount,delete G.discountEligibleTierOrProductIds,G}),this.commit()}async checkIfItemsAvailable(B){let F=[];for(let G of B){let N=await this.checkIfItemAvailable(this.defaultCheckoutLinkKey,G.lineItemId,G.discountId);if(N instanceof E)throw new E(N.code,N.message);F.push(N)}return F}async checkIfItemAvailable(B=this.defaultCheckoutLinkKey,F,G){let N=this.checkoutLinks[B];if(!N)throw new E("checkout_link_not_found",`Unknown checkout link key: ${B}`);let Z=W(N),X={siteKey:this.siteKey,checkoutLinkId:Z.checkoutLinkId,mode:N.mode,lineItemId:F,...G?{discountId:G}:{}};try{let $=await this.post("/check-availability",X);if(typeof $.ok!=="boolean")throw new E("request_failed","Availability check response was invalid.");return $}catch($){if($ instanceof E){if($.details?.length)this.reconcileCartAdjustments($.details);if($.code==="line_item_sold_out"||$.code==="product_unavailable"||$.code==="quantity_exceeded")return{ok:!1,available:$.available??!1,sold_out:$.sold_out??$.code==="line_item_sold_out",quantity_exceeded:$.quantity_exceeded??($.code==="line_item_sold_out"||$.code==="quantity_exceeded")}}throw $}}async validateCheckout(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new E("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=W(F),N=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...N.length>0?{items:N}:{}};try{let X=await this.post("/validate-discounts",Z);if(typeof X.ok!=="boolean")throw new E("request_failed","Cart response was invalid.");return X}catch(X){if(X instanceof E){if(z(X.code))throw this.reconcileCartAdjustments(X.details),new E(X.code,X.message,X.status,X.details);throw new E(X.code,X.message,X.status)}throw new E("request_failed","Failed to validate checkout.",void 0,void 0)}}async resolveCheckoutUrl(B=this.defaultCheckoutLinkKey){let F=this.checkoutLinks[B];if(!F)throw new E("checkout_link_not_found",`Unknown checkout link key: ${B}`);let G=W(F),N=this.itemsForCheckout(B,F),Z={siteKey:this.siteKey,checkoutLinkId:G.checkoutLinkId,mode:F.mode,...G.discountCode?{discountCode:G.discountCode}:{},...N.length>0?{items:N}:{}};try{let X=await this.post("/checkout-url",Z);if(X.ok!==!0||typeof X.checkoutUrl!=="string"||X.checkoutUrl.length===0)throw new E("request_failed","Cart response was invalid.");return Q(X.checkoutUrl,this.siteKey)}catch(X){if(X instanceof E){if(z(X.code))throw this.reconcileCartAdjustments(X.details),new E(X.code,X.message,X.status,X.details);throw new E(X.code,X.message,X.status)}throw new E("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}async checkout(B=this.defaultCheckoutLinkKey){try{let F=await this.resolveCheckoutUrl(B),G=U();if(G)G.location.assign(F);return F}catch(F){if(F instanceof E){if(z(F.code))throw this.reconcileCartAdjustments(F.details),new E(F.code,F.message,F.status,F.details);throw new E(F.code,F.message,F.status)}throw new E("request_failed","Failed to resolve checkout URL.",void 0,void 0)}}itemsForCheckout(B,F){if(F.mode==="static")return[];let G=this.items.filter((N)=>N.checkoutLinkKey===B).map((N)=>({lineItemId:g(N),quantity:N.quantity,...N.discount?{discount:N.discount}:{}}));if(G.length===0)throw new E("cart_empty","Your cart is empty.");return G}normalizeItem(B){let F=Y(B.accessTierId),G=Y(B.productId),N=Y(B.lineItemId)??F??G;if(!B.id||!N||typeof B.unitPrice!=="number")return null;let Z=typeof B.name==="string"&&B.name.trim().length>0?B.name.trim():"Item",X=x(B.discounts);return{...B,id:B.id,lineItemId:N,...F?{accessTierId:F}:{},...G?{productId:G}:{},checkoutLinkKey:B.checkoutLinkKey??this.defaultCheckoutLinkKey,name:Z,unitPrice:B.unitPrice,...X?{discounts:X}:{},quantity:L(B.quantity)}}reconcileCartAdjustments(B){if(!B?.length)return;let F=[...this.items],G=!1;for(let N of B){if(N.discountCode){F=F.map((X)=>{if(!O(X,N))return X;let $={...X,unitPrice:X.fullPrice??X.unitPrice};return $.discounts=q(X.discounts,X.discount),delete $.discount,delete $.discountEligibleTierOrProductIds,G=!0,$});continue}if(N.reason==="line_item_sold_out"||N.reason==="product_unavailable"||N.adjustedQuantity===0){let X=F.length;if(F=F.filter(($)=>!O($,N)),F.length!==X)G=!0;continue}if(N.reason==="quantity_exceeded"&&typeof N.adjustedQuantity==="number"&&N.adjustedQuantity>0){let X=N.adjustedQuantity;F=F.map(($)=>{if(!O($,N))return $;return G=!0,{...$,quantity:X}})}}if(!G)return;this.items=F,this.commit()}async post(B,F){let G=`${this.endpoint}${B}`,N=JSON.stringify(F),Z={"Content-Type":"application/json"},X={body:F,bodyText:N,endpoint:this.endpoint,headers:Z,path:B,siteKey:this.siteKey,url:G},$=typeof this.requestHeaders==="function"?await this.requestHeaders(X):this.requestHeaders,V=p(Z,$),P=await fetch(G,{method:"POST",headers:V,body:N}),J=await P.json().catch(()=>null);if(!P.ok){let C=typeof J?.code==="string"&&J.code.length>0?J.code:"request_failed",w=typeof J?.message==="string"&&J.message.length>0?J.message:"Cart request failed";throw new E(C,w,P.status,h(J),typeof J?.available==="boolean"?J.available:void 0,typeof J?.sold_out==="boolean"?J.sold_out:void 0,typeof J?.quantity_exceeded==="boolean"?J.quantity_exceeded:void 0)}if(!J||typeof J!=="object")throw new E("request_failed","Cart response was invalid.",P.status);return J}commit(){if(this.items=S(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 t(B){return D.init(B)}export{g as resolveLineItemIdForCheckout,t as createBTSCart,k as catalogIdsForCartLine,E as BTSCartError,D as BTSCart};
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@behindthescenes/cart",
3
- "version": "0.0.31",
4
- "generatedAt": "2026-05-22T08:39:12.291Z",
3
+ "version": "0.0.34",
4
+ "generatedAt": "2026-06-02T00:25:30.569Z",
5
5
  "artifacts": {
6
6
  "browser": "browser/browser.js",
7
7
  "cjs": "cjs/index.js",
@@ -1,16 +1,10 @@
1
- import type { BTSCartAvailabilityCheckResponse, BTSCartErrorCode, BTSCartInit, BTSCartItem, BTSCartItemInput, BTSCartListener, BTSCartState } from "./types/index.types";
1
+ import type { BTSCartAvailabilityCheckResponse, BTSCartInit, BTSCartItem, BTSCartItemInput, BTSCartListener, BTSCartState } from "./types/index.types";
2
2
  import { BTSCartError } from "./types/index.types";
3
3
  /**
4
4
  * Prefer catalog UUIDs on checkout-url — matches `resolveDynamicCheckoutCartLineItem` (tier/product
5
5
  * id, then persisted line-item pk).
6
6
  */
7
7
  export declare function resolveLineItemIdForCheckout(item: Pick<BTSCartItem, "lineItemId" | "productId" | "accessTierId">): string;
8
- export type CartAdjustmentDetail = NonNullable<BTSCartError["details"]>[number];
9
- /** Parse `details` from a cart API error body (validate-discounts, checkout-url, check-availability). */
10
- export declare function normalizeCartAdjustmentDetails(json: Record<string, unknown> | null): BTSCartError["details"];
11
- /** Whether a checkout adjustment detail means the line cannot be purchased (qty 0 / unavailable). */
12
- export declare function isSoldOutAdjustmentDetail(detail: CartAdjustmentDetail): boolean;
13
- export declare function isCartAdjustmentErrorCode(code: string): code is BTSCartErrorCode;
14
8
  /** All catalog identifiers for a line — matches API adjustment details and `resolveDynamicCheckoutCartLineItem`. */
15
9
  export declare function catalogIdsForCartLine(item: Pick<BTSCartItem, "lineItemId" | "productId" | "accessTierId">): string[];
16
10
  export declare class BTSCart {
@@ -51,6 +45,8 @@ export declare class BTSCart {
51
45
  removeItem(id: string): void;
52
46
  /** Replace a line quantity, dropping the line when the requested quantity is zero or below. */
53
47
  setQuantity(id: string, quantity: number): void;
48
+ /** Update an existing line without changing its identity, preserving quantity when omitted. */
49
+ updateItem(id: string, patch: Partial<Omit<BTSCartItemInput, "id">>): void;
54
50
  /** Increase a line quantity using a clamped positive integer increment. */
55
51
  increment(id: string, amount?: number): void;
56
52
  /** Decrease a line quantity using a clamped positive integer decrement. */
@@ -69,14 +65,6 @@ export declare class BTSCart {
69
65
  sold_out: boolean;
70
66
  quantity_exceeded: boolean;
71
67
  }>;
72
- /**
73
- * Uses `POST /validate-discounts` (same rules as checkout). Prefer this over
74
- * `checkIfItemAvailable` when `/check-availability` is not deployed.
75
- */
76
- isCatalogLineSoldOut(checkoutLinkKey: string | undefined, catalog: Pick<BTSCartItem, "lineItemId" | "productId" | "accessTierId">, options?: {
77
- discount?: string;
78
- quantity?: number;
79
- }): Promise<boolean>;
80
68
  validateCheckout(checkoutLinkKey?: string): Promise<{
81
69
  ok: boolean;
82
70
  errors?: BTSCartError["details"];
@@ -102,5 +90,5 @@ export declare class BTSCart {
102
90
  }
103
91
  /** Public package factory mirroring the class-based `BTSCart.init` constructor. */
104
92
  export declare function createBTSCart(init: BTSCartInit): BTSCart;
105
- export type { BTSCartCheckoutItemPayload, BTSCartCheckoutLinkConfig, BTSCartCheckoutUrlRequest, BTSCartCheckoutUrlResult, BTSCartErrorCode, BTSCartInit, BTSCartItem, BTSCartItemInput, BTSCartListener, BTSCartMode, BTSCartRequestContext, BTSCartRequestHeaders, BTSCartState, BTSCartStorage, } from "./types/index.types";
93
+ export type { BTSCartAppliedDiscount, BTSCartCheckoutItemPayload, BTSCartCheckoutLinkConfig, BTSCartCheckoutUrlRequest, BTSCartCheckoutUrlResult, BTSCartDiscountKind, BTSCartErrorCode, BTSCartInit, BTSCartItem, BTSCartItemInput, BTSCartListener, BTSCartMode, BTSCartRequestContext, BTSCartRequestHeaders, BTSCartState, BTSCartStorage, } from "./types/index.types";
106
94
  export { BTSCartError } from "./types/index.types";
@@ -16,6 +16,23 @@ export type BTSCartCheckoutLinkConfig = {
16
16
  mode: BTSCartMode;
17
17
  discountCode?: string;
18
18
  };
19
+ export type BTSCartDiscountKind = "amount-off-products" | "amount-off-order" | "buy-x-get-y" | "custom";
20
+ /**
21
+ * Optional discount metadata for embedders that need to render richer cart summaries.
22
+ * These descriptors are persisted in the SDK state but never sent to the BTS cart API.
23
+ */
24
+ export type BTSCartAppliedDiscount = {
25
+ id: string;
26
+ label: string;
27
+ kind?: BTSCartDiscountKind;
28
+ /** Discount token/code backing this descriptor, when it maps to a checkout discount. */
29
+ token?: string;
30
+ /** Display-only discount amount in the embedder's major currency units. */
31
+ amount?: number;
32
+ quantity?: number;
33
+ automatic?: boolean;
34
+ stackable?: boolean;
35
+ };
19
36
  export type BTSCartItemInput = {
20
37
  id: string;
21
38
  /**
@@ -39,6 +56,8 @@ export type BTSCartItemInput = {
39
56
  quantity?: number;
40
57
  /** Internal promo-code ID or customer-facing promo code. Customer codes are resolved by the BTS cart API. */
41
58
  discount?: string;
59
+ /** Optional display metadata for one or more active discounts on the line item. */
60
+ discounts?: BTSCartAppliedDiscount[];
42
61
  /**
43
62
  * Optional catalog identifiers for embedders documenting promo scope. The SDK persists them but
44
63
  * does not clear discounts client-side; the cart API validates promos on checkout-url.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@behindthescenes/cart",
3
- "version": "0.0.31",
3
+ "version": "0.0.34",
4
4
  "description": "Browser cart SDK for BTS external-site checkout links.",
5
5
  "license": "UNLICENSED",
6
6
  "type": "module",